2023-01-03 00:18:19 +00:00
|
|
|
import { Indexable, PluginManifest, Plugin } from "@types";
|
2023-01-10 22:05:40 +00:00
|
|
|
import { navigation } from "@metro/common";
|
2023-01-07 23:05:14 +00:00
|
|
|
import logger from "@lib/logger";
|
2023-01-10 22:05:40 +00:00
|
|
|
import createStorage from "@lib/storage";
|
|
|
|
import PluginSettings from "@/ui/settings/components/PluginSettings";
|
2023-01-03 08:05:16 +00:00
|
|
|
|
|
|
|
type EvaledPlugin = {
|
|
|
|
onLoad?(): void;
|
|
|
|
onUnload(): void;
|
2023-01-05 23:07:58 +00:00
|
|
|
settings: JSX.Element;
|
2023-01-03 08:05:16 +00:00
|
|
|
};
|
2023-01-03 00:18:19 +00:00
|
|
|
|
2023-01-10 08:05:03 +00:00
|
|
|
export const plugins = createStorage<Indexable<Plugin>>("VENDETTA_PLUGINS", async function(parsed) {
|
|
|
|
for (let p of Object.keys(parsed)) {
|
|
|
|
const plugin: Plugin = parsed[p];
|
2023-01-03 00:18:19 +00:00
|
|
|
|
2023-01-10 08:05:03 +00:00
|
|
|
if (parsed[p].update) {
|
|
|
|
await fetchPlugin(plugin.id);
|
2023-01-07 23:05:14 +00:00
|
|
|
} else {
|
2023-01-10 08:05:03 +00:00
|
|
|
plugins[p] = parsed[p];
|
2023-01-07 23:05:14 +00:00
|
|
|
}
|
2023-01-03 00:18:19 +00:00
|
|
|
|
2023-01-10 08:05:03 +00:00
|
|
|
if (parsed[p].enabled && plugins[p]) startPlugin(p);
|
|
|
|
}
|
|
|
|
});
|
2023-01-07 23:05:14 +00:00
|
|
|
const loadedPlugins: Indexable<EvaledPlugin> = {};
|
|
|
|
|
|
|
|
export async function fetchPlugin(id: string) {
|
|
|
|
if (!id.endsWith("/")) id += "/";
|
|
|
|
if (typeof id !== "string" || id in plugins) throw new Error("Plugin ID invalid or taken");
|
2023-01-03 00:18:19 +00:00
|
|
|
|
|
|
|
let pluginManifest: PluginManifest;
|
|
|
|
|
|
|
|
try {
|
2023-01-07 23:05:14 +00:00
|
|
|
pluginManifest = await (await fetch(new URL("manifest.json", id), { cache: "no-store" })).json();
|
2023-01-03 00:18:19 +00:00
|
|
|
} catch {
|
2023-01-07 23:05:14 +00:00
|
|
|
throw new Error(`Failed to fetch manifest for ${id}`);
|
2023-01-03 00:18:19 +00:00
|
|
|
}
|
|
|
|
|
2023-01-03 08:05:16 +00:00
|
|
|
let pluginJs: string;
|
|
|
|
|
2023-01-04 08:01:56 +00:00
|
|
|
// TODO: Remove duplicate error if possible
|
2023-01-03 08:05:16 +00:00
|
|
|
try {
|
2023-01-04 22:39:28 +00:00
|
|
|
// by polymanifest spec, plugins should always specify their main file, but just in case
|
2023-01-07 23:05:14 +00:00
|
|
|
pluginJs = await (await fetch(new URL(pluginManifest.main || "index.js", id), { cache: "no-store" })).text();
|
2023-01-03 08:05:16 +00:00
|
|
|
} catch {
|
2023-01-07 23:05:14 +00:00
|
|
|
throw new Error(`Failed to fetch JS for ${id}`);
|
2023-01-03 08:05:16 +00:00
|
|
|
}
|
|
|
|
|
2023-01-07 23:05:14 +00:00
|
|
|
if (pluginJs.length === 0) throw new Error(`Failed to fetch JS for ${id}`);
|
2023-01-04 08:01:56 +00:00
|
|
|
|
2023-01-03 08:05:16 +00:00
|
|
|
plugins[id] = {
|
|
|
|
id: id,
|
2023-01-03 00:18:19 +00:00
|
|
|
manifest: pluginManifest,
|
2023-01-03 08:05:16 +00:00
|
|
|
enabled: false,
|
2023-01-07 23:05:14 +00:00
|
|
|
update: true,
|
2023-01-03 08:05:16 +00:00
|
|
|
js: pluginJs,
|
2023-01-03 00:18:19 +00:00
|
|
|
};
|
2023-01-03 08:05:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function evalPlugin(plugin: Plugin) {
|
|
|
|
// TODO: Refactor to not depend on own window object
|
2023-01-10 22:05:40 +00:00
|
|
|
const vendettaForPlugins = {
|
|
|
|
...window.vendetta,
|
|
|
|
plugin: {
|
|
|
|
manifest: plugin.manifest,
|
|
|
|
storage: createStorage<Indexable<any>>(plugin.id),
|
|
|
|
showSettings: () => showSettings(plugin),
|
|
|
|
}
|
|
|
|
};
|
2023-01-04 08:01:56 +00:00
|
|
|
const pluginString = `vendetta=>{return ${plugin.js}}\n//# sourceURL=${plugin.id}`;
|
2023-01-03 08:05:16 +00:00
|
|
|
|
2023-01-04 08:01:56 +00:00
|
|
|
const raw = (0, eval)(pluginString)(vendettaForPlugins);
|
|
|
|
const ret = typeof raw == "function" ? raw() : raw;
|
|
|
|
return ret.default || ret;
|
2023-01-03 08:05:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function startPlugin(id: string) {
|
|
|
|
const plugin = plugins[id];
|
|
|
|
if (!plugin) throw new Error("Attempted to start non-existent plugin");
|
|
|
|
|
|
|
|
try {
|
|
|
|
const pluginRet: EvaledPlugin = evalPlugin(plugin);
|
|
|
|
loadedPlugins[id] = pluginRet;
|
|
|
|
pluginRet.onLoad?.();
|
|
|
|
plugin.enabled = true;
|
|
|
|
} catch(e) {
|
|
|
|
logger.error(`Plugin ${plugin.id} errored whilst loading, and will be unloaded`, e);
|
|
|
|
|
|
|
|
try {
|
|
|
|
loadedPlugins[plugin.id]?.onUnload?.();
|
|
|
|
} catch(e2) {
|
|
|
|
logger.error(`Plugin ${plugin.id} errored whilst unloading`, e2);
|
|
|
|
}
|
|
|
|
|
|
|
|
delete loadedPlugins[id];
|
|
|
|
plugin.enabled = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function stopPlugin(id: string) {
|
|
|
|
const plugin = plugins[id];
|
|
|
|
const pluginRet = loadedPlugins[id];
|
|
|
|
if (!plugin) throw new Error("Attempted to stop non-existent plugin");
|
|
|
|
if (!pluginRet) throw new Error("Attempted to stop a non-started plugin");
|
|
|
|
|
|
|
|
try {
|
|
|
|
loadedPlugins[plugin.id]?.onUnload?.();
|
|
|
|
} catch(e) {
|
|
|
|
logger.error(`Plugin ${plugin.id} errored whilst unloading`, e);
|
|
|
|
}
|
|
|
|
|
|
|
|
delete loadedPlugins[id];
|
|
|
|
plugin.enabled = false;
|
|
|
|
}
|
|
|
|
|
2023-01-07 23:05:14 +00:00
|
|
|
export function removePlugin(id: string) {
|
2023-01-07 23:21:29 +00:00
|
|
|
const plugin = plugins[id];
|
|
|
|
if (plugin.enabled) stopPlugin(id);
|
2023-01-07 23:05:14 +00:00
|
|
|
delete plugins[id];
|
|
|
|
}
|
|
|
|
|
2023-01-05 23:07:58 +00:00
|
|
|
export const getSettings = (id: string) => loadedPlugins[id]?.settings;
|
2023-01-10 22:05:40 +00:00
|
|
|
|
|
|
|
export function showSettings(plugin: Plugin) {
|
|
|
|
const settings = getSettings(plugin.id);
|
|
|
|
if (!settings) return logger.error(`Plugin ${plugin.id} is not loaded or has no settings`);
|
|
|
|
|
|
|
|
navigation.push(PluginSettings, {
|
|
|
|
plugin: plugin,
|
|
|
|
children: settings,
|
|
|
|
});
|
|
|
|
}
|