我正在寻找猴子补丁 require() 用我自己的函数替换它的文件加载。我想在内部 require(module_id) 做了类似的事情:
- 将 module_id 转换为文件路径
- 将文件路径加载为字符串
- 将字符串编译成模块对象并正确设置各种全局变量
我希望在不重新实现步骤 1 + 3 的情况下替换步骤 2。查看公共 API,有执行 1-3 的 require() 和执行 1 的 require.resolve()。有没有办法将步骤 2 与第 3 步?
我查看了诸如mockery之类的 require 模拟工具的来源——他们似乎所做的只是用一个函数替换 require() ,该函数拦截某些调用并返回用户提供的对象,并将其他调用传递给本机要求()函数。
对于上下文,我正在尝试编写一个函数 require_at_commit(module_id, git_commit_id),它加载一个模块以及该模块的任何要求,就像它们在给定提交时一样。
我想要这个函数,因为我希望能够编写某些函数 a) 依赖于我的代码库的各个部分,并且 b) 保证在我发展我的代码库时不会改变。我想在不同的时间点“冻结”我的代码,所以认为这可能是一种避免打包我的代码库的 20 个副本的简单方法(另一种方法是使用 "my_code_v1": "git://.. ." 在我的 package.json 中,但我觉得 20 个版本会显得臃肿且缓慢)。
更新:
所以模块加载的源代码在这里:https ://github.com/joyent/node/blob/master/lib/module.js 。具体来说,要执行此类操作,您需要重新实现 Module._load,这非常简单。但是,还有一个更大的障碍,即第 1 步,将 module_id 转换为文件路径,实际上比我想象的要难,因为 resolveFilename 需要能够调用 fs.exists() 才能知道在哪里终止其搜索...所以我不能只替换单个文件,我必须替换整个目录,这意味着将整个 git 修订版导出到一个目录并将 require() 指向该目录可能更容易,而不是覆盖 require() .
更新 2:
最终完全使用了不同的方法......请参阅我在下面添加的答案