Vencord/src/webpack/webpack.ts

237 lines
8 KiB
TypeScript
Raw Normal View History

2022-10-21 23:17:06 +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/>.
*/
2022-09-05 17:54:02 +00:00
import type { WebpackInstance } from "discord-types/other";
2022-10-22 16:18:41 +00:00
2022-10-14 19:34:35 +00:00
import { proxyLazy } from "../utils/proxyLazy";
2022-09-05 17:54:02 +00:00
2022-09-30 22:42:50 +00:00
export let _resolveReady: () => void;
/**
* Fired once a gateway connection to Discord has been established.
* This indicates that the core webpack modules have been initialised
*/
export const onceReady = new Promise<void>(r => _resolveReady = r);
2022-09-05 17:54:02 +00:00
export let wreq: WebpackInstance;
export let cache: WebpackInstance["c"];
2022-08-31 18:47:07 +00:00
export type FilterFn = (mod: any) => boolean;
export const filters = {
byProps: (props: string[]): FilterFn =>
props.length === 1
? m => m[props[0]] !== void 0
: m => props.every(p => m[p] !== void 0),
2022-09-27 14:57:46 +00:00
byDisplayName: (deezNuts: string): FilterFn => m => m.default?.displayName === deezNuts,
byCode: (...code: string[]): FilterFn => m => {
if (typeof m !== "function") return false;
const s = Function.prototype.toString.call(m);
for (const c of code) {
if (!s.includes(c)) return false;
}
return true;
2022-09-28 10:15:30 +00:00
},
2022-08-31 18:47:07 +00:00
};
export const subscriptions = new Map<FilterFn, CallbackFn>();
export const listeners = new Set<CallbackFn>();
export type CallbackFn = (mod: any) => void;
export function _initWebpack(instance: typeof window.webpackChunkdiscord_app) {
2022-09-05 17:54:02 +00:00
if (cache !== void 0) throw "no.";
2022-08-31 18:47:07 +00:00
wreq = instance.push([[Symbol()], {}, r => r]);
2022-09-05 17:54:02 +00:00
cache = wreq.c;
2022-08-31 18:47:07 +00:00
instance.pop();
}
export function find(filter: FilterFn, getDefault = true) {
if (typeof filter !== "function")
2022-10-01 00:27:28 +00:00
throw new Error("Invalid filter. Expected a function got " + typeof filter);
2022-08-31 18:47:07 +00:00
2022-09-05 17:54:02 +00:00
for (const key in cache) {
const mod = cache[key];
2022-09-27 12:34:57 +00:00
if (!mod?.exports) continue;
if (filter(mod.exports))
2022-08-31 18:47:07 +00:00
return mod.exports;
if (typeof mod.exports !== "object") continue;
2022-09-27 12:34:57 +00:00
if (mod.exports.default && filter(mod.exports.default))
2022-08-31 18:47:07 +00:00
return getDefault ? mod.exports.default : mod.exports;
// is 3 is the longest obfuscated export?
// the length check makes search about 20% faster
2022-10-09 20:15:14 +00:00
for (const nestedMod in mod.exports) if (nestedMod.length <= 3) {
2022-09-27 12:34:57 +00:00
const nested = mod.exports[nestedMod];
if (nested && filter(nested)) return nested;
}
2022-08-31 18:47:07 +00:00
}
return null;
}
export function findAll(filter: FilterFn, getDefault = true) {
2022-10-01 00:27:28 +00:00
if (typeof filter !== "function") throw new Error("Invalid filter. Expected a function got " + typeof filter);
2022-08-31 18:47:07 +00:00
const ret = [] as any[];
2022-09-05 17:54:02 +00:00
for (const key in cache) {
const mod = cache[key];
2022-09-30 22:42:50 +00:00
if (!mod?.exports) continue;
if (filter(mod.exports))
ret.push(mod.exports);
else if (typeof mod.exports !== "object")
continue;
if (mod.exports.default && filter(mod.exports.default))
ret.push(getDefault ? mod.exports.default : mod.exports);
2022-10-09 20:15:14 +00:00
else for (const nestedMod in mod.exports) if (nestedMod.length <= 3) {
2022-09-30 22:42:50 +00:00
const nested = mod.exports[nestedMod];
if (nested && filter(nested)) ret.push(nested);
}
2022-08-31 18:47:07 +00:00
}
return ret;
}
2022-10-14 19:34:35 +00:00
/**
* Finds a mangled module by the provided code "code" (must be unique and can be anywhere in the module)
* then maps it into an easily usable module via the specified mappers
* @param code Code snippet
* @param mappers Mappers to create the non mangled exports
* @returns Unmangled exports as specified in mappers
*
* @example mapMangledModule("headerIdIsManaged:", {
* openModal: filters.byCode("headerIdIsManaged:"),
* closeModal: filters.byCode("key==")
* })
*/
export function mapMangledModule<S extends string>(code: string, mappers: Record<S, FilterFn>): Record<S, any> {
const exports = {} as Record<S, any>;
// search every factory function
for (const id in wreq.m) {
const src = wreq.m[id].toString() as string;
if (src.includes(code)) {
const mod = wreq(id as any as number);
outer:
for (const key in mod) {
const member = mod[key];
for (const newName in mappers) {
// if the current mapper matches this module
if (mappers[newName](member)) {
exports[newName] = member;
continue outer;
}
}
}
break;
}
}
return exports;
}
/**
* Same as {@link mapMangledModule} but lazy
*/
export function mapMangledModuleLazy<S extends string>(code: string, mappers: Record<S, FilterFn>): Record<S, any> {
return proxyLazy(() => mapMangledModule(code, mappers));
}
2022-08-31 18:47:07 +00:00
export function findByProps(...props: string[]) {
return find(filters.byProps(props));
}
export function findAllByProps(...props: string[]) {
return findAll(filters.byProps(props));
}
export function findByDisplayName(deezNuts: string) {
return find(filters.byDisplayName(deezNuts));
}
export function waitFor(filter: string | string[] | FilterFn, callback: CallbackFn) {
if (typeof filter === "string") filter = filters.byProps([filter]);
else if (Array.isArray(filter)) filter = filters.byProps(filter);
2022-10-01 00:27:28 +00:00
else if (typeof filter !== "function") throw new Error("filter must be a string, string[] or function, got " + typeof filter);
2022-08-31 18:47:07 +00:00
const existing = find(filter!);
if (existing) return void callback(existing);
subscriptions.set(filter, callback);
}
export function addListener(callback: CallbackFn) {
listeners.add(callback);
}
export function removeListener(callback: CallbackFn) {
listeners.delete(callback);
2022-09-16 19:43:38 +00:00
}
/**
* Search modules by keyword. This searches the factory methods,
* meaning you can search all sorts of things, displayName, methodName, strings somewhere in the code, etc
* @param filters One or more strings or regexes
* @returns Mapping of found modules
*/
export function search(...filters: Array<string | RegExp>) {
const results = {} as Record<number, Function>;
const factories = wreq.m;
outer:
for (const id in factories) {
2022-10-01 23:05:15 +00:00
const factory = factories[id].original ?? factories[id];
2022-09-16 19:43:38 +00:00
const str: string = factory.toString();
for (const filter of filters) {
if (typeof filter === "string" && !str.includes(filter)) continue outer;
if (filter instanceof RegExp && !filter.test(str)) continue outer;
}
results[id] = factory;
}
return results;
}
/**
2022-09-16 20:12:34 +00:00
* Extract a specific module by id into its own Source File. This has no effect on
2022-09-16 19:43:38 +00:00
* the code, it is only useful to be able to look at a specific module without having
* to view a massive file. extract then returns the extracted module so you can jump to it.
2022-09-16 20:59:34 +00:00
* As mentioned above, note that this extracted module is not actually used,
2022-09-16 19:43:38 +00:00
* so putting breakpoints or similar will have no effect.
* @param id The id of the module to extract
*/
export function extract(id: number) {
const mod = wreq.m[id] as Function;
if (!mod) return null;
const code = `
// [EXTRACTED] WebpackModule${id}
// WARNING: This module was extracted to be more easily readable.
// This module is NOT ACTUALLY USED! This means putting breakpoints will have NO EFFECT!!
${mod.toString()}
//# sourceURL=ExtractedWebpackModule${id}
`;
const extracted = (0, eval)(code);
return extracted as Function;
2022-09-16 20:59:34 +00:00
}