Flow.Launcher/Flow.Launcher.Core/Plugin/PluginConfig.cs

140 lines
5.4 KiB
C#
Raw Permalink Normal View History

using System;
2016-01-06 21:34:42 +00:00
using System.Collections.Generic;
using System.Linq;
2014-12-26 11:36:43 +00:00
using System.IO;
using Flow.Launcher.Infrastructure;
2020-04-21 09:12:17 +00:00
using Flow.Launcher.Infrastructure.Logger;
using Flow.Launcher.Plugin;
using System.Text.Json;
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
{
internal abstract class PluginConfig
{
/// <summary>
2020-05-26 01:00:33 +00:00
/// Parse plugin metadata in the given directories
2014-12-26 11:36:43 +00:00
/// </summary>
/// <param name="pluginDirectories"></param>
/// <returns></returns>
public static List<PluginMetadata> Parse(string[] pluginDirectories)
2014-12-26 11:36:43 +00:00
{
2020-05-26 01:00:33 +00:00
var allPluginMetadata = new List<PluginMetadata>();
var directories = pluginDirectories.SelectMany(Directory.EnumerateDirectories);
2014-12-26 11:36:43 +00:00
// todo use linq when diable plugin is implmented since parallel.foreach + list is not thread saft
foreach (var directory in directories)
2014-12-26 11:36:43 +00:00
{
if (File.Exists(Path.Combine(directory, "NeedDelete.txt")))
2014-12-26 11:36:43 +00:00
{
2014-12-29 15:02:50 +00:00
try
{
Directory.Delete(directory, true);
}
2016-01-06 21:34:42 +00:00
catch (Exception e)
2014-12-29 15:02:50 +00:00
{
2017-01-24 00:24:20 +00:00
Log.Exception($"|PluginConfig.ParsePLuginConfigs|Can't delete <{directory}>", e);
2014-12-29 15:02:50 +00:00
}
2014-12-26 11:36:43 +00:00
}
else
2014-12-26 11:36:43 +00:00
{
PluginMetadata metadata = GetPluginMetadata(directory);
if (metadata != null)
{
2020-05-26 01:00:33 +00:00
allPluginMetadata.Add(metadata);
}
2014-12-26 11:36:43 +00:00
}
}
(List<PluginMetadata> uniqueList, List<PluginMetadata> duplicateList) = GetUniqueLatestPluginMetadata(allPluginMetadata);
duplicateList
.ForEach(
x => Log.Warn("PluginConfig",
string.Format("Duplicate plugin name: {0}, id: {1}, version: {2} " +
"not loaded due to version not the highest of the duplicates",
x.Name, x.ID, x.Version),
"GetUniqueLatestPluginMetadata"));
return uniqueList;
}
internal static (List<PluginMetadata>, List<PluginMetadata>) GetUniqueLatestPluginMetadata(List<PluginMetadata> allPluginMetadata)
{
var duplicate_list = new List<PluginMetadata>();
var unique_list = new List<PluginMetadata>();
var duplicateGroups = allPluginMetadata.GroupBy(x => x.ID).Where(g => g.Count() > 1).Select(y => y).ToList();
foreach (var metadata in allPluginMetadata)
{
var duplicatesExist = false;
foreach (var group in duplicateGroups)
{
if (metadata.ID == group.Key)
{
duplicatesExist = true;
// If metadata's version greater than each duplicate's version, CompareTo > 0
var count = group.Where(x => metadata.Version.CompareTo(x.Version) > 0).Count();
// Only add if the meatadata's version is the highest of all duplicates in the group
if (count == group.Count() - 1)
{
unique_list.Add(metadata);
}
else
{
duplicate_list.Add(metadata);
}
}
}
if (!duplicatesExist)
unique_list.Add(metadata);
}
return (unique_list, duplicate_list);
2014-12-26 11:36:43 +00:00
}
private static PluginMetadata GetPluginMetadata(string pluginDirectory)
{
string configPath = Path.Combine(pluginDirectory, Constant.PluginMetadataFileName);
2014-12-26 11:36:43 +00:00
if (!File.Exists(configPath))
{
2017-01-24 00:24:20 +00:00
Log.Error($"|PluginConfig.GetPluginMetadata|Didn't find config file <{configPath}>");
2014-12-26 11:36:43 +00:00
return null;
}
PluginMetadata metadata;
try
{
metadata = JsonSerializer.Deserialize<PluginMetadata>(File.ReadAllText(configPath));
2014-12-26 11:36:43 +00:00
metadata.PluginDirectory = pluginDirectory;
// for plugins which doesn't has ActionKeywords key
metadata.ActionKeywords = metadata.ActionKeywords ?? new List<string> { metadata.ActionKeyword };
// for plugin still use old ActionKeyword
metadata.ActionKeyword = metadata.ActionKeywords?[0];
2014-12-26 11:36:43 +00:00
}
2016-01-06 21:34:42 +00:00
catch (Exception e)
2014-12-26 11:36:43 +00:00
{
2017-01-24 00:24:20 +00:00
Log.Exception($"|PluginConfig.GetPluginMetadata|invalid json for config <{configPath}>", e);
2014-12-26 11:36:43 +00:00
return null;
}
if (!AllowedLanguage.IsAllowed(metadata.Language))
{
2017-01-24 00:24:20 +00:00
Log.Error($"|PluginConfig.GetPluginMetadata|Invalid language <{metadata.Language}> for config <{configPath}>");
2014-12-26 11:36:43 +00:00
return null;
}
if (!File.Exists(metadata.ExecuteFilePath))
{
2017-01-24 00:24:20 +00:00
Log.Error($"|PluginConfig.GetPluginMetadata|execute file path didn't exist <{metadata.ExecuteFilePath}> for conifg <{configPath}");
2014-12-26 11:36:43 +00:00
return null;
}
return metadata;
}
}
}