2022-10-21 23:22:44 +00:00
|
|
|
/*
|
|
|
|
* Vencord, a modification for Discord's desktop app
|
|
|
|
* Copyright (c) 2022 Vendicated and contributors
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2023-06-16 17:07:22 +00:00
|
|
|
import "../suppressExperimentalWarnings.js";
|
2023-05-29 21:56:37 +00:00
|
|
|
import "../checkNodeVersion.js";
|
|
|
|
|
2022-10-23 21:23:52 +00:00
|
|
|
import { exec, execSync } from "child_process";
|
2023-11-09 01:32:34 +00:00
|
|
|
import { constants as FsConstants, readFileSync } from "fs";
|
|
|
|
import { access, readdir, readFile } from "fs/promises";
|
2022-12-25 19:47:35 +00:00
|
|
|
import { join, relative } from "path";
|
2022-10-23 21:23:52 +00:00
|
|
|
import { promisify } from "util";
|
2022-10-16 15:15:15 +00:00
|
|
|
|
2023-06-16 17:07:22 +00:00
|
|
|
// wtf is this assert syntax
|
|
|
|
import PackageJSON from "../../package.json" assert { type: "json" };
|
2023-08-15 23:32:11 +00:00
|
|
|
import { getPluginTarget } from "../utils.mjs";
|
2023-06-16 17:07:22 +00:00
|
|
|
|
|
|
|
export const VERSION = PackageJSON.version;
|
2023-08-25 12:15:02 +00:00
|
|
|
// https://reproducible-builds.org/docs/source-date-epoch/
|
|
|
|
export const BUILD_TIMESTAMP = Number(process.env.SOURCE_DATE_EPOCH) || Date.now();
|
2022-10-23 21:23:52 +00:00
|
|
|
export const watch = process.argv.includes("--watch");
|
2023-11-25 00:32:21 +00:00
|
|
|
export const isDev = watch || process.argv.includes("--dev");
|
2022-10-23 21:23:52 +00:00
|
|
|
export const isStandalone = JSON.stringify(process.argv.includes("--standalone"));
|
2023-08-25 12:15:02 +00:00
|
|
|
export const updaterDisabled = JSON.stringify(process.argv.includes("--disable-updater"));
|
|
|
|
export const gitHash = process.env.VENCORD_HASH || execSync("git rev-parse --short HEAD", { encoding: "utf-8" }).trim();
|
2022-11-07 20:29:31 +00:00
|
|
|
export const banner = {
|
|
|
|
js: `
|
|
|
|
// Vencord ${gitHash}
|
|
|
|
// Standalone: ${isStandalone}
|
|
|
|
// Platform: ${isStandalone === "false" ? process.platform : "Universal"}
|
2023-08-25 12:15:02 +00:00
|
|
|
// Updater disabled: ${updaterDisabled}
|
2022-11-07 20:29:31 +00:00
|
|
|
`.trim()
|
|
|
|
};
|
2022-10-16 15:15:15 +00:00
|
|
|
|
2023-03-11 13:18:32 +00:00
|
|
|
const isWeb = process.argv.slice(0, 2).some(f => f.endsWith("buildWeb.mjs"));
|
|
|
|
|
2023-11-09 01:32:34 +00:00
|
|
|
export function existsAsync(path) {
|
|
|
|
return access(path, FsConstants.F_OK)
|
|
|
|
.then(() => true)
|
|
|
|
.catch(() => false);
|
|
|
|
}
|
|
|
|
|
2022-10-16 15:15:15 +00:00
|
|
|
// https://github.com/evanw/esbuild/issues/619#issuecomment-751995294
|
|
|
|
/**
|
2022-12-25 19:47:35 +00:00
|
|
|
* @type {import("esbuild").Plugin}
|
2022-10-16 15:15:15 +00:00
|
|
|
*/
|
|
|
|
export const makeAllPackagesExternalPlugin = {
|
|
|
|
name: "make-all-packages-external",
|
|
|
|
setup(build) {
|
2022-10-20 17:21:21 +00:00
|
|
|
const filter = /^[^./]|^\.[^./]|^\.\.[^/]/; // Must not start with "/" or "./" or "../"
|
2022-10-16 15:15:15 +00:00
|
|
|
build.onResolve({ filter }, args => ({ path: args.path, external: true }));
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2023-04-05 02:09:42 +00:00
|
|
|
* @type {(kind: "web" | "discordDesktop" | "vencordDesktop") => import("esbuild").Plugin}
|
2022-10-16 15:15:15 +00:00
|
|
|
*/
|
2023-04-05 02:09:42 +00:00
|
|
|
export const globPlugins = kind => ({
|
2022-10-16 15:15:15 +00:00
|
|
|
name: "glob-plugins",
|
|
|
|
setup: build => {
|
2022-10-23 21:23:52 +00:00
|
|
|
const filter = /^~plugins$/;
|
|
|
|
build.onResolve({ filter }, args => {
|
2022-10-16 15:15:15 +00:00
|
|
|
return {
|
|
|
|
namespace: "import-plugins",
|
|
|
|
path: args.path
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
2022-10-23 21:23:52 +00:00
|
|
|
build.onLoad({ filter, namespace: "import-plugins" }, async () => {
|
2023-06-13 00:36:25 +00:00
|
|
|
const pluginDirs = ["plugins/_api", "plugins/_core", "plugins", "userplugins"];
|
2022-10-16 15:15:15 +00:00
|
|
|
let code = "";
|
|
|
|
let plugins = "\n";
|
2022-10-17 21:46:52 +00:00
|
|
|
let i = 0;
|
|
|
|
for (const dir of pluginDirs) {
|
2023-11-09 01:32:34 +00:00
|
|
|
if (!await existsAsync(`./src/${dir}`)) continue;
|
2022-10-17 21:46:52 +00:00
|
|
|
const files = await readdir(`./src/${dir}`);
|
|
|
|
for (const file of files) {
|
2023-06-13 00:36:25 +00:00
|
|
|
if (file.startsWith("_") || file.startsWith(".")) continue;
|
2023-03-10 23:25:32 +00:00
|
|
|
if (file === "index.ts") continue;
|
2023-06-13 00:36:25 +00:00
|
|
|
|
2023-08-15 23:32:11 +00:00
|
|
|
const target = getPluginTarget(file);
|
|
|
|
if (target) {
|
|
|
|
if (target === "dev" && !watch) continue;
|
|
|
|
if (target === "web" && kind === "discordDesktop") continue;
|
|
|
|
if (target === "desktop" && kind === "web") continue;
|
|
|
|
if (target === "discordDesktop" && kind !== "discordDesktop") continue;
|
|
|
|
if (target === "vencordDesktop" && kind !== "vencordDesktop") continue;
|
2023-03-11 13:18:32 +00:00
|
|
|
}
|
2023-03-10 23:25:32 +00:00
|
|
|
|
2022-10-17 21:46:52 +00:00
|
|
|
const mod = `p${i}`;
|
2023-01-25 02:42:01 +00:00
|
|
|
code += `import ${mod} from "./${dir}/${file.replace(/\.tsx?$/, "")}";\n`;
|
2022-10-17 21:46:52 +00:00
|
|
|
plugins += `[${mod}.name]:${mod},\n`;
|
|
|
|
i++;
|
2022-10-16 15:15:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
code += `export default {${plugins}};`;
|
|
|
|
return {
|
|
|
|
contents: code,
|
2022-10-17 21:46:52 +00:00
|
|
|
resolveDir: "./src"
|
2022-10-16 15:15:15 +00:00
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
2023-04-05 02:09:42 +00:00
|
|
|
});
|
2022-10-16 15:15:15 +00:00
|
|
|
|
|
|
|
/**
|
2022-12-25 19:47:35 +00:00
|
|
|
* @type {import("esbuild").Plugin}
|
2022-10-16 15:15:15 +00:00
|
|
|
*/
|
|
|
|
export const gitHashPlugin = {
|
|
|
|
name: "git-hash-plugin",
|
|
|
|
setup: build => {
|
2022-10-23 21:23:52 +00:00
|
|
|
const filter = /^~git-hash$/;
|
2022-10-16 15:15:15 +00:00
|
|
|
build.onResolve({ filter }, args => ({
|
|
|
|
namespace: "git-hash", path: args.path
|
|
|
|
}));
|
|
|
|
build.onLoad({ filter, namespace: "git-hash" }, () => ({
|
|
|
|
contents: `export default "${gitHash}"`
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
};
|
2022-10-22 02:41:33 +00:00
|
|
|
|
2022-10-23 21:23:52 +00:00
|
|
|
/**
|
2022-12-25 19:47:35 +00:00
|
|
|
* @type {import("esbuild").Plugin}
|
2022-10-23 21:23:52 +00:00
|
|
|
*/
|
|
|
|
export const gitRemotePlugin = {
|
|
|
|
name: "git-remote-plugin",
|
|
|
|
setup: build => {
|
|
|
|
const filter = /^~git-remote$/;
|
|
|
|
build.onResolve({ filter }, args => ({
|
|
|
|
namespace: "git-remote", path: args.path
|
|
|
|
}));
|
|
|
|
build.onLoad({ filter, namespace: "git-remote" }, async () => {
|
2023-08-25 12:15:02 +00:00
|
|
|
let remote = process.env.VENCORD_REMOTE;
|
|
|
|
if (!remote) {
|
2023-08-25 12:33:33 +00:00
|
|
|
const res = await promisify(exec)("git remote get-url origin", { encoding: "utf-8" });
|
2023-08-25 12:15:02 +00:00
|
|
|
remote = res.stdout.trim()
|
|
|
|
.replace("https://github.com/", "")
|
|
|
|
.replace("git@github.com:", "")
|
|
|
|
.replace(/.git$/, "");
|
|
|
|
}
|
2022-10-23 21:23:52 +00:00
|
|
|
|
|
|
|
return { contents: `export default "${remote}"` };
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-10-22 02:41:33 +00:00
|
|
|
/**
|
2022-12-25 19:47:35 +00:00
|
|
|
* @type {import("esbuild").Plugin}
|
2022-10-22 02:41:33 +00:00
|
|
|
*/
|
|
|
|
export const fileIncludePlugin = {
|
|
|
|
name: "file-include-plugin",
|
|
|
|
setup: build => {
|
2022-10-23 21:23:52 +00:00
|
|
|
const filter = /^~fileContent\/.+$/;
|
2022-10-22 02:41:33 +00:00
|
|
|
build.onResolve({ filter }, args => ({
|
|
|
|
namespace: "include-file",
|
|
|
|
path: args.path,
|
|
|
|
pluginData: {
|
|
|
|
path: join(args.resolveDir, args.path.slice("include-file/".length))
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
build.onLoad({ filter, namespace: "include-file" }, async ({ pluginData: { path } }) => {
|
|
|
|
const [name, format] = path.split(";");
|
|
|
|
return {
|
|
|
|
contents: `export default ${JSON.stringify(await readFile(name, format ?? "utf-8"))}`
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-12-25 19:47:35 +00:00
|
|
|
const styleModule = readFileSync("./scripts/build/module/style.js", "utf-8");
|
|
|
|
/**
|
|
|
|
* @type {import("esbuild").Plugin}
|
|
|
|
*/
|
|
|
|
export const stylePlugin = {
|
|
|
|
name: "style-plugin",
|
|
|
|
setup: ({ onResolve, onLoad }) => {
|
|
|
|
onResolve({ filter: /\.css\?managed$/, namespace: "file" }, ({ path, resolveDir }) => ({
|
|
|
|
path: relative(process.cwd(), join(resolveDir, path.replace("?managed", ""))),
|
|
|
|
namespace: "managed-style",
|
|
|
|
}));
|
|
|
|
onLoad({ filter: /\.css$/, namespace: "managed-style" }, async ({ path }) => {
|
|
|
|
const css = await readFile(path, "utf-8");
|
|
|
|
const name = relative(process.cwd(), path).replaceAll("\\", "/");
|
|
|
|
|
|
|
|
return {
|
|
|
|
loader: "js",
|
|
|
|
contents: styleModule
|
|
|
|
.replaceAll("STYLE_SOURCE", JSON.stringify(css))
|
|
|
|
.replaceAll("STYLE_NAME", JSON.stringify(name))
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-12-15 11:38:24 +00:00
|
|
|
/**
|
|
|
|
* @type {import("esbuild").Plugin}
|
|
|
|
*/
|
|
|
|
export const translationPlugin = {
|
|
|
|
name: "translation-plugin",
|
|
|
|
setup: ({ onResolve, onLoad }) => {
|
|
|
|
const filter = /^~translations$/;
|
|
|
|
|
|
|
|
onResolve({ filter }, ({ path }) => ({
|
|
|
|
namespace: "translations", path
|
|
|
|
}));
|
|
|
|
onLoad({ filter, namespace: "translations" }, async () => {
|
|
|
|
const translations = {};
|
|
|
|
const locales = await readdir("./translations");
|
|
|
|
|
|
|
|
for (const locale of locales) {
|
|
|
|
const translationBundles = await readdir(`./translations/${locale}`);
|
|
|
|
|
|
|
|
for (const bundle of translationBundles) {
|
|
|
|
const name = bundle.replace(/\.ftl$/, "");
|
|
|
|
|
|
|
|
// we map this in reverse order to the file structure as it's more logical in the code to do it this
|
|
|
|
// way (translations are retrieved by bundle name, not locale, but on the fs it makes more sense to
|
|
|
|
// sort them by locale)
|
|
|
|
translations[name] ??= {};
|
|
|
|
translations[name][locale] = await readFile(`./translations/${locale}/${bundle}`, "utf-8");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
contents: `export default ${JSON.stringify(translations)}`,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-10-22 02:41:33 +00:00
|
|
|
/**
|
2022-10-29 18:27:48 +00:00
|
|
|
* @type {import("esbuild").BuildOptions}
|
2022-10-22 02:41:33 +00:00
|
|
|
*/
|
|
|
|
export const commonOpts = {
|
|
|
|
logLevel: "info",
|
|
|
|
bundle: true,
|
|
|
|
watch,
|
|
|
|
minify: !watch,
|
|
|
|
sourcemap: watch ? "inline" : "",
|
|
|
|
legalComments: "linked",
|
2022-11-07 20:29:31 +00:00
|
|
|
banner,
|
2023-12-15 11:38:24 +00:00
|
|
|
plugins: [fileIncludePlugin, gitHashPlugin, gitRemotePlugin, stylePlugin, translationPlugin],
|
|
|
|
external: ["~plugins", "~git-hash", "~git-remote", "~translations", "/assets/*"],
|
2022-10-29 18:27:48 +00:00
|
|
|
inject: ["./scripts/build/inject/react.mjs"],
|
|
|
|
jsxFactory: "VencordCreateElement",
|
|
|
|
jsxFragment: "VencordFragment",
|
|
|
|
// Work around https://github.com/evanw/esbuild/issues/2460
|
|
|
|
tsconfig: "./scripts/build/tsconfig.esbuild.json"
|
2022-10-22 02:41:33 +00:00
|
|
|
};
|