From 605c99ba7074c3dbfeb441f608d77a2bc50e4629 Mon Sep 17 00:00:00 2001 From: Jesse Plamondon-Willard Date: Mon, 31 Oct 2016 20:46:23 -0400 Subject: format & document config code --- src/StardewModdingAPI/Config.cs | 134 ++++++++++++++++---------------------- src/StardewModdingAPI/Manifest.cs | 63 +++++++++--------- src/StardewModdingAPI/Program.cs | 30 +++++---- 3 files changed, 104 insertions(+), 123 deletions(-) (limited to 'src/StardewModdingAPI') diff --git a/src/StardewModdingAPI/Config.cs b/src/StardewModdingAPI/Config.cs index 7e5c4046..ff8ade90 100644 --- a/src/StardewModdingAPI/Config.cs +++ b/src/StardewModdingAPI/Config.cs @@ -1,8 +1,4 @@ -/* - Copyright 2016 Zoey (Zoryn) -*/ - -using System; +using System; using System.IO; using System.Linq; using Newtonsoft.Json; @@ -10,119 +6,107 @@ using Newtonsoft.Json.Linq; namespace StardewModdingAPI { + /// A dynamic configuration class for a mod. public abstract class Config { + /********* + ** Accessors + *********/ + /// The full path to the configuration file. [JsonIgnore] public virtual string ConfigLocation { get; protected internal set; } + /// The directory path containing the configuration file. [JsonIgnore] - public virtual string ConfigDir => Path.GetDirectoryName(ConfigLocation); + public virtual string ConfigDir => Path.GetDirectoryName(this.ConfigLocation); + + /********* + ** Public methods + *********/ + /// Construct an instance of the config class. + /// The config class type. public virtual Config Instance() where T : Config => Activator.CreateInstance(); - /// - /// Loads the config from the json blob on disk, updating and re-writing to the disk if needed. - /// - /// - /// + /// Load the config from the JSON file, saving it to disk if needed. + /// The config class type. public virtual T LoadConfig() where T : Config { - if (string.IsNullOrEmpty(ConfigLocation)) + // validate + if (string.IsNullOrEmpty(this.ConfigLocation)) { Log.AsyncR("A config tried to load without specifying a location on the disk."); return null; } - T ret = null; - - if (!File.Exists(ConfigLocation)) + // read or generate config + T returnValue; + if (!File.Exists(this.ConfigLocation)) { - //no config exists, generate default values - var c = GenerateDefaultConfig(); - c.ConfigLocation = ConfigLocation; - ret = c; + T config = this.GenerateDefaultConfig(); + config.ConfigLocation = this.ConfigLocation; + returnValue = config; } else { try { - //try to load the config from a json blob on disk T config = JsonConvert.DeserializeObject(File.ReadAllText(this.ConfigLocation)); config.ConfigLocation = this.ConfigLocation; - - //update the config with default values if needed - ret = config.UpdateConfig(); + returnValue = config.UpdateConfig(); } catch (Exception ex) { - Log.AsyncR($"Invalid JSON ({GetType().Name}): {ConfigLocation} \n{ex}"); - return GenerateDefaultConfig(); + Log.AsyncR($"Invalid JSON ({this.GetType().Name}): {this.ConfigLocation} \n{ex}"); + return this.GenerateDefaultConfig(); } } - ret.WriteConfig(); - return ret; + returnValue.WriteConfig(); + return returnValue; } /// Get the default config values. public abstract T GenerateDefaultConfig() where T : Config; - /// - /// Merges a default-value config with the user-config on disk. - /// - /// - /// + /// Get the current configuration with missing values defaulted. + /// The config class type. public virtual T UpdateConfig() where T : Config { try { - //default config - var b = JObject.FromObject(this.Instance().GenerateDefaultConfig()); - - //user config - var u = JObject.FromObject(this); - - //overwrite default values with user values - b.Merge(u, new JsonMergeSettings {MergeArrayHandling = MergeArrayHandling.Replace}); + // get default + user config + JObject defaultConfig = JObject.FromObject(this.Instance().GenerateDefaultConfig()); + JObject currentConfig = JObject.FromObject(this); + defaultConfig.Merge(currentConfig, new JsonMergeSettings { MergeArrayHandling = MergeArrayHandling.Replace }); - //cast json object to config - var c = b.ToObject(); + // cast json object to config + T config = defaultConfig.ToObject(); - //re-write the location on disk to the object - c.ConfigLocation = this.ConfigLocation; + // update location + config.ConfigLocation = this.ConfigLocation; - return c; + return config; } catch (Exception ex) { - Log.AsyncR("An error occured when updating a config: " + ex); + Log.AsyncR($"An error occured when updating a config: {ex}"); return this as T; } } } + /// Provides extension methods for classes. public static class ConfigExtensions { - /// - /// Initializes an instance of any class that inherits from Config. - /// This method performs the loading, saving, and merging of the config on the disk and in memory at a default state. - /// This method should not be used to re-load or to re-save a config. - /// NOTE: You MUST set your config EQUAL to the return of this method! - /// - /// - /// - /// - /// + /// Initialise the configuration. That includes loading, saving, and merging the config file and in memory at a default state. This method should not be used to reload or to resave a config. NOTE: You MUST set your config EQUAL to the return of this method! + /// The config class type. + /// The base configuration to initialise. + /// The base configuration file path. public static T InitializeConfig(this T baseConfig, string configLocation) where T : Config { if (baseConfig == null) - { baseConfig = Activator.CreateInstance(); - /* - Log.AsyncR("A config tried to initialize whilst being null."); - return null; - */ - } if (string.IsNullOrEmpty(configLocation)) { @@ -131,14 +115,12 @@ namespace StardewModdingAPI } baseConfig.ConfigLocation = configLocation; - var c = baseConfig.LoadConfig(); - - return c; + return baseConfig.LoadConfig(); } - /// - /// Writes a config to a json blob on the disk specified in the config's properties. - /// + /// Writes the configuration to the JSON file. + /// The config class type. + /// The base configuration to initialise. public static void WriteConfig(this T baseConfig) where T : Config { if (string.IsNullOrEmpty(baseConfig?.ConfigLocation) || string.IsNullOrEmpty(baseConfig.ConfigDir)) @@ -147,22 +129,20 @@ namespace StardewModdingAPI return; } - var s = JsonConvert.SerializeObject(baseConfig, Formatting.Indented); - + string json = JsonConvert.SerializeObject(baseConfig, Formatting.Indented); if (!Directory.Exists(baseConfig.ConfigDir)) Directory.CreateDirectory(baseConfig.ConfigDir); - if (!File.Exists(baseConfig.ConfigLocation) || !File.ReadAllText(baseConfig.ConfigLocation).SequenceEqual(s)) - File.WriteAllText(baseConfig.ConfigLocation, s); + if (!File.Exists(baseConfig.ConfigLocation) || !File.ReadAllText(baseConfig.ConfigLocation).SequenceEqual(json)) + File.WriteAllText(baseConfig.ConfigLocation, json); } - /// - /// Re-reads the json blob on the disk and merges its values with a default config. - /// NOTE: You MUST set your config EQUAL to the return of this method! - /// + /// Rereads the JSON file and merges its values with a default config. NOTE: You MUST set your config EQUAL to the return of this method! + /// The config class type. + /// The base configuration to initialise. public static T ReloadConfig(this T baseConfig) where T : Config { return baseConfig.LoadConfig(); } } -} \ No newline at end of file +} diff --git a/src/StardewModdingAPI/Manifest.cs b/src/StardewModdingAPI/Manifest.cs index 5eabc01b..ed43ab7d 100644 --- a/src/StardewModdingAPI/Manifest.cs +++ b/src/StardewModdingAPI/Manifest.cs @@ -6,74 +6,71 @@ using Newtonsoft.Json.Linq; namespace StardewModdingAPI { + /// A manifest which describes a mod for SMAPI. public class Manifest : Config { - /// - /// The name of your mod. - /// + /********* + ** Accessors + *********/ + /// The mod name. public virtual string Name { get; set; } - /// - /// The name of the mod's authour. - /// + /// The mod author's name. public virtual string Authour { get; set; } - /// - /// The version of the mod. - /// + /// The mod version. public virtual Version Version { get; set; } - /// - /// A description of the mod. - /// + /// A brief description of the mod. public virtual string Description { get; set; } - /// - /// The unique ID of the mod. It doesn't *need* to be anything. - /// + /// The unique mod ID. public virtual string UniqueID { get; set; } - /// - /// Whether or not the mod uses per-save-config files. - /// + /// Whether the mod uses per-save config files. public virtual bool PerSaveConfigs { get; set; } - /// - /// The name of the DLL in the directory that has the Entry() method. - /// + /// The name of the DLL in the directory that has the method. public virtual string EntryDll { get; set; } + + /********* + ** Public methods + *********/ + /// Get the default config values. public override T GenerateDefaultConfig() { - Name = ""; - Authour = ""; - Version = new Version(0, 0, 0, ""); - Description = ""; - UniqueID = Guid.NewGuid().ToString(); - PerSaveConfigs = false; - EntryDll = ""; + this.Name = ""; + this.Authour = ""; + this.Version = new Version(0, 0, 0, ""); + this.Description = ""; + this.UniqueID = Guid.NewGuid().ToString(); + this.PerSaveConfigs = false; + this.EntryDll = ""; return this as T; } + /// Load the config from the JSON file, saving it to disk if needed. + /// The config class type. public override T LoadConfig() { - if (File.Exists(ConfigLocation)) + if (File.Exists(this.ConfigLocation)) { try { - Manifest m = JsonConvert.DeserializeObject(File.ReadAllText(ConfigLocation)); + Manifest m = JsonConvert.DeserializeObject(File.ReadAllText(this.ConfigLocation)); } catch { //Invalid json blob. Try to remove version? try { - JObject j = JObject.Parse(File.ReadAllText(ConfigLocation)); + JObject j = JObject.Parse(File.ReadAllText(this.ConfigLocation)); if (!j.GetValue("Version").Contains("{")) { Log.AsyncC("INVALID JSON VERSION. TRYING TO REMOVE SO A NEW CAN BE AUTO-GENERATED"); j.Remove("Version"); - File.WriteAllText(ConfigLocation, j.ToString()); + File.WriteAllText(this.ConfigLocation, j.ToString()); } } catch (Exception) @@ -86,4 +83,4 @@ namespace StardewModdingAPI return base.LoadConfig(); } } -} \ No newline at end of file +} diff --git a/src/StardewModdingAPI/Program.cs b/src/StardewModdingAPI/Program.cs index 0099b045..b08d17c2 100644 --- a/src/StardewModdingAPI/Program.cs +++ b/src/StardewModdingAPI/Program.cs @@ -38,7 +38,7 @@ namespace StardewModdingAPI public static Texture2D DebugPixel { get; private set; } // ReSharper disable once PossibleNullReferenceException - public static int BuildType => (int) StardewProgramType.GetField("buildType", BindingFlags.Public | BindingFlags.Static).GetValue(null); + public static int BuildType => (int)StardewProgramType.GetField("buildType", BindingFlags.Public | BindingFlags.Static).GetValue(null); ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// @@ -215,35 +215,39 @@ namespace StardewModdingAPI Log.AsyncY("LOADING MODS"); foreach (string directory in Directory.GetDirectories(ModPath)) { - foreach (string manifestFile in Directory.GetFiles(directory, "manifest.json")) + foreach (string manifestPath in Directory.GetFiles(directory, "manifest.json")) { - if (manifestFile.Contains("StardewInjector")) + if (manifestPath.Contains("StardewInjector")) continue; - Log.AsyncG("Found Manifest: " + manifestFile); - var manifest = new Manifest(); + + // read manifest + Log.AsyncG($"Found Manifest: {manifestPath}"); + Manifest manifest = new Manifest(); try { - string t = File.ReadAllText(manifestFile); - if (string.IsNullOrEmpty(t)) + // read manifest text + string json = File.ReadAllText(manifestPath); + if (string.IsNullOrEmpty(json)) { - Log.AsyncR($"Failed to read mod manifest '{manifestFile}'. Manifest is empty!"); + Log.AsyncR($"Failed to read mod manifest '{manifestPath}'. Manifest is empty!"); continue; } - manifest = manifest.InitializeConfig(manifestFile); - + // deserialise manifest + manifest = manifest.InitializeConfig(manifestPath); if (string.IsNullOrEmpty(manifest.EntryDll)) { - Log.AsyncR($"Failed to read mod manifest '{manifestFile}'. EntryDll is empty!"); + Log.AsyncR($"Failed to read mod manifest '{manifestPath}'. EntryDll is empty!"); continue; } } catch (Exception ex) { - Log.AsyncR($"Failed to read mod manifest '{manifestFile}'. Exception details:\n" + ex); + Log.AsyncR($"Failed to read mod manifest '{manifestPath}'. Exception details:\n" + ex); continue; } - string targDir = Path.GetDirectoryName(manifestFile); + + string targDir = Path.GetDirectoryName(manifestPath); string psDir = Path.Combine(targDir, "psconfigs"); Log.AsyncY($"Created psconfigs directory @{psDir}"); try -- cgit