|
|
- using IPA.Config;
- using IPA.Loader.Features;
- using IPA.Logging;
- using IPA.Utilities;
- using Mono.Cecil;
- using Newtonsoft.Json;
- using System;
- using System.Collections.Generic;
- using System.IO;
- using System.Linq;
- using System.Reflection;
- using System.Threading.Tasks;
- using Version = SemVer.Version;
-
- namespace IPA.Loader
- {
- /// <summary>
- /// A type to manage the loading of plugins.
- /// </summary>
- public class PluginLoader
- {
- internal static Task LoadTask() => Task.Run(() =>
- {
- LoadMetadata();
- Resolve();
- ComputeLoadOrder();
- });
-
- /// <summary>
- /// A class which describes
- /// </summary>
- public class PluginMetadata
- {
- /// <summary>
- /// The assembly the plugin was loaded from.
- /// </summary>
- public Assembly Assembly { get; internal set; }
-
- /// <summary>
- /// The TypeDefinition for the main type of the plugin.
- /// </summary>
- public TypeDefinition PluginType { get; internal set; }
-
- /// <summary>
- /// The human readable name of the plugin.
- /// </summary>
- public string Name { get; internal set; }
-
- /// <summary>
- /// The ModSaber ID of the plugin, or null if it doesn't have one.
- /// </summary>
- public string Id { get; internal set; }
-
- /// <summary>
- /// The version of the plugin.
- /// </summary>
- public Version Version { get; internal set; }
-
- /// <summary>
- /// The file the plugin was loaded from.
- /// </summary>
- public FileInfo File { get; internal set; }
-
- // ReSharper disable once UnusedAutoPropertyAccessor.Global
- /// <summary>
- /// The features this plugin requests.
- /// </summary>
- public IReadOnlyList<Feature> Features => InternalFeatures;
-
- internal List<Feature> InternalFeatures = new List<Feature>();
-
- private PluginManifest manifest;
-
- internal PluginManifest Manifest
- {
- get => manifest;
- set
- {
- manifest = value;
- Name = value.Name;
- Version = value.Version;
- Id = value.Id;
- }
- }
-
- /// <inheritdoc />
- public override string ToString() => $"{Name}({Id}@{Version})({PluginType?.FullName}) from '{LoneFunctions.GetRelativePath(File?.FullName, BeatSaber.InstallPath)}'";
- }
-
- /// <summary>
- /// A container object for all the data relating to a plugin.
- /// </summary>
- public class PluginInfo
- {
- internal IBeatSaberPlugin Plugin { get; set; }
-
- /// <summary>
- /// Metadata for the plugin.
- /// </summary>
- public PluginMetadata Metadata { get; internal set; } = new PluginMetadata();
- }
-
- internal static List<PluginMetadata> PluginsMetadata = new List<PluginMetadata>();
-
- internal static void LoadMetadata()
- {
- string[] plugins = Directory.GetFiles(BeatSaber.PluginsPath, "*.dll");
-
- try
- {
- var selfMeta = new PluginMetadata
- {
- Assembly = Assembly.GetExecutingAssembly(),
- File = new FileInfo(Path.Combine(BeatSaber.InstallPath, "IPA.exe")),
- PluginType = null
- };
-
- string manifest;
- using (var manifestReader =
- new StreamReader(
- selfMeta.Assembly.GetManifestResourceStream(typeof(PluginLoader), "manifest.json") ??
- throw new InvalidOperationException()))
- manifest = manifestReader.ReadToEnd();
-
- selfMeta.Manifest = JsonConvert.DeserializeObject<PluginManifest>(manifest);
-
- PluginsMetadata.Add(selfMeta);
- }
- catch (Exception e)
- {
- Logger.loader.Critical("Error loading own manifest");
- Logger.loader.Critical(e);
- }
-
- foreach (var plugin in plugins)
- {
- try
- {
- var metadata = new PluginMetadata
- {
- File = new FileInfo(Path.Combine(BeatSaber.PluginsPath, plugin))
- };
-
- var pluginModule = AssemblyDefinition.ReadAssembly(plugin, new ReaderParameters
- {
- ReadingMode = ReadingMode.Immediate,
- ReadWrite = false,
- AssemblyResolver = new CecilLibLoader()
- }).MainModule;
-
- var iBeatSaberPlugin = pluginModule.ImportReference(typeof(IBeatSaberPlugin));
- foreach (var type in pluginModule.Types)
- {
- foreach (var inter in type.Interfaces)
- {
- var ifType = inter.InterfaceType;
-
- if (iBeatSaberPlugin.FullName == ifType.FullName)
- {
- metadata.PluginType = type;
- break;
- }
- }
-
- if (metadata.PluginType != null) break;
- }
-
- if (metadata.PluginType == null)
- {
- Logger.loader.Warn($"Could not find plugin type for {Path.GetFileName(plugin)}");
- continue;
- }
-
- foreach (var resource in pluginModule.Resources)
- {
- if (!(resource is EmbeddedResource embedded) ||
- embedded.Name != $"{metadata.PluginType.Namespace}.manifest.json") continue;
-
- string manifest;
- using (var manifestReader = new StreamReader(embedded.GetResourceStream()))
- manifest = manifestReader.ReadToEnd();
-
- metadata.Manifest = JsonConvert.DeserializeObject<PluginManifest>(manifest);
- break;
- }
-
- Logger.loader.Debug($"Adding info for {Path.GetFileName(plugin)}");
- PluginsMetadata.Add(metadata);
- }
- catch (Exception e)
- {
- Logger.loader.Error($"Could not load data for plugin {Path.GetFileName(plugin)}");
- Logger.loader.Error(e);
- }
- }
- }
-
- internal static void Resolve()
- { // resolves duplicates and conflicts, etc
- PluginsMetadata.Sort((a, b) => a.Version.CompareTo(b.Version));
-
- var ids = new HashSet<string>();
- var ignore = new HashSet<PluginMetadata>();
- var resolved = new List<PluginMetadata>(PluginsMetadata.Count);
- foreach (var meta in PluginsMetadata)
- {
- if (meta.Id != null)
- {
- if (ids.Contains(meta.Id))
- {
- Logger.loader.Warn($"Found duplicates of {meta.Id}, using newest");
- ignore.Add(meta);
- continue; // because of sorted order, hightest order will always be the first one
- }
-
- bool processedLater = false;
- foreach (var meta2 in PluginsMetadata)
- {
- if (ignore.Contains(meta2)) continue;
- if (meta == meta2)
- {
- processedLater = true;
- continue;
- }
-
- if (!meta2.Manifest.Conflicts.ContainsKey(meta.Id)) continue;
-
- var range = meta2.Manifest.Conflicts[meta.Id];
- if (!range.IsSatisfied(meta.Version)) continue;
-
- Logger.loader.Warn($"{meta.Id}@{meta.Version} conflicts with {meta2.Name}");
-
- if (processedLater)
- {
- Logger.loader.Warn($"Ignoring {meta2.Name}");
- ignore.Add(meta2);
- }
- else
- {
- Logger.loader.Warn($"Ignoring {meta.Name}");
- ignore.Add(meta);
- break;
- }
- }
- }
-
- if (ignore.Contains(meta)) continue;
- if (meta.Id != null) ids.Add(meta.Id);
-
- resolved.Add(meta);
- }
-
- PluginsMetadata = resolved;
- }
-
- internal static void ComputeLoadOrder()
- {
- PluginsMetadata.Sort((a, b) =>
- {
- if (a.Id == b.Id) return 0;
- if (a.Id != null)
- {
- if (b.Manifest.Dependencies.ContainsKey(a.Id) || b.Manifest.LoadAfter.Contains(a.Id)) return -1;
- if (b.Manifest.LoadBefore.Contains(a.Id)) return 1;
- }
- if (b.Id != null)
- {
- if (a.Manifest.Dependencies.ContainsKey(b.Id) || a.Manifest.LoadAfter.Contains(b.Id)) return 1;
- if (a.Manifest.LoadBefore.Contains(b.Id)) return -1;
- }
-
- return 0;
- });
-
- var metadata = new List<PluginMetadata>();
- var pluginsToLoad = new Dictionary<string, Version>();
- foreach (var meta in PluginsMetadata)
- {
- bool load = true;
- foreach (var dep in meta.Manifest.Dependencies)
- {
- if (pluginsToLoad.ContainsKey(dep.Key) && dep.Value.IsSatisfied(pluginsToLoad[dep.Key])) continue;
-
- load = false;
- Logger.loader.Warn($"{meta.Name} is missing dependency {dep.Key}@{dep.Value}");
- }
-
- if (load)
- {
- metadata.Add(meta);
- if (meta.Id != null)
- pluginsToLoad.Add(meta.Id, meta.Version);
- }
- }
-
- PluginsMetadata = metadata;
- }
-
- internal static void Load(PluginMetadata meta)
- {
- if (meta.Assembly == null)
- meta.Assembly = Assembly.LoadFrom(meta.File.FullName);
- }
-
- internal static PluginInfo InitPlugin(PluginMetadata meta)
- {
- if (meta.PluginType == null)
- return new PluginInfo()
- {
- Metadata = meta,
- Plugin = null
- };
-
- var info = new PluginInfo();
-
- try
- {
- Load(meta);
-
- var type = meta.Assembly.GetType(meta.PluginType.FullName);
- var instance = (IBeatSaberPlugin)Activator.CreateInstance(type);
-
- info.Metadata = meta;
- info.Plugin = instance;
-
- {
- var init = type.GetMethod("Init", BindingFlags.Instance | BindingFlags.Public);
- if (init != null)
- {
- var initArgs = new List<object>();
- var initParams = init.GetParameters();
-
- Logger modLogger = null;
- IModPrefs modPrefs = null;
- IConfigProvider cfgProvider = null;
-
- foreach (var param in initParams)
- {
- var paramType = param.ParameterType;
- if (paramType.IsAssignableFrom(typeof(Logger)))
- {
- if (modLogger == null) modLogger = new StandardLogger(meta.Name);
- initArgs.Add(modLogger);
- }
- else if (paramType.IsAssignableFrom(typeof(IModPrefs)))
- {
- if (modPrefs == null) modPrefs = new ModPrefs(instance);
- initArgs.Add(modPrefs);
- }
- else if (paramType.IsAssignableFrom(typeof(IConfigProvider)))
- {
- if (cfgProvider == null)
- {
- cfgProvider = Config.Config.GetProviderFor(Path.Combine("UserData", $"{meta.Name}"), param);
- cfgProvider.Load();
- }
- initArgs.Add(cfgProvider);
- }
- else
- initArgs.Add(paramType.GetDefault());
- }
-
- init.Invoke(instance, initArgs.ToArray());
- }
- }
- }
- catch (AmbiguousMatchException)
- {
- Logger.loader.Error($"Only one Init allowed per plugin (ambiguous match in {meta.Name})");
- return null;
- }
- catch (Exception e)
- {
- Logger.loader.Error($"Could not init plugin {meta.Name}: {e}");
- return null;
- }
-
- return info;
- }
-
- internal static List<PluginInfo> LoadPlugins() => PluginsMetadata.Select(InitPlugin).Where(p => p != null).ToList();
- }
- }
|