TypeLoader.cs 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Linq;
  5. using System.Reflection;
  6. using System.Text;
  7. using BepInEx.Configuration;
  8. using BepInEx.Logging;
  9. using Mono.Cecil;
  10. namespace BepInEx.Bootstrap
  11. {
  12. /// <summary>
  13. /// A cacheable metadata item. Can be used with <see cref="TypeLoader.LoadAssemblyCache{T}"/> and <see cref="TypeLoader.SaveAssemblyCache{T}"/> to cache plugin metadata.
  14. /// </summary>
  15. public interface ICacheable
  16. {
  17. /// <summary>
  18. /// Serialize the object into a binary format.
  19. /// </summary>
  20. /// <param name="bw"></param>
  21. void Save(BinaryWriter bw);
  22. /// <summary>
  23. /// Loads the object from binary format.
  24. /// </summary>
  25. /// <param name="br"></param>
  26. void Load(BinaryReader br);
  27. }
  28. /// <summary>
  29. /// A cached assembly.
  30. /// </summary>
  31. /// <typeparam name="T"></typeparam>
  32. public class CachedAssembly<T> where T : ICacheable
  33. {
  34. /// <summary>
  35. /// List of cached items inside the assembly.
  36. /// </summary>
  37. public List<T> CacheItems { get; set; }
  38. /// <summary>
  39. /// Timestamp of the assembly. Used to check the age of the cache.
  40. /// </summary>
  41. public long Timestamp { get; set; }
  42. }
  43. /// <summary>
  44. /// Provides methods for loading specified types from an assembly.
  45. /// </summary>
  46. public static class TypeLoader
  47. {
  48. /// <summary>
  49. /// Default assembly resolved used by the <see cref="TypeLoader"/>
  50. /// </summary>
  51. public static readonly DefaultAssemblyResolver CecilResolver;
  52. /// <summary>
  53. /// Default reader parameters used by <see cref="TypeLoader"/>
  54. /// </summary>
  55. public static readonly ReaderParameters ReaderParameters;
  56. public static HashSet<string> SearchDirectories = new HashSet<string>();
  57. static TypeLoader()
  58. {
  59. CecilResolver = new DefaultAssemblyResolver();
  60. ReaderParameters = new ReaderParameters { AssemblyResolver = CecilResolver };
  61. CecilResolver.ResolveFailure += CecilResolveOnFailure;
  62. }
  63. public static AssemblyDefinition CecilResolveOnFailure(object sender, AssemblyNameReference reference)
  64. {
  65. if (!Utility.TryParseAssemblyName(reference.FullName, out var name))
  66. return null;
  67. if (Utility.TryResolveDllAssembly(name, Paths.BepInExAssemblyDirectory, ReaderParameters, out var assembly) ||
  68. Utility.TryResolveDllAssembly(name, Paths.PluginPath, ReaderParameters, out assembly))
  69. return assembly;
  70. foreach (var dir in SearchDirectories)
  71. {
  72. if (Utility.TryResolveDllAssembly(name, dir, ReaderParameters, out assembly))
  73. return assembly;
  74. }
  75. return AssemblyResolve?.Invoke(sender, reference);
  76. }
  77. /// <summary>
  78. /// Event fired when <see cref="TypeLoader"/> fails to resolve a type during type loading.
  79. /// </summary>
  80. public static event AssemblyResolveEventHandler AssemblyResolve;
  81. /// <summary>
  82. /// Looks up assemblies in the given directory and locates all types that can be loaded and collects their metadata.
  83. /// </summary>
  84. /// <typeparam name="T">The specific base type to search for.</typeparam>
  85. /// <param name="directory">The directory to search for assemblies.</param>
  86. /// <param name="typeSelector">A function to check if a type should be selected and to build the type metadata.</param>
  87. /// <param name="assemblyFilter">A filter function to quickly determine if the assembly can be loaded.</param>
  88. /// <param name="cacheName">The name of the cache to get cached types from.</param>
  89. /// <returns>A dictionary of all assemblies in the directory and the list of type metadatas of types that match the selector.</returns>
  90. public static Dictionary<string, List<T>> FindPluginTypes<T>(string directory, Func<TypeDefinition, string, T> typeSelector, Func<AssemblyDefinition, bool> assemblyFilter = null, string cacheName = null) where T : ICacheable, new()
  91. {
  92. var result = new Dictionary<string, List<T>>();
  93. Dictionary<string, CachedAssembly<T>> cache = null;
  94. if (cacheName != null)
  95. cache = LoadAssemblyCache<T>(cacheName);
  96. foreach (string dll in Directory.GetFiles(Path.GetFullPath(directory), "*.dll", SearchOption.AllDirectories))
  97. try
  98. {
  99. if (cache != null && cache.TryGetValue(dll, out var cacheEntry))
  100. {
  101. long lastWrite = File.GetLastWriteTimeUtc(dll).Ticks;
  102. if (lastWrite == cacheEntry.Timestamp)
  103. {
  104. result[dll] = cacheEntry.CacheItems;
  105. continue;
  106. }
  107. }
  108. var ass = AssemblyDefinition.ReadAssembly(dll, ReaderParameters);
  109. Logger.LogDebug($"Examining '{dll}'");
  110. if (!assemblyFilter?.Invoke(ass) ?? false)
  111. {
  112. result[dll] = new List<T>();
  113. ass.Dispose();
  114. continue;
  115. }
  116. var matches = ass.MainModule.Types
  117. .Select(t => typeSelector(t, dll))
  118. .Where(t => t != null).ToList();
  119. result[dll] = matches;
  120. ass.Dispose();
  121. }
  122. catch (BadImageFormatException e)
  123. {
  124. Logger.LogDebug($"Skipping loading {dll} because it's not a valid .NET assembly. Full error: {e.Message}");
  125. }
  126. catch (Exception e)
  127. {
  128. Logger.LogError(e.ToString());
  129. }
  130. if (cacheName != null)
  131. SaveAssemblyCache(cacheName, result);
  132. return result;
  133. }
  134. /// <summary>
  135. /// Loads an index of type metadatas from a cache.
  136. /// </summary>
  137. /// <param name="cacheName">Name of the cache</param>
  138. /// <typeparam name="T">Cacheable item</typeparam>
  139. /// <returns>Cached type metadatas indexed by the path of the assembly that defines the type. If no cache is defined, return null.</returns>
  140. public static Dictionary<string, CachedAssembly<T>> LoadAssemblyCache<T>(string cacheName) where T : ICacheable, new()
  141. {
  142. if (!EnableAssemblyCache.Value)
  143. return null;
  144. var result = new Dictionary<string, CachedAssembly<T>>();
  145. try
  146. {
  147. string path = Path.Combine(Paths.CachePath, $"{cacheName}_typeloader.dat");
  148. if (!File.Exists(path))
  149. return null;
  150. using (var br = new BinaryReader(File.OpenRead(path)))
  151. {
  152. int entriesCount = br.ReadInt32();
  153. for (var i = 0; i < entriesCount; i++)
  154. {
  155. string entryIdentifier = br.ReadString();
  156. long entryDate = br.ReadInt64();
  157. int itemsCount = br.ReadInt32();
  158. var items = new List<T>();
  159. for (var j = 0; j < itemsCount; j++)
  160. {
  161. var entry = new T();
  162. entry.Load(br);
  163. items.Add(entry);
  164. }
  165. result[entryIdentifier] = new CachedAssembly<T> { Timestamp = entryDate, CacheItems = items };
  166. }
  167. }
  168. }
  169. catch (Exception e)
  170. {
  171. Logger.LogWarning($"Failed to load cache \"{cacheName}\"; skipping loading cache. Reason: {e.Message}.");
  172. }
  173. return result;
  174. }
  175. /// <summary>
  176. /// Saves indexed type metadata into a cache.
  177. /// </summary>
  178. /// <param name="cacheName">Name of the cache</param>
  179. /// <param name="entries">List of plugin metadatas indexed by the path to the assembly that contains the types</param>
  180. /// <typeparam name="T">Cacheable item</typeparam>
  181. public static void SaveAssemblyCache<T>(string cacheName, Dictionary<string, List<T>> entries) where T : ICacheable
  182. {
  183. if (!EnableAssemblyCache.Value)
  184. return;
  185. try
  186. {
  187. if (!Directory.Exists(Paths.CachePath))
  188. Directory.CreateDirectory(Paths.CachePath);
  189. string path = Path.Combine(Paths.CachePath, $"{cacheName}_typeloader.dat");
  190. using (var bw = new BinaryWriter(File.OpenWrite(path)))
  191. {
  192. bw.Write(entries.Count);
  193. foreach (var kv in entries)
  194. {
  195. bw.Write(kv.Key);
  196. bw.Write(File.GetLastWriteTimeUtc(kv.Key).Ticks);
  197. bw.Write(kv.Value.Count);
  198. foreach (var item in kv.Value)
  199. item.Save(bw);
  200. }
  201. }
  202. }
  203. catch (Exception e)
  204. {
  205. Logger.LogWarning($"Failed to save cache \"{cacheName}\"; skipping saving cache. Reason: {e.Message}.");
  206. }
  207. }
  208. /// <summary>
  209. /// Converts TypeLoadException to a readable string.
  210. /// </summary>
  211. /// <param name="ex">TypeLoadException</param>
  212. /// <returns>Readable representation of the exception</returns>
  213. public static string TypeLoadExceptionToString(ReflectionTypeLoadException ex)
  214. {
  215. var sb = new StringBuilder();
  216. foreach (var exSub in ex.LoaderExceptions)
  217. {
  218. sb.AppendLine(exSub.Message);
  219. if (exSub is FileNotFoundException exFileNotFound)
  220. {
  221. if (!string.IsNullOrEmpty(exFileNotFound.FusionLog))
  222. {
  223. sb.AppendLine("Fusion Log:");
  224. sb.AppendLine(exFileNotFound.FusionLog);
  225. }
  226. }
  227. else if (exSub is FileLoadException exLoad)
  228. {
  229. if (!string.IsNullOrEmpty(exLoad.FusionLog))
  230. {
  231. sb.AppendLine("Fusion Log:");
  232. sb.AppendLine(exLoad.FusionLog);
  233. }
  234. }
  235. sb.AppendLine();
  236. }
  237. return sb.ToString();
  238. }
  239. #region Config
  240. private static readonly ConfigEntry<bool> EnableAssemblyCache = ConfigFile.CoreConfig.Bind(
  241. "Caching", "EnableAssemblyCache",
  242. true,
  243. "Enable/disable assembly metadata cache\nEnabling this will speed up discovery of plugins and patchers by caching the metadata of all types BepInEx discovers.");
  244. #endregion
  245. }
  246. }