Created
August 10, 2012 04:54
-
-
Save FLYBYME/3311158 to your computer and use it in GitHub Desktop.
require for raft paas
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// Copyright Joyent, Inc. and other Node contributors. | |
// | |
// Permission is hereby granted, free of charge, to any person obtaining a | |
// copy of this software and associated documentation files (the | |
// "Software"), to deal in the Software without restriction, including | |
// without limitation the rights to use, copy, modify, merge, publish, | |
// distribute, sublicense, and/or sell copies of the Software, and to permit | |
// persons to whom the Software is furnished to do so, subject to the | |
// following conditions: | |
// | |
// The above copyright notice and this permission notice shall be included | |
// in all copies or substantial portions of the Software. | |
// | |
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS | |
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | |
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN | |
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, | |
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR | |
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE | |
// USE OR OTHER DEALINGS IN THE SOFTWARE. | |
var Script = process.binding('evals').NodeScript; | |
var runInThisContext = Script.runInThisContext; | |
var runInNewContext = Script.runInNewContext; | |
var assert = require('assert').ok; | |
var fs = 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 thier own context. | |
Module._contextLoad = true; | |
Module._cache = {}; | |
Module._pathCache = {}; | |
Module._extensions = {}; | |
var modulePaths = []; | |
Module.globalPaths = []; | |
Module.wrapper = ['(function (exports, require, module, __filename, __dirname, raft) { ', '\n});']; | |
Module.wrap = function(script) { | |
return Module.wrapper[0] + script + Module.wrapper[1]; | |
}; | |
var path = require('path'); | |
Module._debug = function() { | |
}; | |
if (process.env.NODE_DEBUG && /module/.test(process.env.NODE_DEBUG)) { | |
Module._debug = function(x) { | |
console.error(x); | |
}; | |
} | |
// We use this alias for the preprocessor that filters it out | |
var 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) { | |
var fs = require('fs'); | |
try { | |
return fs.statSync(path); | |
} catch (ex) { | |
} | |
return false; | |
} | |
// check if the directory is a package.json dir | |
var packageCache = {}; | |
function readPackage(requestPath) { | |
if (hasOwnProperty(packageCache, requestPath)) { | |
return packageCache[requestPath]; | |
} | |
var fs = require('fs'); | |
try { | |
var jsonPath = path.resolve(requestPath, 'package.json'); | |
var json = fs.readFileSync(jsonPath, 'utf8'); | |
} catch (e) { | |
return false; | |
} | |
try { | |
var pkg = packageCache[requestPath] = JSON.parse(json); | |
} 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 || !pkg.main) | |
return false; | |
var filename = path.resolve(requestPath, pkg.main); | |
return tryFile(filename) || 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) { | |
var fs = require('fs'); | |
var 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]); | |
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) { | |
// try to join the request to the path | |
filename = tryFile(basePath); | |
if (!filename && !trailingSlash) { | |
// 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' ? /[\/\\]/ : /\//; | |
// yes, '/' works on both, but let's be a little canonical. | |
var joiner = 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(joiner); | |
paths.push(dir); | |
} | |
return paths; | |
}; | |
Module._resolveLookupPaths = function(request, parent) { | |
if (fs.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)]]; | |
}; | |
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 (fs.exists(filename)) { | |
// REPL is a special case, because it needs the real require. | |
debug('load native module ' + request); | |
return Module.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 (fs.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; | |
}; | |
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; | |
}; | |
Module.prototype.require = function(path) { | |
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; | |
// 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.raft = Module.raft || {}; | |
sandbox.root = root; | |
return runInNewContext(content, sandbox, filename, true); | |
} | |
debug('load root module'); | |
// root module | |
global.require = require; | |
global.exports = self.exports; | |
global.__filename = filename; | |
global.__dirname = dirname; | |
global.raft = Module.raft || {}; | |
global.module = self; | |
return runInThisContext(content, filename, true); | |
} | |
// create wrapper function | |
var wrapper = Module.wrap(content); | |
var compiledWrapper = runInThisContext(wrapper, filename, true); | |
if (global.v8debug) { | |
if (!resolvedArgv) { | |
resolvedArgv = Module._resolveFilename(process.argv[1], null); | |
} | |
// 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 = require('fs').readFileSync(filename, 'utf8'); | |
module._compile(stripBOM(content), filename); | |
}; | |
// Native extension for .json | |
Module._extensions['.json'] = function(module, filename) { | |
var content = require('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'] = function(module, filename) { | |
process.dlopen(filename, module.exports); | |
}; | |
// bootstrap main module. | |
Module.runMain = function() { | |
// Load the main module--the command line argument. | |
Module._load(process.argv[1], null, true); | |
}; | |
Module._initPaths = function() { | |
var paths = [path.resolve(process.execPath, '..', '..', 'lib', 'node')]; | |
if (process.env['HOME']) { | |
paths.unshift(path.resolve(process.env['HOME'], '.node_libraries')); | |
paths.unshift(path.resolve(process.env['HOME'], '.node_modules')); | |
} | |
if (process.env['NODE_PATH']) { | |
var splitter = process.platform === 'win32' ? ';' : ':'; | |
paths = process.env['NODE_PATH'].split(splitter).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', '.'); | |
}; | |
// bootstrap main module. | |
Module._loadRaft = function(r) { | |
Module.raft = r | |
}; | |
Module._initPaths(); | |
// backwards compatibility | |
Module.Module = Module; | |
/* | |
* | |
* | |
* | |
*/ | |
Module._loadRaft({ | |
test : true, | |
a : function(s) { | |
console.log(s) | |
} | |
}) | |
Module._load('./test') |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
console.log('test') | |
console.log(raft) | |
var a = require('./test1') | |
console.log(a) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
console.log('test1') | |
console.log(raft) | |
raft.a('sdasdasdasd') | |
module.exports = 'sadasdasd' | |
console.log('log', exports) |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment