2024-05-03 02:18:12 +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/>.
* /
import { proxyLazy } from "@utils/lazy" ;
import { LazyComponent } from "@utils/lazyReact" ;
import { Logger } from "@utils/Logger" ;
import { canonicalizeMatch } from "@utils/patches" ;
import { proxyInner , proxyInnerValue } from "@utils/proxyInner" ;
import type { WebpackInstance } from "discord-types/other" ;
import { traceFunction } from "../debug/Tracer" ;
const logger = new Logger ( "Webpack" ) ;
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 ) ;
export let wreq : WebpackInstance ;
export let cache : WebpackInstance [ "c" ] ;
export type FilterFn = ( mod : any ) = > boolean ;
export const filters = {
byProps : ( . . . props : string [ ] ) : FilterFn = > {
const filter = props . length === 1
? m = > m ? . [ props [ 0 ] ] !== void 0
: m = > props . every ( p = > m ? . [ p ] !== void 0 ) ;
// @ts-ignore
filter . $ $vencordProps = [ "byProps" , . . . props ] ;
return filter ;
} ,
byCode : ( . . . code : string [ ] ) : FilterFn = > {
const filter = 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 ;
} ;
filter . $ $vencordProps = [ "byCode" , . . . code ] ;
return filter ;
} ,
byStoreName : ( name : string ) : FilterFn = > {
const filter = m = > m ? . constructor ? . displayName === name ;
filter . $ $vencordProps = [ "byStoreName" , name ] ;
return filter ;
} ,
componentByCode : ( . . . code : string [ ] ) : FilterFn = > {
const filter = filters . byCode ( . . . code ) ;
const wrapper = m = > {
if ( filter ( m ) ) return true ;
if ( ! m ? . $ $typeof ) return false ;
if ( m ? . type && m . type . render ) return filter ( m . type . render ) ; // memo + forwardRef
if ( m ? . type ) return filter ( m . type ) ; // memos
if ( m ? . render ) return filter ( m . render ) ; // forwardRefs
return false ;
} ;
wrapper . $ $vencordProps = [ "componentByCode" , . . . code ] ;
return wrapper ;
}
} ;
export type ModCallbackFn = ( mod : any ) = > void ;
export type ModCallbackFnWithId = ( mod : any , id : string ) = > void ;
export const waitForSubscriptions = new Map < FilterFn , ModCallbackFn > ( ) ;
export const moduleListeners = new Set < ModCallbackFnWithId > ( ) ;
export const factoryListeners = new Set < ( factory : ( module : any , exports : any , require : WebpackInstance ) = > void ) = > void > ( ) ;
export const beforeInitListeners = new Set < ( wreq : WebpackInstance ) = > void > ( ) ;
export function _initWebpack ( webpackRequire : WebpackInstance ) {
wreq = webpackRequire ;
cache = webpackRequire . c ;
}
let devToolsOpen = false ;
if ( IS_DEV && IS_DISCORD_DESKTOP ) {
// At this point in time, DiscordNative has not been exposed yet, so setImmediate is needed
setTimeout ( ( ) = > {
DiscordNative /* just to make sure */ ? . window . setDevtoolsCallbacks ( ( ) = > devToolsOpen = true , ( ) = > devToolsOpen = false ) ;
} , 0 ) ;
}
export const webpackSearchHistory = [ ] as Array < [ "waitFor" | "find" | "findComponent" | "findExportedComponent" | "findComponentByCode" | "findByProps" | "findByCode" | "findStore" | "extractAndLoadChunks" | "webpackDependantLazy" | "webpackDependantLazyComponent" , any [ ] ] > ;
function printFilter ( filter : FilterFn ) {
if ( "$$vencordProps" in filter ) {
const props = filter . $ $vencordProps as string [ ] ;
return ` ${ props [ 0 ] } ( ${ props . slice ( 1 ) . map ( arg = > ` " ${ arg } " ` ) . join ( ", " ) } ) ` ;
}
return filter . toString ( ) ;
}
/ * *
* Wait for the first module that matches the provided filter to be required ,
* then call the callback with the module as the first argument .
*
* If the module is already required , the callback will be called immediately .
* @param filter A function that takes a module and returns a boolean
* @param callback A function that takes the found module as its first argument
* /
export function waitFor ( filter : FilterFn , callback : ModCallbackFn , { isIndirect = false } : { isIndirect? : boolean ; } = { } ) {
if ( typeof filter !== "function" )
throw new Error ( "Invalid filter. Expected a function got " + typeof filter ) ;
if ( typeof callback !== "function" )
throw new Error ( "Invalid callback. Expected a function got " + typeof callback ) ;
2024-05-07 04:13:46 +00:00
if ( IS_DEV && ! isIndirect ) {
const originalCallback = callback ;
let callbackCalled = false ;
callback = function ( ) {
callbackCalled = true ;
// @ts-ignore
originalCallback ( . . . arguments ) ;
} ;
// @ts-ignore
callback . $ $vencordCallbackCalled = ( ) = > callbackCalled ;
webpackSearchHistory . push ( [ "waitFor" , [ callback , filter ] ] ) ;
}
2024-05-03 02:18:12 +00:00
if ( cache != null ) {
const existing = cacheFind ( filter ) ;
if ( existing ) return callback ( existing ) ;
}
waitForSubscriptions . set ( filter , callback ) ;
}
/ * *
* Find the first module that matches the filter .
*
* The way this works internally is :
* Wait for the first module that matches the provided filter to be required ,
* then call the callback with the module as the first argument .
*
* If the module is already required , the callback will be called immediately .
*
* The callback must return a value that will be used as the proxy inner value .
*
2024-05-07 09:23:36 +00:00
* If no callback is specified , the default callback will assign the proxy inner value to all the module .
2024-05-03 02:18:12 +00:00
* @param filter A function that takes a module and returns a boolean
2024-05-07 09:23:36 +00:00
* @param callback A function that takes the found module as its first argument and returns something to use as the proxy inner value . Useful if you want to use a value from the module , instead of all of it . Defaults to the module itself
2024-05-03 02:18:12 +00:00
* @returns A proxy that has the callback return value as its true value , or the callback return value if the callback was called when the function was called
* /
export function find < T = any > ( filter : FilterFn , callback : ( mod : any ) = > any = m = > m , { isIndirect = false } : { isIndirect? : boolean ; } = { } ) {
if ( typeof filter !== "function" )
throw new Error ( "Invalid filter. Expected a function got " + typeof filter ) ;
if ( typeof callback !== "function" )
throw new Error ( "Invalid callback. Expected a function got " + typeof callback ) ;
2024-05-04 20:49:20 +00:00
const [ proxy , setInnerValue ] = proxyInner < T > ( ` Webpack find matched no module. Filter: ${ printFilter ( filter ) } ` , "Webpack find with proxy called on a primitive value. This can happen if you try to destructure a primitive in the top level definition of the find." ) ;
2024-05-03 02:18:12 +00:00
waitFor ( filter , mod = > setInnerValue ( callback ( mod ) ) , { isIndirect : true } ) ;
2024-05-07 04:13:46 +00:00
if ( IS_DEV && ! isIndirect ) {
webpackSearchHistory . push ( [ "find" , [ proxy , filter ] ] ) ;
}
2024-05-03 02:18:12 +00:00
if ( proxy [ proxyInnerValue ] != null ) return proxy [ proxyInnerValue ] as T ;
return proxy ;
}
/ * *
* Find the first component that matches the filter .
* @param filter A function that takes a module and returns a boolean
* @param parse A function that takes the found component as its first argument and returns a component . Useful if you want to wrap the found component in something . Defaults to the original component
* @returns The component if found , or a noop component
* /
export function findComponent < T extends object = any > ( filter : FilterFn , parse : ( component : any ) = > React . ComponentType < T > = m = > m , { isIndirect = false } : { isIndirect? : boolean ; } = { } ) {
if ( typeof filter !== "function" )
throw new Error ( "Invalid filter. Expected a function got " + typeof filter ) ;
if ( typeof parse !== "function" )
throw new Error ( "Invalid component parse. Expected a function got " + typeof parse ) ;
2024-05-07 04:13:46 +00:00
let InnerComponent = null as React . ComponentType < T > | null ;
2024-05-07 02:22:36 +00:00
let findFailedLogged = false ;
const WrapperComponent = ( props : T ) = > {
if ( InnerComponent === null && ! findFailedLogged ) {
findFailedLogged = true ;
2024-05-03 02:18:12 +00:00
logger . error ( ` Webpack find matched no module. Filter: ${ printFilter ( filter ) } ` ) ;
}
2024-05-07 02:22:36 +00:00
return InnerComponent && < InnerComponent { ...props } / > ;
2024-05-03 02:18:12 +00:00
} ;
waitFor ( filter , ( v : any ) = > {
const parsedComponent = parse ( v ) ;
InnerComponent = parsedComponent ;
Object . assign ( InnerComponent , parsedComponent ) ;
} , { isIndirect : true } ) ;
2024-05-07 04:13:46 +00:00
if ( IS_DEV ) {
WrapperComponent . $ $vencordInner = ( ) = > InnerComponent ;
if ( ! isIndirect ) {
webpackSearchHistory . push ( [ "findComponent" , [ WrapperComponent , filter ] ] ) ;
}
}
2024-05-07 02:22:36 +00:00
if ( InnerComponent !== null ) return InnerComponent ;
2024-05-03 02:18:12 +00:00
return WrapperComponent ;
}
/ * *
* Find the first component that is exported by the first prop name .
*
* @example findExportedComponent ( "FriendRow" )
* @example findExportedComponent ( "FriendRow" , "Friend" , FriendRow = > React . memo ( FriendRow ) )
*
* @param props A list of prop names to search the exports for
* @param parse A function that takes the found component as its first argument and returns a component . Useful if you want to wrap the found component in something . Defaults to the original component
* @returns The component if found , or a noop component
* /
export function findExportedComponent < T extends object = any > ( . . . props : string [ ] | [ . . . string [ ] , ( component : any ) = > React . ComponentType < T > ] ) {
const parse = ( typeof props . at ( - 1 ) === "function" ? props . pop ( ) : m = > m ) as ( component : any ) = > React . ComponentType < T > ;
const newProps = props as string [ ] ;
const filter = filters . byProps ( . . . newProps ) ;
2024-05-07 04:13:46 +00:00
let InnerComponent = null as React . ComponentType < T > | null ;
2024-05-07 02:22:36 +00:00
let findFailedLogged = false ;
const WrapperComponent = ( props : T ) = > {
if ( InnerComponent === null && ! findFailedLogged ) {
findFailedLogged = true ;
2024-05-03 02:18:12 +00:00
logger . error ( ` Webpack find matched no module. Filter: ${ printFilter ( filter ) } ` ) ;
}
2024-05-07 02:22:36 +00:00
return InnerComponent && < InnerComponent { ...props } / > ;
2024-05-03 02:18:12 +00:00
} ;
waitFor ( filter , ( v : any ) = > {
const parsedComponent = parse ( v [ newProps [ 0 ] ] ) ;
InnerComponent = parsedComponent ;
Object . assign ( InnerComponent , parsedComponent ) ;
} , { isIndirect : true } ) ;
2024-05-07 04:13:46 +00:00
if ( IS_DEV ) {
WrapperComponent . $ $vencordInner = ( ) = > InnerComponent ;
webpackSearchHistory . push ( [ "findExportedComponent" , [ WrapperComponent , . . . props ] ] ) ;
}
2024-05-07 02:22:36 +00:00
if ( InnerComponent !== null ) return InnerComponent ;
2024-05-03 02:18:12 +00:00
return WrapperComponent as React . ComponentType < T > ;
}
/ * *
2024-05-07 09:28:43 +00:00
* Find the first component in a default export that includes all the given code .
2024-05-03 02:18:12 +00:00
*
* @example findComponentByCode ( ".Messages.USER_SETTINGS_PROFILE_COLOR_SELECT_COLOR" )
* @example findComponentByCode ( ".Messages.USER_SETTINGS_PROFILE_COLOR_SELECT_COLOR" , ".BACKGROUND_PRIMARY)" , ColorPicker = > React . memo ( ColorPicker ) )
*
* @param code A list of code to search each export for
* @param parse A function that takes the found component as its first argument and returns a component . Useful if you want to wrap the found component in something . Defaults to the original component
* @returns The component if found , or a noop component
* /
export function findComponentByCode < T extends object = any > ( . . . code : string [ ] | [ . . . string [ ] , ( component : any ) = > React . ComponentType < T > ] ) {
const parse = ( typeof code . at ( - 1 ) === "function" ? code . pop ( ) : m = > m ) as ( component : any ) = > React . ComponentType < T > ;
const newCode = code as string [ ] ;
2024-05-07 04:13:46 +00:00
const ComponentResult = findComponent < T > ( filters . componentByCode ( . . . newCode ) , parse , { isIndirect : true } ) ;
if ( IS_DEV ) {
webpackSearchHistory . push ( [ "findComponentByCode" , [ ComponentResult , . . . code ] ] ) ;
}
2024-05-03 02:18:12 +00:00
2024-05-07 04:13:46 +00:00
return ComponentResult ;
2024-05-03 02:18:12 +00:00
}
/ * *
2024-05-07 09:23:36 +00:00
* Find the first module or default export that includes all the given props .
2024-05-03 02:18:12 +00:00
*
* @param props A list of props to search the exports for
* /
export function findByProps < T = any > ( . . . props : string [ ] ) {
2024-05-07 04:13:46 +00:00
const result = find < T > ( filters . byProps ( . . . props ) , m = > m , { isIndirect : true } ) ;
2024-05-03 02:18:12 +00:00
2024-05-07 04:13:46 +00:00
if ( IS_DEV ) {
webpackSearchHistory . push ( [ "findByProps" , [ result , . . . props ] ] ) ;
}
return result ;
2024-05-03 02:18:12 +00:00
}
/ * *
2024-05-07 09:28:43 +00:00
* Find the first default export that includes all the given code .
2024-05-03 02:18:12 +00:00
*
* @param code A list of code to search each export for
* /
export function findByCode < T = any > ( . . . code : string [ ] ) {
2024-05-07 04:13:46 +00:00
const result = find < T > ( filters . byCode ( . . . code ) , m = > m , { isIndirect : true } ) ;
if ( IS_DEV ) {
webpackSearchHistory . push ( [ "findByCode" , [ result , . . . code ] ] ) ;
}
2024-05-03 02:18:12 +00:00
2024-05-07 04:13:46 +00:00
return result ;
2024-05-03 02:18:12 +00:00
}
/ * *
2024-05-07 09:23:36 +00:00
* Find a store by its name .
2024-05-03 02:18:12 +00:00
*
* @param name The store name
* /
export function findStore < T = any > ( name : string ) {
2024-05-07 04:13:46 +00:00
const result = find < T > ( filters . byStoreName ( name ) , m = > m , { isIndirect : true } ) ;
if ( IS_DEV ) {
webpackSearchHistory . push ( [ "findStore" , [ result , name ] ] ) ;
}
2024-05-03 02:18:12 +00:00
2024-05-07 04:13:46 +00:00
return result ;
2024-05-03 02:18:12 +00:00
}
/ * *
* Find the first already required module that matches the filter .
* @param filter A function that takes a module and returns a boolean
* @returns The found module or null
* /
export const cacheFind = traceFunction ( "find" , function find ( filter : FilterFn ) {
if ( typeof filter !== "function" )
throw new Error ( "Invalid filter. Expected a function got " + typeof filter ) ;
for ( const key in cache ) {
const mod = cache [ key ] ;
if ( ! mod ? . exports || mod . exports === window ) continue ;
if ( filter ( mod . exports ) ) {
return mod . exports ;
}
if ( mod . exports . default && filter ( mod . exports . default ) ) {
return mod . exports . default ;
}
}
return null ;
} ) ;
export function cacheFindAll ( filter : FilterFn ) {
if ( typeof filter !== "function" )
throw new Error ( "Invalid filter. Expected a function got " + typeof filter ) ;
const ret = [ ] as any [ ] ;
for ( const key in cache ) {
const mod = cache [ key ] ;
if ( ! mod ? . exports ) continue ;
if ( filter ( mod . exports ) ) {
ret . push ( mod . exports ) ;
}
if ( mod . exports . default && filter ( mod . exports . default ) ) {
ret . push ( mod . exports . default ) ;
}
}
return ret ;
}
/ * *
2024-05-07 09:23:36 +00:00
* Same as { @link cacheFind } but in bulk .
2024-05-03 02:18:12 +00:00
* @param filterFns Array of filters . Please note that this array will be modified in place , so if you still
* need it afterwards , pass a copy .
* @returns Array of results in the same order as the passed filters
* /
export const cacheFindBulk = traceFunction ( "findBulk" , function findBulk ( . . . filterFns : FilterFn [ ] ) {
if ( ! Array . isArray ( filterFns ) )
throw new Error ( "Invalid filters. Expected function[] got " + typeof filterFns ) ;
const { length } = filterFns ;
if ( length === 0 )
throw new Error ( "Expected at least two filters." ) ;
if ( length === 1 ) {
if ( IS_DEV ) {
throw new Error ( "bulk called with only one filter. Use find" ) ;
}
return cacheFind ( filterFns [ 0 ] ) ;
}
let found = 0 ;
const results = Array ( length ) ;
outer :
for ( const key in cache ) {
const mod = cache [ key ] ;
if ( ! mod ? . exports ) continue ;
for ( let j = 0 ; j < length ; j ++ ) {
const filter = filterFns [ j ] ;
if ( filter ( mod . exports ) ) {
results [ j ] = mod . exports ;
filterFns . splice ( j -- , 1 ) ;
if ( ++ found === length ) break outer ;
break ;
}
if ( mod . exports . default && filter ( mod . exports . default ) ) {
results [ j ] = mod . exports . default ;
filterFns . splice ( j -- , 1 ) ;
if ( ++ found === length ) break outer ;
break ;
}
}
}
if ( found !== length ) {
const err = new Error ( ` Got ${ length } filters, but only found ${ found } modules! ` ) ;
if ( ! IS_DEV ) {
logger . warn ( err ) ;
return results ;
}
if ( ! devToolsOpen ) {
throw err ; // Strict behaviour in DevBuilds to fail early and make sure the issue is found
}
}
return results ;
} ) ;
/ * *
2024-05-07 09:23:36 +00:00
* Find the id of the first module factory that includes all the given code .
2024-05-03 02:18:12 +00:00
* @returns string or null
* /
export const findModuleId = traceFunction ( "findModuleId" , function findModuleId ( . . . code : string [ ] ) {
outer :
for ( const id in wreq . m ) {
const str = wreq . m [ id ] . toString ( ) ;
for ( const c of code ) {
if ( ! str . includes ( c ) ) continue outer ;
}
return id ;
}
const err = new Error ( "Didn't find module with code(s):\n" + code . join ( "\n" ) ) ;
if ( ! IS_DEV ) {
logger . warn ( err ) ;
return null ;
}
if ( ! devToolsOpen ) {
throw err ; // Strict behaviour in DevBuilds to fail early and make sure the issue is found
}
return null ;
} ) ;
/ * *
2024-05-07 09:23:36 +00:00
* Find the first module factory that includes all the given code .
2024-05-03 02:18:12 +00:00
* @returns The module factory or null
* /
export function findModuleFactory ( . . . code : string [ ] ) {
const id = findModuleId ( . . . code ) ;
if ( ! id ) return null ;
return wreq . m [ id ] ;
}
/ * *
* This is just a wrapper around { @link proxyLazy } to make our reporter test for your webpack finds .
*
* Wraps the result of factory in a Proxy you can consume as if it wasn ' t lazy .
2024-05-07 09:23:36 +00:00
* On first property access , the factory is evaluated .
2024-05-03 02:18:12 +00:00
* @param factory Factory returning the result
* @param attempts How many times to try to evaluate the factory before giving up
* @returns Result of factory function
* /
export function webpackDependantLazy < T = any > ( factory : ( ) = > any , attempts? : number ) {
if ( IS_DEV ) webpackSearchHistory . push ( [ "webpackDependantLazy" , [ factory ] ] ) ;
return proxyLazy < T > ( factory , attempts ) ;
}
/ * *
* This is just a wrapper around { @link LazyComponent } to make our reporter test for your webpack finds .
*
* A lazy component . The factory method is called on first render .
* @param factory Function returning a Component
* @param attempts How many times to try to get the component before giving up
* @returns Result of factory function
* /
export function webpackDependantLazyComponent < T extends object = any > ( factory : ( ) = > any , attempts? : number ) {
if ( IS_DEV ) webpackSearchHistory . push ( [ "webpackDependantLazyComponent" , [ factory ] ] ) ;
return LazyComponent < T > ( factory , attempts ) ;
}
2024-05-08 00:33:05 +00:00
function deprecatedRedirect < T extends ( ...args : any [ ] ) = > any > ( oldMethod : string , newMethod : string , redirect : T ) : T {
return ( ( . . . args : Parameters < T > ) = > {
logger . warn ( ` Method ${ oldMethod } is deprecated. Use ${ newMethod } instead. For more information read https://github.com/Vendicated/Vencord/pull/2409#issue-2277161516 ` ) ;
return redirect ( . . . args ) ;
} ) as T ;
}
2024-05-03 08:35:53 +00:00
/ * *
* @deprecated Use { @link webpackDependantLazy } instead
2024-05-03 22:39:21 +00:00
*
2024-05-03 08:35:53 +00:00
* This is just a wrapper around { @link proxyLazy } to make our reporter test for your webpack finds .
*
* Wraps the result of factory in a Proxy you can consume as if it wasn ' t lazy .
2024-05-07 09:23:36 +00:00
* On first property access , the factory is evaluated .
2024-05-03 08:35:53 +00:00
* @param factory Factory returning the result
* @param attempts How many times to try to evaluate the factory before giving up
* @returns Result of factory function
* /
2024-05-08 00:33:05 +00:00
export const proxyLazyWebpack = deprecatedRedirect ( "proxyLazyWebpack" , "webpackDependantLazy" , webpackDependantLazy ) ;
2024-05-03 08:35:53 +00:00
/ * *
* @deprecated Use { @link webpackDependantLazyComponent } instead
2024-05-03 22:39:21 +00:00
*
2024-05-03 08:35:53 +00:00
* This is just a wrapper around { @link LazyComponent } to make our reporter test for your webpack finds .
*
* A lazy component . The factory method is called on first render .
* @param factory Function returning a Component
* @param attempts How many times to try to get the component before giving up
* @returns Result of factory function
* /
2024-05-08 00:33:05 +00:00
export const LazyComponentWebpack = deprecatedRedirect ( "LazyComponentWebpack" , "webpackDependantLazyComponent" , webpackDependantLazyComponent ) ;
2024-05-03 08:35:53 +00:00
/ * *
* @deprecated Use { @link find } instead
2024-05-03 22:39:21 +00:00
*
2024-05-03 08:35:53 +00:00
* Find the first module that matches the filter , lazily
* /
2024-05-08 00:33:05 +00:00
export const findLazy = deprecatedRedirect ( "findLazy" , "find" , find ) ;
2024-05-03 08:35:53 +00:00
/ * *
* @deprecated Use { @link findByProps } instead
2024-05-03 22:39:21 +00:00
*
2024-05-03 08:35:53 +00:00
* Find the first module that has the specified properties , lazily
* /
2024-05-08 00:33:05 +00:00
export const findByPropsLazy = deprecatedRedirect ( "findByPropsLazy" , "findByProps" , findByProps ) ;
2024-05-03 08:35:53 +00:00
/ * *
* @deprecated Use { @link findByCode } instead
2024-05-03 22:39:21 +00:00
*
2024-05-03 08:35:53 +00:00
* Find the first function that includes all the given code , lazily
* /
2024-05-08 00:33:05 +00:00
export const findByCodeLazy = deprecatedRedirect ( "findByCodeLazy" , "findByCode" , findByCode ) ;
2024-05-03 08:35:53 +00:00
/ * *
* @deprecated Use { @link findStore } instead
2024-05-03 22:39:21 +00:00
*
2024-05-03 08:35:53 +00:00
* Find a store by its displayName , lazily
* /
2024-05-08 00:33:05 +00:00
export const findStoreLazy = deprecatedRedirect ( "findStoreLazy" , "findStore" , findStore ) ;
2024-05-03 08:35:53 +00:00
/ * *
* @deprecated Use { @link findComponent } instead
2024-05-03 22:39:21 +00:00
*
2024-05-03 08:35:53 +00:00
* Finds the first component that matches the filter , lazily .
* /
2024-05-08 00:33:05 +00:00
export const findComponentLazy = deprecatedRedirect ( "findComponentLazy" , "findComponent" , findComponent ) ;
2024-05-03 08:35:53 +00:00
/ * *
* @deprecated Use { @link findComponentByCode } instead
2024-05-03 22:39:21 +00:00
*
2024-05-03 08:35:53 +00:00
* Finds the first component that includes all the given code , lazily
* /
2024-05-08 00:33:05 +00:00
export const findComponentByCodeLazy = deprecatedRedirect ( "findComponentByCodeLazy" , "findComponentByCode" , findComponentByCode ) ;
2024-05-03 08:35:53 +00:00
/ * *
* @deprecated Use { @link findExportedComponent } instead
2024-05-03 22:39:21 +00:00
*
2024-05-03 08:35:53 +00:00
* Finds the first component that is exported by the first prop name , lazily
* /
2024-05-08 00:33:05 +00:00
export const findExportedComponentLazy = deprecatedRedirect ( "findExportedComponentLazy" , "findExportedComponent" , findExportedComponent ) ;
2024-05-03 08:35:53 +00:00
/ * *
* @deprecated Use { @link cacheFindAll } instead
* /
2024-05-08 00:33:05 +00:00
export const findAll = deprecatedRedirect ( "findAll" , "cacheFindAll" , cacheFindAll ) ;
2024-05-03 08:35:53 +00:00
/ * *
* @deprecated Use { @link cacheFindBulk } instead
2024-05-03 22:39:21 +00:00
*
2024-05-03 08:35:53 +00:00
* Same as { @link cacheFind } but in bulk
* @param filterFns Array of filters . Please note that this array will be modified in place , so if you still
* need it afterwards , pass a copy .
* @returns Array of results in the same order as the passed filters
* /
2024-05-08 00:33:05 +00:00
export const findBulk = deprecatedRedirect ( "findBulk" , "cacheFindBulk" , cacheFindBulk ) ;
2024-05-03 08:35:53 +00:00
2024-05-03 02:18:12 +00:00
const DefaultExtractAndLoadChunksRegex = /(?:Promise\.all\((\[\i\.\i\(".+?"\).+?\])\)|Promise\.resolve\(\)).then\(\i\.bind\(\i,"(.+?)"\)\)/ ;
/ * *
2024-05-07 09:23:36 +00:00
* Extract and load chunks using their entry point .
2024-05-03 02:18:12 +00:00
* @param code An array of all the code the module factory containing the lazy chunk loading must include
* @param matcher A RegExp that returns the chunk ids array as the first capture group and the entry point id as the second . Defaults to a matcher that captures the lazy chunk loading found in the module factory
* @returns A promise that resolves when the chunks were loaded
* /
export async function extractAndLoadChunks ( code : string [ ] , matcher : RegExp = DefaultExtractAndLoadChunksRegex ) {
const module = findModuleFactory ( . . . code ) ;
if ( ! module ) {
const err = new Error ( "extractAndLoadChunks: Couldn't find module factory" ) ;
logger . warn ( err , "Code:" , code , "Matcher:" , matcher ) ;
return ;
}
const match = module . toString ( ) . match ( canonicalizeMatch ( matcher ) ) ;
if ( ! match ) {
const err = new Error ( "extractAndLoadChunks: Couldn't find chunk loading in module factory code" ) ;
logger . warn ( err , "Code:" , code , "Matcher:" , matcher ) ;
// Strict behaviour in DevBuilds to fail early and make sure the issue is found
if ( IS_DEV && ! devToolsOpen )
throw err ;
return ;
}
const [ , rawChunkIds , entryPointId ] = match ;
if ( Number . isNaN ( entryPointId ) ) {
const err = new Error ( "extractAndLoadChunks: Matcher didn't return a capturing group with the chunk ids array, or the entry point id returned as the second group wasn't a number" ) ;
logger . warn ( err , "Code:" , code , "Matcher:" , matcher ) ;
// Strict behaviour in DevBuilds to fail early and make sure the issue is found
if ( IS_DEV && ! devToolsOpen )
throw err ;
return ;
}
if ( rawChunkIds ) {
const chunkIds = Array . from ( rawChunkIds . matchAll ( /\("(.+?)"\)/g ) ) . map ( ( m : any ) = > m [ 1 ] ) ;
await Promise . all ( chunkIds . map ( id = > wreq . e ( id ) ) ) ;
}
wreq ( entryPointId ) ;
}
/ * *
* This is just a wrapper around { @link extractAndLoadChunks } to make our reporter test for your webpack finds .
*
2024-05-07 09:23:36 +00:00
* Extract and load chunks using their entry point .
2024-05-03 02:18:12 +00:00
* @param code An array of all the code the module factory containing the lazy chunk loading must include
* @param matcher A RegExp that returns the chunk ids array as the first capture group and the entry point id as the second . Defaults to a matcher that captures the lazy chunk loading found in the module factory
* @returns A function that returns a promise that resolves when the chunks were loaded , on first call
* /
export function extractAndLoadChunksLazy ( code : string [ ] , matcher = DefaultExtractAndLoadChunksRegex ) {
if ( IS_DEV ) webpackSearchHistory . push ( [ "extractAndLoadChunks" , [ code , matcher ] ] ) ;
return ( ) = > extractAndLoadChunks ( code , matcher ) ;
}
/ * *
* 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 ) {
const factory = factories [ id ] ;
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 ;
}
/ * *
* Extract a specific module by id into its own Source File . This has no effect on
* 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 .
* As mentioned above , note that this extracted module is not actually used ,
* so putting breakpoints or similar will have no effect .
* @param id The id of the module to extract
* /
export function extract ( id : string | 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!!
0 , $ { mod . toString ( ) }
//# sourceURL=ExtractedWebpackModule${id}
` ;
const extracted = ( 0 , eval ) ( code ) ;
return extracted as Function ;
}