mirror of
https://github.com/microsoft/PowerToys.git
synced 2025-12-16 11:48:06 +01:00
Do not load plugin when it is disabled (#10515)
This commit is contained in:
@@ -37,7 +37,7 @@ namespace PowerLauncher
|
|||||||
private ThemeManager _themeManager;
|
private ThemeManager _themeManager;
|
||||||
private SettingWindowViewModel _settingsVM;
|
private SettingWindowViewModel _settingsVM;
|
||||||
private StringMatcher _stringMatcher;
|
private StringMatcher _stringMatcher;
|
||||||
private SettingsWatcher _settingsWatcher;
|
private SettingsReader _settingsReader;
|
||||||
|
|
||||||
[STAThread]
|
[STAThread]
|
||||||
public static void Main()
|
public static void Main()
|
||||||
@@ -103,6 +103,9 @@ namespace PowerLauncher
|
|||||||
_mainVM = new MainViewModel(_settings);
|
_mainVM = new MainViewModel(_settings);
|
||||||
_mainWindow = new MainWindow(_settings, _mainVM);
|
_mainWindow = new MainWindow(_settings, _mainVM);
|
||||||
API = new PublicAPIInstance(_settingsVM, _mainVM, _themeManager);
|
API = new PublicAPIInstance(_settingsVM, _mainVM, _themeManager);
|
||||||
|
_settingsReader = new SettingsReader(_settings, _themeManager);
|
||||||
|
_settingsReader.ReadSettings();
|
||||||
|
|
||||||
PluginManager.InitializePlugins(API);
|
PluginManager.InitializePlugins(API);
|
||||||
|
|
||||||
Current.MainWindow = _mainWindow;
|
Current.MainWindow = _mainWindow;
|
||||||
@@ -113,7 +116,7 @@ namespace PowerLauncher
|
|||||||
|
|
||||||
RegisterExitEvents();
|
RegisterExitEvents();
|
||||||
|
|
||||||
_settingsWatcher = new SettingsWatcher(_settings, _themeManager);
|
_settingsReader.ReadSettingsOnChange();
|
||||||
|
|
||||||
_mainVM.MainWindowVisibility = Visibility.Visible;
|
_mainVM.MainWindowVisibility = Visibility.Visible;
|
||||||
_mainVM.ColdStartFix();
|
_mainVM.ColdStartFix();
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ using System.IO.Abstractions;
|
|||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
using PowerLauncher.Properties;
|
||||||
using Wox.Infrastructure;
|
using Wox.Infrastructure;
|
||||||
using Wox.Infrastructure.Storage;
|
using Wox.Infrastructure.Storage;
|
||||||
using Wox.Plugin;
|
using Wox.Plugin;
|
||||||
@@ -49,7 +50,10 @@ namespace PowerLauncher.Plugin
|
|||||||
{
|
{
|
||||||
if (_allPlugins == null)
|
if (_allPlugins == null)
|
||||||
{
|
{
|
||||||
_allPlugins = PluginsLoader.Plugins(PluginConfig.Parse(Directories));
|
_allPlugins = PluginConfig.Parse(Directories)
|
||||||
|
.Where(x => x.Language.ToUpperInvariant() == AllowedLanguage.CSharp)
|
||||||
|
.Select(x => new PluginPair(x))
|
||||||
|
.ToList();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -119,23 +123,15 @@ namespace PowerLauncher.Plugin
|
|||||||
var failedPlugins = new ConcurrentQueue<PluginPair>();
|
var failedPlugins = new ConcurrentQueue<PluginPair>();
|
||||||
Parallel.ForEach(AllPlugins, pair =>
|
Parallel.ForEach(AllPlugins, pair =>
|
||||||
{
|
{
|
||||||
try
|
if (pair.Metadata.Disabled)
|
||||||
{
|
{
|
||||||
var milliseconds = Stopwatch.Debug($"PluginManager.InitializePlugins - Init method time cost for <{pair.Metadata.Name}>", () =>
|
return;
|
||||||
{
|
|
||||||
pair.Plugin.Init(new PluginInitContext
|
|
||||||
{
|
|
||||||
CurrentPluginMetadata = pair.Metadata,
|
|
||||||
API = API,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
pair.Metadata.InitTime += milliseconds;
|
|
||||||
Log.Info($"Total init cost for <{pair.Metadata.Name}> is <{pair.Metadata.InitTime}ms>", MethodBase.GetCurrentMethod().DeclaringType);
|
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
|
||||||
|
pair.LoadPlugin(API);
|
||||||
|
|
||||||
|
if (!pair.IsPluginLoaded)
|
||||||
{
|
{
|
||||||
Log.Exception($"Fail to Init plugin: {pair.Metadata.Name}", e, MethodBase.GetCurrentMethod().DeclaringType);
|
|
||||||
pair.Metadata.Disabled = true;
|
|
||||||
failedPlugins.Enqueue(pair);
|
failedPlugins.Enqueue(pair);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -145,7 +141,8 @@ namespace PowerLauncher.Plugin
|
|||||||
if (failedPlugins.Any())
|
if (failedPlugins.Any())
|
||||||
{
|
{
|
||||||
var failed = string.Join(",", failedPlugins.Select(x => x.Metadata.Name));
|
var failed = string.Join(",", failedPlugins.Select(x => x.Metadata.Name));
|
||||||
API.ShowMsg($"Fail to Init Plugins", $"Plugins: {failed} - fail to load and would be disabled, please contact plugin creator for help", string.Empty, false);
|
var description = string.Format(CultureInfo.CurrentCulture, Resources.FailedToInitializePluginsDescription, failed);
|
||||||
|
API.ShowMsg(Resources.FailedToInitializePluginsTitle, description, string.Empty, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -162,6 +159,11 @@ namespace PowerLauncher.Plugin
|
|||||||
throw new ArgumentNullException(nameof(pair));
|
throw new ArgumentNullException(nameof(pair));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!pair.IsPluginLoaded)
|
||||||
|
{
|
||||||
|
return new List<Result>();
|
||||||
|
}
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
List<Result> results = null;
|
List<Result> results = null;
|
||||||
|
|||||||
@@ -1,91 +0,0 @@
|
|||||||
// Copyright (c) Microsoft Corporation
|
|
||||||
// The Microsoft Corporation licenses this file to you under the MIT license.
|
|
||||||
// See the LICENSE file in the project root for more information.
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Reflection;
|
|
||||||
using System.Runtime.Loader;
|
|
||||||
using Wox.Infrastructure;
|
|
||||||
using Wox.Plugin;
|
|
||||||
using Wox.Plugin.Logger;
|
|
||||||
|
|
||||||
namespace PowerLauncher.Plugin
|
|
||||||
{
|
|
||||||
public static class PluginsLoader
|
|
||||||
{
|
|
||||||
public const string PATH = "PATH";
|
|
||||||
|
|
||||||
public static List<PluginPair> Plugins(List<PluginMetadata> metadatas)
|
|
||||||
{
|
|
||||||
var csharpPlugins = CSharpPlugins(metadatas).ToList();
|
|
||||||
return csharpPlugins;
|
|
||||||
}
|
|
||||||
|
|
||||||
[System.Diagnostics.CodeAnalysis.SuppressMessage("Design", "CA1031:Do not catch general exception types", Justification = "Suppressing this to enable FxCop. We are logging the exception, and going forward general exceptions should not be caught")]
|
|
||||||
public static IEnumerable<PluginPair> CSharpPlugins(List<PluginMetadata> source)
|
|
||||||
{
|
|
||||||
var plugins = new List<PluginPair>();
|
|
||||||
var metadatas = source
|
|
||||||
.Where(o => o.Language.ToUpperInvariant() == AllowedLanguage.CSharp)
|
|
||||||
.ToList();
|
|
||||||
|
|
||||||
foreach (var metadata in metadatas)
|
|
||||||
{
|
|
||||||
var milliseconds = Stopwatch.Debug($"PluginsLoader.CSharpPlugins - Constructor init cost for {metadata.Name}", () =>
|
|
||||||
{
|
|
||||||
#if DEBUG
|
|
||||||
var assembly = AssemblyLoadContext.Default.LoadFromAssemblyPath(metadata.ExecuteFilePath);
|
|
||||||
var types = assembly.GetTypes();
|
|
||||||
var type = types.First(o => o.IsClass && !o.IsAbstract && o.GetInterfaces().Contains(typeof(IPlugin)));
|
|
||||||
var plugin = (IPlugin)Activator.CreateInstance(type);
|
|
||||||
#else
|
|
||||||
Assembly assembly;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
assembly = AssemblyLoadContext.Default.LoadFromAssemblyPath(metadata.ExecuteFilePath);
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Log.Exception($"Couldn't load assembly for {metadata.Name}", e, MethodBase.GetCurrentMethod().DeclaringType);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var types = assembly.GetTypes();
|
|
||||||
Type type;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
type = types.First(o => o.IsClass && !o.IsAbstract && o.GetInterfaces().Contains(typeof(IPlugin)));
|
|
||||||
}
|
|
||||||
catch (InvalidOperationException e)
|
|
||||||
{
|
|
||||||
Log.Exception($"Can't find class implement IPlugin for <{metadata.Name}>", e, MethodBase.GetCurrentMethod().DeclaringType);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
IPlugin plugin;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
plugin = (IPlugin)Activator.CreateInstance(type);
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Log.Exception($"Can't create instance for <{metadata.Name}>", e, MethodBase.GetCurrentMethod().DeclaringType);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
PluginPair pair = new PluginPair
|
|
||||||
{
|
|
||||||
Plugin = plugin,
|
|
||||||
Metadata = metadata,
|
|
||||||
};
|
|
||||||
plugins.Add(pair);
|
|
||||||
});
|
|
||||||
metadata.InitTime += milliseconds;
|
|
||||||
}
|
|
||||||
|
|
||||||
return plugins;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -132,6 +132,24 @@ namespace PowerLauncher.Properties {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Looks up a localized string similar to Plugins: {0} - fail to load and would be disabled, please contact plugins creator for help.
|
||||||
|
/// </summary>
|
||||||
|
public static string FailedToInitializePluginsDescription {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("FailedToInitializePluginsDescription", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Looks up a localized string similar to Fail to initialize plugins.
|
||||||
|
/// </summary>
|
||||||
|
public static string FailedToInitializePluginsTitle {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("FailedToInitializePluginsTitle", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Last execution time: {0}.
|
/// Looks up a localized string similar to Last execution time: {0}.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|||||||
@@ -185,4 +185,10 @@
|
|||||||
<data name="deseralization_error_message" xml:space="preserve">
|
<data name="deseralization_error_message" xml:space="preserve">
|
||||||
<value>Settings will be reset to default and program will continue to function.</value>
|
<value>Settings will be reset to default and program will continue to function.</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="FailedToInitializePluginsDescription" xml:space="preserve">
|
||||||
|
<value>Plugins: {0} - fail to load and would be disabled, please contact plugins creator for help</value>
|
||||||
|
</data>
|
||||||
|
<data name="FailedToInitializePluginsTitle" xml:space="preserve">
|
||||||
|
<value>Fail to initialize plugins</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
@@ -22,29 +22,23 @@ using JsonException = System.Text.Json.JsonException;
|
|||||||
namespace PowerLauncher
|
namespace PowerLauncher
|
||||||
{
|
{
|
||||||
// Watch for /Local/Microsoft/PowerToys/Launcher/Settings.json changes
|
// Watch for /Local/Microsoft/PowerToys/Launcher/Settings.json changes
|
||||||
public class SettingsWatcher : BaseModel
|
public class SettingsReader : BaseModel
|
||||||
{
|
{
|
||||||
private readonly ISettingsUtils _settingsUtils;
|
private readonly ISettingsUtils _settingsUtils;
|
||||||
|
|
||||||
private const int MaxRetries = 10;
|
private const int MaxRetries = 10;
|
||||||
private static readonly object _watcherSyncObject = new object();
|
private static readonly object _readSyncObject = new object();
|
||||||
private readonly IFileSystemWatcher _watcher;
|
|
||||||
private readonly PowerToysRunSettings _settings;
|
private readonly PowerToysRunSettings _settings;
|
||||||
|
|
||||||
private readonly ThemeManager _themeManager;
|
private readonly ThemeManager _themeManager;
|
||||||
|
|
||||||
public SettingsWatcher(PowerToysRunSettings settings, ThemeManager themeManager)
|
private IFileSystemWatcher _watcher;
|
||||||
|
|
||||||
|
public SettingsReader(PowerToysRunSettings settings, ThemeManager themeManager)
|
||||||
{
|
{
|
||||||
_settingsUtils = new SettingsUtils();
|
_settingsUtils = new SettingsUtils();
|
||||||
_settings = settings;
|
_settings = settings;
|
||||||
_themeManager = themeManager;
|
_themeManager = themeManager;
|
||||||
|
|
||||||
// Set up watcher
|
|
||||||
_watcher = Microsoft.PowerToys.Settings.UI.Library.Utilities.Helper.GetFileWatcher(PowerLauncherSettings.ModuleName, "settings.json", OverloadSettings);
|
|
||||||
|
|
||||||
// Load initial settings file
|
|
||||||
OverloadSettings();
|
|
||||||
|
|
||||||
// Apply theme at startup
|
// Apply theme at startup
|
||||||
_themeManager.ChangeTheme(_settings.Theme, true);
|
_themeManager.ChangeTheme(_settings.Theme, true);
|
||||||
}
|
}
|
||||||
@@ -61,9 +55,14 @@ namespace PowerLauncher
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void OverloadSettings()
|
public void ReadSettingsOnChange()
|
||||||
{
|
{
|
||||||
Monitor.Enter(_watcherSyncObject);
|
_watcher = Microsoft.PowerToys.Settings.UI.Library.Utilities.Helper.GetFileWatcher(PowerLauncherSettings.ModuleName, "settings.json", ReadSettings);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ReadSettings()
|
||||||
|
{
|
||||||
|
Monitor.Enter(_readSyncObject);
|
||||||
var retry = true;
|
var retry = true;
|
||||||
var retryCount = 0;
|
var retryCount = 0;
|
||||||
while (retry)
|
while (retry)
|
||||||
@@ -86,16 +85,7 @@ namespace PowerLauncher
|
|||||||
foreach (var setting in overloadSettings.Plugins)
|
foreach (var setting in overloadSettings.Plugins)
|
||||||
{
|
{
|
||||||
var plugin = PluginManager.AllPlugins.FirstOrDefault(x => x.Metadata.ID == setting.Id);
|
var plugin = PluginManager.AllPlugins.FirstOrDefault(x => x.Metadata.ID == setting.Id);
|
||||||
if (plugin != null)
|
plugin?.Update(setting, App.API);
|
||||||
{
|
|
||||||
plugin.Metadata.Disabled = setting.Disabled;
|
|
||||||
plugin.Metadata.ActionKeyword = setting.ActionKeyword;
|
|
||||||
plugin.Metadata.IsGlobal = setting.IsGlobal;
|
|
||||||
if (plugin.Plugin is ISettingProvider)
|
|
||||||
{
|
|
||||||
(plugin.Plugin as ISettingProvider).UpdateSettings(setting);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -168,7 +158,7 @@ namespace PowerLauncher
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Monitor.Exit(_watcherSyncObject);
|
Monitor.Exit(_readSyncObject);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static string ConvertHotkey(HotkeySettings hotkey)
|
private static string ConvertHotkey(HotkeySettings hotkey)
|
||||||
@@ -3,6 +3,14 @@
|
|||||||
// See the LICENSE file in the project root for more information.
|
// See the LICENSE file in the project root for more information.
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
|
using System.Diagnostics;
|
||||||
|
using System.Globalization;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Runtime.Loader;
|
||||||
|
using Microsoft.PowerToys.Settings.UI.Library;
|
||||||
|
using Wox.Plugin.Logger;
|
||||||
|
using Wox.Plugin.Properties;
|
||||||
|
|
||||||
namespace Wox.Plugin
|
namespace Wox.Plugin
|
||||||
{
|
{
|
||||||
@@ -12,6 +20,76 @@ namespace Wox.Plugin
|
|||||||
|
|
||||||
public PluginMetadata Metadata { get; internal set; }
|
public PluginMetadata Metadata { get; internal set; }
|
||||||
|
|
||||||
|
public PluginPair(PluginMetadata metadata)
|
||||||
|
{
|
||||||
|
this.Metadata = metadata;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool IsPluginLoaded { get; set; }
|
||||||
|
|
||||||
|
public void LoadPlugin(IPublicAPI api)
|
||||||
|
{
|
||||||
|
if (Metadata.Disabled)
|
||||||
|
{
|
||||||
|
Log.Info($"Do not load {Metadata.Name} as it is disabled.", GetType());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (IsPluginLoaded)
|
||||||
|
{
|
||||||
|
Log.Info($"Plugin {Metadata.Name} is already loaded", GetType());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var stopWatch = new Stopwatch();
|
||||||
|
stopWatch.Start();
|
||||||
|
if (!CreatePluginInstance())
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!InitPlugin(api))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
stopWatch.Stop();
|
||||||
|
IsPluginLoaded = true;
|
||||||
|
Metadata.InitTime += stopWatch.ElapsedMilliseconds;
|
||||||
|
Log.Info($"Total load cost for <{Metadata.Name}> is <{Metadata.InitTime}ms>", GetType());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Update(PowerLauncherPluginSettings setting, IPublicAPI api)
|
||||||
|
{
|
||||||
|
if (setting == null || api == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Metadata.Disabled && !setting.Disabled)
|
||||||
|
{
|
||||||
|
Metadata.Disabled = false;
|
||||||
|
LoadPlugin(api);
|
||||||
|
if (!IsPluginLoaded)
|
||||||
|
{
|
||||||
|
var title = string.Format(CultureInfo.CurrentCulture, Resources.FailedToLoadPluginTitle, Metadata.Name);
|
||||||
|
api.ShowMsg(title, Resources.FailedToLoadPluginDescription, string.Empty, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Metadata.Disabled = setting.Disabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
Metadata.ActionKeyword = setting.ActionKeyword;
|
||||||
|
Metadata.IsGlobal = setting.IsGlobal;
|
||||||
|
if (Plugin is ISettingProvider)
|
||||||
|
{
|
||||||
|
(Plugin as ISettingProvider).UpdateSettings(setting);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public override string ToString()
|
public override string ToString()
|
||||||
{
|
{
|
||||||
return Metadata.Name;
|
return Metadata.Name;
|
||||||
@@ -36,5 +114,69 @@ namespace Wox.Plugin
|
|||||||
var hashcode = Metadata.ID?.GetHashCode(StringComparison.Ordinal) ?? 0;
|
var hashcode = Metadata.ID?.GetHashCode(StringComparison.Ordinal) ?? 0;
|
||||||
return hashcode;
|
return hashcode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private bool CreatePluginInstance()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
_assembly = AssemblyLoadContext.Default.LoadFromAssemblyPath(Metadata.ExecuteFilePath);
|
||||||
|
}
|
||||||
|
#pragma warning disable CA1031 // Do not catch general exception types
|
||||||
|
catch (Exception e)
|
||||||
|
#pragma warning restore CA1031 // Do not catch general exception types
|
||||||
|
{
|
||||||
|
Log.Exception($"Couldn't load assembly for {Metadata.Name}", e, MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var types = _assembly.GetTypes();
|
||||||
|
Type type;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
type = types.First(o => o.IsClass && !o.IsAbstract && o.GetInterfaces().Contains(typeof(IPlugin)));
|
||||||
|
}
|
||||||
|
catch (InvalidOperationException e)
|
||||||
|
{
|
||||||
|
Log.Exception($"Can't find class implement IPlugin for <{Metadata.Name}>", e, MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
Plugin = (IPlugin)Activator.CreateInstance(type);
|
||||||
|
}
|
||||||
|
#pragma warning disable CA1031 // Do not catch general exception types
|
||||||
|
catch (Exception e)
|
||||||
|
#pragma warning restore CA1031 // Do not catch general exception types
|
||||||
|
{
|
||||||
|
Log.Exception($"Can't create instance for <{Metadata.Name}>", e, MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool InitPlugin(IPublicAPI api)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
Plugin.Init(new PluginInitContext
|
||||||
|
{
|
||||||
|
CurrentPluginMetadata = Metadata,
|
||||||
|
API = api,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
#pragma warning disable CA1031 // Do not catch general exception types
|
||||||
|
catch (Exception e)
|
||||||
|
#pragma warning restore CA1031 // Do not catch general exception types
|
||||||
|
{
|
||||||
|
Log.Exception($"Fail to Init plugin: {Metadata.Name}", e, GetType());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Assembly _assembly;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -59,5 +59,23 @@ namespace Wox.Plugin.Properties {
|
|||||||
resourceCulture = value;
|
resourceCulture = value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Looks up a localized string similar to Please contact plugin creator for help.
|
||||||
|
/// </summary>
|
||||||
|
public static string FailedToLoadPluginDescription {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("FailedToLoadPluginDescription", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Looks up a localized string similar to Fail to Load {0} Plugin.
|
||||||
|
/// </summary>
|
||||||
|
public static string FailedToLoadPluginTitle {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("FailedToLoadPluginTitle", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -117,4 +117,10 @@
|
|||||||
<resheader name="writer">
|
<resheader name="writer">
|
||||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||||
</resheader>
|
</resheader>
|
||||||
|
<data name="FailedToLoadPluginDescription" xml:space="preserve">
|
||||||
|
<value>Please contact plugin creator for help</value>
|
||||||
|
</data>
|
||||||
|
<data name="FailedToLoadPluginTitle" xml:space="preserve">
|
||||||
|
<value>Fail to Load {0} Plugin</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
@@ -45,10 +45,10 @@ namespace Wox.Test
|
|||||||
var results = new List<Result>() { result };
|
var results = new List<Result>() { result };
|
||||||
var pluginMock = new Mock<IPlugin>();
|
var pluginMock = new Mock<IPlugin>();
|
||||||
pluginMock.Setup(r => r.Query(query)).Returns(results);
|
pluginMock.Setup(r => r.Query(query)).Returns(results);
|
||||||
var pluginPair = new PluginPair
|
var pluginPair = new PluginPair(metadata)
|
||||||
{
|
{
|
||||||
Plugin = pluginMock.Object,
|
Plugin = pluginMock.Object,
|
||||||
Metadata = metadata,
|
IsPluginLoaded = true,
|
||||||
};
|
};
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
|
|||||||
@@ -28,10 +28,7 @@ namespace Wox.Test
|
|||||||
// Arrange
|
// Arrange
|
||||||
PluginManager.SetAllPlugins(new List<PluginPair>()
|
PluginManager.SetAllPlugins(new List<PluginPair>()
|
||||||
{
|
{
|
||||||
new PluginPair
|
new PluginPair(new PluginMetadata() { ActionKeyword = ">" }),
|
||||||
{
|
|
||||||
Metadata = new PluginMetadata() { ActionKeyword = ">" },
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
string searchQuery = "> file.txt file2 file3";
|
string searchQuery = "> file.txt file2 file3";
|
||||||
@@ -51,7 +48,7 @@ namespace Wox.Test
|
|||||||
string searchQuery = "file.txt file2 file3";
|
string searchQuery = "file.txt file2 file3";
|
||||||
PluginManager.SetAllPlugins(new List<PluginPair>()
|
PluginManager.SetAllPlugins(new List<PluginPair>()
|
||||||
{
|
{
|
||||||
new PluginPair { Metadata = new PluginMetadata() { Disabled = false, IsGlobal = true } },
|
new PluginPair(new PluginMetadata() { Disabled = false, IsGlobal = true }),
|
||||||
});
|
});
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
@@ -66,7 +63,7 @@ namespace Wox.Test
|
|||||||
public void QueryBuildShouldGenerateSameSearchQueryWithOrWithoutSpaceAfterActionKeyword()
|
public void QueryBuildShouldGenerateSameSearchQueryWithOrWithoutSpaceAfterActionKeyword()
|
||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
var plugin = new PluginPair { Metadata = new PluginMetadata() { ActionKeyword = "a" } };
|
var plugin = new PluginPair(new PluginMetadata() { ActionKeyword = "a" });
|
||||||
PluginManager.SetAllPlugins(new List<PluginPair>()
|
PluginManager.SetAllPlugins(new List<PluginPair>()
|
||||||
{
|
{
|
||||||
plugin,
|
plugin,
|
||||||
@@ -93,8 +90,8 @@ namespace Wox.Test
|
|||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
string searchQuery = "abcdefgh";
|
string searchQuery = "abcdefgh";
|
||||||
var firstPlugin = new PluginPair { Metadata = new PluginMetadata { ActionKeyword = "ab", ID = "plugin1" } };
|
var firstPlugin = new PluginPair(new PluginMetadata { ActionKeyword = "ab", ID = "plugin1" });
|
||||||
var secondPlugin = new PluginPair { Metadata = new PluginMetadata { ActionKeyword = "abcd", ID = "plugin2" } };
|
var secondPlugin = new PluginPair(new PluginMetadata { ActionKeyword = "abcd", ID = "plugin2" });
|
||||||
PluginManager.SetAllPlugins(new List<PluginPair>()
|
PluginManager.SetAllPlugins(new List<PluginPair>()
|
||||||
{
|
{
|
||||||
firstPlugin,
|
firstPlugin,
|
||||||
@@ -117,8 +114,8 @@ namespace Wox.Test
|
|||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
string searchQuery = "abcd efgh";
|
string searchQuery = "abcd efgh";
|
||||||
var firstPlugin = new PluginPair { Metadata = new PluginMetadata { ActionKeyword = "ab", ID = "plugin1" } };
|
var firstPlugin = new PluginPair(new PluginMetadata { ActionKeyword = "ab", ID = "plugin1" });
|
||||||
var secondPlugin = new PluginPair { Metadata = new PluginMetadata { ActionKeyword = "abcd", ID = "plugin2" } };
|
var secondPlugin = new PluginPair(new PluginMetadata { ActionKeyword = "abcd", ID = "plugin2" });
|
||||||
PluginManager.SetAllPlugins(new List<PluginPair>()
|
PluginManager.SetAllPlugins(new List<PluginPair>()
|
||||||
{
|
{
|
||||||
firstPlugin,
|
firstPlugin,
|
||||||
@@ -142,8 +139,8 @@ namespace Wox.Test
|
|||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
string searchQuery = "!efgh";
|
string searchQuery = "!efgh";
|
||||||
var firstPlugin = new PluginPair { Metadata = new PluginMetadata { ActionKeyword = "!", ID = "plugin1" } };
|
var firstPlugin = new PluginPair(new PluginMetadata { ActionKeyword = "!", ID = "plugin1" });
|
||||||
var secondPlugin = new PluginPair { Metadata = new PluginMetadata { ActionKeyword = "!", ID = "plugin2" } };
|
var secondPlugin = new PluginPair(new PluginMetadata { ActionKeyword = "!", ID = "plugin2" });
|
||||||
PluginManager.SetAllPlugins(new List<PluginPair>()
|
PluginManager.SetAllPlugins(new List<PluginPair>()
|
||||||
{
|
{
|
||||||
firstPlugin,
|
firstPlugin,
|
||||||
|
|||||||
Reference in New Issue
Block a user