mirror of
https://github.com/nodejs/node.git
synced 2025-04-30 15:41:06 +00:00

Many of the util.is*() methods used to check data types simply compare against a single value or the result of typeof. This commit replaces calls to these methods with equivalent checks. This commit does not touch calls to the more complex methods (isRegExp(), isDate(), etc.). Fixes: https://github.com/iojs/io.js/issues/607 PR-URL: https://github.com/iojs/io.js/pull/647 Reviewed-By: Ben Noordhuis <info@bnoordhuis.nl>
528 lines
14 KiB
JavaScript
528 lines
14 KiB
JavaScript
'use strict';
|
|
|
|
const NativeModule = require('native_module');
|
|
const util = NativeModule.require('util');
|
|
const runInThisContext = require('vm').runInThisContext;
|
|
const runInNewContext = require('vm').runInNewContext;
|
|
const assert = require('assert').ok;
|
|
const fs = NativeModule.require('fs');
|
|
|
|
|
|
// If obj.hasOwnProperty has been overridden, then calling
|
|
// obj.hasOwnProperty(prop) will break.
|
|
// See: https://github.com/joyent/node/issues/1707
|
|
function hasOwnProperty(obj, prop) {
|
|
return Object.prototype.hasOwnProperty.call(obj, prop);
|
|
}
|
|
|
|
|
|
function Module(id, parent) {
|
|
this.id = id;
|
|
this.exports = {};
|
|
this.parent = parent;
|
|
if (parent && parent.children) {
|
|
parent.children.push(this);
|
|
}
|
|
|
|
this.filename = null;
|
|
this.loaded = false;
|
|
this.children = [];
|
|
}
|
|
module.exports = Module;
|
|
|
|
// Set the environ variable NODE_MODULE_CONTEXTS=1 to make node load all
|
|
// modules in their own context.
|
|
Module._contextLoad = (+process.env['NODE_MODULE_CONTEXTS'] > 0);
|
|
Module._cache = {};
|
|
Module._pathCache = {};
|
|
Module._extensions = {};
|
|
var modulePaths = [];
|
|
Module.globalPaths = [];
|
|
|
|
Module.wrapper = NativeModule.wrapper;
|
|
Module.wrap = NativeModule.wrap;
|
|
|
|
const path = NativeModule.require('path');
|
|
|
|
Module._debug = util.debuglog('module');
|
|
|
|
|
|
// We use this alias for the preprocessor that filters it out
|
|
const debug = Module._debug;
|
|
|
|
|
|
// given a module name, and a list of paths to test, returns the first
|
|
// matching file in the following precedence.
|
|
//
|
|
// require("a.<ext>")
|
|
// -> a.<ext>
|
|
//
|
|
// require("a")
|
|
// -> a
|
|
// -> a.<ext>
|
|
// -> a/index.<ext>
|
|
|
|
function statPath(path) {
|
|
try {
|
|
return fs.statSync(path);
|
|
} catch (ex) {}
|
|
return false;
|
|
}
|
|
|
|
// check if the directory is a package.json dir
|
|
const packageMainCache = {};
|
|
|
|
function readPackage(requestPath) {
|
|
if (hasOwnProperty(packageMainCache, requestPath)) {
|
|
return packageMainCache[requestPath];
|
|
}
|
|
|
|
try {
|
|
var jsonPath = path.resolve(requestPath, 'package.json');
|
|
var json = fs.readFileSync(jsonPath, 'utf8');
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
|
|
try {
|
|
var pkg = packageMainCache[requestPath] = JSON.parse(json).main;
|
|
} catch (e) {
|
|
e.path = jsonPath;
|
|
e.message = 'Error parsing ' + jsonPath + ': ' + e.message;
|
|
throw e;
|
|
}
|
|
return pkg;
|
|
}
|
|
|
|
function tryPackage(requestPath, exts) {
|
|
var pkg = readPackage(requestPath);
|
|
|
|
if (!pkg) return false;
|
|
|
|
var filename = path.resolve(requestPath, pkg);
|
|
return tryFile(filename, null) || tryExtensions(filename, exts) ||
|
|
tryExtensions(path.resolve(filename, 'index'), exts);
|
|
}
|
|
|
|
// In order to minimize unnecessary lstat() calls,
|
|
// this cache is a list of known-real paths.
|
|
// Set to an empty object to reset.
|
|
Module._realpathCache = {};
|
|
|
|
// check if the file exists and is not a directory
|
|
function tryFile(requestPath, stats) {
|
|
stats = stats || statPath(requestPath);
|
|
if (stats && !stats.isDirectory()) {
|
|
return fs.realpathSync(requestPath, Module._realpathCache);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// given a path check a the file exists with any of the set extensions
|
|
function tryExtensions(p, exts) {
|
|
for (var i = 0, EL = exts.length; i < EL; i++) {
|
|
var filename = tryFile(p + exts[i], null);
|
|
|
|
if (filename) {
|
|
return filename;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
Module._findPath = function(request, paths) {
|
|
var exts = Object.keys(Module._extensions);
|
|
|
|
if (request.charAt(0) === '/') {
|
|
paths = [''];
|
|
}
|
|
|
|
var trailingSlash = (request.slice(-1) === '/');
|
|
|
|
var cacheKey = JSON.stringify({request: request, paths: paths});
|
|
if (Module._pathCache[cacheKey]) {
|
|
return Module._pathCache[cacheKey];
|
|
}
|
|
|
|
// For each path
|
|
for (var i = 0, PL = paths.length; i < PL; i++) {
|
|
var basePath = path.resolve(paths[i], request);
|
|
var filename;
|
|
|
|
if (!trailingSlash) {
|
|
var stats = statPath(basePath);
|
|
// try to join the request to the path
|
|
filename = tryFile(basePath, stats);
|
|
|
|
if (!filename && stats && stats.isDirectory()) {
|
|
filename = tryPackage(basePath, exts);
|
|
}
|
|
|
|
if (!filename) {
|
|
// try it with each of the extensions
|
|
filename = tryExtensions(basePath, exts);
|
|
}
|
|
}
|
|
|
|
if (!filename) {
|
|
filename = tryPackage(basePath, exts);
|
|
}
|
|
|
|
if (!filename) {
|
|
// try it with each of the extensions at "index"
|
|
filename = tryExtensions(path.resolve(basePath, 'index'), exts);
|
|
}
|
|
|
|
if (filename) {
|
|
Module._pathCache[cacheKey] = filename;
|
|
return filename;
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
|
|
// 'from' is the __dirname of the module.
|
|
Module._nodeModulePaths = function(from) {
|
|
// guarantee that 'from' is absolute.
|
|
from = path.resolve(from);
|
|
|
|
// note: this approach *only* works when the path is guaranteed
|
|
// to be absolute. Doing a fully-edge-case-correct path.split
|
|
// that works on both Windows and Posix is non-trivial.
|
|
var splitRe = process.platform === 'win32' ? /[\/\\]/ : /\//;
|
|
var paths = [];
|
|
var parts = from.split(splitRe);
|
|
|
|
for (var tip = parts.length - 1; tip >= 0; tip--) {
|
|
// don't search in .../node_modules/node_modules
|
|
if (parts[tip] === 'node_modules') continue;
|
|
var dir = parts.slice(0, tip + 1).concat('node_modules').join(path.sep);
|
|
paths.push(dir);
|
|
}
|
|
|
|
return paths;
|
|
};
|
|
|
|
|
|
Module._resolveLookupPaths = function(request, parent) {
|
|
if (NativeModule.exists(request)) {
|
|
return [request, []];
|
|
}
|
|
|
|
var start = request.substring(0, 2);
|
|
if (start !== './' && start !== '..') {
|
|
var paths = modulePaths;
|
|
if (parent) {
|
|
if (!parent.paths) parent.paths = [];
|
|
paths = parent.paths.concat(paths);
|
|
}
|
|
return [request, paths];
|
|
}
|
|
|
|
// with --eval, parent.id is not set and parent.filename is null
|
|
if (!parent || !parent.id || !parent.filename) {
|
|
// make require('./path/to/foo') work - normally the path is taken
|
|
// from realpath(__filename) but with eval there is no filename
|
|
var mainPaths = ['.'].concat(modulePaths);
|
|
mainPaths = Module._nodeModulePaths('.').concat(mainPaths);
|
|
return [request, mainPaths];
|
|
}
|
|
|
|
// Is the parent an index module?
|
|
// We can assume the parent has a valid extension,
|
|
// as it already has been accepted as a module.
|
|
var isIndex = /^index\.\w+?$/.test(path.basename(parent.filename));
|
|
var parentIdPath = isIndex ? parent.id : path.dirname(parent.id);
|
|
var id = path.resolve(parentIdPath, request);
|
|
|
|
// make sure require('./path') and require('path') get distinct ids, even
|
|
// when called from the toplevel js file
|
|
if (parentIdPath === '.' && id.indexOf('/') === -1) {
|
|
id = './' + id;
|
|
}
|
|
|
|
debug('RELATIVE: requested:' + request +
|
|
' set ID to: ' + id + ' from ' + parent.id);
|
|
|
|
return [id, [path.dirname(parent.filename)]];
|
|
};
|
|
|
|
|
|
// Check the cache for the requested file.
|
|
// 1. If a module already exists in the cache: return its exports object.
|
|
// 2. If the module is native: call `NativeModule.require()` with the
|
|
// filename and return the result.
|
|
// 3. Otherwise, create a new module for the file and save it to the cache.
|
|
// Then have it load the file contents before returning its exports
|
|
// object.
|
|
Module._load = function(request, parent, isMain) {
|
|
if (parent) {
|
|
debug('Module._load REQUEST ' + (request) + ' parent: ' + parent.id);
|
|
}
|
|
|
|
var filename = Module._resolveFilename(request, parent);
|
|
|
|
var cachedModule = Module._cache[filename];
|
|
if (cachedModule) {
|
|
return cachedModule.exports;
|
|
}
|
|
|
|
if (NativeModule.exists(filename)) {
|
|
// REPL is a special case, because it needs the real require.
|
|
if (filename == 'repl') {
|
|
var replModule = new Module('repl');
|
|
replModule._compile(NativeModule.getSource('repl'), 'repl.js');
|
|
NativeModule._cache.repl = replModule;
|
|
return replModule.exports;
|
|
}
|
|
|
|
debug('load native module ' + request);
|
|
return NativeModule.require(filename);
|
|
}
|
|
|
|
var module = new Module(filename, parent);
|
|
|
|
if (isMain) {
|
|
process.mainModule = module;
|
|
module.id = '.';
|
|
}
|
|
|
|
Module._cache[filename] = module;
|
|
|
|
var hadException = true;
|
|
|
|
try {
|
|
module.load(filename);
|
|
hadException = false;
|
|
} finally {
|
|
if (hadException) {
|
|
delete Module._cache[filename];
|
|
}
|
|
}
|
|
|
|
return module.exports;
|
|
};
|
|
|
|
Module._resolveFilename = function(request, parent) {
|
|
if (NativeModule.exists(request)) {
|
|
return request;
|
|
}
|
|
|
|
var resolvedModule = Module._resolveLookupPaths(request, parent);
|
|
var id = resolvedModule[0];
|
|
var paths = resolvedModule[1];
|
|
|
|
// look up the filename first, since that's the cache key.
|
|
debug('looking for ' + JSON.stringify(id) +
|
|
' in ' + JSON.stringify(paths));
|
|
|
|
var filename = Module._findPath(request, paths);
|
|
if (!filename) {
|
|
var err = new Error("Cannot find module '" + request + "'");
|
|
err.code = 'MODULE_NOT_FOUND';
|
|
throw err;
|
|
}
|
|
return filename;
|
|
};
|
|
|
|
|
|
// Given a file name, pass it to the proper extension handler.
|
|
Module.prototype.load = function(filename) {
|
|
debug('load ' + JSON.stringify(filename) +
|
|
' for module ' + JSON.stringify(this.id));
|
|
|
|
assert(!this.loaded);
|
|
this.filename = filename;
|
|
this.paths = Module._nodeModulePaths(path.dirname(filename));
|
|
|
|
var extension = path.extname(filename) || '.js';
|
|
if (!Module._extensions[extension]) extension = '.js';
|
|
Module._extensions[extension](this, filename);
|
|
this.loaded = true;
|
|
};
|
|
|
|
|
|
// Loads a module at the given file path. Returns that module's
|
|
// `exports` property.
|
|
Module.prototype.require = function(path) {
|
|
assert(path, 'missing path');
|
|
assert(typeof path === 'string', 'path must be a string');
|
|
return Module._load(path, this);
|
|
};
|
|
|
|
|
|
// Resolved path to process.argv[1] will be lazily placed here
|
|
// (needed for setting breakpoint when called with --debug-brk)
|
|
var resolvedArgv;
|
|
|
|
|
|
// Run the file contents in the correct scope or sandbox. Expose
|
|
// the correct helper variables (require, module, exports) to
|
|
// the file.
|
|
// Returns exception, if any.
|
|
Module.prototype._compile = function(content, filename) {
|
|
var self = this;
|
|
// remove shebang
|
|
content = content.replace(/^\#\!.*/, '');
|
|
|
|
function require(path) {
|
|
return self.require(path);
|
|
}
|
|
|
|
require.resolve = function(request) {
|
|
return Module._resolveFilename(request, self);
|
|
};
|
|
|
|
Object.defineProperty(require, 'paths', { get: function() {
|
|
throw new Error('require.paths is removed. Use ' +
|
|
'node_modules folders, or the NODE_PATH ' +
|
|
'environment variable instead.');
|
|
}});
|
|
|
|
require.main = process.mainModule;
|
|
|
|
// Enable support to add extra extension types
|
|
require.extensions = Module._extensions;
|
|
require.registerExtension = function() {
|
|
throw new Error('require.registerExtension() removed. Use ' +
|
|
'require.extensions instead.');
|
|
};
|
|
|
|
require.cache = Module._cache;
|
|
|
|
var dirname = path.dirname(filename);
|
|
|
|
if (Module._contextLoad) {
|
|
if (self.id !== '.') {
|
|
debug('load submodule');
|
|
// not root module
|
|
var sandbox = {};
|
|
for (var k in global) {
|
|
sandbox[k] = global[k];
|
|
}
|
|
sandbox.require = require;
|
|
sandbox.exports = self.exports;
|
|
sandbox.__filename = filename;
|
|
sandbox.__dirname = dirname;
|
|
sandbox.module = self;
|
|
sandbox.global = sandbox;
|
|
sandbox.root = root;
|
|
|
|
return runInNewContext(content, sandbox, { filename: filename });
|
|
}
|
|
|
|
debug('load root module');
|
|
// root module
|
|
global.require = require;
|
|
global.exports = self.exports;
|
|
global.__filename = filename;
|
|
global.__dirname = dirname;
|
|
global.module = self;
|
|
|
|
return runInThisContext(content, { filename: filename });
|
|
}
|
|
|
|
// create wrapper function
|
|
var wrapper = Module.wrap(content);
|
|
|
|
var compiledWrapper = runInThisContext(wrapper, { filename: filename });
|
|
if (global.v8debug) {
|
|
if (!resolvedArgv) {
|
|
// we enter the repl if we're not given a filename argument.
|
|
if (process.argv[1]) {
|
|
resolvedArgv = Module._resolveFilename(process.argv[1], null);
|
|
} else {
|
|
resolvedArgv = 'repl';
|
|
}
|
|
}
|
|
|
|
// Set breakpoint on module start
|
|
if (filename === resolvedArgv) {
|
|
global.v8debug.Debug.setBreakPoint(compiledWrapper, 0, 0);
|
|
}
|
|
}
|
|
var args = [self.exports, require, self, filename, dirname];
|
|
return compiledWrapper.apply(self.exports, args);
|
|
};
|
|
|
|
|
|
function stripBOM(content) {
|
|
// Remove byte order marker. This catches EF BB BF (the UTF-8 BOM)
|
|
// because the buffer-to-string conversion in `fs.readFileSync()`
|
|
// translates it to FEFF, the UTF-16 BOM.
|
|
if (content.charCodeAt(0) === 0xFEFF) {
|
|
content = content.slice(1);
|
|
}
|
|
return content;
|
|
}
|
|
|
|
|
|
// Native extension for .js
|
|
Module._extensions['.js'] = function(module, filename) {
|
|
var content = fs.readFileSync(filename, 'utf8');
|
|
module._compile(stripBOM(content), filename);
|
|
};
|
|
|
|
|
|
// Native extension for .json
|
|
Module._extensions['.json'] = function(module, filename) {
|
|
var content = fs.readFileSync(filename, 'utf8');
|
|
try {
|
|
module.exports = JSON.parse(stripBOM(content));
|
|
} catch (err) {
|
|
err.message = filename + ': ' + err.message;
|
|
throw err;
|
|
}
|
|
};
|
|
|
|
|
|
//Native extension for .node
|
|
Module._extensions['.node'] = process.dlopen;
|
|
|
|
|
|
// bootstrap main module.
|
|
Module.runMain = function() {
|
|
// Load the main module--the command line argument.
|
|
Module._load(process.argv[1], null, true);
|
|
// Handle any nextTicks added in the first tick of the program
|
|
process._tickCallback();
|
|
};
|
|
|
|
Module._initPaths = function() {
|
|
const isWindows = process.platform === 'win32';
|
|
|
|
if (isWindows) {
|
|
var homeDir = process.env.USERPROFILE;
|
|
} else {
|
|
var homeDir = process.env.HOME;
|
|
}
|
|
|
|
var paths = [path.resolve(process.execPath, '..', '..', 'lib', 'node')];
|
|
|
|
if (homeDir) {
|
|
paths.unshift(path.resolve(homeDir, '.node_libraries'));
|
|
paths.unshift(path.resolve(homeDir, '.node_modules'));
|
|
}
|
|
|
|
var nodePath = process.env['NODE_PATH'];
|
|
if (nodePath) {
|
|
paths = nodePath.split(path.delimiter).concat(paths);
|
|
}
|
|
|
|
modulePaths = paths;
|
|
|
|
// clone as a read-only copy, for introspection.
|
|
Module.globalPaths = modulePaths.slice(0);
|
|
};
|
|
|
|
// bootstrap repl
|
|
Module.requireRepl = function() {
|
|
return Module._load('repl', '.');
|
|
};
|
|
|
|
Module._initPaths();
|
|
|
|
// backwards compatibility
|
|
Module.Module = Module;
|