You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

616 lines
23 KiB

  1. using IPA.Loader.Features;
  2. using IPA.Logging;
  3. using IPA.Utilities;
  4. using Mono.Cecil;
  5. using Newtonsoft.Json;
  6. using System;
  7. using System.Collections.Generic;
  8. using System.IO;
  9. using System.Linq;
  10. using System.Reflection;
  11. using System.Text.RegularExpressions;
  12. using System.Threading.Tasks;
  13. using Version = SemVer.Version;
  14. namespace IPA.Loader
  15. {
  16. /// <summary>
  17. /// A type to manage the loading of plugins.
  18. /// </summary>
  19. public class PluginLoader
  20. {
  21. internal static Task LoadTask() => Task.Run(() =>
  22. {
  23. LoadMetadata();
  24. Resolve();
  25. FilterDisabled();
  26. ComputeLoadOrder();
  27. });
  28. /// <summary>
  29. /// A class which describes a loaded plugin.
  30. /// </summary>
  31. public class PluginMetadata
  32. {
  33. /// <summary>
  34. /// The assembly the plugin was loaded from.
  35. /// </summary>
  36. public Assembly Assembly { get; internal set; }
  37. /// <summary>
  38. /// The TypeDefinition for the main type of the plugin.
  39. /// </summary>
  40. public TypeDefinition PluginType { get; internal set; }
  41. /// <summary>
  42. /// The human readable name of the plugin.
  43. /// </summary>
  44. public string Name { get; internal set; }
  45. /// <summary>
  46. /// The BeatMods ID of the plugin, or null if it doesn't have one.
  47. /// </summary>
  48. public string Id { get; internal set; }
  49. /// <summary>
  50. /// The version of the plugin.
  51. /// </summary>
  52. public Version Version { get; internal set; }
  53. /// <summary>
  54. /// The file the plugin was loaded from.
  55. /// </summary>
  56. public FileInfo File { get; internal set; }
  57. // ReSharper disable once UnusedAutoPropertyAccessor.Global
  58. /// <summary>
  59. /// The features this plugin requests.
  60. /// </summary>
  61. public IReadOnlyList<Feature> Features => InternalFeatures;
  62. internal readonly List<Feature> InternalFeatures = new List<Feature>();
  63. internal bool IsSelf;
  64. internal bool IsBare;
  65. private PluginManifest manifest;
  66. internal HashSet<PluginMetadata> Dependencies { get; } = new HashSet<PluginMetadata>();
  67. internal PluginManifest Manifest
  68. {
  69. get => manifest;
  70. set
  71. {
  72. manifest = value;
  73. Name = value.Name;
  74. Version = value.Version;
  75. Id = value.Id;
  76. }
  77. }
  78. /// <inheritdoc />
  79. public override string ToString() => $"{Name}({Id}@{Version})({PluginType?.FullName}) from '{Utils.GetRelativePath(File?.FullName, BeatSaber.InstallPath)}'";
  80. }
  81. /// <summary>
  82. /// A container object for all the data relating to a plugin.
  83. /// </summary>
  84. public class PluginInfo
  85. {
  86. internal IBeatSaberPlugin Plugin { get; set; }
  87. /// <summary>
  88. /// Metadata for the plugin.
  89. /// </summary>
  90. public PluginMetadata Metadata { get; internal set; } = new PluginMetadata();
  91. }
  92. internal static List<PluginMetadata> PluginsMetadata = new List<PluginMetadata>();
  93. private static readonly Regex embeddedTextDescriptionPattern = new Regex(@"#!\[(.+)\]", RegexOptions.Compiled | RegexOptions.Singleline);
  94. internal static void LoadMetadata()
  95. {
  96. string[] plugins = Directory.GetFiles(BeatSaber.PluginsPath, "*.dll");
  97. try
  98. {
  99. var selfMeta = new PluginMetadata
  100. {
  101. Assembly = Assembly.GetExecutingAssembly(),
  102. File = new FileInfo(Path.Combine(BeatSaber.InstallPath, "IPA.exe")),
  103. PluginType = null,
  104. IsSelf = true
  105. };
  106. string manifest;
  107. using (var manifestReader =
  108. new StreamReader(
  109. selfMeta.Assembly.GetManifestResourceStream(typeof(PluginLoader), "manifest.json") ??
  110. throw new InvalidOperationException()))
  111. manifest = manifestReader.ReadToEnd();
  112. selfMeta.Manifest = JsonConvert.DeserializeObject<PluginManifest>(manifest);
  113. PluginsMetadata.Add(selfMeta);
  114. }
  115. catch (Exception e)
  116. {
  117. Logger.loader.Critical("Error loading own manifest");
  118. Logger.loader.Critical(e);
  119. }
  120. foreach (var plugin in plugins)
  121. {
  122. try
  123. {
  124. var metadata = new PluginMetadata
  125. {
  126. File = new FileInfo(Path.Combine(BeatSaber.PluginsPath, plugin)),
  127. IsSelf = false
  128. };
  129. var pluginModule = AssemblyDefinition.ReadAssembly(plugin, new ReaderParameters
  130. {
  131. ReadingMode = ReadingMode.Immediate,
  132. ReadWrite = false,
  133. AssemblyResolver = new CecilLibLoader()
  134. }).MainModule;
  135. var iBeatSaberPlugin = pluginModule.ImportReference(typeof(IBeatSaberPlugin));
  136. foreach (var type in pluginModule.Types)
  137. {
  138. foreach (var inter in type.Interfaces)
  139. {
  140. var ifType = inter.InterfaceType;
  141. if (iBeatSaberPlugin.FullName == ifType.FullName)
  142. {
  143. metadata.PluginType = type;
  144. break;
  145. }
  146. }
  147. if (metadata.PluginType != null) break;
  148. }
  149. if (metadata.PluginType == null)
  150. {
  151. Logger.loader.Notice(
  152. #if DIRE_LOADER_WARNINGS
  153. $"Could not find plugin type for {Path.GetFileName(plugin)}"
  154. #else
  155. $"New plugin type not present in {Path.GetFileName(plugin)}; maybe an old plugin?"
  156. #endif
  157. );
  158. continue;
  159. }
  160. foreach (var resource in pluginModule.Resources)
  161. {
  162. if (!(resource is EmbeddedResource embedded) ||
  163. embedded.Name != $"{metadata.PluginType.Namespace}.manifest.json") continue;
  164. string manifest;
  165. using (var manifestReader = new StreamReader(embedded.GetResourceStream()))
  166. manifest = manifestReader.ReadToEnd();
  167. metadata.Manifest = JsonConvert.DeserializeObject<PluginManifest>(manifest);
  168. break;
  169. }
  170. if (metadata.Manifest == null)
  171. {
  172. Logger.loader.Error("Could not find manifest.json in namespace " +
  173. $"{metadata.PluginType.Namespace} for {Path.GetFileName(plugin)}");
  174. continue;
  175. }
  176. Logger.loader.Debug($"Adding info for {Path.GetFileName(plugin)}");
  177. PluginsMetadata.Add(metadata);
  178. }
  179. catch (Exception e)
  180. {
  181. Logger.loader.Error($"Could not load data for plugin {Path.GetFileName(plugin)}");
  182. Logger.loader.Error(e);
  183. }
  184. foreach (var meta in PluginsMetadata)
  185. { // process description include
  186. var lines = meta.Manifest.Description.Split('\n');
  187. var m = embeddedTextDescriptionPattern.Match(lines[0]);
  188. if (m.Success)
  189. {
  190. if (meta.IsBare)
  191. {
  192. Logger.loader.Warn($"Bare manifest cannot specify description file");
  193. meta.Manifest.Description = string.Join("\n", lines.Skip(1)); // ignore first line
  194. continue;
  195. }
  196. var name = m.Groups[1].Value;
  197. string description;
  198. if (!meta.IsSelf)
  199. {
  200. var resc = meta.PluginType.Module.Resources.Select(r => r as EmbeddedResource)
  201. .Where(r => r != null)
  202. .FirstOrDefault(r => r.Name == name);
  203. if (resc == null)
  204. {
  205. Logger.loader.Warn($"Could not find description file for plugin {meta.Name} ({name}); ignoring include");
  206. meta.Manifest.Description = string.Join("\n", lines.Skip(1)); // ignore first line
  207. continue;
  208. }
  209. using (var reader = new StreamReader(resc.GetResourceStream()))
  210. description = reader.ReadToEnd();
  211. }
  212. else
  213. {
  214. using (var descriptionReader =
  215. new StreamReader(
  216. meta.Assembly.GetManifestResourceStream(name) ??
  217. throw new InvalidOperationException()))
  218. description = descriptionReader.ReadToEnd();
  219. }
  220. meta.Manifest.Description = description;
  221. }
  222. }
  223. }
  224. IEnumerable<string> bareManifests = Directory.GetFiles(BeatSaber.PluginsPath, "*.json");
  225. bareManifests = bareManifests.Concat(Directory.GetFiles(BeatSaber.PluginsPath, "*.manifest"));
  226. foreach (var manifest in bareManifests)
  227. {
  228. try
  229. {
  230. var metadata = new PluginMetadata
  231. {
  232. File = new FileInfo(Path.Combine(BeatSaber.PluginsPath, manifest)),
  233. IsSelf = false,
  234. IsBare = true,
  235. };
  236. metadata.Manifest = JsonConvert.DeserializeObject<PluginManifest>(File.ReadAllText(manifest));
  237. Logger.loader.Debug($"Adding info for bare manifest {Path.GetFileName(manifest)}");
  238. PluginsMetadata.Add(metadata);
  239. }
  240. catch (Exception e)
  241. {
  242. Logger.loader.Error($"Could not load data for bare manifest {Path.GetFileName(manifest)}");
  243. Logger.loader.Error(e);
  244. }
  245. }
  246. }
  247. // keep track of these for the updater; it should still be able to update mods not loaded
  248. // TODO: add ignore reason
  249. internal static HashSet<PluginMetadata> ignoredPlugins = new HashSet<PluginMetadata>();
  250. internal static void Resolve()
  251. { // resolves duplicates and conflicts, etc
  252. PluginsMetadata.Sort((a, b) => b.Version.CompareTo(a.Version));
  253. var ids = new HashSet<string>();
  254. var ignore = new HashSet<PluginMetadata>();
  255. var resolved = new List<PluginMetadata>(PluginsMetadata.Count);
  256. foreach (var meta in PluginsMetadata)
  257. {
  258. if (meta.Id != null)
  259. {
  260. if (ids.Contains(meta.Id))
  261. {
  262. Logger.loader.Warn($"Found duplicates of {meta.Id}, using newest");
  263. ignore.Add(meta);
  264. ignoredPlugins.Add(meta);
  265. continue; // because of sorted order, hightest order will always be the first one
  266. }
  267. bool processedLater = false;
  268. foreach (var meta2 in PluginsMetadata)
  269. {
  270. if (ignore.Contains(meta2)) continue;
  271. if (meta == meta2)
  272. {
  273. processedLater = true;
  274. continue;
  275. }
  276. if (!meta2.Manifest.Conflicts.ContainsKey(meta.Id)) continue;
  277. var range = meta2.Manifest.Conflicts[meta.Id];
  278. if (!range.IsSatisfied(meta.Version)) continue;
  279. Logger.loader.Warn($"{meta.Id}@{meta.Version} conflicts with {meta2.Name}");
  280. if (processedLater)
  281. {
  282. Logger.loader.Warn($"Ignoring {meta2.Name}");
  283. ignore.Add(meta2);
  284. }
  285. else
  286. {
  287. Logger.loader.Warn($"Ignoring {meta.Name}");
  288. ignore.Add(meta);
  289. break;
  290. }
  291. }
  292. }
  293. if (ignore.Contains(meta))
  294. {
  295. ignoredPlugins.Add(meta);
  296. continue;
  297. }
  298. if (meta.Id != null)
  299. ids.Add(meta.Id);
  300. resolved.Add(meta);
  301. }
  302. PluginsMetadata = resolved;
  303. }
  304. private static void FilterDisabled()
  305. { // TODO: move disabled to a seperate list from ignored
  306. var enabled = new List<PluginMetadata>(PluginsMetadata.Count);
  307. var disabled = DisabledConfig.Ref.Value.DisabledModIds;
  308. foreach (var meta in PluginsMetadata)
  309. {
  310. if (disabled.Contains(meta.Id ?? meta.Name))
  311. ignoredPlugins.Add(meta);
  312. else
  313. enabled.Add(meta);
  314. }
  315. PluginsMetadata = enabled;
  316. }
  317. internal static void ComputeLoadOrder()
  318. {
  319. #if DEBUG
  320. Logger.loader.Debug(string.Join(", ", PluginsMetadata.Select(p => p.ToString())));
  321. #endif
  322. bool InsertInto(HashSet<PluginMetadata> root, PluginMetadata meta, bool isRoot = false)
  323. { // this is slow, and hella recursive
  324. bool inserted = false;
  325. foreach (var sr in root)
  326. {
  327. inserted = inserted || InsertInto(sr.Dependencies, meta);
  328. if (meta.Id != null)
  329. if (sr.Manifest.Dependencies.ContainsKey(meta.Id) || sr.Manifest.LoadAfter.Contains(meta.Id))
  330. inserted = inserted || sr.Dependencies.Add(meta);
  331. if (sr.Id != null)
  332. if (meta.Manifest.LoadBefore.Contains(sr.Id))
  333. inserted = inserted || sr.Dependencies.Add(meta);
  334. }
  335. if (isRoot)
  336. {
  337. foreach (var sr in root)
  338. {
  339. InsertInto(meta.Dependencies, sr);
  340. if (sr.Id != null)
  341. if (meta.Manifest.Dependencies.ContainsKey(sr.Id) || meta.Manifest.LoadAfter.Contains(sr.Id))
  342. meta.Dependencies.Add(sr);
  343. if (meta.Id != null)
  344. if (sr.Manifest.LoadBefore.Contains(meta.Id))
  345. meta.Dependencies.Add(sr);
  346. }
  347. root.Add(meta);
  348. }
  349. return inserted;
  350. }
  351. var pluginTree = new HashSet<PluginMetadata>();
  352. foreach (var meta in PluginsMetadata)
  353. InsertInto(pluginTree, meta, true);
  354. void DeTree(List<PluginMetadata> into, HashSet<PluginMetadata> tree)
  355. {
  356. foreach (var st in tree)
  357. if (!into.Contains(st))
  358. {
  359. DeTree(into, st.Dependencies);
  360. into.Add(st);
  361. }
  362. }
  363. var deTreed = new List<PluginMetadata>();
  364. DeTree(deTreed, pluginTree);
  365. #if DEBUG
  366. Logger.loader.Debug(string.Join(", ", deTreed.Select(p => p.ToString())));
  367. #endif
  368. var metadata = new List<PluginMetadata>();
  369. var pluginsToLoad = new Dictionary<string, Version>();
  370. foreach (var meta in deTreed)
  371. {
  372. bool load = true;
  373. foreach (var dep in meta.Manifest.Dependencies)
  374. {
  375. #if DEBUG
  376. Logger.loader.Debug($"Looking for dependency {dep.Key} with version range {dep.Value.Intersect(new SemVer.Range("*.*.*"))}");
  377. #endif
  378. if (pluginsToLoad.ContainsKey(dep.Key) && dep.Value.IsSatisfied(pluginsToLoad[dep.Key]))
  379. continue;
  380. load = false;
  381. Logger.loader.Warn($"{meta.Name} is missing dependency {dep.Key}@{dep.Value}");
  382. }
  383. if (load)
  384. {
  385. metadata.Add(meta);
  386. if (meta.Id != null)
  387. pluginsToLoad.Add(meta.Id, meta.Version);
  388. }
  389. else
  390. ignoredPlugins.Add(meta);
  391. }
  392. PluginsMetadata = metadata;
  393. }
  394. internal static void InitFeatures()
  395. {
  396. var parsedFeatures = PluginsMetadata.Select(m =>
  397. Tuple.Create(m,
  398. m.Manifest.Features.Select(f =>
  399. Tuple.Create(f, Ref.Create<Feature.FeatureParse?>(null))
  400. ).ToList()
  401. )
  402. ).ToList();
  403. while (DefineFeature.NewFeature)
  404. {
  405. DefineFeature.NewFeature = false;
  406. foreach (var plugin in parsedFeatures)
  407. for (var i = 0; i < plugin.Item2.Count; i++)
  408. {
  409. var feature = plugin.Item2[i];
  410. var success = Feature.TryParseFeature(feature.Item1, plugin.Item1, out var featureObj,
  411. out var exception, out var valid, out var parsed, feature.Item2.Value);
  412. if (!success && !valid && featureObj == null && exception == null) // no feature of type found
  413. feature.Item2.Value = parsed;
  414. else if (success)
  415. {
  416. if (valid && featureObj.StoreOnPlugin)
  417. plugin.Item1.InternalFeatures.Add(featureObj);
  418. else if (!valid)
  419. Logger.features.Warn(
  420. $"Feature not valid on {plugin.Item1.Name}: {featureObj.InvalidMessage}");
  421. plugin.Item2.RemoveAt(i--);
  422. }
  423. else
  424. {
  425. Logger.features.Error($"Error parsing feature definition on {plugin.Item1.Name}");
  426. Logger.features.Error(exception);
  427. plugin.Item2.RemoveAt(i--);
  428. }
  429. }
  430. foreach (var plugin in PluginsMetadata)
  431. foreach (var feature in plugin.Features)
  432. feature.Evaluate();
  433. }
  434. foreach (var plugin in parsedFeatures)
  435. {
  436. if (plugin.Item2.Count <= 0) continue;
  437. Logger.features.Warn($"On plugin {plugin.Item1.Name}:");
  438. foreach (var feature in plugin.Item2)
  439. Logger.features.Warn($" Feature not found with name {feature.Item1}");
  440. }
  441. }
  442. internal static void ReleaseAll()
  443. {
  444. ignoredPlugins = new HashSet<PluginMetadata>();
  445. PluginsMetadata = new List<PluginMetadata>();
  446. Feature.Reset();
  447. GC.Collect();
  448. }
  449. internal static void Load(PluginMetadata meta)
  450. {
  451. if (meta.Assembly == null && meta.PluginType != null)
  452. meta.Assembly = Assembly.LoadFrom(meta.File.FullName);
  453. }
  454. internal static PluginInfo InitPlugin(PluginMetadata meta)
  455. {
  456. if (meta.PluginType == null)
  457. return new PluginInfo()
  458. {
  459. Metadata = meta,
  460. Plugin = null
  461. };
  462. var info = new PluginInfo();
  463. if (meta.Manifest.GameVersion != BeatSaber.GameVersion)
  464. Logger.loader.Warn($"Mod {meta.Name} developed for game version {meta.Manifest.GameVersion}, so it may not work properly.");
  465. try
  466. {
  467. Load(meta);
  468. Feature denyingFeature = null;
  469. if (!meta.Features.All(f => (denyingFeature = f).BeforeLoad(meta)))
  470. {
  471. Logger.loader.Warn(
  472. $"Feature {denyingFeature?.GetType()} denied plugin {meta.Name} from loading! {denyingFeature?.InvalidMessage}");
  473. ignoredPlugins.Add(meta);
  474. return null;
  475. }
  476. var type = meta.Assembly.GetType(meta.PluginType.FullName);
  477. var instance = (IBeatSaberPlugin)Activator.CreateInstance(type);
  478. info.Metadata = meta;
  479. info.Plugin = instance;
  480. var init = type.GetMethod("Init", BindingFlags.Instance | BindingFlags.Public);
  481. if (init != null)
  482. {
  483. denyingFeature = null;
  484. if (!meta.Features.All(f => (denyingFeature = f).BeforeInit(info)))
  485. {
  486. Logger.loader.Warn(
  487. $"Feature {denyingFeature?.GetType()} denied plugin {meta.Name} from initializing! {denyingFeature?.InvalidMessage}");
  488. ignoredPlugins.Add(meta);
  489. return null;
  490. }
  491. PluginInitInjector.Inject(init, info);
  492. }
  493. foreach (var feature in meta.Features)
  494. try
  495. {
  496. feature.AfterInit(info);
  497. }
  498. catch (Exception e)
  499. {
  500. Logger.loader.Critical($"Feature errored in {nameof(Feature.AfterInit)}: {e}");
  501. }
  502. }
  503. catch (AmbiguousMatchException)
  504. {
  505. Logger.loader.Error($"Only one Init allowed per plugin (ambiguous match in {meta.Name})");
  506. // not adding to ignoredPlugins here because this should only happen in a development context
  507. // if someone fucks this up on release thats on them
  508. return null;
  509. }
  510. catch (Exception e)
  511. {
  512. Logger.loader.Error($"Could not init plugin {meta.Name}: {e}");
  513. ignoredPlugins.Add(meta);
  514. return null;
  515. }
  516. return info;
  517. }
  518. internal static List<PluginInfo> LoadPlugins()
  519. {
  520. InitFeatures();
  521. return PluginsMetadata.Select(InitPlugin).Where(p => p != null).ToList();
  522. }
  523. }
  524. }