Equicord/src/equicordplugins/webpackTarball/index.tsx

159 lines
5.6 KiB
TypeScript
Raw Normal View History

2024-07-18 02:14:43 -04:00
/*
* Vencord, a Discord client mod
* Copyright (c) 2024 Vendicated and contributors
* SPDX-License-Identifier: GPL-3.0-or-later
*/
import { definePluginSettings } from "@api/Settings";
import { Devs, WEBPACK_CHUNK } from "@utils/constants";
import { makeLazy } from "@utils/lazy";
import { closeModal, ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalProps, ModalRoot, openModal } from "@utils/modal";
import definePlugin, { OptionType } from "@utils/types";
import { findByProps, wreq } from "@webpack";
import { Button, Flex, Forms, Switch, Text, Timestamp, useState } from "@webpack/common";
import TarFile from "./tar";
import * as Webpack from "./webpack";
export const settings = definePluginSettings({
patched: {
type: OptionType.BOOLEAN,
default: true,
description: "Include patched modules",
restartNeeded: true,
},
});
export default definePlugin({
name: "WebpackTarball",
description: "Converts Discord's webpack sources into a tarball.",
authors: [Devs.Kyuuhachi],
settings,
toolboxActions: {
"Webpack Tarball"() {
const key = openModal(props => (
<TarModal
modalProps={props}
close={() => closeModal(key)}
/>
));
}
},
});
export const getBuildNumber = makeLazy(() => {
try {
const metrics = findByProps("_getMetricWithDefaults")._flush.toString();
const [, builtAt, buildNumber] = metrics.match(/\{built_at:"(\d+)",build_number:"(\d+)"\}/);
return { buildNumber, builtAt: new Date(Number(builtAt)) };
} catch (e) {
console.error("failed to get build number:", e);
return { buildNumber: "unknown", builtAt: new Date() };
}
});
async function saveTar(patched: boolean) {
const tar = new TarFile();
const { buildNumber, builtAt } = getBuildNumber();
const mtime = (builtAt.getTime() / 1000) | 0;
const root = patched ? `vencord-${buildNumber}` : `discord-${buildNumber}`;
for (const [id, module] of Object.entries(wreq.m)) {
const patchedSrc = Function.toString.call(module);
const originalSrc = module.toString();
if (patched && patchedSrc !== originalSrc)
tar.addTextFile(
`${root}/${id}.v.js`,
`webpack[${JSON.stringify(id)}] = ${patchedSrc}\n`,
{ mtime },
);
tar.addTextFile(
`${root}/${id}.js`,
`webpack[${JSON.stringify(id)}] = ${originalSrc}\n`,
{ mtime },
);
}
tar.save(`${root}.tar`);
}
function TarModal({ modalProps, close }: { modalProps: ModalProps; close(): void; }) {
const { buildNumber, builtAt } = getBuildNumber();
const [, rerender] = useState({});
const [isLoading, setLoading] = useState(false);
const paths = Webpack.getChunkPaths(wreq);
const status = Object.entries(Webpack.getLoadedChunks(wreq))
.filter(([k]) => wreq.o(paths, k))
.map(([, v]) => v);
const loading = status.length;
const loaded = status.filter(v => v === 0 || v === undefined).length;
const errored = status.filter(v => v === undefined).length;
const all = Object.keys(paths).length;
const { patched } = settings.use(["patched"]);
return (
<ModalRoot {...modalProps}>
<ModalHeader>
<Forms.FormTitle tag="h2">
Webpack Tarball
</Forms.FormTitle>
<Text variant="text-md/normal">
<Timestamp timestamp={new Date(builtAt)} isInline={false}>
{"Build number "}
{buildNumber}
</Timestamp>
</Text>
<ModalCloseButton onClick={close} />
</ModalHeader>
<ModalContent>
<div style={{ marginTop: "8px", marginBottom: "24px" }}>
<Forms.FormTitle>
Lazy chunks
</Forms.FormTitle>
<Flex align={Flex.Align.CENTER}>
<Text
variant="text-md/normal"
style={{ flexGrow: 1 }}
>
{loaded}/{all}
{errored ? ` (${errored} errors)` : null}
</Text>
<Button
disabled={loading === all || isLoading}
onClick={async () => {
setLoading(true);
// @ts-ignore
await Webpack.protectWebpack(window[WEBPACK_CHUNK], async () => {
await Webpack.forceLoadAll(wreq, rerender);
});
}}
>
{loaded === all ? "Loaded" : loading === all ? "Loading" : "Load all"}
</Button>
</Flex>
</div>
<Switch
value={patched}
onChange={v => settings.store.patched = v}
hideBorder
>
{settings.def.patched.description}
</Switch>
</ModalContent>
<ModalFooter>
<Button
onClick={() => {
saveTar(patched);
close();
}}
>
Create
</Button>
</ModalFooter>
</ModalRoot>
);
}