You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

371 lines
10 KiB

4 years ago
  1. 'use strict';
  2. const Module = require('module');
  3. const crypto = require('crypto');
  4. const fs = require('fs');
  5. const path = require('path');
  6. const vm = require('vm');
  7. const os = require('os');
  8. const hasOwnProperty = Object.prototype.hasOwnProperty;
  9. //------------------------------------------------------------------------------
  10. // FileSystemBlobStore
  11. //------------------------------------------------------------------------------
  12. class FileSystemBlobStore {
  13. constructor(directory, prefix) {
  14. const name = prefix ? slashEscape(prefix + '.') : '';
  15. this._blobFilename = path.join(directory, name + 'BLOB');
  16. this._mapFilename = path.join(directory, name + 'MAP');
  17. this._lockFilename = path.join(directory, name + 'LOCK');
  18. this._directory = directory;
  19. this._load();
  20. }
  21. has(key, invalidationKey) {
  22. if (hasOwnProperty.call(this._memoryBlobs, key)) {
  23. return this._invalidationKeys[key] === invalidationKey;
  24. } else if (hasOwnProperty.call(this._storedMap, key)) {
  25. return this._storedMap[key][0] === invalidationKey;
  26. }
  27. return false;
  28. }
  29. get(key, invalidationKey) {
  30. if (hasOwnProperty.call(this._memoryBlobs, key)) {
  31. if (this._invalidationKeys[key] === invalidationKey) {
  32. return this._memoryBlobs[key];
  33. }
  34. } else if (hasOwnProperty.call(this._storedMap, key)) {
  35. const mapping = this._storedMap[key];
  36. if (mapping[0] === invalidationKey) {
  37. return this._storedBlob.slice(mapping[1], mapping[2]);
  38. }
  39. }
  40. }
  41. set(key, invalidationKey, buffer) {
  42. this._invalidationKeys[key] = invalidationKey;
  43. this._memoryBlobs[key] = buffer;
  44. this._dirty = true;
  45. }
  46. delete(key) {
  47. if (hasOwnProperty.call(this._memoryBlobs, key)) {
  48. this._dirty = true;
  49. delete this._memoryBlobs[key];
  50. }
  51. if (hasOwnProperty.call(this._invalidationKeys, key)) {
  52. this._dirty = true;
  53. delete this._invalidationKeys[key];
  54. }
  55. if (hasOwnProperty.call(this._storedMap, key)) {
  56. this._dirty = true;
  57. delete this._storedMap[key];
  58. }
  59. }
  60. isDirty() {
  61. return this._dirty;
  62. }
  63. save() {
  64. const dump = this._getDump();
  65. const blobToStore = Buffer.concat(dump[0]);
  66. const mapToStore = JSON.stringify(dump[1]);
  67. try {
  68. mkdirpSync(this._directory);
  69. fs.writeFileSync(this._lockFilename, 'LOCK', {flag: 'wx'});
  70. } catch (error) {
  71. // Swallow the exception if we fail to acquire the lock.
  72. return false;
  73. }
  74. try {
  75. fs.writeFileSync(this._blobFilename, blobToStore);
  76. fs.writeFileSync(this._mapFilename, mapToStore);
  77. } finally {
  78. fs.unlinkSync(this._lockFilename);
  79. }
  80. return true;
  81. }
  82. _load() {
  83. try {
  84. this._storedBlob = fs.readFileSync(this._blobFilename);
  85. this._storedMap = JSON.parse(fs.readFileSync(this._mapFilename));
  86. } catch (e) {
  87. this._storedBlob = Buffer.alloc(0);
  88. this._storedMap = {};
  89. }
  90. this._dirty = false;
  91. this._memoryBlobs = {};
  92. this._invalidationKeys = {};
  93. }
  94. _getDump() {
  95. const buffers = [];
  96. const newMap = {};
  97. let offset = 0;
  98. function push(key, invalidationKey, buffer) {
  99. buffers.push(buffer);
  100. newMap[key] = [invalidationKey, offset, offset + buffer.length];
  101. offset += buffer.length;
  102. }
  103. for (const key of Object.keys(this._memoryBlobs)) {
  104. const buffer = this._memoryBlobs[key];
  105. const invalidationKey = this._invalidationKeys[key];
  106. push(key, invalidationKey, buffer);
  107. }
  108. for (const key of Object.keys(this._storedMap)) {
  109. if (hasOwnProperty.call(newMap, key)) continue;
  110. const mapping = this._storedMap[key];
  111. const buffer = this._storedBlob.slice(mapping[1], mapping[2]);
  112. push(key, mapping[0], buffer);
  113. }
  114. return [buffers, newMap];
  115. }
  116. }
  117. //------------------------------------------------------------------------------
  118. // NativeCompileCache
  119. //------------------------------------------------------------------------------
  120. class NativeCompileCache {
  121. constructor() {
  122. this._cacheStore = null;
  123. this._previousModuleCompile = null;
  124. }
  125. setCacheStore(cacheStore) {
  126. this._cacheStore = cacheStore;
  127. }
  128. install() {
  129. const self = this;
  130. const hasRequireResolvePaths = typeof require.resolve.paths === 'function';
  131. this._previousModuleCompile = Module.prototype._compile;
  132. Module.prototype._compile = function(content, filename) {
  133. const mod = this;
  134. function require(id) {
  135. return mod.require(id);
  136. }
  137. // https://github.com/nodejs/node/blob/v10.15.3/lib/internal/modules/cjs/helpers.js#L28
  138. function resolve(request, options) {
  139. return Module._resolveFilename(request, mod, false, options);
  140. }
  141. require.resolve = resolve;
  142. // https://github.com/nodejs/node/blob/v10.15.3/lib/internal/modules/cjs/helpers.js#L37
  143. // resolve.resolve.paths was added in v8.9.0
  144. if (hasRequireResolvePaths) {
  145. resolve.paths = function paths(request) {
  146. return Module._resolveLookupPaths(request, mod, true);
  147. };
  148. }
  149. require.main = process.mainModule;
  150. // Enable support to add extra extension types
  151. require.extensions = Module._extensions;
  152. require.cache = Module._cache;
  153. const dirname = path.dirname(filename);
  154. const compiledWrapper = self._moduleCompile(filename, content);
  155. // We skip the debugger setup because by the time we run, node has already
  156. // done that itself.
  157. // `Buffer` is included for Electron.
  158. // See https://github.com/zertosh/v8-compile-cache/pull/10#issuecomment-518042543
  159. const args = [mod.exports, require, mod, filename, dirname, process, global, Buffer];
  160. return compiledWrapper.apply(mod.exports, args);
  161. };
  162. }
  163. uninstall() {
  164. Module.prototype._compile = this._previousModuleCompile;
  165. }
  166. _moduleCompile(filename, content) {
  167. // https://github.com/nodejs/node/blob/v7.5.0/lib/module.js#L511
  168. // Remove shebang
  169. var contLen = content.length;
  170. if (contLen >= 2) {
  171. if (content.charCodeAt(0) === 35/*#*/ &&
  172. content.charCodeAt(1) === 33/*!*/) {
  173. if (contLen === 2) {
  174. // Exact match
  175. content = '';
  176. } else {
  177. // Find end of shebang line and slice it off
  178. var i = 2;
  179. for (; i < contLen; ++i) {
  180. var code = content.charCodeAt(i);
  181. if (code === 10/*\n*/ || code === 13/*\r*/) break;
  182. }
  183. if (i === contLen) {
  184. content = '';
  185. } else {
  186. // Note that this actually includes the newline character(s) in the
  187. // new output. This duplicates the behavior of the regular
  188. // expression that was previously used to replace the shebang line
  189. content = content.slice(i);
  190. }
  191. }
  192. }
  193. }
  194. // create wrapper function
  195. var wrapper = Module.wrap(content);
  196. var invalidationKey = crypto
  197. .createHash('sha1')
  198. .update(content, 'utf8')
  199. .digest('hex');
  200. var buffer = this._cacheStore.get(filename, invalidationKey);
  201. var script = new vm.Script(wrapper, {
  202. filename: filename,
  203. lineOffset: 0,
  204. displayErrors: true,
  205. cachedData: buffer,
  206. produceCachedData: true,
  207. });
  208. if (script.cachedDataProduced) {
  209. this._cacheStore.set(filename, invalidationKey, script.cachedData);
  210. } else if (script.cachedDataRejected) {
  211. this._cacheStore.delete(filename);
  212. }
  213. var compiledWrapper = script.runInThisContext({
  214. filename: filename,
  215. lineOffset: 0,
  216. columnOffset: 0,
  217. displayErrors: true,
  218. });
  219. return compiledWrapper;
  220. }
  221. }
  222. //------------------------------------------------------------------------------
  223. // utilities
  224. //
  225. // https://github.com/substack/node-mkdirp/blob/f2003bb/index.js#L55-L98
  226. // https://github.com/zertosh/slash-escape/blob/e7ebb99/slash-escape.js
  227. //------------------------------------------------------------------------------
  228. function mkdirpSync(p_) {
  229. _mkdirpSync(path.resolve(p_), 0o777);
  230. }
  231. function _mkdirpSync(p, mode) {
  232. try {
  233. fs.mkdirSync(p, mode);
  234. } catch (err0) {
  235. if (err0.code === 'ENOENT') {
  236. _mkdirpSync(path.dirname(p));
  237. _mkdirpSync(p);
  238. } else {
  239. try {
  240. const stat = fs.statSync(p);
  241. if (!stat.isDirectory()) { throw err0; }
  242. } catch (err1) {
  243. throw err0;
  244. }
  245. }
  246. }
  247. }
  248. function slashEscape(str) {
  249. const ESCAPE_LOOKUP = {
  250. '\\': 'zB',
  251. ':': 'zC',
  252. '/': 'zS',
  253. '\x00': 'z0',
  254. 'z': 'zZ',
  255. };
  256. const ESCAPE_REGEX = /[\\:/\x00z]/g; // eslint-disable-line no-control-regex
  257. return str.replace(ESCAPE_REGEX, match => ESCAPE_LOOKUP[match]);
  258. }
  259. function supportsCachedData() {
  260. const script = new vm.Script('""', {produceCachedData: true});
  261. // chakracore, as of v1.7.1.0, returns `false`.
  262. return script.cachedDataProduced === true;
  263. }
  264. function getCacheDir() {
  265. const v8_compile_cache_cache_dir = process.env.V8_COMPILE_CACHE_CACHE_DIR;
  266. if (v8_compile_cache_cache_dir) {
  267. return v8_compile_cache_cache_dir;
  268. }
  269. // Avoid cache ownership issues on POSIX systems.
  270. const dirname = typeof process.getuid === 'function'
  271. ? 'v8-compile-cache-' + process.getuid()
  272. : 'v8-compile-cache';
  273. const version = typeof process.versions.v8 === 'string'
  274. ? process.versions.v8
  275. : typeof process.versions.chakracore === 'string'
  276. ? 'chakracore-' + process.versions.chakracore
  277. : 'node-' + process.version;
  278. const cacheDir = path.join(os.tmpdir(), dirname, version);
  279. return cacheDir;
  280. }
  281. function getParentName() {
  282. // `module.parent.filename` is undefined or null when:
  283. // * node -e 'require("v8-compile-cache")'
  284. // * node -r 'v8-compile-cache'
  285. // * Or, requiring from the REPL.
  286. const parentName = module.parent && typeof module.parent.filename === 'string'
  287. ? module.parent.filename
  288. : process.cwd();
  289. return parentName;
  290. }
  291. //------------------------------------------------------------------------------
  292. // main
  293. //------------------------------------------------------------------------------
  294. if (!process.env.DISABLE_V8_COMPILE_CACHE && supportsCachedData()) {
  295. const cacheDir = getCacheDir();
  296. const prefix = getParentName();
  297. const blobStore = new FileSystemBlobStore(cacheDir, prefix);
  298. const nativeCompileCache = new NativeCompileCache();
  299. nativeCompileCache.setCacheStore(blobStore);
  300. nativeCompileCache.install();
  301. process.once('exit', () => {
  302. if (blobStore.isDirty()) {
  303. blobStore.save();
  304. }
  305. nativeCompileCache.uninstall();
  306. });
  307. }
  308. module.exports.__TEST__ = {
  309. FileSystemBlobStore,
  310. NativeCompileCache,
  311. mkdirpSync,
  312. slashEscape,
  313. supportsCachedData,
  314. getCacheDir,
  315. getParentName,
  316. };