using System; using System.Collections.Generic; using System.Linq; namespace StardewModdingAPI.Toolkit.Framework.ModData { /// The parsed mod metadata from SMAPI's internal mod list. public class ModDataRecord { /********* ** Accessors *********/ /// The mod's default display name. public string DisplayName { get; } /// The mod's current unique ID. public string ID { get; } /// The former mod IDs (if any). public string[] FormerIDs { get; } /// Maps local versions to a semantic version for update checks. public IDictionary MapLocalVersions { get; } /// Maps remote versions to a semantic version for update checks. public IDictionary MapRemoteVersions { get; } /// The versioned field data. public ModDataField[] Fields { get; } /********* ** Public methods *********/ /// Construct an instance. /// The mod's default display name. /// The raw data model. internal ModDataRecord(string displayName, ModDataModel model) { this.DisplayName = displayName; this.ID = model.ID; this.FormerIDs = model.GetFormerIDs().ToArray(); this.MapLocalVersions = new Dictionary(model.MapLocalVersions, StringComparer.InvariantCultureIgnoreCase); this.MapRemoteVersions = new Dictionary(model.MapRemoteVersions, StringComparer.InvariantCultureIgnoreCase); this.Fields = model.GetFields().ToArray(); } /// Get whether the mod has (or previously had) the given ID. /// The mod ID. public bool HasID(string id) { // try main ID if (this.ID.Equals(id, StringComparison.InvariantCultureIgnoreCase)) return true; // try former IDs foreach (string formerID in this.FormerIDs) { if (formerID.Equals(id, StringComparison.InvariantCultureIgnoreCase)) return true; } return false; } /// Get a semantic local version for update checks. /// The remote version to normalise. public ISemanticVersion GetLocalVersionForUpdateChecks(ISemanticVersion version) { return this.MapLocalVersions != null && this.MapLocalVersions.TryGetValue(version.ToString(), out string newVersion) ? new SemanticVersion(newVersion) : version; } /// Get a semantic remote version for update checks. /// The remote version to normalise. public string GetRemoteVersionForUpdateChecks(string version) { // normalise version if possible if (SemanticVersion.TryParse(version, out ISemanticVersion parsed)) version = parsed.ToString(); // fetch remote version return this.MapRemoteVersions != null && this.MapRemoteVersions.TryGetValue(version, out string newVersion) ? newVersion : version; } /// Get the possible mod IDs. public IEnumerable GetIDs() { return this.FormerIDs .Concat(new[] { this.ID }) .Where(p => !string.IsNullOrWhiteSpace(p)) .Select(p => p.Trim()) .Distinct(); } /// Get a parsed representation of the which match a given manifest. /// The manifest to match. public ModDataRecordVersionedFields GetVersionedFields(IManifest manifest) { ModDataRecordVersionedFields parsed = new ModDataRecordVersionedFields { DisplayName = this.DisplayName, DataRecord = this }; foreach (ModDataField field in this.Fields.Where(field => field.IsMatch(manifest))) { switch (field.Key) { // update key case ModDataFieldKey.UpdateKey: parsed.UpdateKey = field.Value; break; // alternative URL case ModDataFieldKey.AlternativeUrl: parsed.AlternativeUrl = field.Value; break; // status case ModDataFieldKey.Status: parsed.Status = (ModStatus)Enum.Parse(typeof(ModStatus), field.Value, ignoreCase: true); parsed.StatusUpperVersion = field.UpperVersion; break; // status reason phrase case ModDataFieldKey.StatusReasonPhrase: parsed.StatusReasonPhrase = field.Value; break; } } return parsed; } } }