Files
se-launcher/CringePlugins/Loader/PluginInstance.cs
pas2704 36af9a722a
All checks were successful
Build / Compute Version (push) Successful in 5s
Build / Build Nuget package (CringeBootstrap.Abstractions) (push) Successful in 1m55s
Build / Build Nuget package (CringePlugins) (push) Successful in 2m24s
Build / Build Nuget package (NuGet) (push) Successful in 2m6s
Build / Build Nuget package (SharedCringe) (push) Successful in 2m47s
Build / Build Launcher (push) Successful in 3m45s
Wrap plugins in PluginWrapper to handle fatal errors in plugins
2024-11-09 03:42:25 -05:00

98 lines
3.3 KiB
C#

using System.Collections.Immutable;
using System.Runtime.Loader;
using CringeBootstrap.Abstractions;
using CringePlugins.Utils;
using NLog;
using SharedCringe.Loader;
using VRage.Plugins;
namespace CringePlugins.Loader;
internal sealed class PluginInstance
{
public bool HasConfig => _openConfigAction != null;
private readonly string _entrypointPath;
private PluginAssemblyLoadContext? _context;
private IPlugin? _instance;
private Action? _openConfigAction;
private IHandleInputPlugin? _handleInputInstance;
private PluginWrapper? _wrappedInstance;
private static readonly ILogger Log = LogManager.GetCurrentClassLogger();
public PluginMetadata Metadata { get; }
public PluginInstance(PluginMetadata metadata, string entrypointPath)
{
_entrypointPath = entrypointPath;
Metadata = metadata;
}
public PluginInstance(string entrypointPath) : this(PluginMetadata.ReadFromEntrypoint(entrypointPath), entrypointPath)
{
}
public void Instantiate(ImmutableArray<DerivedAssemblyLoadContext>.Builder contextBuilder)
{
if (AssemblyLoadContext.GetLoadContext(typeof(PluginInstance).Assembly) is not ICoreLoadContext parentContext)
throw new NotSupportedException("Plugin instantiation is not supported in this context");
_context = new PluginAssemblyLoadContext(parentContext, _entrypointPath);
contextBuilder.Add(_context);
var entrypoint = _context.LoadEntrypoint();
var plugins = IntrospectionContext.Global.CollectDerivedTypes<IPlugin>(entrypoint.GetMainModule()).ToArray();
if (plugins.Length == 0)
throw new InvalidOperationException("Entrypoint does not contain any plugins");
if (plugins.Length > 1)
throw new InvalidOperationException("Entrypoint contains multiple plugins");
_instance = (IPlugin) Activator.CreateInstance(plugins[0])!;
var openConfigMethod = plugins[0].GetMethod("OpenConfigDialog");
if (openConfigMethod is not null)
{
if (openConfigMethod.ReturnType != typeof(void) || openConfigMethod.IsStatic || openConfigMethod.GetParameters().Length > 0)
{
Log.Error("Plugin has OpenConfigDialog method with incorrect signature: {Name}, v{Version} - {Source}",
Metadata.Name, Metadata.Version, Metadata.Source);
}
else
{
_openConfigAction = openConfigMethod.CreateDelegate<Action>(_instance);
}
}
_handleInputInstance = _instance as IHandleInputPlugin;
_wrappedInstance = new PluginWrapper(Metadata.Name, _instance);
}
public void RegisterLifetime()
{
if (_instance is null)
throw new InvalidOperationException("Must call Instantiate first");
MyPlugins.m_plugins.Add(_wrappedInstance);
if (_handleInputInstance is not null)
MyPlugins.m_handleInputPlugins.Add(_wrappedInstance);
}
public void OpenConfig()
{
if (_openConfigAction is null)
throw new InvalidOperationException("Plugin does not have OpenConfigDialog method");
try
{
_openConfigAction();
}
catch (Exception ex)
{
Log.Error(ex, "Error opening config: {Exception}");
}
}
}