2016-07-24 21:26:24 +02:00
|
|
|
|
using System;
|
2017-04-01 19:27:05 +02:00
|
|
|
|
using System.Collections.Generic;
|
2019-05-26 14:55:12 +02:00
|
|
|
|
using System.Diagnostics.CodeAnalysis;
|
2016-07-23 18:36:58 +02:00
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Text;
|
2019-05-26 14:55:12 +02:00
|
|
|
|
using TweetLib.Core.Features.Plugins.Enums;
|
|
|
|
|
using TweetLib.Core.Features.Plugins.Events;
|
2021-12-28 15:25:19 +01:00
|
|
|
|
using TweetLib.Utils.Collections;
|
|
|
|
|
using TweetLib.Utils.Data;
|
|
|
|
|
using TweetLib.Utils.Static;
|
2016-07-23 18:36:58 +02:00
|
|
|
|
|
2021-08-07 07:48:29 +02:00
|
|
|
|
namespace TweetLib.Core.Features.Plugins {
|
|
|
|
|
[SuppressMessage("ReSharper", "UnusedMember.Global")]
|
2022-01-19 21:53:42 +01:00
|
|
|
|
sealed class PluginBridge {
|
2021-12-28 15:43:21 +01:00
|
|
|
|
private readonly Dictionary<int, Plugin> tokens = new ();
|
|
|
|
|
private readonly Random rand = new ();
|
2021-08-07 07:48:29 +02:00
|
|
|
|
|
2021-12-28 15:43:21 +01:00
|
|
|
|
private readonly FileCache fileCache = new ();
|
|
|
|
|
private readonly TwoKeyDictionary<int, string, InjectedString> notificationInjections = new (4, 1);
|
2021-08-07 07:48:29 +02:00
|
|
|
|
|
2021-12-28 15:25:19 +01:00
|
|
|
|
internal IEnumerable<InjectedString> NotificationInjections => notificationInjections.InnerValues;
|
2021-08-07 07:48:29 +02:00
|
|
|
|
internal ISet<Plugin> WithConfigureFunction { get; } = new HashSet<Plugin>();
|
|
|
|
|
|
|
|
|
|
public PluginBridge(PluginManager manager) {
|
|
|
|
|
manager.Reloaded += manager_Reloaded;
|
|
|
|
|
manager.Config.PluginChangedState += Config_PluginChangedState;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal int GetTokenFromPlugin(Plugin plugin) {
|
|
|
|
|
foreach (var kvp in tokens) {
|
|
|
|
|
if (kvp.Value.Equals(plugin)) {
|
|
|
|
|
return kvp.Key;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int token, attempts = 1000;
|
|
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
token = rand.Next();
|
|
|
|
|
} while (tokens.ContainsKey(token) && --attempts >= 0);
|
|
|
|
|
|
|
|
|
|
if (attempts < 0) {
|
|
|
|
|
token = -tokens.Count - 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tokens[token] = plugin;
|
|
|
|
|
return token;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal Plugin? GetPluginFromToken(int token) {
|
2022-06-09 00:41:43 +02:00
|
|
|
|
return tokens.TryGetValue(token, out var plugin) ? plugin : null;
|
2021-08-07 07:48:29 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Event handlers
|
|
|
|
|
|
2022-06-09 00:41:43 +02:00
|
|
|
|
private void manager_Reloaded(object? sender, PluginErrorEventArgs e) {
|
2021-08-07 07:48:29 +02:00
|
|
|
|
tokens.Clear();
|
|
|
|
|
fileCache.Clear();
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-09 00:41:43 +02:00
|
|
|
|
private void Config_PluginChangedState(object? sender, PluginChangedStateEventArgs e) {
|
2021-08-07 07:48:29 +02:00
|
|
|
|
if (!e.IsEnabled) {
|
|
|
|
|
int token = GetTokenFromPlugin(e.Plugin);
|
|
|
|
|
|
|
|
|
|
fileCache.Remove(token);
|
|
|
|
|
notificationInjections.Remove(token);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Utility methods
|
|
|
|
|
|
|
|
|
|
private string GetFullPathOrThrow(int token, PluginFolder folder, string path) {
|
|
|
|
|
Plugin? plugin = GetPluginFromToken(token);
|
|
|
|
|
string fullPath = plugin == null ? string.Empty : plugin.GetFullPathIfSafe(folder, path);
|
|
|
|
|
|
|
|
|
|
if (fullPath.Length == 0) {
|
|
|
|
|
throw folder switch {
|
|
|
|
|
PluginFolder.Data => new ArgumentException("File path has to be relative to the plugin data folder."),
|
|
|
|
|
PluginFolder.Root => new ArgumentException("File path has to be relative to the plugin root folder."),
|
|
|
|
|
_ => new ArgumentException($"Invalid folder type {folder}, this is a TweetDuck error.")
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
return fullPath;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string ReadFileUnsafe(int token, PluginFolder folder, string path, bool readCached) {
|
|
|
|
|
string fullPath = GetFullPathOrThrow(token, folder, path);
|
|
|
|
|
|
2022-06-09 00:41:43 +02:00
|
|
|
|
if (readCached && fileCache.TryGetValue(token, folder, path, out var cachedContents)) {
|
2021-08-07 07:48:29 +02:00
|
|
|
|
return cachedContents;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
return fileCache[token, folder, path] = File.ReadAllText(fullPath, Encoding.UTF8);
|
|
|
|
|
} catch (FileNotFoundException) {
|
|
|
|
|
throw new FileNotFoundException("File not found.");
|
|
|
|
|
} catch (DirectoryNotFoundException) {
|
|
|
|
|
throw new DirectoryNotFoundException("Directory not found.");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Public methods
|
|
|
|
|
|
|
|
|
|
public void WriteFile(int token, string path, string contents) {
|
|
|
|
|
string fullPath = GetFullPathOrThrow(token, PluginFolder.Data, path);
|
|
|
|
|
|
|
|
|
|
FileUtils.CreateDirectoryForFile(fullPath);
|
|
|
|
|
File.WriteAllText(fullPath, contents, Encoding.UTF8);
|
|
|
|
|
fileCache[token, PluginFolder.Data, path] = contents;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string ReadFile(int token, string path, bool cache) {
|
|
|
|
|
return ReadFileUnsafe(token, PluginFolder.Data, path, cache);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void DeleteFile(int token, string path) {
|
|
|
|
|
string fullPath = GetFullPathOrThrow(token, PluginFolder.Data, path);
|
|
|
|
|
|
|
|
|
|
fileCache.Remove(token, PluginFolder.Data, path);
|
|
|
|
|
File.Delete(fullPath);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool CheckFileExists(int token, string path) {
|
|
|
|
|
return File.Exists(GetFullPathOrThrow(token, PluginFolder.Data, path));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string ReadFileRoot(int token, string path) {
|
|
|
|
|
return ReadFileUnsafe(token, PluginFolder.Root, path, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool CheckFileExistsRoot(int token, string path) {
|
|
|
|
|
return File.Exists(GetFullPathOrThrow(token, PluginFolder.Root, path));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void InjectIntoNotificationsBefore(int token, string key, string search, string html) {
|
2021-12-28 15:25:19 +01:00
|
|
|
|
notificationInjections[token, key] = new InjectedString(InjectedString.Position.Before, search, html);
|
2021-08-07 07:48:29 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void InjectIntoNotificationsAfter(int token, string key, string search, string html) {
|
2021-12-28 15:25:19 +01:00
|
|
|
|
notificationInjections[token, key] = new InjectedString(InjectedString.Position.After, search, html);
|
2021-08-07 07:48:29 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void SetConfigurable(int token) {
|
|
|
|
|
Plugin? plugin = GetPluginFromToken(token);
|
|
|
|
|
|
|
|
|
|
if (plugin != null) {
|
|
|
|
|
WithConfigureFunction.Add(plugin);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private sealed class FileCache {
|
2021-12-28 15:43:21 +01:00
|
|
|
|
private readonly TwoKeyDictionary<int, string, string> cache = new (4, 2);
|
2021-08-07 07:48:29 +02:00
|
|
|
|
|
|
|
|
|
public string this[int token, PluginFolder folder, string path] {
|
|
|
|
|
set => cache[token, Key(folder, path)] = value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Clear() {
|
|
|
|
|
cache.Clear();
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-09 00:41:43 +02:00
|
|
|
|
public bool TryGetValue(int token, PluginFolder folder, string path, [MaybeNullWhen(false)] out string contents) {
|
2021-08-07 07:48:29 +02:00
|
|
|
|
return cache.TryGetValue(token, Key(folder, path), out contents);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Remove(int token) {
|
|
|
|
|
cache.Remove(token);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Remove(int token, PluginFolder folder, string path) {
|
|
|
|
|
cache.Remove(token, Key(folder, path));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static string Key(PluginFolder folder, string path) {
|
|
|
|
|
string prefix = folder switch {
|
|
|
|
|
PluginFolder.Root => "root/",
|
|
|
|
|
PluginFolder.Data => "data/",
|
|
|
|
|
_ => throw new InvalidOperationException($"Invalid folder type {folder}, this is a TweetDuck error.")
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return prefix + path.Replace('\\', '/').Trim();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-07-23 18:36:58 +02:00
|
|
|
|
}
|