123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351 |
- 'use strict';
- const Module = require('module');
- const crypto = require('crypto');
- const fs = require('fs');
- const path = require('path');
- const vm = require('vm');
- const os = require('os');
- const hasOwnProperty = Object.prototype.hasOwnProperty;
- //------------------------------------------------------------------------------
- // FileSystemBlobStore
- //------------------------------------------------------------------------------
- class FileSystemBlobStore {
- constructor(directory, prefix) {
- const name = prefix ? slashEscape(prefix + '.') : '';
- this._blobFilename = path.join(directory, name + 'BLOB');
- this._mapFilename = path.join(directory, name + 'MAP');
- this._lockFilename = path.join(directory, name + 'LOCK');
- this._directory = directory;
- this._load();
- }
- has(key, invalidationKey) {
- if (hasOwnProperty.call(this._memoryBlobs, key)) {
- return this._invalidationKeys[key] === invalidationKey;
- } else if (hasOwnProperty.call(this._storedMap, key)) {
- return this._storedMap[key][0] === invalidationKey;
- }
- return false;
- }
- get(key, invalidationKey) {
- if (hasOwnProperty.call(this._memoryBlobs, key)) {
- if (this._invalidationKeys[key] === invalidationKey) {
- return this._memoryBlobs[key];
- }
- } else if (hasOwnProperty.call(this._storedMap, key)) {
- const mapping = this._storedMap[key];
- if (mapping[0] === invalidationKey) {
- return this._storedBlob.slice(mapping[1], mapping[2]);
- }
- }
- }
- set(key, invalidationKey, buffer) {
- this._invalidationKeys[key] = invalidationKey;
- this._memoryBlobs[key] = buffer;
- this._dirty = true;
- }
- delete(key) {
- if (hasOwnProperty.call(this._memoryBlobs, key)) {
- this._dirty = true;
- delete this._memoryBlobs[key];
- }
- if (hasOwnProperty.call(this._invalidationKeys, key)) {
- this._dirty = true;
- delete this._invalidationKeys[key];
- }
- if (hasOwnProperty.call(this._storedMap, key)) {
- this._dirty = true;
- delete this._storedMap[key];
- }
- }
- isDirty() {
- return this._dirty;
- }
- save() {
- const dump = this._getDump();
- const blobToStore = Buffer.concat(dump[0]);
- const mapToStore = JSON.stringify(dump[1]);
- try {
- mkdirpSync(this._directory);
- fs.writeFileSync(this._lockFilename, 'LOCK', {flag: 'wx'});
- } catch (error) {
- // Swallow the exception if we fail to acquire the lock.
- return false;
- }
- try {
- fs.writeFileSync(this._blobFilename, blobToStore);
- fs.writeFileSync(this._mapFilename, mapToStore);
- } catch (error) {
- throw error;
- } finally {
- fs.unlinkSync(this._lockFilename);
- }
- return true;
- }
- _load() {
- try {
- this._storedBlob = fs.readFileSync(this._blobFilename);
- this._storedMap = JSON.parse(fs.readFileSync(this._mapFilename));
- } catch (e) {
- this._storedBlob = Buffer.alloc(0);
- this._storedMap = {};
- }
- this._dirty = false;
- this._memoryBlobs = {};
- this._invalidationKeys = {};
- }
- _getDump() {
- const buffers = [];
- const newMap = {};
- let offset = 0;
- function push(key, invalidationKey, buffer) {
- buffers.push(buffer);
- newMap[key] = [invalidationKey, offset, offset + buffer.length];
- offset += buffer.length;
- }
- for (const key of Object.keys(this._memoryBlobs)) {
- const buffer = this._memoryBlobs[key];
- const invalidationKey = this._invalidationKeys[key];
- push(key, invalidationKey, buffer);
- }
- for (const key of Object.keys(this._storedMap)) {
- if (hasOwnProperty.call(newMap, key)) continue;
- const mapping = this._storedMap[key];
- const buffer = this._storedBlob.slice(mapping[1], mapping[2]);
- push(key, mapping[0], buffer);
- }
- return [buffers, newMap];
- }
- }
- //------------------------------------------------------------------------------
- // NativeCompileCache
- //------------------------------------------------------------------------------
- class NativeCompileCache {
- constructor() {
- this._cacheStore = null;
- this._previousModuleCompile = null;
- }
- setCacheStore(cacheStore) {
- this._cacheStore = cacheStore;
- }
- install() {
- const self = this;
- this._previousModuleCompile = Module.prototype._compile;
- Module.prototype._compile = function(content, filename) {
- const mod = this;
- function require(id) {
- return mod.require(id);
- }
- require.resolve = function(request) {
- return Module._resolveFilename(request, mod);
- };
- require.main = process.mainModule;
- // Enable support to add extra extension types
- require.extensions = Module._extensions;
- require.cache = Module._cache;
- const dirname = path.dirname(filename);
- const compiledWrapper = self._moduleCompile(filename, content);
- // We skip the debugger setup because by the time we run, node has already
- // done that itself.
- const args = [mod.exports, require, mod, filename, dirname, process, global];
- return compiledWrapper.apply(mod.exports, args);
- };
- }
- uninstall() {
- Module.prototype._compile = this._previousModuleCompile;
- }
- _moduleCompile(filename, content) {
- // https://github.com/nodejs/node/blob/v7.5.0/lib/module.js#L511
- // Remove shebang
- var contLen = content.length;
- if (contLen >= 2) {
- if (content.charCodeAt(0) === 35/*#*/ &&
- content.charCodeAt(1) === 33/*!*/) {
- if (contLen === 2) {
- // Exact match
- content = '';
- } else {
- // Find end of shebang line and slice it off
- var i = 2;
- for (; i < contLen; ++i) {
- var code = content.charCodeAt(i);
- if (code === 10/*\n*/ || code === 13/*\r*/) break;
- }
- if (i === contLen) {
- content = '';
- } else {
- // Note that this actually includes the newline character(s) in the
- // new output. This duplicates the behavior of the regular
- // expression that was previously used to replace the shebang line
- content = content.slice(i);
- }
- }
- }
- }
- // create wrapper function
- var wrapper = Module.wrap(content);
- var invalidationKey = crypto
- .createHash('sha1')
- .update(content, 'utf8')
- .digest('hex');
- var buffer = this._cacheStore.get(filename, invalidationKey);
- var script = new vm.Script(wrapper, {
- filename: filename,
- lineOffset: 0,
- displayErrors: true,
- cachedData: buffer,
- produceCachedData: true,
- });
- if (script.cachedDataProduced) {
- this._cacheStore.set(filename, invalidationKey, script.cachedData);
- } else if (script.cachedDataRejected) {
- this._cacheStore.delete(filename);
- }
- var compiledWrapper = script.runInThisContext({
- filename: filename,
- lineOffset: 0,
- columnOffset: 0,
- displayErrors: true,
- });
- return compiledWrapper;
- }
- }
- //------------------------------------------------------------------------------
- // utilities
- //
- // https://github.com/substack/node-mkdirp/blob/f2003bb/index.js#L55-L98
- // https://github.com/zertosh/slash-escape/blob/e7ebb99/slash-escape.js
- //------------------------------------------------------------------------------
- function mkdirpSync(p_) {
- _mkdirpSync(path.resolve(p_), parseInt('0777', 8) & ~process.umask());
- }
- function _mkdirpSync(p, mode) {
- try {
- fs.mkdirSync(p, mode);
- } catch (err0) {
- if (err0.code === 'ENOENT') {
- _mkdirpSync(path.dirname(p));
- _mkdirpSync(p);
- } else {
- try {
- const stat = fs.statSync(p);
- if (!stat.isDirectory()) { throw err0; }
- } catch (err1) {
- throw err0;
- }
- }
- }
- }
- function slashEscape(str) {
- const ESCAPE_LOOKUP = {
- '\\': 'zB',
- ':': 'zC',
- '/': 'zS',
- '\x00': 'z0',
- 'z': 'zZ',
- };
- return str.replace(/[\\:\/\x00z]/g, match => (ESCAPE_LOOKUP[match]));
- }
- function supportsCachedData() {
- const script = new vm.Script('""', {produceCachedData: true});
- // chakracore, as of v1.7.1.0, returns `false`.
- return script.cachedDataProduced === true;
- }
- function getCacheDir() {
- // Avoid cache ownership issues on POSIX systems.
- const dirname = typeof process.getuid === 'function'
- ? 'v8-compile-cache-' + process.getuid()
- : 'v8-compile-cache';
- const version = typeof process.versions.v8 === 'string'
- ? process.versions.v8
- : typeof process.versions.chakracore === 'string'
- ? 'chakracore-' + process.versions.chakracore
- : 'node-' + process.version;
- const cacheDir = path.join(os.tmpdir(), dirname, version);
- return cacheDir;
- }
- function getParentName() {
- // `module.parent.filename` is undefined or null when:
- // * node -e 'require("v8-compile-cache")'
- // * node -r 'v8-compile-cache'
- // * Or, requiring from the REPL.
- const parentName = module.parent && typeof module.parent.filename === 'string'
- ? module.parent.filename
- : process.cwd();
- return parentName;
- }
- //------------------------------------------------------------------------------
- // main
- //------------------------------------------------------------------------------
- if (!process.env.DISABLE_V8_COMPILE_CACHE && supportsCachedData()) {
- const cacheDir = getCacheDir();
- const prefix = getParentName();
- const blobStore = new FileSystemBlobStore(cacheDir, prefix);
- const nativeCompileCache = new NativeCompileCache();
- nativeCompileCache.setCacheStore(blobStore);
- nativeCompileCache.install();
- process.once('exit', code => {
- if (blobStore.isDirty()) {
- blobStore.save();
- }
- nativeCompileCache.uninstall();
- });
- }
- module.exports.__TEST__ = {
- FileSystemBlobStore,
- NativeCompileCache,
- mkdirpSync,
- slashEscape,
- supportsCachedData,
- getCacheDir,
- getParentName,
- };
|