From 426c949ee4752650b0e7e0dc4b18fcb296fcd514 Mon Sep 17 00:00:00 2001 From: Nuckyz <61953774+Nuckyz@users.noreply.github.com> Date: Wed, 19 Jun 2024 23:49:56 -0300 Subject: [PATCH] Future proof ContextMenuAPI against mangled export --- src/api/ContextMenu.ts | 8 +-- src/plugins/_api/contextMenu.ts | 56 ++++++++++++++++-- src/webpack/patchWebpack.ts | 44 +++++++------- src/webpack/webpack.ts | 101 ++++++++++++++++++-------------- 4 files changed, 136 insertions(+), 73 deletions(-) diff --git a/src/api/ContextMenu.ts b/src/api/ContextMenu.ts index fdd4facf4..66ea75a06 100644 --- a/src/api/ContextMenu.ts +++ b/src/api/ContextMenu.ts @@ -121,7 +121,7 @@ export function findGroupChildrenByChildId(id: string | string[], children: Arra } interface ContextMenuProps { - contextMenuApiArguments?: Array; + contextMenuAPIArguments?: Array; navId: string; children: Array; "aria-label": string; @@ -135,7 +135,7 @@ export function _usePatchContextMenu(props: ContextMenuProps) { children: cloneMenuChildren(props.children), }; - props.contextMenuApiArguments ??= []; + props.contextMenuAPIArguments ??= []; const contextMenuPatches = navPatches.get(props.navId); if (!Array.isArray(props.children)) props.children = [props.children]; @@ -143,7 +143,7 @@ export function _usePatchContextMenu(props: ContextMenuProps) { if (contextMenuPatches) { for (const patch of contextMenuPatches) { try { - patch(props.children, ...props.contextMenuApiArguments); + patch(props.children, ...props.contextMenuAPIArguments); } catch (err) { ContextMenuLogger.error(`Patch for ${props.navId} errored,`, err); } @@ -152,7 +152,7 @@ export function _usePatchContextMenu(props: ContextMenuProps) { for (const patch of globalPatches) { try { - patch(props.navId, props.children, ...props.contextMenuApiArguments); + patch(props.navId, props.children, ...props.contextMenuAPIArguments); } catch (err) { ContextMenuLogger.error("Global patch errored,", err); } diff --git a/src/plugins/_api/contextMenu.ts b/src/plugins/_api/contextMenu.ts index 01619546d..c877a3d3a 100644 --- a/src/plugins/_api/contextMenu.ts +++ b/src/plugins/_api/contextMenu.ts @@ -18,6 +18,38 @@ import { Devs } from "@utils/constants"; import definePlugin from "@utils/types"; +import { filters, waitFor, waitForSubscriptions } from "@webpack"; + +/** + * The last var name which the ContextMenu module was WebpackRequire'd and assigned to + */ +let lastVarName = ""; + +/** + * The key exporting the ContextMenu module "Menu" + */ +let exportKey: PropertyKey = ""; + +/** + * The id of the module exporting the ContextMenu module "Menu" + */ +let modId: PropertyKey = ""; + +let mangledCallback: (...args: any[]) => any; +waitFor(filters.byCode("Menu API only allows Items and groups of Items as children."), mangledCallback = (_, modInfo) => { + exportKey = modInfo.exportKey; + modId = modInfo.id; + + waitForSubscriptions.delete(nonMangledCallback); +}); + +let nonMangledCallback: (...args: any[]) => any; +waitFor(filters.byProps("Menu", "MenuItem"), nonMangledCallback = (_, modInfo) => { + exportKey = "Menu"; + modId = modInfo.id; + + waitForSubscriptions.delete(mangledCallback); +}); export default definePlugin({ name: "ContextMenuAPI", @@ -34,12 +66,26 @@ export default definePlugin({ } }, { - find: ".Menu,{", + find: "navId:", all: true, - replacement: { - match: /Menu,{(?<=\.jsxs?\)\(\i\.Menu,{)/g, - replace: "$&contextMenuApiArguments:typeof arguments!=='undefined'?arguments:[]," - } + noWarn: true, + replacement: [ + { + get match() { + return RegExp(`${String(modId)}(?<=(\\i)=.+?)`); + }, + replace: (id, varName) => { + lastVarName = varName; + return id; + } + }, + { + get match() { + return RegExp(`${String(exportKey)},{(?<=${lastVarName}\\.${String(exportKey)},{)`, "g"); + }, + replace: "$&contextMenuAPIArguments:typeof arguments!=='undefined'?arguments:[]," + } + ] } ] }); diff --git a/src/webpack/patchWebpack.ts b/src/webpack/patchWebpack.ts index 48f1b8147..da5603675 100644 --- a/src/webpack/patchWebpack.ts +++ b/src/webpack/patchWebpack.ts @@ -24,7 +24,7 @@ import { WebpackInstance } from "discord-types/other"; import { traceFunction } from "../debug/Tracer"; import { patches } from "../plugins"; -import { _initWebpack, beforeInitListeners, factoryListeners, moduleListeners, subscriptions, wreq } from "."; +import { _initWebpack, beforeInitListeners, factoryListeners, moduleListeners, waitForSubscriptions, wreq } from "."; const logger = new Logger("WebpackInterceptor", "#8caaee"); const initCallbackRegex = canonicalizeMatch(/{return \i\(".+?"\)}/); @@ -204,8 +204,7 @@ function patchFactories(factories: Record string, original: any, (...args: any[]): void; }; diff --git a/src/webpack/webpack.ts b/src/webpack/webpack.ts index b536063e8..93c721289 100644 --- a/src/webpack/webpack.ts +++ b/src/webpack/webpack.ts @@ -68,11 +68,21 @@ export const filters = { } }; -export type CallbackFn = (mod: any, id: string) => void; +export type ModListenerInfo = { + id: PropertyKey; + factory: (module: any, exports: any, require: WebpackInstance) => void; +}; + +export type ModCallbackInfo = ModListenerInfo & { + exportKey: PropertyKey; +}; + +export type ModListenerFn = (module: any, info: ModListenerInfo) => void; +export type ModCallbackFn = (module: any, info: ModCallbackInfo) => void; -export const subscriptions = new Map(); -export const moduleListeners = new Set(); export const factoryListeners = new Set<(factory: (module: any, exports: any, require: WebpackInstance) => void) => void>(); +export const moduleListeners = new Set(); +export const waitForSubscriptions = new Map(); export const beforeInitListeners = new Set<(wreq: WebpackInstance) => void>(); export function _initWebpack(webpackRequire: WebpackInstance) { @@ -106,7 +116,7 @@ export const find = traceFunction("find", function find(filter: FilterFn, { isIn for (const key in cache) { const mod = cache[key]; - if (!mod.loaded || !mod?.exports) continue; + if (!mod?.loaded || mod?.exports == null) continue; if (filter(mod.exports)) { return isWaitFor ? [mod.exports, key] : mod.exports; @@ -114,16 +124,13 @@ export const find = traceFunction("find", function find(filter: FilterFn, { isIn if (typeof mod.exports !== "object") continue; - if (mod.exports.default && filter(mod.exports.default)) { - const found = mod.exports.default; - return isWaitFor ? [found, key] : found; + if (mod.exports.default != null && filter(mod.exports.default)) { + return isWaitFor ? [mod.exports.default, key] : mod.exports.default; } - // the length check makes search about 20% faster - for (const nestedMod in mod.exports) if (nestedMod.length <= 3) { - const nested = mod.exports[nestedMod]; - if (nested && filter(nested)) { - return isWaitFor ? [nested, key] : nested; + for (const key in mod.exports) if (key.length <= 3) { + if (mod.exports[key] != null && filter(mod.exports[key])) { + return isWaitFor ? [mod.exports[key], key] : mod.exports[key]; } } } @@ -142,18 +149,25 @@ export function findAll(filter: FilterFn) { const ret = [] as any[]; for (const key in cache) { const mod = cache[key]; - if (!mod.loaded || !mod?.exports) continue; + if (!mod?.loaded || mod?.exports == null) continue; - if (filter(mod.exports)) + if (filter(mod.exports)) { ret.push(mod.exports); - else if (typeof mod.exports !== "object") continue; + } - if (mod.exports.default && filter(mod.exports.default)) + if (typeof mod.exports !== "object") continue; + + if (mod.exports.default != null && filter(mod.exports.default)) { ret.push(mod.exports.default); - else for (const nestedMod in mod.exports) if (nestedMod.length <= 3) { - const nested = mod.exports[nestedMod]; - if (nested && filter(nested)) ret.push(nested); + continue; + } + + for (const key in mod.exports) if (key.length <= 3) { + if (mod.exports[key] && filter(mod.exports[key])) { + ret.push(mod.exports[key]); + break; + } } } @@ -190,7 +204,7 @@ export const findBulk = traceFunction("findBulk", function findBulk(...filterFns outer: for (const key in cache) { const mod = cache[key]; - if (!mod.loaded || !mod?.exports) continue; + if (!mod.loaded || mod?.exports == null) continue; for (let j = 0; j < length; j++) { const filter = filters[j]; @@ -204,26 +218,23 @@ export const findBulk = traceFunction("findBulk", function findBulk(...filterFns break; } - if (typeof mod.exports !== "object") - continue; + if (typeof mod.exports !== "object") continue; if (mod.exports.default && filter(mod.exports.default)) { results[j] = mod.exports.default; filters[j] = undefined; if (++found === length) break outer; - break; + continue; } - for (const nestedMod in mod.exports) - if (nestedMod.length <= 3) { - const nested = mod.exports[nestedMod]; - if (nested && filter(nested)) { - results[j] = nested; - filters[j] = undefined; - if (++found === length) break outer; - continue outer; - } + for (const key in mod.exports) if (key.length <= 3) { + if (mod.exports[key] && filter(mod.exports[key])) { + results[j] = mod.exports[key]; + filters[j] = undefined; + if (++found === length) break outer; + break; } + } } } @@ -293,7 +304,7 @@ export const lazyWebpackSearchHistory = [] as Array<["find" | "findByProps" | "f * Note that the example below exists already as an api, see {@link findByPropsLazy} * @example const mod = proxyLazy(() => findByProps("blah")); console.log(mod.blah); */ -export function proxyLazyWebpack(factory: () => any, attempts?: number) { +export function proxyLazyWebpack(factory: () => T, attempts?: number) { if (IS_REPORTER) lazyWebpackSearchHistory.push(["proxyLazyWebpack", [factory]]); return proxyLazy(factory, attempts); @@ -446,25 +457,27 @@ export function findExportedComponentLazy(...props: stri * }) */ export const mapMangledModule = traceFunction("mapMangledModule", function mapMangledModule(code: string, mappers: Record): Record { - const exports = {} as Record; + const mapping = {} as Record; const id = findModuleId(code); - if (id === null) - return exports; + if (id === null) return mapping; + + const exports = wreq(id as any); - const mod = wreq(id as any); outer: - for (const key in mod) { - const member = mod[key]; + for (const key in exports) { + const value = exports[key]; + for (const newName in mappers) { // if the current mapper matches this module - if (mappers[newName](member)) { - exports[newName] = member; + if (mappers[newName](value)) { + mapping[newName] = value; continue outer; } } } - return exports; + + return mapping; }); /** @@ -570,7 +583,7 @@ export function extractAndLoadChunksLazy(code: string[], matcher = DefaultExtrac * Wait for a module that matches the provided filter to be registered, * then call the callback with the module as the first argument */ -export function waitFor(filter: string | string[] | FilterFn, callback: CallbackFn, { isIndirect = false }: { isIndirect?: boolean; } = {}) { +export function waitFor(filter: string | string[] | FilterFn, callback: ModCallbackFn, { isIndirect = false }: { isIndirect?: boolean; } = {}) { if (IS_REPORTER && !isIndirect) lazyWebpackSearchHistory.push(["waitFor", Array.isArray(filter) ? filter : [filter]]); if (typeof filter === "string") @@ -585,7 +598,7 @@ export function waitFor(filter: string | string[] | FilterFn, callback: Callback if (existing) return void callback(existing, id); } - subscriptions.set(filter, callback); + waitForSubscriptions.set(filter, callback); } /**