Flow.Launcher/Flow.Launcher.Core/Plugin/PluginsLoader.cs

168 lines
6.8 KiB
C#
Raw Normal View History

using System;
2014-12-26 11:36:43 +00:00
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
2020-05-20 23:02:51 +00:00
using System.Threading.Tasks;
using System.Windows;
using CommunityToolkit.Mvvm.DependencyInjection;
2022-11-21 22:20:27 +00:00
using Flow.Launcher.Core.ExternalPlugins.Environments;
2023-04-25 12:04:08 +00:00
#pragma warning disable IDE0005
2020-04-21 09:12:17 +00:00
using Flow.Launcher.Infrastructure.Logger;
2023-04-25 12:04:08 +00:00
#pragma warning restore IDE0005
2020-04-21 09:12:17 +00:00
using Flow.Launcher.Infrastructure.UserSettings;
using Flow.Launcher.Plugin;
2014-12-26 11:36:43 +00:00
2020-04-21 09:12:17 +00:00
namespace Flow.Launcher.Core.Plugin
2014-12-26 11:36:43 +00:00
{
public static class PluginsLoader
2014-12-26 11:36:43 +00:00
{
2025-04-08 13:53:00 +00:00
private static readonly string ClassName = nameof(PluginsLoader);
// We should not initialize API in static constructor because it will create another API instance
private static IPublicAPI api = null;
private static IPublicAPI API => api ??= Ioc.Default.GetRequiredService<IPublicAPI>();
2016-05-12 01:45:35 +00:00
public static List<PluginPair> Plugins(List<PluginMetadata> metadatas, PluginsSettings settings)
{
var dotnetPlugins = DotNetPlugins(metadatas);
2023-11-04 02:16:51 +00:00
2022-11-21 22:20:27 +00:00
var pythonEnv = new PythonEnvironment(metadatas, settings);
2023-03-26 06:12:21 +00:00
var pythonV2Env = new PythonV2Environment(metadatas, settings);
2022-11-21 22:20:27 +00:00
var tsEnv = new TypeScriptEnvironment(metadatas, settings);
var jsEnv = new JavaScriptEnvironment(metadatas, settings);
2023-11-04 02:16:51 +00:00
var tsV2Env = new TypeScriptV2Environment(metadatas, settings);
var jsV2Env = new JavaScriptV2Environment(metadatas, settings);
2022-11-21 22:20:27 +00:00
var pythonPlugins = pythonEnv.Setup();
2023-03-26 06:12:21 +00:00
var pythonV2Plugins = pythonV2Env.Setup();
2022-11-21 22:20:27 +00:00
var tsPlugins = tsEnv.Setup();
var jsPlugins = jsEnv.Setup();
2023-11-04 02:16:51 +00:00
var tsV2Plugins = tsV2Env.Setup();
var jsV2Plugins = jsV2Env.Setup();
var executablePlugins = ExecutablePlugins(metadatas);
2023-11-04 02:16:51 +00:00
var executableV2Plugins = ExecutableV2Plugins(metadatas);
2022-10-23 09:05:12 +00:00
var plugins = dotnetPlugins
2023-11-04 02:16:51 +00:00
.Concat(pythonPlugins)
.Concat(pythonV2Plugins)
.Concat(tsPlugins)
.Concat(jsPlugins)
.Concat(tsV2Plugins)
.Concat(jsV2Plugins)
.Concat(executablePlugins)
.Concat(executableV2Plugins)
.ToList();
return plugins;
}
2025-03-30 05:57:35 +00:00
private static IEnumerable<PluginPair> DotNetPlugins(List<PluginMetadata> source)
2014-12-26 11:36:43 +00:00
{
var erroredPlugins = new List<string>();
2014-12-26 11:36:43 +00:00
var plugins = new List<PluginPair>();
var metadatas = source.Where(o => AllowedLanguage.IsDotNet(o.Language));
2014-12-26 11:36:43 +00:00
foreach (var metadata in metadatas)
2014-12-26 11:36:43 +00:00
{
2025-04-08 13:53:00 +00:00
var milliseconds = API.StopwatchLogDebug(ClassName, $"Constructor init cost for {metadata.Name}", () =>
2016-11-30 01:08:29 +00:00
{
Assembly assembly = null;
2021-03-23 09:25:46 +00:00
IAsyncPlugin plugin = null;
try
{
var assemblyLoader = new PluginAssemblyLoader(metadata.ExecuteFilePath);
assembly = assemblyLoader.LoadAssemblyAndDependencies();
2021-03-23 09:25:46 +00:00
var type = assemblyLoader.FromAssemblyGetTypeOfInterface(assembly,
typeof(IAsyncPlugin));
2021-03-23 09:25:46 +00:00
plugin = Activator.CreateInstance(type) as IAsyncPlugin;
metadata.AssemblyName = assembly.GetName().Name;
}
#if DEBUG
2025-02-24 03:35:41 +00:00
catch (Exception)
{
throw;
}
#else
catch (Exception e) when (assembly == null)
{
Log.Exception(ClassName, $"Couldn't load assembly for the plugin: {metadata.Name}", e);
}
catch (InvalidOperationException e)
{
Log.Exception(ClassName, $"Can't find the required IPlugin interface for the plugin: <{metadata.Name}>", e);
}
catch (ReflectionTypeLoadException e)
{
Log.Exception(ClassName, $"The GetTypes method was unable to load assembly types for the plugin: <{metadata.Name}>", e);
}
catch (Exception e)
{
Log.Exception(ClassName, $"The following plugin has errored and can not be loaded: <{metadata.Name}>", e);
}
2021-03-23 09:25:46 +00:00
#endif
if (plugin == null)
{
erroredPlugins.Add(metadata.Name);
return;
}
2021-03-23 09:25:46 +00:00
2023-11-04 02:16:51 +00:00
plugins.Add(new PluginPair { Plugin = plugin, Metadata = metadata });
});
2016-11-30 01:08:29 +00:00
metadata.InitTime += milliseconds;
}
if (erroredPlugins.Count > 0)
{
2025-02-24 03:35:41 +00:00
var errorPluginString = string.Join(Environment.NewLine, erroredPlugins);
2020-05-21 20:48:03 +00:00
var errorMessage = "The following "
+ (erroredPlugins.Count > 1 ? "plugins have " : "plugin has ")
+ "errored and cannot be loaded:";
2020-05-21 20:48:03 +00:00
_ = Task.Run(() =>
{
Ioc.Default.GetRequiredService<IPublicAPI>().ShowMsgBox($"{errorMessage}{Environment.NewLine}{Environment.NewLine}" +
$"{errorPluginString}{Environment.NewLine}{Environment.NewLine}" +
$"Please refer to the logs for more information", "",
MessageBoxButton.OK, MessageBoxImage.Warning);
});
}
return plugins;
}
2025-03-30 05:57:35 +00:00
private static IEnumerable<PluginPair> ExecutablePlugins(IEnumerable<PluginMetadata> source)
2016-05-07 15:56:29 +00:00
{
2020-06-24 22:55:20 +00:00
return source
.Where(o => o.Language.Equals(AllowedLanguage.Executable, StringComparison.OrdinalIgnoreCase))
.Select(metadata =>
2020-06-24 22:55:20 +00:00
{
2025-03-23 04:47:57 +00:00
return new PluginPair
{
Plugin = new ExecutablePlugin(metadata.ExecuteFilePath),
Metadata = metadata
};
2020-06-24 22:55:20 +00:00
});
2016-05-07 15:56:29 +00:00
}
2023-11-04 02:16:51 +00:00
2025-03-30 05:57:35 +00:00
private static IEnumerable<PluginPair> ExecutableV2Plugins(IEnumerable<PluginMetadata> source)
2023-11-04 02:16:51 +00:00
{
return source
.Where(o => o.Language.Equals(AllowedLanguage.ExecutableV2, StringComparison.OrdinalIgnoreCase))
.Select(metadata =>
2023-11-04 02:16:51 +00:00
{
2025-03-23 04:47:57 +00:00
return new PluginPair
{
Plugin = new ExecutablePlugin(metadata.ExecuteFilePath),
Metadata = metadata
};
2023-11-04 02:16:51 +00:00
});
}
2014-12-26 11:36:43 +00:00
}
}