diff --git a/docs/release-notes.md b/docs/release-notes.md
index 4596a525..f3f2efa4 100644
--- a/docs/release-notes.md
+++ b/docs/release-notes.md
@@ -23,7 +23,7 @@
* Fixed `.pdb` files ignored for error stack traces for mods rewritten by SMAPI.
* For SMAPI developers:
- * When deploying web services to a single-instance app, the MongoDB server can now be replaced with in-memory storage.
+ * Eliminated MongoDB storage in the web services, which complicated the code unnecessarily. The app still uses an abstract interface for storage, so we can wrap a distributed cache in the future if needed.
* Merged the separate legacy redirects app on AWS into the main app on Azure.
## 3.5
diff --git a/docs/technical/web.md b/docs/technical/web.md
index ef591aee..d21b87ac 100644
--- a/docs/technical/web.md
+++ b/docs/technical/web.md
@@ -352,7 +352,6 @@ your machine, with no external dependencies aside from the actual mod sites.
--------------------------- | -----------
`AzureBlobConnectionString` | The connection string for the Azure Blob storage account. Defaults to using the system's temporary file folder if not specified.
`GitHubUsername`
`GitHubPassword` | The GitHub credentials with which to query GitHub release info. Defaults to anonymous requests if not specified.
- `Storage` | How to storage cached wiki/mod data. `InMemory` is recommended in most cases, or `MongoInMemory` to test the MongoDB storage code. See [production environment](#production-environment) for more info on `Mongo`.
2. Launch `SMAPI.Web` from Visual Studio to run a local version of the site.
@@ -385,23 +384,4 @@ Initial setup:
`Site:BetaBlurb` | If `Site:BetaEnabled` is true and there's a beta version of SMAPI in its GitHub releases, this is shown on the beta download button as explanatory subtext.
`Site:SupporterList` | A list of Patreon supports to credit on the download page.
-To enable distributed servers:
-
-1. Launch an empty MongoDB server (e.g. using [MongoDB Atlas](https://www.mongodb.com/cloud/atlas))
- for mod data.
-2. Add these application settings in the App Services environment:
-
- property name | description
- ------------------------------- | -----------------
- `Storage:Mode` | Set to `Mongo`.
- `Storage:ConnectionString` | Set to the connection string for the MongoDB instance.
-
- Optional settings:
-
- property name | description
- ------------------------------- | -----------------
- `Storage:Database` | Set to the MongoDB database name (defaults to `smapi`).
-
-To deploy updates:
-1. [Deploy the web project from Visual Studio](https://docs.microsoft.com/en-us/visualstudio/deployment/quickstart-deploy-to-azure).
-2. If the MongoDB schema changed, delete the MongoDB database. (It'll be recreated automatically.)
+To deploy updates, just [redeploy the web project from Visual Studio](https://docs.microsoft.com/en-us/visualstudio/deployment/quickstart-deploy-to-azure).
diff --git a/src/SMAPI.Web/BackgroundService.cs b/src/SMAPI.Web/BackgroundService.cs
index 275622fe..64bd5ca5 100644
--- a/src/SMAPI.Web/BackgroundService.cs
+++ b/src/SMAPI.Web/BackgroundService.cs
@@ -84,7 +84,7 @@ namespace StardewModdingAPI.Web
public static async Task UpdateWikiAsync()
{
WikiModList wikiCompatList = await new ModToolkit().GetWikiCompatibilityListAsync();
- BackgroundService.WikiCache.SaveWikiData(wikiCompatList.StableVersion, wikiCompatList.BetaVersion, wikiCompatList.Mods, out _, out _);
+ BackgroundService.WikiCache.SaveWikiData(wikiCompatList.StableVersion, wikiCompatList.BetaVersion, wikiCompatList.Mods);
}
/// Remove mods which haven't been requested in over 48 hours.
diff --git a/src/SMAPI.Web/Controllers/ModsApiController.cs b/src/SMAPI.Web/Controllers/ModsApiController.cs
index 6032186f..b9d7c32d 100644
--- a/src/SMAPI.Web/Controllers/ModsApiController.cs
+++ b/src/SMAPI.Web/Controllers/ModsApiController.cs
@@ -12,6 +12,7 @@ using StardewModdingAPI.Toolkit.Framework.Clients.Wiki;
using StardewModdingAPI.Toolkit.Framework.ModData;
using StardewModdingAPI.Toolkit.Framework.UpdateData;
using StardewModdingAPI.Web.Framework;
+using StardewModdingAPI.Web.Framework.Caching;
using StardewModdingAPI.Web.Framework.Caching.Mods;
using StardewModdingAPI.Web.Framework.Caching.Wiki;
using StardewModdingAPI.Web.Framework.Clients.Chucklefish;
@@ -90,7 +91,7 @@ namespace StardewModdingAPI.Web.Controllers
return new ModEntryModel[0];
// fetch wiki data
- WikiModEntry[] wikiData = this.WikiCache.GetWikiMods().Select(p => p.GetModel()).ToArray();
+ WikiModEntry[] wikiData = this.WikiCache.GetWikiMods().Select(p => p.Data).ToArray();
IDictionary mods = new Dictionary(StringComparer.CurrentCultureIgnoreCase);
foreach (ModSearchEntryModel mod in model.Mods)
{
@@ -283,27 +284,30 @@ namespace StardewModdingAPI.Web.Controllers
/// Whether to allow non-standard versions.
private async Task GetInfoForUpdateKeyAsync(UpdateKey updateKey, bool allowNonStandardVersions)
{
- // get mod
- if (!this.ModCache.TryGetMod(updateKey.Repository, updateKey.ID, out CachedMod mod) || this.ModCache.IsStale(mod.LastUpdated, mod.FetchStatus == RemoteModStatus.TemporaryError ? this.Config.Value.ErrorCacheMinutes : this.Config.Value.SuccessCacheMinutes))
+ // get from cache
+ if (this.ModCache.TryGetMod(updateKey.Repository, updateKey.ID, out Cached cachedMod) && !this.ModCache.IsStale(cachedMod.LastUpdated, cachedMod.Data.Status == RemoteModStatus.TemporaryError ? this.Config.Value.ErrorCacheMinutes : this.Config.Value.SuccessCacheMinutes))
+ return cachedMod.Data;
+
+ // fetch from mod site
{
// get site
if (!this.Repositories.TryGetValue(updateKey.Repository, out IModRepository repository))
return new ModInfoModel().SetError(RemoteModStatus.DoesNotExist, $"There's no mod site with key '{updateKey.Repository}'. Expected one of [{string.Join(", ", this.Repositories.Keys)}].");
// fetch mod
- ModInfoModel result = await repository.GetModInfoAsync(updateKey.ID);
- if (result.Error == null)
+ ModInfoModel mod = await repository.GetModInfoAsync(updateKey.ID);
+ if (mod.Error == null)
{
- if (result.Version == null)
- result.SetError(RemoteModStatus.InvalidData, $"The update key '{updateKey}' matches a mod with no version number.");
- else if (!SemanticVersion.TryParse(result.Version, allowNonStandardVersions, out _))
- result.SetError(RemoteModStatus.InvalidData, $"The update key '{updateKey}' matches a mod with invalid semantic version '{result.Version}'.");
+ if (mod.Version == null)
+ mod.SetError(RemoteModStatus.InvalidData, $"The update key '{updateKey}' matches a mod with no version number.");
+ else if (!SemanticVersion.TryParse(mod.Version, allowNonStandardVersions, out _))
+ mod.SetError(RemoteModStatus.InvalidData, $"The update key '{updateKey}' matches a mod with invalid semantic version '{mod.Version}'.");
}
// cache mod
- this.ModCache.SaveMod(repository.VendorKey, updateKey.ID, result, out mod);
+ this.ModCache.SaveMod(repository.VendorKey, updateKey.ID, mod);
+ return mod;
}
- return mod.GetModel();
}
/// Get update keys based on the available mod metadata, while maintaining the precedence order.
diff --git a/src/SMAPI.Web/Controllers/ModsController.cs b/src/SMAPI.Web/Controllers/ModsController.cs
index b621ded0..24e36709 100644
--- a/src/SMAPI.Web/Controllers/ModsController.cs
+++ b/src/SMAPI.Web/Controllers/ModsController.cs
@@ -2,6 +2,7 @@ using System.Linq;
using System.Text.RegularExpressions;
using Microsoft.AspNetCore.Mvc;
using Microsoft.Extensions.Options;
+using StardewModdingAPI.Web.Framework.Caching;
using StardewModdingAPI.Web.Framework.Caching.Wiki;
using StardewModdingAPI.Web.Framework.ConfigModels;
using StardewModdingAPI.Web.ViewModels;
@@ -51,16 +52,16 @@ namespace StardewModdingAPI.Web.Controllers
public ModListModel FetchData()
{
// fetch cached data
- if (!this.Cache.TryGetWikiMetadata(out CachedWikiMetadata metadata))
+ if (!this.Cache.TryGetWikiMetadata(out Cached metadata))
return new ModListModel();
// build model
return new ModListModel(
- stableVersion: metadata.StableVersion,
- betaVersion: metadata.BetaVersion,
+ stableVersion: metadata.Data.StableVersion,
+ betaVersion: metadata.Data.BetaVersion,
mods: this.Cache
.GetWikiMods()
- .Select(mod => new ModModel(mod.GetModel()))
+ .Select(mod => new ModModel(mod.Data))
.OrderBy(p => Regex.Replace(p.Name.ToLower(), "[^a-z0-9]", "")), // ignore case, spaces, and special characters when sorting
lastUpdated: metadata.LastUpdated,
isStale: this.Cache.IsStale(metadata.LastUpdated, this.StaleMinutes)
diff --git a/src/SMAPI.Web/Framework/Caching/Cached.cs b/src/SMAPI.Web/Framework/Caching/Cached.cs
new file mode 100644
index 00000000..52041a16
--- /dev/null
+++ b/src/SMAPI.Web/Framework/Caching/Cached.cs
@@ -0,0 +1,37 @@
+using System;
+
+namespace StardewModdingAPI.Web.Framework.Caching
+{
+ /// A cache entry.
+ /// The cached value type.
+ internal class Cached
+ {
+ /*********
+ ** Accessors
+ *********/
+ /// The cached data.
+ public T Data { get; set; }
+
+ /// When the data was last updated.
+ public DateTimeOffset LastUpdated { get; set; }
+
+ /// When the data was last requested through the mod API.
+ public DateTimeOffset LastRequested { get; set; }
+
+
+ /*********
+ ** Public methods
+ *********/
+ /// Construct an empty instance.
+ public Cached() { }
+
+ /// Construct an instance.
+ /// The cached data.
+ public Cached(T data)
+ {
+ this.Data = data;
+ this.LastUpdated = DateTimeOffset.UtcNow;
+ this.LastRequested = DateTimeOffset.UtcNow;
+ }
+ }
+}
diff --git a/src/SMAPI.Web/Framework/Caching/Mods/CachedMod.cs b/src/SMAPI.Web/Framework/Caching/Mods/CachedMod.cs
deleted file mode 100644
index 96eca847..00000000
--- a/src/SMAPI.Web/Framework/Caching/Mods/CachedMod.cs
+++ /dev/null
@@ -1,107 +0,0 @@
-using System;
-using System.Diagnostics.CodeAnalysis;
-using MongoDB.Bson;
-using MongoDB.Bson.Serialization.Attributes;
-using StardewModdingAPI.Toolkit.Framework.UpdateData;
-using StardewModdingAPI.Web.Framework.ModRepositories;
-
-namespace StardewModdingAPI.Web.Framework.Caching.Mods
-{
- /// The model for cached mod data.
- internal class CachedMod
- {
- /*********
- ** Accessors
- *********/
- /****
- ** Tracking
- ****/
- /// The internal MongoDB ID.
- [SuppressMessage("ReSharper", "InconsistentNaming", Justification = "Named per MongoDB conventions.")]
- [BsonIgnoreIfDefault]
- public ObjectId _id { get; set; }
-
- /// When the data was last updated.
- public DateTimeOffset LastUpdated { get; set; }
-
- /// When the data was last requested through the web API.
- public DateTimeOffset LastRequested { get; set; }
-
- /****
- ** Metadata
- ****/
- /// The mod site on which the mod is found.
- public ModRepositoryKey Site { get; set; }
-
- /// The mod's unique ID within the .
- public string ID { get; set; }
-
- /// The mod availability status on the remote site.
- public RemoteModStatus FetchStatus { get; set; }
-
- /// The error message providing more info for the , if applicable.
- public string FetchError { get; set; }
-
-
- /****
- ** Mod info
- ****/
- /// The mod's display name.
- public string Name { get; set; }
-
- /// The mod's latest version.
- public string MainVersion { get; set; }
-
- /// The mod's latest optional or prerelease version, if newer than .
- public string PreviewVersion { get; set; }
-
- /// The URL for the mod page.
- public string Url { get; set; }
-
- /// The license URL, if available.
- public string LicenseUrl { get; set; }
-
- /// The license name, if available.
- public string LicenseName { get; set; }
-
-
- /*********
- ** Accessors
- *********/
- /// Construct an instance.
- public CachedMod() { }
-
- /// Construct an instance.
- /// The mod site on which the mod is found.
- /// The mod's unique ID within the .
- /// The mod data.
- public CachedMod(ModRepositoryKey site, string id, ModInfoModel mod)
- {
- // tracking
- this.LastUpdated = DateTimeOffset.UtcNow;
- this.LastRequested = DateTimeOffset.UtcNow;
-
- // metadata
- this.Site = site;
- this.ID = id;
- this.FetchStatus = mod.Status;
- this.FetchError = mod.Error;
-
- // mod info
- this.Name = mod.Name;
- this.MainVersion = mod.Version;
- this.PreviewVersion = mod.PreviewVersion;
- this.Url = mod.Url;
- this.LicenseUrl = mod.LicenseUrl;
- this.LicenseName = mod.LicenseName;
- }
-
- /// Get the API model for the cached data.
- public ModInfoModel GetModel()
- {
- return new ModInfoModel(name: this.Name, version: this.MainVersion, url: this.Url, previewVersion: this.PreviewVersion)
- .SetLicense(this.LicenseUrl, this.LicenseName)
- .SetError(this.FetchStatus, this.FetchError);
- }
- }
-}
diff --git a/src/SMAPI.Web/Framework/Caching/Mods/IModCacheRepository.cs b/src/SMAPI.Web/Framework/Caching/Mods/IModCacheRepository.cs
index 08749f3b..004202f9 100644
--- a/src/SMAPI.Web/Framework/Caching/Mods/IModCacheRepository.cs
+++ b/src/SMAPI.Web/Framework/Caching/Mods/IModCacheRepository.cs
@@ -15,14 +15,13 @@ namespace StardewModdingAPI.Web.Framework.Caching.Mods
/// The mod's unique ID within the .
/// The fetched mod.
/// Whether to update the mod's 'last requested' date.
- bool TryGetMod(ModRepositoryKey site, string id, out CachedMod mod, bool markRequested = true);
+ bool TryGetMod(ModRepositoryKey site, string id, out Cached mod, bool markRequested = true);
/// Save data fetched for a mod.
/// The mod site on which the mod is found.
/// The mod's unique ID within the .
/// The mod data.
- /// The stored mod record.
- void SaveMod(ModRepositoryKey site, string id, ModInfoModel mod, out CachedMod cachedMod);
+ void SaveMod(ModRepositoryKey site, string id, ModInfoModel mod);
/// Delete data for mods which haven't been requested within a given time limit.
/// The minimum age for which to remove mods.
diff --git a/src/SMAPI.Web/Framework/Caching/Mods/ModCacheMemoryRepository.cs b/src/SMAPI.Web/Framework/Caching/Mods/ModCacheMemoryRepository.cs
index 9c5a217e..62461116 100644
--- a/src/SMAPI.Web/Framework/Caching/Mods/ModCacheMemoryRepository.cs
+++ b/src/SMAPI.Web/Framework/Caching/Mods/ModCacheMemoryRepository.cs
@@ -13,7 +13,7 @@ namespace StardewModdingAPI.Web.Framework.Caching.Mods
** Fields
*********/
/// The cached mod data indexed by {site key}:{ID}.
- private readonly IDictionary Mods = new Dictionary(StringComparer.InvariantCultureIgnoreCase);
+ private readonly IDictionary> Mods = new Dictionary>(StringComparer.InvariantCultureIgnoreCase);
/*********
@@ -24,19 +24,20 @@ namespace StardewModdingAPI.Web.Framework.Caching.Mods
/// The mod's unique ID within the .
/// The fetched mod.
/// Whether to update the mod's 'last requested' date.
- public bool TryGetMod(ModRepositoryKey site, string id, out CachedMod mod, bool markRequested = true)
+ public bool TryGetMod(ModRepositoryKey site, string id, out Cached mod, bool markRequested = true)
{
// get mod
- if (!this.Mods.TryGetValue(this.GetKey(site, id), out mod))
+ if (!this.Mods.TryGetValue(this.GetKey(site, id), out var cachedMod))
+ {
+ mod = null;
return false;
+ }
// bump 'last requested'
if (markRequested)
- {
- mod.LastRequested = DateTimeOffset.UtcNow;
- mod = this.SaveMod(mod);
- }
+ cachedMod.LastRequested = DateTimeOffset.UtcNow;
+ mod = cachedMod;
return true;
}
@@ -44,11 +45,10 @@ namespace StardewModdingAPI.Web.Framework.Caching.Mods
/// The mod site on which the mod is found.
/// The mod's unique ID within the .
/// The mod data.
- /// The stored mod record.
- public void SaveMod(ModRepositoryKey site, string id, ModInfoModel mod, out CachedMod cachedMod)
+ public void SaveMod(ModRepositoryKey site, string id, ModInfoModel mod)
{
string key = this.GetKey(site, id);
- cachedMod = this.SaveMod(new CachedMod(site, id, mod));
+ this.Mods[key] = new Cached(mod);
}
/// Delete data for mods which haven't been requested within a given time limit.
@@ -66,14 +66,6 @@ namespace StardewModdingAPI.Web.Framework.Caching.Mods
this.Mods.Remove(key);
}
- /// Save data fetched for a mod.
- /// The mod data.
- public CachedMod SaveMod(CachedMod mod)
- {
- string key = this.GetKey(mod.Site, mod.ID);
- return this.Mods[key] = mod;
- }
-
/*********
** Private methods
@@ -81,7 +73,7 @@ namespace StardewModdingAPI.Web.Framework.Caching.Mods
/// Get a cache key.
/// The mod site.
/// The mod ID.
- public string GetKey(ModRepositoryKey site, string id)
+ private string GetKey(ModRepositoryKey site, string id)
{
return $"{site}:{id.Trim()}".ToLower();
}
diff --git a/src/SMAPI.Web/Framework/Caching/Mods/ModCacheMongoRepository.cs b/src/SMAPI.Web/Framework/Caching/Mods/ModCacheMongoRepository.cs
deleted file mode 100644
index f105baab..00000000
--- a/src/SMAPI.Web/Framework/Caching/Mods/ModCacheMongoRepository.cs
+++ /dev/null
@@ -1,105 +0,0 @@
-using System;
-using MongoDB.Driver;
-using StardewModdingAPI.Toolkit.Framework.UpdateData;
-using StardewModdingAPI.Web.Framework.ModRepositories;
-
-namespace StardewModdingAPI.Web.Framework.Caching.Mods
-{
- /// Manages cached mod data in MongoDB.
- internal class ModCacheMongoRepository : BaseCacheRepository, IModCacheRepository
- {
- /*********
- ** Fields
- *********/
- /// The collection for cached mod data.
- private readonly IMongoCollection Mods;
-
-
- /*********
- ** Public methods
- *********/
- /// Construct an instance.
- /// The authenticated MongoDB database.
- public ModCacheMongoRepository(IMongoDatabase database)
- {
- // get collections
- this.Mods = database.GetCollection("mods");
-
- // add indexes if needed
- this.Mods.Indexes.CreateOne(new CreateIndexModel(Builders.IndexKeys.Ascending(p => p.ID).Ascending(p => p.Site)));
- }
-
-
- /*********
- ** Public methods
- *********/
- /// Get the cached mod data.
- /// The mod site to search.
- /// The mod's unique ID within the .
- /// The fetched mod.
- /// Whether to update the mod's 'last requested' date.
- public bool TryGetMod(ModRepositoryKey site, string id, out CachedMod mod, bool markRequested = true)
- {
- // get mod
- id = this.NormalizeId(id);
- mod = this.Mods.Find(entry => entry.ID == id && entry.Site == site).FirstOrDefault();
- if (mod == null)
- return false;
-
- // bump 'last requested'
- if (markRequested)
- {
- mod.LastRequested = DateTimeOffset.UtcNow;
- mod = this.SaveMod(mod);
- }
-
- return true;
- }
-
- /// Save data fetched for a mod.
- /// The mod site on which the mod is found.
- /// The mod's unique ID within the .
- /// The mod data.
- /// The stored mod record.
- public void SaveMod(ModRepositoryKey site, string id, ModInfoModel mod, out CachedMod cachedMod)
- {
- id = this.NormalizeId(id);
-
- cachedMod = this.SaveMod(new CachedMod(site, id, mod));
- }
-
- /// Delete data for mods which haven't been requested within a given time limit.
- /// The minimum age for which to remove mods.
- public void RemoveStaleMods(TimeSpan age)
- {
- DateTimeOffset minDate = DateTimeOffset.UtcNow.Subtract(age);
- this.Mods.DeleteMany(p => p.LastRequested < minDate);
- }
-
- /// Save data fetched for a mod.
- /// The mod data.
- public CachedMod SaveMod(CachedMod mod)
- {
- string id = this.NormalizeId(mod.ID);
-
- this.Mods.ReplaceOne(
- entry => entry.ID == id && entry.Site == mod.Site,
- mod,
- new ReplaceOptions { IsUpsert = true }
- );
-
- return mod;
- }
-
-
- /*********
- ** Private methods
- *********/
- /// Normalize a mod ID for case-insensitive search.
- /// The mod ID.
- public string NormalizeId(string id)
- {
- return id.Trim().ToLower();
- }
- }
-}
diff --git a/src/SMAPI.Web/Framework/Caching/UtcDateTimeOffsetSerializer.cs b/src/SMAPI.Web/Framework/Caching/UtcDateTimeOffsetSerializer.cs
deleted file mode 100644
index 6a103e37..00000000
--- a/src/SMAPI.Web/Framework/Caching/UtcDateTimeOffsetSerializer.cs
+++ /dev/null
@@ -1,40 +0,0 @@
-using System;
-using MongoDB.Bson;
-using MongoDB.Bson.Serialization;
-using MongoDB.Bson.Serialization.Serializers;
-
-namespace StardewModdingAPI.Web.Framework.Caching
-{
- /// Serializes to a UTC date field instead of the default array.
- public class UtcDateTimeOffsetSerializer : StructSerializerBase
- {
- /*********
- ** Fields
- *********/
- /// The underlying date serializer.
- private static readonly DateTimeSerializer DateTimeSerializer = new DateTimeSerializer(DateTimeKind.Utc, BsonType.DateTime);
-
-
- /*********
- ** Public methods
- *********/
- /// Deserializes a value.
- /// The deserialization context.
- /// The deserialization args.
- /// A deserialized value.
- public override DateTimeOffset Deserialize(BsonDeserializationContext context, BsonDeserializationArgs args)
- {
- DateTime date = UtcDateTimeOffsetSerializer.DateTimeSerializer.Deserialize(context, args);
- return new DateTimeOffset(date, TimeSpan.Zero);
- }
-
- /// Serializes a value.
- /// The serialization context.
- /// The serialization args.
- /// The object.
- public override void Serialize(BsonSerializationContext context, BsonSerializationArgs args, DateTimeOffset value)
- {
- UtcDateTimeOffsetSerializer.DateTimeSerializer.Serialize(context, args, value.UtcDateTime);
- }
- }
-}
diff --git a/src/SMAPI.Web/Framework/Caching/Wiki/CachedWikiMod.cs b/src/SMAPI.Web/Framework/Caching/Wiki/CachedWikiMod.cs
deleted file mode 100644
index 7e7c99bc..00000000
--- a/src/SMAPI.Web/Framework/Caching/Wiki/CachedWikiMod.cs
+++ /dev/null
@@ -1,230 +0,0 @@
-using System;
-using System.Collections.Generic;
-using System.Diagnostics.CodeAnalysis;
-using MongoDB.Bson;
-using MongoDB.Bson.Serialization.Attributes;
-using MongoDB.Bson.Serialization.Options;
-using StardewModdingAPI.Toolkit;
-using StardewModdingAPI.Toolkit.Framework.Clients.Wiki;
-
-namespace StardewModdingAPI.Web.Framework.Caching.Wiki
-{
- /// The model for cached wiki mods.
- internal class CachedWikiMod
- {
- /*********
- ** Accessors
- *********/
- /****
- ** Tracking
- ****/
- /// The internal MongoDB ID.
- [SuppressMessage("ReSharper", "InconsistentNaming", Justification = "Named per MongoDB conventions.")]
- public ObjectId _id { get; set; }
-
- /// When the data was last updated.
- public DateTimeOffset LastUpdated { get; set; }
-
- /****
- ** Mod info
- ****/
- /// The mod's unique ID. If the mod has alternate/old IDs, they're listed in latest to newest order.
- public string[] ID { get; set; }
-
- /// The mod's display name. If the mod has multiple names, the first one is the most canonical name.
- public string[] Name { get; set; }
-
- /// The mod's author name. If the author has multiple names, the first one is the most canonical name.
- public string[] Author { get; set; }
-
- /// The mod ID on Nexus.
- public int? NexusID { get; set; }
-
- /// The mod ID in the Chucklefish mod repo.
- public int? ChucklefishID { get; set; }
-
- /// The mod ID in the CurseForge mod repo.
- public int? CurseForgeID { get; set; }
-
- /// The mod key in the CurseForge mod repo (used in mod page URLs).
- public string CurseForgeKey { get; set; }
-
- /// The mod ID in the ModDrop mod repo.
- public int? ModDropID { get; set; }
-
- /// The GitHub repository in the form 'owner/repo'.
- public string GitHubRepo { get; set; }
-
- /// The URL to a non-GitHub source repo.
- public string CustomSourceUrl { get; set; }
-
- /// The custom mod page URL (if applicable).
- public string CustomUrl { get; set; }
-
- /// The name of the mod which loads this content pack, if applicable.
- public string ContentPackFor { get; set; }
-
- /// The human-readable warnings for players about this mod.
- public string[] Warnings { get; set; }
-
- /// The URL of the pull request which submits changes for an unofficial update to the author, if any.
- public string PullRequestUrl { get; set; }
-
- /// Special notes intended for developers who maintain unofficial updates or submit pull requests.
- public string DevNote { get; set; }
-
- /// The link anchor for the mod entry in the wiki compatibility list.
- public string Anchor { get; set; }
-
- /****
- ** Stable compatibility
- ****/
- /// The compatibility status.
- public WikiCompatibilityStatus MainStatus { get; set; }
-
- /// The human-readable summary of the compatibility status or workaround, without HTML formatting.
- public string MainSummary { get; set; }
-
- /// The game or SMAPI version which broke this mod (if applicable).
- public string MainBrokeIn { get; set; }
-
- /// The version of the latest unofficial update, if applicable.
- public string MainUnofficialVersion { get; set; }
-
- /// The URL to the latest unofficial update, if applicable.
- public string MainUnofficialUrl { get; set; }
-
- /****
- ** Beta compatibility
- ****/
- /// The compatibility status.
- public WikiCompatibilityStatus? BetaStatus { get; set; }
-
- /// The human-readable summary of the compatibility status or workaround, without HTML formatting.
- public string BetaSummary { get; set; }
-
- /// The game or SMAPI version which broke this mod (if applicable).
- public string BetaBrokeIn { get; set; }
-
- /// The version of the latest unofficial update, if applicable.
- public string BetaUnofficialVersion { get; set; }
-
- /// The URL to the latest unofficial update, if applicable.
- public string BetaUnofficialUrl { get; set; }
-
- /****
- ** Version maps
- ****/
- /// Maps local versions to a semantic version for update checks.
- [BsonDictionaryOptions(Representation = DictionaryRepresentation.ArrayOfArrays)]
- public IDictionary MapLocalVersions { get; set; }
-
- /// Maps remote versions to a semantic version for update checks.
- [BsonDictionaryOptions(Representation = DictionaryRepresentation.ArrayOfArrays)]
- public IDictionary MapRemoteVersions { get; set; }
-
-
- /*********
- ** Accessors
- *********/
- /// Construct an instance.
- public CachedWikiMod() { }
-
- /// Construct an instance.
- /// The mod data.
- public CachedWikiMod(WikiModEntry mod)
- {
- // tracking
- this.LastUpdated = DateTimeOffset.UtcNow;
-
- // mod info
- this.ID = mod.ID;
- this.Name = mod.Name;
- this.Author = mod.Author;
- this.NexusID = mod.NexusID;
- this.ChucklefishID = mod.ChucklefishID;
- this.CurseForgeID = mod.CurseForgeID;
- this.CurseForgeKey = mod.CurseForgeKey;
- this.ModDropID = mod.ModDropID;
- this.GitHubRepo = mod.GitHubRepo;
- this.CustomSourceUrl = mod.CustomSourceUrl;
- this.CustomUrl = mod.CustomUrl;
- this.ContentPackFor = mod.ContentPackFor;
- this.PullRequestUrl = mod.PullRequestUrl;
- this.Warnings = mod.Warnings;
- this.DevNote = mod.DevNote;
- this.Anchor = mod.Anchor;
-
- // stable compatibility
- this.MainStatus = mod.Compatibility.Status;
- this.MainSummary = mod.Compatibility.Summary;
- this.MainBrokeIn = mod.Compatibility.BrokeIn;
- this.MainUnofficialVersion = mod.Compatibility.UnofficialVersion?.ToString();
- this.MainUnofficialUrl = mod.Compatibility.UnofficialUrl;
-
- // beta compatibility
- this.BetaStatus = mod.BetaCompatibility?.Status;
- this.BetaSummary = mod.BetaCompatibility?.Summary;
- this.BetaBrokeIn = mod.BetaCompatibility?.BrokeIn;
- this.BetaUnofficialVersion = mod.BetaCompatibility?.UnofficialVersion?.ToString();
- this.BetaUnofficialUrl = mod.BetaCompatibility?.UnofficialUrl;
-
- // version maps
- this.MapLocalVersions = mod.MapLocalVersions;
- this.MapRemoteVersions = mod.MapRemoteVersions;
- }
-
- /// Reconstruct the original model.
- public WikiModEntry GetModel()
- {
- var mod = new WikiModEntry
- {
- ID = this.ID,
- Name = this.Name,
- Author = this.Author,
- NexusID = this.NexusID,
- ChucklefishID = this.ChucklefishID,
- CurseForgeID = this.CurseForgeID,
- CurseForgeKey = this.CurseForgeKey,
- ModDropID = this.ModDropID,
- GitHubRepo = this.GitHubRepo,
- CustomSourceUrl = this.CustomSourceUrl,
- CustomUrl = this.CustomUrl,
- ContentPackFor = this.ContentPackFor,
- Warnings = this.Warnings,
- PullRequestUrl = this.PullRequestUrl,
- DevNote = this.DevNote,
- Anchor = this.Anchor,
-
- // stable compatibility
- Compatibility = new WikiCompatibilityInfo
- {
- Status = this.MainStatus,
- Summary = this.MainSummary,
- BrokeIn = this.MainBrokeIn,
- UnofficialVersion = this.MainUnofficialVersion != null ? new SemanticVersion(this.MainUnofficialVersion) : null,
- UnofficialUrl = this.MainUnofficialUrl
- },
-
- // version maps
- MapLocalVersions = this.MapLocalVersions,
- MapRemoteVersions = this.MapRemoteVersions
- };
-
- // beta compatibility
- if (this.BetaStatus != null)
- {
- mod.BetaCompatibility = new WikiCompatibilityInfo
- {
- Status = this.BetaStatus.Value,
- Summary = this.BetaSummary,
- BrokeIn = this.BetaBrokeIn,
- UnofficialVersion = this.BetaUnofficialVersion != null ? new SemanticVersion(this.BetaUnofficialVersion) : null,
- UnofficialUrl = this.BetaUnofficialUrl
- };
- }
-
- return mod;
- }
- }
-}
diff --git a/src/SMAPI.Web/Framework/Caching/Wiki/IWikiCacheRepository.cs b/src/SMAPI.Web/Framework/Caching/Wiki/IWikiCacheRepository.cs
index 02097f52..2ab7ea5a 100644
--- a/src/SMAPI.Web/Framework/Caching/Wiki/IWikiCacheRepository.cs
+++ b/src/SMAPI.Web/Framework/Caching/Wiki/IWikiCacheRepository.cs
@@ -1,6 +1,5 @@
using System;
using System.Collections.Generic;
-using System.Linq.Expressions;
using StardewModdingAPI.Toolkit.Framework.Clients.Wiki;
namespace StardewModdingAPI.Web.Framework.Caching.Wiki
@@ -13,18 +12,16 @@ namespace StardewModdingAPI.Web.Framework.Caching.Wiki
*********/
/// Get the cached wiki metadata.
/// The fetched metadata.
- bool TryGetWikiMetadata(out CachedWikiMetadata metadata);
+ bool TryGetWikiMetadata(out Cached metadata);
/// Get the cached wiki mods.
/// A filter to apply, if any.
- IEnumerable GetWikiMods(Expression> filter = null);
+ IEnumerable> GetWikiMods(Func filter = null);
/// Save data fetched from the wiki compatibility list.
/// The current stable Stardew Valley version.
/// The current beta Stardew Valley version.
/// The mod data.
- /// The stored metadata record.
- /// The stored mod records.
- void SaveWikiData(string stableVersion, string betaVersion, IEnumerable mods, out CachedWikiMetadata cachedMetadata, out CachedWikiMod[] cachedMods);
+ void SaveWikiData(string stableVersion, string betaVersion, IEnumerable mods);
}
}
diff --git a/src/SMAPI.Web/Framework/Caching/Wiki/WikiCacheMemoryRepository.cs b/src/SMAPI.Web/Framework/Caching/Wiki/WikiCacheMemoryRepository.cs
index 4621f5e3..064a7c3c 100644
--- a/src/SMAPI.Web/Framework/Caching/Wiki/WikiCacheMemoryRepository.cs
+++ b/src/SMAPI.Web/Framework/Caching/Wiki/WikiCacheMemoryRepository.cs
@@ -1,7 +1,6 @@
using System;
using System.Collections.Generic;
using System.Linq;
-using System.Linq.Expressions;
using StardewModdingAPI.Toolkit.Framework.Clients.Wiki;
namespace StardewModdingAPI.Web.Framework.Caching.Wiki
@@ -13,10 +12,10 @@ namespace StardewModdingAPI.Web.Framework.Caching.Wiki
** Fields
*********/
/// The saved wiki metadata.
- private CachedWikiMetadata Metadata;
+ private Cached Metadata;
/// The cached wiki data.
- private CachedWikiMod[] Mods = new CachedWikiMod[0];
+ private Cached[] Mods = new Cached[0];
/*********
@@ -24,7 +23,7 @@ namespace StardewModdingAPI.Web.Framework.Caching.Wiki
*********/
/// Get the cached wiki metadata.
/// The fetched metadata.
- public bool TryGetWikiMetadata(out CachedWikiMetadata metadata)
+ public bool TryGetWikiMetadata(out Cached metadata)
{
metadata = this.Metadata;
return metadata != null;
@@ -32,23 +31,23 @@ namespace StardewModdingAPI.Web.Framework.Caching.Wiki
/// Get the cached wiki mods.
/// A filter to apply, if any.
- public IEnumerable GetWikiMods(Expression> filter = null)
+ public IEnumerable> GetWikiMods(Func filter = null)
{
- return filter != null
- ? this.Mods.Where(filter.Compile())
- : this.Mods.ToArray();
+ foreach (var mod in this.Mods)
+ {
+ if (filter == null || filter(mod.Data))
+ yield return mod;
+ }
}
/// Save data fetched from the wiki compatibility list.
/// The current stable Stardew Valley version.
/// The current beta Stardew Valley version.
/// The mod data.
- /// The stored metadata record.
- /// The stored mod records.
- public void SaveWikiData(string stableVersion, string betaVersion, IEnumerable mods, out CachedWikiMetadata cachedMetadata, out CachedWikiMod[] cachedMods)
+ public void SaveWikiData(string stableVersion, string betaVersion, IEnumerable mods)
{
- this.Metadata = cachedMetadata = new CachedWikiMetadata(stableVersion, betaVersion);
- this.Mods = cachedMods = mods.Select(mod => new CachedWikiMod(mod)).ToArray();
+ this.Metadata = new Cached(new WikiMetadata(stableVersion, betaVersion));
+ this.Mods = mods.Select(mod => new Cached(mod)).ToArray();
}
}
}
diff --git a/src/SMAPI.Web/Framework/Caching/Wiki/WikiCacheMongoRepository.cs b/src/SMAPI.Web/Framework/Caching/Wiki/WikiCacheMongoRepository.cs
deleted file mode 100644
index 07e7c721..00000000
--- a/src/SMAPI.Web/Framework/Caching/Wiki/WikiCacheMongoRepository.cs
+++ /dev/null
@@ -1,73 +0,0 @@
-using System;
-using System.Collections.Generic;
-using System.Linq;
-using System.Linq.Expressions;
-using MongoDB.Driver;
-using StardewModdingAPI.Toolkit.Framework.Clients.Wiki;
-
-namespace StardewModdingAPI.Web.Framework.Caching.Wiki
-{
- /// Manages cached wiki data in MongoDB.
- internal class WikiCacheMongoRepository : BaseCacheRepository, IWikiCacheRepository
- {
- /*********
- ** Fields
- *********/
- /// The collection for wiki metadata.
- private readonly IMongoCollection Metadata;
-
- /// The collection for wiki mod data.
- private readonly IMongoCollection Mods;
-
-
- /*********
- ** Public methods
- *********/
- /// Construct an instance.
- /// The authenticated MongoDB database.
- public WikiCacheMongoRepository(IMongoDatabase database)
- {
- // get collections
- this.Metadata = database.GetCollection("wiki-metadata");
- this.Mods = database.GetCollection("wiki-mods");
-
- // add indexes if needed
- this.Mods.Indexes.CreateOne(new CreateIndexModel(Builders.IndexKeys.Ascending(p => p.ID)));
- }
-
- /// Get the cached wiki metadata.
- /// The fetched metadata.
- public bool TryGetWikiMetadata(out CachedWikiMetadata metadata)
- {
- metadata = this.Metadata.Find("{}").FirstOrDefault();
- return metadata != null;
- }
-
- /// Get the cached wiki mods.
- /// A filter to apply, if any.
- public IEnumerable GetWikiMods(Expression> filter = null)
- {
- return filter != null
- ? this.Mods.Find(filter).ToList()
- : this.Mods.Find("{}").ToList();
- }
-
- /// Save data fetched from the wiki compatibility list.
- /// The current stable Stardew Valley version.
- /// The current beta Stardew Valley version.
- /// The mod data.
- /// The stored metadata record.
- /// The stored mod records.
- public void SaveWikiData(string stableVersion, string betaVersion, IEnumerable mods, out CachedWikiMetadata cachedMetadata, out CachedWikiMod[] cachedMods)
- {
- cachedMetadata = new CachedWikiMetadata(stableVersion, betaVersion);
- cachedMods = mods.Select(mod => new CachedWikiMod(mod)).ToArray();
-
- this.Mods.DeleteMany("{}");
- this.Mods.InsertMany(cachedMods);
-
- this.Metadata.DeleteMany("{}");
- this.Metadata.InsertOne(cachedMetadata);
- }
- }
-}
diff --git a/src/SMAPI.Web/Framework/Caching/Wiki/CachedWikiMetadata.cs b/src/SMAPI.Web/Framework/Caching/Wiki/WikiMetadata.cs
similarity index 59%
rename from src/SMAPI.Web/Framework/Caching/Wiki/CachedWikiMetadata.cs
rename to src/SMAPI.Web/Framework/Caching/Wiki/WikiMetadata.cs
index 6a560eb4..c04de4a5 100644
--- a/src/SMAPI.Web/Framework/Caching/Wiki/CachedWikiMetadata.cs
+++ b/src/SMAPI.Web/Framework/Caching/Wiki/WikiMetadata.cs
@@ -1,22 +1,11 @@
-using System;
-using System.Diagnostics.CodeAnalysis;
-using MongoDB.Bson;
-
namespace StardewModdingAPI.Web.Framework.Caching.Wiki
{
/// The model for cached wiki metadata.
- internal class CachedWikiMetadata
+ internal class WikiMetadata
{
/*********
** Accessors
*********/
- /// The internal MongoDB ID.
- [SuppressMessage("ReSharper", "InconsistentNaming", Justification = "Named per MongoDB conventions.")]
- public ObjectId _id { get; set; }
-
- /// When the data was last updated.
- public DateTimeOffset LastUpdated { get; set; }
-
/// The current stable Stardew Valley version.
public string StableVersion { get; set; }
@@ -28,16 +17,15 @@ namespace StardewModdingAPI.Web.Framework.Caching.Wiki
** Public methods
*********/
/// Construct an instance.
- public CachedWikiMetadata() { }
+ public WikiMetadata() { }
/// Construct an instance.
/// The current stable Stardew Valley version.
/// The current beta Stardew Valley version.
- public CachedWikiMetadata(string stableVersion, string betaVersion)
+ public WikiMetadata(string stableVersion, string betaVersion)
{
this.StableVersion = stableVersion;
this.BetaVersion = betaVersion;
- this.LastUpdated = DateTimeOffset.UtcNow;
}
}
}
diff --git a/src/SMAPI.Web/Framework/ConfigModels/StorageConfig.cs b/src/SMAPI.Web/Framework/ConfigModels/StorageConfig.cs
deleted file mode 100644
index 61cc4855..00000000
--- a/src/SMAPI.Web/Framework/ConfigModels/StorageConfig.cs
+++ /dev/null
@@ -1,18 +0,0 @@
-namespace StardewModdingAPI.Web.Framework.ConfigModels
-{
- /// The config settings for cache storage.
- internal class StorageConfig
- {
- /*********
- ** Accessors
- *********/
- /// The storage mechanism to use.
- public StorageMode Mode { get; set; }
-
- /// The connection string for the storage mechanism, if applicable.
- public string ConnectionString { get; set; }
-
- /// The database name for the storage mechanism, if applicable.
- public string Database { get; set; }
- }
-}
diff --git a/src/SMAPI.Web/Framework/ConfigModels/StorageMode.cs b/src/SMAPI.Web/Framework/ConfigModels/StorageMode.cs
deleted file mode 100644
index 4c2ea801..00000000
--- a/src/SMAPI.Web/Framework/ConfigModels/StorageMode.cs
+++ /dev/null
@@ -1,15 +0,0 @@
-namespace StardewModdingAPI.Web.Framework.ConfigModels
-{
- /// Indicates a storage mechanism to use.
- internal enum StorageMode
- {
- /// Store data in a hosted MongoDB instance.
- Mongo,
-
- /// Store data in an in-memory MongoDB instance. This is useful for testing MongoDB storage locally, but will likely fail when deployed since it needs permission to open a local port.
- MongoInMemory,
-
- /// Store data in-memory. This is suitable for local testing or single-instance servers, but will cause issues when distributed across multiple servers.
- InMemory
- }
-}
diff --git a/src/SMAPI.Web/SMAPI.Web.csproj b/src/SMAPI.Web/SMAPI.Web.csproj
index 7ed79ea3..c6c0f774 100644
--- a/src/SMAPI.Web/SMAPI.Web.csproj
+++ b/src/SMAPI.Web/SMAPI.Web.csproj
@@ -15,14 +15,11 @@
-
-
-
diff --git a/src/SMAPI.Web/Startup.cs b/src/SMAPI.Web/Startup.cs
index dee2edc2..586b0c3c 100644
--- a/src/SMAPI.Web/Startup.cs
+++ b/src/SMAPI.Web/Startup.cs
@@ -1,9 +1,7 @@
-using System;
using System.Collections.Generic;
using System.Net;
using Hangfire;
using Hangfire.MemoryStorage;
-using Hangfire.Mongo;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.AspNetCore.Rewrite;
@@ -11,13 +9,9 @@ using Microsoft.AspNetCore.Routing;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Options;
-using Mongo2Go;
-using MongoDB.Bson.Serialization;
-using MongoDB.Driver;
using Newtonsoft.Json;
using StardewModdingAPI.Toolkit.Serialization;
using StardewModdingAPI.Web.Framework;
-using StardewModdingAPI.Web.Framework.Caching;
using StardewModdingAPI.Web.Framework.Caching.Mods;
using StardewModdingAPI.Web.Framework.Caching.Wiki;
using StardewModdingAPI.Web.Framework.Clients.Chucklefish;
@@ -68,13 +62,10 @@ namespace StardewModdingAPI.Web
.Configure(this.Configuration.GetSection("BackgroundServices"))
.Configure(this.Configuration.GetSection("ModCompatibilityList"))
.Configure(this.Configuration.GetSection("ModUpdateCheck"))
- .Configure(this.Configuration.GetSection("Storage"))
.Configure(this.Configuration.GetSection("Site"))
.Configure(options => options.ConstraintMap.Add("semanticVersion", typeof(VersionConstraint)))
.AddLogging()
.AddMemoryCache();
- StorageConfig storageConfig = this.Configuration.GetSection("Storage").Get();
- StorageMode storageMode = storageConfig.Mode;
// init MVC
services
@@ -85,39 +76,8 @@ namespace StardewModdingAPI.Web
.AddRazorPages();
// init storage
- switch (storageMode)
- {
- case StorageMode.InMemory:
- services.AddSingleton(new ModCacheMemoryRepository());
- services.AddSingleton(new WikiCacheMemoryRepository());
- break;
-
- case StorageMode.Mongo:
- case StorageMode.MongoInMemory:
- {
- // local MongoDB instance
- services.AddSingleton(_ => storageMode == StorageMode.MongoInMemory
- ? MongoDbRunner.Start()
- : throw new NotSupportedException($"The in-memory MongoDB runner isn't available in storage mode {storageMode}.")
- );
-
- // MongoDB
- services.AddSingleton(serv =>
- {
- BsonSerializer.RegisterSerializer(new UtcDateTimeOffsetSerializer());
- return new MongoClient(this.GetMongoDbConnectionString(serv, storageConfig))
- .GetDatabase(storageConfig.Database);
- });
-
- // repositories
- services.AddSingleton(serv => new ModCacheMongoRepository(serv.GetRequiredService()));
- services.AddSingleton(serv => new WikiCacheMongoRepository(serv.GetRequiredService()));
- }
- break;
-
- default:
- throw new NotSupportedException($"Unhandled storage mode '{storageMode}'.");
- }
+ services.AddSingleton(new ModCacheMemoryRepository());
+ services.AddSingleton(new WikiCacheMemoryRepository());
// init Hangfire
services
@@ -126,24 +86,8 @@ namespace StardewModdingAPI.Web
config
.SetDataCompatibilityLevel(CompatibilityLevel.Version_170)
.UseSimpleAssemblyNameTypeSerializer()
- .UseRecommendedSerializerSettings();
-
- switch (storageMode)
- {
- case StorageMode.InMemory:
- config.UseMemoryStorage();
- break;
-
- case StorageMode.MongoInMemory:
- case StorageMode.Mongo:
- string connectionString = this.GetMongoDbConnectionString(serv, storageConfig);
- config.UseMongoStorage(MongoClientSettings.FromConnectionString(connectionString), $"{storageConfig.Database}-hangfire", new MongoStorageOptions
- {
- MigrationOptions = new MongoMigrationOptions(MongoMigrationStrategy.Drop),
- CheckConnection = false // error on startup takes down entire process
- });
- break;
- }
+ .UseRecommendedSerializerSettings()
+ .UseMemoryStorage();
});
// init background service
@@ -254,20 +198,6 @@ namespace StardewModdingAPI.Web
settings.NullValueHandling = NullValueHandling.Ignore;
}
- /// Get the MongoDB connection string for the given storage configuration.
- /// The service provider.
- /// The storage configuration
- /// There's no MongoDB instance in the given storage mode.
- private string GetMongoDbConnectionString(IServiceProvider services, StorageConfig storageConfig)
- {
- return storageConfig.Mode switch
- {
- StorageMode.Mongo => storageConfig.ConnectionString,
- StorageMode.MongoInMemory => services.GetRequiredService().ConnectionString,
- _ => throw new NotSupportedException($"There's no MongoDB instance in storage mode {storageConfig.Mode}.")
- };
- }
-
/// Get the redirect rules to apply.
private RewriteOptions GetRedirectRules()
{
diff --git a/src/SMAPI.Web/appsettings.Development.json b/src/SMAPI.Web/appsettings.Development.json
index 41c00e79..3aa69285 100644
--- a/src/SMAPI.Web/appsettings.Development.json
+++ b/src/SMAPI.Web/appsettings.Development.json
@@ -17,12 +17,6 @@
"NexusApiKey": null
},
- "Storage": {
- "Mode": "MongoInMemory",
- "ConnectionString": null,
- "Database": "smapi-edge"
- },
-
"BackgroundServices": {
"Enabled": true
}