diff options
Diffstat (limited to 'src')
| -rw-r--r-- | src/StardewModdingAPI.Tests/Core/ModResolverTests.cs | 14 | ||||
| -rw-r--r-- | src/StardewModdingAPI.Web/Controllers/ModsController.cs | 6 | ||||
| -rw-r--r-- | src/StardewModdingAPI/Constants.cs | 9 | ||||
| -rw-r--r-- | src/StardewModdingAPI/Framework/ModLoading/ModResolver.cs | 31 | ||||
| -rw-r--r-- | src/StardewModdingAPI/Framework/Models/Manifest.cs | 10 | ||||
| -rw-r--r-- | src/StardewModdingAPI/Framework/Models/ModDataDefaults.cs | 18 | ||||
| -rw-r--r-- | src/StardewModdingAPI/Framework/Models/ModDataRecord.cs | 4 | ||||
| -rw-r--r-- | src/StardewModdingAPI/IManifest.cs | 10 | ||||
| -rw-r--r-- | src/StardewModdingAPI/Program.cs | 23 | ||||
| -rw-r--r-- | src/StardewModdingAPI/StardewModdingAPI.config.json | 494 | ||||
| -rw-r--r-- | src/StardewModdingAPI/StardewModdingAPI.csproj | 1 |
11 files changed, 294 insertions, 326 deletions
diff --git a/src/StardewModdingAPI.Tests/Core/ModResolverTests.cs b/src/StardewModdingAPI.Tests/Core/ModResolverTests.cs index 6a3fded6..051ffe99 100644 --- a/src/StardewModdingAPI.Tests/Core/ModResolverTests.cs +++ b/src/StardewModdingAPI.Tests/Core/ModResolverTests.cs @@ -119,7 +119,7 @@ namespace StardewModdingAPI.Tests.Core [Test(Description = "Assert that validation doesn't fail if there are no mods installed.")] public void ValidateManifests_NoMods_DoesNothing() { - new ModResolver().ValidateManifests(new ModMetadata[0], apiVersion: new SemanticVersion("1.0")); + new ModResolver().ValidateManifests(new ModMetadata[0], apiVersion: new SemanticVersion("1.0"), vendorModUrls: new Dictionary<string, string>()); } [Test(Description = "Assert that validation skips manifests that have already failed without calling any other properties.")] @@ -130,7 +130,7 @@ namespace StardewModdingAPI.Tests.Core mock.Setup(p => p.Status).Returns(ModMetadataStatus.Failed); // act - new ModResolver().ValidateManifests(new[] { mock.Object }, apiVersion: new SemanticVersion("1.0")); + new ModResolver().ValidateManifests(new[] { mock.Object }, apiVersion: new SemanticVersion("1.0"), vendorModUrls: new Dictionary<string, string>()); // assert mock.VerifyGet(p => p.Status, Times.Once, "The validation did not check the manifest status."); @@ -148,7 +148,7 @@ namespace StardewModdingAPI.Tests.Core }); // act - new ModResolver().ValidateManifests(new[] { mock.Object }, apiVersion: new SemanticVersion("1.0")); + new ModResolver().ValidateManifests(new[] { mock.Object }, apiVersion: new SemanticVersion("1.0"), vendorModUrls: new Dictionary<string, string>()); // assert mock.Verify(p => p.SetStatus(ModMetadataStatus.Failed, It.IsAny<string>()), Times.Once, "The validation did not fail the metadata."); @@ -163,7 +163,7 @@ namespace StardewModdingAPI.Tests.Core this.SetupMetadataForValidation(mock); // act - new ModResolver().ValidateManifests(new[] { mock.Object }, apiVersion: new SemanticVersion("1.0")); + new ModResolver().ValidateManifests(new[] { mock.Object }, apiVersion: new SemanticVersion("1.0"), vendorModUrls: new Dictionary<string, string>()); // assert mock.Verify(p => p.SetStatus(ModMetadataStatus.Failed, It.IsAny<string>()), Times.Once, "The validation did not fail the metadata."); @@ -177,7 +177,7 @@ namespace StardewModdingAPI.Tests.Core this.SetupMetadataForValidation(mock); // act - new ModResolver().ValidateManifests(new[] { mock.Object }, apiVersion: new SemanticVersion("1.0")); + new ModResolver().ValidateManifests(new[] { mock.Object }, apiVersion: new SemanticVersion("1.0"), vendorModUrls: new Dictionary<string, string>()); // assert mock.Verify(p => p.SetStatus(ModMetadataStatus.Failed, It.IsAny<string>()), Times.Once, "The validation did not fail the metadata."); @@ -194,7 +194,7 @@ namespace StardewModdingAPI.Tests.Core this.SetupMetadataForValidation(mod); // act - new ModResolver().ValidateManifests(new[] { modA.Object, modB.Object }, apiVersion: new SemanticVersion("1.0")); + new ModResolver().ValidateManifests(new[] { modA.Object, modB.Object }, apiVersion: new SemanticVersion("1.0"), vendorModUrls: new Dictionary<string, string>()); // assert modA.Verify(p => p.SetStatus(ModMetadataStatus.Failed, It.IsAny<string>()), Times.Once, "The validation did not fail the first mod with a unique ID."); @@ -220,7 +220,7 @@ namespace StardewModdingAPI.Tests.Core mock.Setup(p => p.DirectoryPath).Returns(modFolder); // act - new ModResolver().ValidateManifests(new[] { mock.Object }, apiVersion: new SemanticVersion("1.0")); + new ModResolver().ValidateManifests(new[] { mock.Object }, apiVersion: new SemanticVersion("1.0"), vendorModUrls: new Dictionary<string, string>()); // assert // if Moq doesn't throw a method-not-setup exception, the validation didn't override the status. diff --git a/src/StardewModdingAPI.Web/Controllers/ModsController.cs b/src/StardewModdingAPI.Web/Controllers/ModsController.cs index f29de45a..26cdfa31 100644 --- a/src/StardewModdingAPI.Web/Controllers/ModsController.cs +++ b/src/StardewModdingAPI.Web/Controllers/ModsController.cs @@ -81,7 +81,7 @@ namespace StardewModdingAPI.Web.Controllers [HttpGet] public async Task<IDictionary<string, ModInfoModel>> GetAsync(string modKeys) { - string[] modKeysArray = modKeys?.Split(',').Select(p => p.Trim()).ToArray(); + string[] modKeysArray = modKeys?.Split(',').ToArray(); if (modKeysArray == null || !modKeysArray.Any()) return new Dictionary<string, ModInfoModel>(); @@ -154,8 +154,8 @@ namespace StardewModdingAPI.Web.Controllers } // parse - vendorKey = parts[0]; - modID = parts[1]; + vendorKey = parts[0].Trim(); + modID = parts[1].Trim(); return true; } } diff --git a/src/StardewModdingAPI/Constants.cs b/src/StardewModdingAPI/Constants.cs index fc780f40..5b1d7737 100644 --- a/src/StardewModdingAPI/Constants.cs +++ b/src/StardewModdingAPI/Constants.cs @@ -1,4 +1,5 @@ using System; +using System.Collections.Generic; using System.IO; using System.Linq; using System.Reflection; @@ -86,6 +87,14 @@ namespace StardewModdingAPI Platform.Mono; #endif + /// <summary>Maps vendor keys (like <c>Nexus</c>) to their mod URL template (where <c>{0}</c> is the mod ID) during mod compatibility checks. This doesn't affect update checks, which defer to the remote web API.</summary> + internal static readonly IDictionary<string, string> VendorModUrls = new Dictionary<string, string>(StringComparer.InvariantCultureIgnoreCase) + { + ["Chucklefish"] = "https://community.playstarbound.com/resources/{0}", + ["Nexus"] = "http://nexusmods.com/stardewvalley/mods/{0}", + ["GitHub"] = "https://github.com/{0}/releases" + }; + /********* ** Internal methods diff --git a/src/StardewModdingAPI/Framework/ModLoading/ModResolver.cs b/src/StardewModdingAPI/Framework/ModLoading/ModResolver.cs index 4d20e7c8..d0ef1b08 100644 --- a/src/StardewModdingAPI/Framework/ModLoading/ModResolver.cs +++ b/src/StardewModdingAPI/Framework/ModLoading/ModResolver.cs @@ -61,13 +61,9 @@ namespace StardewModdingAPI.Framework.ModLoading dataRecord = dataRecords.FirstOrDefault(record => record.ID.Matches(key, manifest)); } - // apply defaults - if (dataRecord?.Defaults != null) - { - manifest.ChucklefishID = manifest.ChucklefishID ?? dataRecord.Defaults.ChucklefishID; - manifest.GitHubProject = manifest.GitHubProject ?? dataRecord.Defaults.GitHubProject; - manifest.NexusID = manifest.NexusID ?? dataRecord.Defaults.NexusID; - } + // add default update keys + if (manifest != null && manifest.UpdateKeys == null && dataRecord?.UpdateKeys != null) + manifest.UpdateKeys = dataRecord.UpdateKeys; // build metadata string displayName = !string.IsNullOrWhiteSpace(manifest?.Name) @@ -84,7 +80,8 @@ namespace StardewModdingAPI.Framework.ModLoading /// <summary>Validate manifest metadata.</summary> /// <param name="mods">The mod manifests to validate.</param> /// <param name="apiVersion">The current SMAPI version.</param> - public void ValidateManifests(IEnumerable<IModMetadata> mods, ISemanticVersion apiVersion) + /// <param name="vendorModUrls">Maps vendor keys (like <c>Nexus</c>) to their mod URL template (where <c>{0}</c> is the mod ID).</param> + public void ValidateManifests(IEnumerable<IModMetadata> mods, ISemanticVersion apiVersion, IDictionary<string, string> vendorModUrls) { mods = mods.ToArray(); @@ -110,12 +107,18 @@ namespace StardewModdingAPI.Framework.ModLoading // get update URLs List<string> updateUrls = new List<string>(); - if (!string.IsNullOrWhiteSpace(mod.Manifest.ChucklefishID)) - updateUrls.Add($"https://community.playstarbound.com/resources/{mod.Manifest.ChucklefishID}"); - if (!string.IsNullOrWhiteSpace(mod.Manifest.NexusID)) - updateUrls.Add($"http://nexusmods.com/stardewvalley/mods/{mod.Manifest.NexusID}"); - if (!string.IsNullOrWhiteSpace(mod.Manifest.GitHubProject)) - updateUrls.Add($"https://github.com/{mod.Manifest.GitHubProject}/releases"); + foreach (string key in mod.Manifest.UpdateKeys ?? new string[0]) + { + string[] parts = key.Split(new[] { ':' }, 2); + if (parts.Length != 2) + continue; + + string vendorKey = parts[0].Trim(); + string modID = parts[1].Trim(); + + if (vendorModUrls.TryGetValue(vendorKey, out string urlTemplate)) + updateUrls.Add(string.Format(urlTemplate, modID)); + } if (mod.DataRecord.AlternativeUrl != null) updateUrls.Add(mod.DataRecord.AlternativeUrl); diff --git a/src/StardewModdingAPI/Framework/Models/Manifest.cs b/src/StardewModdingAPI/Framework/Models/Manifest.cs index c891644f..a051354c 100644 --- a/src/StardewModdingAPI/Framework/Models/Manifest.cs +++ b/src/StardewModdingAPI/Framework/Models/Manifest.cs @@ -34,14 +34,8 @@ namespace StardewModdingAPI.Framework.Models [JsonConverter(typeof(SFieldConverter))] public IManifestDependency[] Dependencies { get; set; } - /// <summary>The mod's unique ID in the Chucklefish mod site (if any), used for update checks.</summary> - public string ChucklefishID { get; set; } - - /// <summary>The mod's unique ID in Nexus Mods (if any), used for update checks.</summary> - public string NexusID { get; set; } - - /// <summary>The mod's organisation and project name on GitHub (if any), used for update checks.</summary> - public string GitHubProject { get; set; } + /// <summary>The namespaced mod IDs to query for updates (like <c>Nexus:541</c>).</summary> + public string[] UpdateKeys { get; set; } /// <summary>The unique mod ID.</summary> public string UniqueID { get; set; } diff --git a/src/StardewModdingAPI/Framework/Models/ModDataDefaults.cs b/src/StardewModdingAPI/Framework/Models/ModDataDefaults.cs deleted file mode 100644 index e0ab94b8..00000000 --- a/src/StardewModdingAPI/Framework/Models/ModDataDefaults.cs +++ /dev/null @@ -1,18 +0,0 @@ -namespace StardewModdingAPI.Framework.Models -{ - /// <summary>Default values for support fields to inject into the manifest.</summary> - internal class ModDataDefaults - { - /********* - ** Accessors - *********/ - /// <summary>The mod's unique ID in the Chucklefish mod site (if any), used for update checks.</summary> - public string ChucklefishID { get; set; } - - /// <summary>The mod's unique ID in Nexus Mods (if any), used for update checks.</summary> - public string NexusID { get; set; } - - /// <summary>The mod's organisation and project name on GitHub (if any), used for update checks.</summary> - public string GitHubProject { get; set; } - } -} diff --git a/src/StardewModdingAPI/Framework/Models/ModDataRecord.cs b/src/StardewModdingAPI/Framework/Models/ModDataRecord.cs index 0d033e82..c6a12188 100644 --- a/src/StardewModdingAPI/Framework/Models/ModDataRecord.cs +++ b/src/StardewModdingAPI/Framework/Models/ModDataRecord.cs @@ -15,8 +15,8 @@ namespace StardewModdingAPI.Framework.Models [JsonConverter(typeof(SFieldConverter))] public ModDataID ID { get; set; } - /// <summary>Default values for support fields to inject into the manifest.</summary> - public ModDataDefaults Defaults { get; set; } + /// <summary>A value to inject into <see cref="IManifest.UpdateKeys"/> field if it's not already set.</summary> + public string[] UpdateKeys { get; set; } /// <summary>The URL where the player can get an unofficial or alternative version of the mod if the official version isn't compatible.</summary> public string AlternativeUrl { get; set; } diff --git a/src/StardewModdingAPI/IManifest.cs b/src/StardewModdingAPI/IManifest.cs index 9513834a..befef901 100644 --- a/src/StardewModdingAPI/IManifest.cs +++ b/src/StardewModdingAPI/IManifest.cs @@ -32,14 +32,8 @@ namespace StardewModdingAPI /// <summary>The other mods that must be loaded before this mod.</summary> IManifestDependency[] Dependencies { get; } - /// <summary>The mod's unique ID in the Chucklefish mod site (if any), used for update checks.</summary> - string ChucklefishID { get; } - - /// <summary>The mod's unique ID in Nexus Mods (if any), used for update checks.</summary> - string NexusID { get; } - - /// <summary>The mod's organisation and project name on GitHub (if any), used for update checks.</summary> - string GitHubProject { get; } + /// <summary>The namespaced mod IDs to query for updates (like <c>Nexus:541</c>).</summary> + string[] UpdateKeys { get; set; } /// <summary>Any manifest fields which didn't match a valid field.</summary> IDictionary<string, object> ExtraFields { get; } diff --git a/src/StardewModdingAPI/Program.cs b/src/StardewModdingAPI/Program.cs index 0c4648ab..3575add6 100644 --- a/src/StardewModdingAPI/Program.cs +++ b/src/StardewModdingAPI/Program.cs @@ -347,7 +347,7 @@ namespace StardewModdingAPI // load manifests IModMetadata[] mods = resolver.ReadManifests(Constants.ModPath, new JsonHelper(), this.Settings.ModData).ToArray(); - resolver.ValidateManifests(mods, Constants.ApiVersion); + resolver.ValidateManifests(mods, Constants.ApiVersion, Constants.VendorModUrls); // process dependencies mods = resolver.ProcessDependencies(mods).ToArray(); @@ -530,25 +530,12 @@ namespace StardewModdingAPI } // add update keys - bool hasUpdateKeys = false; - if (!string.IsNullOrWhiteSpace(mod.Manifest.ChucklefishID)) + if (mod.Manifest.UpdateKeys?.Any() == true) { - hasUpdateKeys = true; - modsByKey[$"Chucklefish:{mod.Manifest.ChucklefishID}"] = mod; + foreach (string updateKey in mod.Manifest.UpdateKeys) + modsByKey[updateKey] = mod; } - if (!string.IsNullOrWhiteSpace(mod.Manifest.NexusID)) - { - hasUpdateKeys = true; - modsByKey[$"Nexus:{mod.Manifest.NexusID}"] = mod; - } - if (!string.IsNullOrWhiteSpace(mod.Manifest.GitHubProject)) - { - hasUpdateKeys = true; - modsByKey[$"GitHub:{mod.Manifest.GitHubProject}"] = mod; - } - - // log - if (!hasUpdateKeys) + else this.VerboseLog($" {mod.DisplayName}: no update keys."); } diff --git a/src/StardewModdingAPI/StardewModdingAPI.config.json b/src/StardewModdingAPI/StardewModdingAPI.config.json index 9003095c..54fe52c5 100644 --- a/src/StardewModdingAPI/StardewModdingAPI.config.json +++ b/src/StardewModdingAPI/StardewModdingAPI.config.json @@ -48,8 +48,8 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha * or a JSON structure containing any of three manifest fields (ID, Name, and Author) to * match. * - * - 'Defaults' specifies fields to inject into the mod's manifest if they're not already set. - * Supported fields: ChucklefishID, GitHubProject, and NexusID. + * - 'UpdateKeys' specifies the value of the equivalent manifest field if it's not already set. + * This is used to enable update checks for older mods that haven't been updated to use it yet. * * - 'AlternativeUrl' specifies a URL where the player can find an unofficial update or * alternative if the mod is no longer compatible. @@ -73,7 +73,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // AccessChestAnywhere "ID": "AccessChestAnywhere", - "Defaults": { "NexusID": 257 }, + "UpdateKeys": [ "Nexus:257" ], "AlternativeUrl": "https://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~1.1": { "Status": "AssumeBroken" } // broke in SDV 1.1 @@ -85,7 +85,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // AdjustArtisanPrices "ID": "1e36d4ca-c7ef-4dfb-9927-d27a6c3c8bdc", - "Defaults": { "ChucklefishID": 3532 }, + "UpdateKeys": [ "Chucklefish:3532" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~0.1": { "Status": "AssumeBroken" } // broke in SMAPI 1.9 @@ -94,12 +94,12 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Adjust Monster "ID": "mmanlapat.AdjustMonster", - "Defaults": { "NexusID": 1161 } + "UpdateKeys": [ "Nexus:1161" ] }, { // Advanced Location Loader "ID": "Entoarox.AdvancedLocationLoader", - //"Defaults": { "ChucklefishID": 3619 }, // Entoarox opted out of mod update checks + //"UpdateKeys": [ "Chucklefish:3619" ], // Entoarox opted out of mod update checks "Compatibility": { "~1.2.10": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 } @@ -107,12 +107,12 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Adventure Shop Inventory "ID": "HammurabiAdventureShopInventory", - "Defaults": { "ChucklefishID": 4608 } + "UpdateKeys": [ "Chucklefish:4608" ] }, { // AgingMod "ID": "skn.AgingMod", - "Defaults": { "NexusID": 1129 }, + "UpdateKeys": [ "Nexus:1129" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~1.0": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 @@ -121,17 +121,17 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // All Crops All Seasons "ID": "29ee8246-d67b-4242-a340-35a9ae0d5dd7 | community.AllCropsAllSeasons", // changed in 1.3 - "Defaults": { "NexusID": 170 } + "UpdateKeys": [ "Nexus:170" ] }, { // All Professions "ID": "8c37b1a7-4bfb-4916-9d8a-9533e6363ea3 | community.AllProfessions", // changed in 1.2 - "Defaults": { "NexusID": 174 } + "UpdateKeys": [ "Nexus:174" ] }, { // Almighty Tool "ID": "AlmightyTool.dll | 439", // changed in 1.2.1 - "Defaults": { "NexusID": 439 }, + "UpdateKeys": [ "Nexus:439" ], "Compatibility": { "~1.1.1": { "Status": "AssumeBroken" } // broke in SDV 1.2 }, @@ -152,7 +152,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Animal Sitter "ID": "AnimalSitter.dll | jwdred.AnimalSitter", // changed in 1.0.9 - "Defaults": { "NexusID": 581 }, + "UpdateKeys": [ "Nexus:581" ], "Compatibility": { "~1.0.8": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 } @@ -160,7 +160,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // A Tapper's Dream "ID": "ddde5195-8f85-4061-90cc-0d4fd5459358", - "Defaults": { "NexusID": 260 }, + "UpdateKeys": [ "Nexus:260" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~1.4": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 @@ -169,7 +169,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Auto Animal Doors "ID": "AaronTaggart.AutoAnimalDoors", - "Defaults": { "NexusID": 1019 }, + "UpdateKeys": [ "Nexus:1019" ], "MapRemoteVersions": { "1.1.1": "1.1" // manifest not updated } @@ -177,22 +177,22 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Auto-Eat "ID": "BALANCEMOD_AutoEat | Permamiss.AutoEat", // changed in 1.1.1 - "Defaults": { "NexusID": 643 } + "UpdateKeys": [ "Nexus:643" ] }, { // AutoGate "ID": "AutoGate", - "Defaults": { "NexusID": 820 } + "UpdateKeys": [ "Nexus:820" ] }, { // Automate "ID": "Pathoschild.Automate", - "Defaults": { "NexusID": 1063 } + "UpdateKeys": [ "Nexus:1063" ] }, { // Automated Doors "ID": "1abcfa07-2cf4-4dc3-a6e9-6068b642112b | azah.automated-doors", // changed in 1.4.1 - "Defaults": { "GitHubProject": "azah/AutomatedDoors" }, + "UpdateKeys": [ "GitHub:azah/AutomatedDoors" ], "MapLocalVersions": { "1.4.1-1": "1.4.1" } @@ -200,12 +200,12 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // AutoSpeed "ID": "{ID:'4be88c18-b6f3-49b0-ba96-f94b1a5be890', Name:'AutoSpeed'} | Omegasis.AutoSpeed", // changed in 1.4; disambiguate from other Alpha_Omegasis mods - "Defaults": { "NexusID": 443 } + "UpdateKeys": [ "Nexus:443" ] }, { // Basic Sprinkler Improved "ID": "lrsk_sdvm_bsi.0117171308", - "Defaults": { "NexusID": 833 }, + "UpdateKeys": [ "Nexus:833" ], "MapRemoteVersions": { "1.0.2": "1.0.1-release" // manifest not updated } @@ -213,22 +213,22 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Better Quality More Seasons "ID": "SB_BQMS", - "Defaults": { "NexusID": 935 } + "UpdateKeys": [ "Nexus:935" ] }, { // Better Quarry "ID": "BetterQuarry", - "Defaults": { "NexusID": 771 } + "UpdateKeys": [ "Nexus:771" ] }, { // Better Ranching "ID": "BetterRanching", - "Defaults": { "NexusID": 859 } + "UpdateKeys": [ "Nexus:859" ] }, { // Better Shipping Box "ID": "Kithio:BetterShippingBox", - "Defaults": { "ChucklefishID": 4302 }, + "UpdateKeys": [ "Chucklefish:4302" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~1.0.2": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 @@ -240,7 +240,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Better Sprinklers "ID": "SPDSprinklersMod | Speeder.BetterSprinklers", // changed in 2.3 - "Defaults": { "NexusID": 41 }, + "UpdateKeys": [ "Nexus:41" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~2.3.1-pathoschild-update": { "Status": "AssumeBroken" } // broke in SDV 1.2 @@ -249,12 +249,12 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Billboard Anywhere "ID": "{ID:'7ad4f6f7-c3de-4729-a40f-7a11d2b2a358', Name:'Billboard Anywhere'} | Omegasis.BillboardAnywhere", // changed in 1.4; disambiguate from other mods by Alpha_Omegasis - "Defaults": { "NexusID": 492 } + "UpdateKeys": [ "Nexus:492" ] }, { // Birthday Mail "ID": "005e02dc-d900-425c-9c68-1ff55c5a295d | KathrynHazuka.BirthdayMail", // changed in 1.2.3-pathoschild-update - "Defaults": { "NexusID": 276 }, + "UpdateKeys": [ "Nexus:276" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~1.2.2": { "Status": "AssumeBroken" } // broke in SDV 1.2 @@ -263,12 +263,12 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Breed Like Rabbits "ID": "dycedarger.breedlikerabbits", - "Defaults": { "NexusID": 948 } + "UpdateKeys": [ "Nexus:948" ] }, { // Build Endurance "ID": "{ID:'4be88c18-b6f3-49b0-ba96-f94b1a5be890', Name:'BuildEndurance'} | Omegasis.BuildEndurance", // changed in 1.4; disambiguate from other Alpha_Omegasis mods - "Defaults": { "NexusID": 445 }, + "UpdateKeys": [ "Nexus:445" ], "Compatibility": { "~1.3": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 } @@ -276,7 +276,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Build Health "ID": "{ID:'4be88c18-b6f3-49b0-ba96-f94b1a5be890', Name:'BuildHealth'} | Omegasis.BuildHealth", // changed in 1.4; disambiguate from other Alpha_Omegasis mods - "Defaults": { "NexusID": 446 }, + "UpdateKeys": [ "Nexus:446" ], "Compatibility": { "~1.3": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 } @@ -284,7 +284,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Buy Cooking Recipes "ID": "Denifia.BuyRecipes", - "Defaults": { "NexusID": 1126 }, + "UpdateKeys": [ "Nexus:1126" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~1.0": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 @@ -293,7 +293,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Buy Back Collectables "ID": "BuyBackCollectables | Omegasis.BuyBackCollectables", // changed in 1.4 - "Defaults": { "NexusID": 507 }, + "UpdateKeys": [ "Nexus:507" ], "Compatibility": { "~1.3": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 } @@ -301,12 +301,12 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Carry Chest "ID": "spacechase0.CarryChest", - "Defaults": { "NexusID": 1333 } + "UpdateKeys": [ "Nexus:1333" ] }, { // Casks Anywhere "ID": "CasksAnywhere", - "Defaults": { "NexusID": 878 }, + "UpdateKeys": [ "Nexus:878" ], "MapLocalVersions": { "1.1-alpha": "1.1" } @@ -314,12 +314,12 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Categorize Chests "ID": "CategorizeChests", - "Defaults": { "NexusID": 1300 } + "UpdateKeys": [ "Nexus:1300" ] }, { // ChefsCloset "ID": "Duder.ChefsCloset", - "Defaults": { "NexusID": 1030 }, + "UpdateKeys": [ "Nexus:1030" ], "MapLocalVersions": { "1.3-1": "1.3" } @@ -327,7 +327,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Chest Label System "ID": "SPDChestLabel | Speeder.ChestLabel", // changed in 1.5.1-pathoschild-update - "Defaults": { "NexusID": 242 }, + "UpdateKeys": [ "Nexus:242" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~1.6": { "Status": "AssumeBroken" } // broke in SDV 1.1 @@ -336,7 +336,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Chest Pooling "ID": "ChestPooling.dll | mralbobo.ChestPooling", // changed in 1.3 - "Defaults": { "GitHubProject": "mralbobo/stardew-chest-pooling" }, + "UpdateKeys": [ "GitHub:mralbobo/stardew-chest-pooling" ], "Compatibility": { "~1.2": { "Status": "AssumeBroken" } // broke in SDV 1.2 } @@ -344,7 +344,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Chests Anywhere "ID": "ChestsAnywhere | Pathoschild.ChestsAnywhere", // changed in 1.9 - "Defaults": { "NexusID": 518 }, + "UpdateKeys": [ "Nexus:518" ], "Compatibility": { "~1.9-beta": { "Status": "AssumeBroken" } // broke in SDV 1.2 } @@ -352,7 +352,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Choose Baby Gender "ID": "ChooseBabyGender.dll", - "Defaults": { "NexusID": 590 }, + "UpdateKeys": [ "Nexus:590" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~1.0.2": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 @@ -361,7 +361,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // CJB Automation "ID": "CJBAutomation", - "Defaults": { "NexusID": 211 }, + "UpdateKeys": [ "Nexus:211" ], "AlternativeUrl": "http://www.nexusmods.com/stardewvalley/mods/1063", "Compatibility": { "~1.4": { "Status": "AssumeBroken" } // broke in SDV 1.2 @@ -370,7 +370,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // CJB Cheats Menu "ID": "CJBCheatsMenu | CJBok.CheatsMenu", // changed in 1.14 - "Defaults": { "NexusID": 4 }, + "UpdateKeys": [ "Nexus:4" ], "Compatibility": { "~1.12": { "Status": "AssumeBroken" } // broke in SDV 1.1 } @@ -378,7 +378,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // CJB Item Spawner "ID": "CJBItemSpawner | CJBok.ItemSpawner", // changed in 1.7 - "Defaults": { "NexusID": 93 }, + "UpdateKeys": [ "Nexus:93" ], "Compatibility": { "~1.5": { "Status": "AssumeBroken" } // broke in SDV 1.1 } @@ -386,7 +386,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // CJB Show Item Sell Price "ID": "CJBShowItemSellPrice | CJBok.ShowItemSellPrice", // changed in 1.7 - "Defaults": { "NexusID": 5 }, + "UpdateKeys": [ "Nexus:5" ], "Compatibility": { "~1.6": { "Status": "AssumeBroken" } // broke in SDV 1.2 } @@ -394,12 +394,12 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Clean Farm "ID": "tstaples.CleanFarm", - "Defaults": { "NexusID": 794 } + "UpdateKeys": [ "Nexus:794" ] }, { // Climates of Ferngill "ID": "KoihimeNakamura.ClimatesOfFerngill", - "Defaults": { "NexusID": 604 }, + "UpdateKeys": [ "Nexus:604" ], "Compatibility": { "~1.0": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 } @@ -407,7 +407,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Cold Weather Haley "ID": "LordXamon.ColdWeatherHaleyPRO", - "Defaults": { "NexusID": 1169 }, + "UpdateKeys": [ "Nexus:1169" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~1.0": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 @@ -426,7 +426,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Combat with Farm Implements "ID": "SPDFarmingImplementsInCombat", - "Defaults": { "NexusID": 313 }, + "UpdateKeys": [ "Nexus:313" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~1.0": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 @@ -435,12 +435,12 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Community Bundle Item Tooltip "ID": "musbah.bundleTooltip", - "Defaults": { "NexusID": 1329 } + "UpdateKeys": [ "Nexus:1329" ] }, { // Configurable Machines "ID": "21da6619-dc03-4660-9794-8e5b498f5b97", - "Defaults": { "NexusID": 280 }, + "UpdateKeys": [ "Nexus:280" ], "MapLocalVersions": { "1.2-beta": "1.2" } @@ -448,7 +448,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Configurable Shipping Dates "ID": "ConfigurableShippingDates", - "Defaults": { "NexusID": 675 }, + "UpdateKeys": [ "Nexus:675" ], "AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0", "Compatibility": { "~1.1.1": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 @@ -457,7 +457,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Cooking Skill "ID": "CookingSkill | spacechase0.CookingSkill", // changed in 1.0.4–6 - "Defaults": { "NexusID": 522 }, + "UpdateKeys": [ "Nexus:522" ], "Compatibility": { "~1.0.6": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 } @@ -465,7 +465,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // CrabNet "ID": "CrabNet.dll | jwdred.CrabNet", // changed in 1.0.5 - "Defaults": { "NexusID": 584 }, + "UpdateKeys": [ "Nexus:584" ], "Compatibility": { "~1.0.4": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 } @@ -473,37 +473,37 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Current Location "ID": "CurrentLocation102120161203", - "Defaults": { "NexusID": 638 } + "UpdateKeys": [ "Nexus:638" ] }, { // Custom Critters "ID": "spacechase0.CustomCritters", - "Defaults": { "NexusID": 1255 } + "UpdateKeys": [ "Nexus:1255" ] }, { // Custom Element Handler "ID": "Platonymous.CustomElementHandler", - "Defaults": { "NexusID": 1068 } + "UpdateKeys": [ "Nexus:1068" ] }, { // Custom Farming "ID": "Platonymous.CustomFarming", - "Defaults": { "NexusID": 991 } + "UpdateKeys": [ "Nexus:991" ] }, { // Custom Farm Types "ID": "spacechase0.CustomFarmTypes", - "Defaults": { "NexusID": 1140 } + "UpdateKeys": [ "Nexus:1140" ] }, { // Custom Furniture "ID": "Platonymous.CustomFurniture", - "Defaults": { "NexusID": 1254 } + "UpdateKeys": [ "Nexus:1254" ] }, { // Customize Exterior "ID": "CustomizeExterior | spacechase0.CustomizeExterior", // changed in 1.0.3 - "Defaults": { "NexusID": 1099 }, + "UpdateKeys": [ "Nexus:1099" ], "Compatibility": { "~1.0.2": { "Status": "AssumeBroken" } // broke in SMAPI 2.0 } @@ -511,7 +511,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Customizable Cart Redux "ID": "KoihimeNakamura.CCR", - "Defaults": { "NexusID": 1402 }, + "UpdateKeys": [ "Nexus:1402" ], "MapLocalVersions": { "1.1-20170917": "1.1" } @@ -519,7 +519,7 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha { // Customizable Traveling Cart Days "ID": "TravelingCartYyeahdude", - "Defaults": { "NexusID": 567 }, + "UpdateKeys": [ "Nexus:567" ], "AlternativeUr |
