Description
Extracting these ideas into a separate thread, since #94 is not focused on the challenges of using the helper functions in multiple loaders.
You could use them in a sorta-compositional way by having one loader override these helpers, so that when the next loader imports them they get the custom versions.
I'm worried that we don't realize how complex this is, and if anyone sits down to attempt it in a loader, they'll realize they don't like it, and they'll propose that we offer a better API for composition.
Maybe I'm thinking about this wrong, but to customize a single export of a native module, you need to redirect incoming resolutions to a virtual wrapper, but allow the wrapper to itself import the un-wrapped native version. (or a wrapped version from a subsequent loader)
// pseudo-code
// Goals:
// Wrap a single helper function
// Be compatible with other loaders before or after us in the chain that also wrap helper functions
const guid = `${ Math.random() }`; // avoid conflicts with other loaders
function resolve(specifier, ctx, next) {
if(specifier === 'node:module') return next(`node:module-wrapper-${guid}`, ctx);
if(specifier === `node:module-next-${guid}`) return next(`node:module`, ctx);
return next(specifier, ctx);
}
function load(url, ctx, next) {
if(url === `node:module-next-${guid}`) return next(`node:module`, ctx);
if(url === `node:module-wrapper-${guid}`) return wrapperText;
return next(url, ctx);
}
const wrapperText = `
export * from 'node:module-next-${guid}';
import {functionToWrap as _functionToWrap} from 'node:module-next-${guid}';
export functionToWrap(a) {
if(conditional) return _functionToWrap(a + 1);
return _functionToWrap(a);
}
`;