UNPKG

babel-core

Version:

Turn ES6 code into readable vanilla ES5 with source maps

740 lines (577 loc) 18.3 kB
"use strict"; var _interopRequireWildcard = function (obj) { return obj && obj.__esModule ? obj : { "default": obj }; }; var _interopRequire = function (obj) { return obj && obj.__esModule ? obj["default"] : obj; }; var _classCallCheck = function (instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }; var includes = _interopRequire(require("lodash/collection/includes")); var traverse = _interopRequire(require("./index")); var defaults = _interopRequire(require("lodash/object/defaults")); var messages = _interopRequireWildcard(require("../messages")); var Binding = _interopRequire(require("./binding")); var globals = _interopRequire(require("globals")); var flatten = _interopRequire(require("lodash/array/flatten")); var extend = _interopRequire(require("lodash/object/extend")); var object = _interopRequire(require("../helpers/object")); var each = _interopRequire(require("lodash/collection/each")); var t = _interopRequireWildcard(require("../types")); var functionVariableVisitor = { enter: function enter(node, parent, scope, state) { var _this = this; if (t.isFor(node)) { each(t.FOR_INIT_KEYS, function (key) { var declar = _this.get(key); if (declar.isVar()) state.scope.registerBinding("var", declar); }); } // this block is a function so we'll stop since none of the variables // declared within are accessible if (this.isFunction()) return this.skip(); // function identifier doesn't belong to this scope if (state.blockId && node === state.blockId) return; // delegate block scope handling to the `blockVariableVisitor` if (this.isBlockScoped()) return; // this will be hit again once we traverse into it after this iteration if (this.isExportDeclaration() && t.isDeclaration(node.declaration)) return; // we've ran into a declaration! if (this.isDeclaration()) state.scope.registerDeclaration(this); } }; var programReferenceVisitor = { enter: function enter(node, parent, scope, state) { if (t.isReferencedIdentifier(node, parent) && !scope.hasBinding(node.name)) { state.addGlobal(node); } else if (t.isLabeledStatement(node)) { state.addGlobal(node); } else if (t.isAssignmentExpression(node)) { scope.registerConstantViolation(this.get("left"), this.get("right")); } else if (t.isUpdateExpression(node)) { scope.registerConstantViolation(this.get("argument"), null); } else if (t.isUnaryExpression(node) && node.operator === "delete") { scope.registerConstantViolation(this.get("left"), null); } } }; var blockVariableVisitor = { enter: function enter(node, parent, scope, state) { if (this.isFunctionDeclaration() || this.isBlockScoped()) { state.registerDeclaration(this); } else if (this.isScope()) { this.skip(); } } }; var Scope = (function () { /** * This searches the current "scope" and collects all references/bindings * within. */ function Scope(path, parent, file) { _classCallCheck(this, Scope); if (parent && parent.block === path.node) { return parent; } var cached = path.getData("scope"); if (cached && cached.parent === parent) { return cached; } else { path.setData("scope", this); } this.parent = parent; this.file = parent ? parent.file : file; this.parentBlock = path.parent; this.block = path.node; this.path = path; this.crawl(); } Scope.globals = flatten([globals.builtin, globals.browser, globals.node].map(Object.keys)); Scope.contextVariables = ["this", "arguments", "super"]; /** * Description */ Scope.prototype.traverse = (function (_traverse) { var _traverseWrapper = function traverse(_x, _x2, _x3) { return _traverse.apply(this, arguments); }; _traverseWrapper.toString = function () { return _traverse.toString(); }; return _traverseWrapper; })(function (node, opts, state) { traverse(node, opts, this, state); }); /** * Description */ Scope.prototype.generateTemp = function generateTemp() { var name = arguments[0] === undefined ? "temp" : arguments[0]; var id = this.generateUidIdentifier(name); this.push({ id: id }); return id; }; /** * Description */ Scope.prototype.generateUidIdentifier = function generateUidIdentifier(name) { return t.identifier(this.generateUid(name)); }; /** * Description */ Scope.prototype.generateUid = function generateUid(name) { name = t.toIdentifier(name).replace(/^_+/, ""); var uid; var i = 0; do { uid = this._generateUid(name, i); i++; } while (this.hasBinding(uid) || this.hasGlobal(uid) || this.hasUid(uid)); this.getFunctionParent().uids[uid] = true; return uid; }; Scope.prototype._generateUid = function _generateUid(name, i) { var id = name; if (i > 1) id += i; return "_" + id; }; /** * Description */ Scope.prototype.hasUid = function hasUid(name) { var scope = this; do { if (scope.uids[name]) return true; scope = scope.parent; } while (scope); return false; }; /* * Description */ Scope.prototype.generateUidBasedOnNode = function generateUidBasedOnNode(parent, defaultName) { var node = parent; if (t.isAssignmentExpression(parent)) { node = parent.left; } else if (t.isVariableDeclarator(parent)) { node = parent.id; } else if (t.isProperty(node)) { node = node.key; } var parts = []; var add = (function (_add) { var _addWrapper = function add(_x) { return _add.apply(this, arguments); }; _addWrapper.toString = function () { return _add.toString(); }; return _addWrapper; })(function (node) { if (t.isModuleDeclaration(node)) { if (node.specifiers && node.specifiers.length) { for (var i = 0; i < node.specifiers.length; i++) { add(node.specifiers[i]); } } else { add(node.source); } } else if (t.isModuleSpecifier(node)) { add(node.local); } else if (t.isMemberExpression(node)) { add(node.object); add(node.property); } else if (t.isIdentifier(node)) { parts.push(node.name); } else if (t.isLiteral(node)) { parts.push(node.value); } else if (t.isCallExpression(node)) { add(node.callee); } else if (t.isObjectExpression(node) || t.isObjectPattern(node)) { for (var i = 0; i < node.properties.length; i++) { var prop = node.properties[i]; add(prop.key || prop.argument); } } }); add(node); var id = parts.join("$"); id = id.replace(/^_/, "") || defaultName || "ref"; return this.generateUidIdentifier(id); }; /** * Description */ Scope.prototype.generateMemoisedReference = function generateMemoisedReference(node, dontPush) { if (t.isThisExpression(node) || t.isSuper(node)) { return null; } if (t.isIdentifier(node) && this.hasBinding(node.name)) { return null; } var id = this.generateUidBasedOnNode(node); if (!dontPush) this.push({ id: id }); return id; }; /** * Description */ Scope.prototype.checkBlockScopedCollisions = function checkBlockScopedCollisions(kind, name, id) { var local = this.getOwnBindingInfo(name); if (!local) return; if (kind === "param") return; if (kind === "hoisted" && local.kind === "let") return; if (local.kind === "let" || local.kind === "const" || local.kind === "module" || local.kind === "param") { throw this.file.errorWithNode(id, messages.get("scopeDuplicateDeclaration", name), TypeError); } }; /** * Description */ Scope.prototype.rename = function rename(oldName, newName, block) { if (!newName) newName = this.generateUidIdentifier(oldName).name; var info = this.getBinding(oldName); if (!info) return; var binding = info.identifier; var scope = info.scope; scope.traverse(block || scope.block, { enter: function enter(node, parent, scope) { if (t.isReferencedIdentifier(node, parent) && node.name === oldName) { node.name = newName; } else if (t.isDeclaration(node)) { var ids = this.getBindingIdentifiers(); for (var name in ids) { if (name === oldName) ids[name].name = newName; } } else if (this.isScope()) { if (!scope.bindingIdentifierEquals(oldName, binding)) { this.skip(); } } } }); if (!block) { scope.removeOwnBinding(oldName); scope.bindings[newName] = info; binding.name = newName; } }; /** * Description */ Scope.prototype.dump = function dump() { var scope = this; do { console.log(scope.block.type, "Bindings:", Object.keys(scope.bindings)); } while (scope = scope.parent); console.log("-------------"); }; /** * Description */ Scope.prototype.toArray = function toArray(node, i) { var file = this.file; if (t.isIdentifier(node)) { var binding = this.getBinding(node.name); if (binding && binding.isTypeGeneric("Array", { inference: false })) return node; } if (t.isArrayExpression(node)) { return node; } if (t.isIdentifier(node, { name: "arguments" })) { return t.callExpression(t.memberExpression(file.addHelper("slice"), t.identifier("call")), [node]); } var helperName = "to-array"; var args = [node]; if (i === true) { helperName = "to-consumable-array"; } else if (i) { args.push(t.literal(i)); helperName = "sliced-to-array"; if (this.file.isLoose("es6.forOf")) helperName += "-loose"; } return t.callExpression(file.addHelper(helperName), args); }; /** * Description */ Scope.prototype.registerDeclaration = function registerDeclaration(path) { var node = path.node; if (t.isFunctionDeclaration(node)) { this.registerBinding("hoisted", path); } else if (t.isVariableDeclaration(node)) { var declarations = path.get("declarations"); for (var i = 0; i < declarations.length; i++) { this.registerBinding(node.kind, declarations[i]); } } else if (t.isClassDeclaration(node)) { this.registerBinding("let", path); } else if (t.isImportDeclaration(node) || t.isExportDeclaration(node)) { this.registerBinding("module", path); } else { this.registerBinding("unknown", path); } }; /** * Description */ Scope.prototype.registerConstantViolation = function registerConstantViolation(left, right) { var ids = left.getBindingIdentifiers(); for (var name in ids) { var binding = this.getBinding(name); if (!binding) continue; if (right) { var rightType = right.typeAnnotation; if (rightType && binding.isCompatibleWithType(rightType)) continue; } binding.reassign(); } }; /** * Description */ Scope.prototype.registerBinding = function registerBinding(kind, path) { if (!kind) throw new ReferenceError("no `kind`"); var ids = path.getBindingIdentifiers(); for (var name in ids) { var id = ids[name]; this.checkBlockScopedCollisions(kind, name, id); this.bindings[name] = new Binding({ identifier: id, scope: this, path: path, kind: kind }); } }; /** * Description */ Scope.prototype.addGlobal = function addGlobal(node) { this.globals[node.name] = node; }; /** * Description */ Scope.prototype.hasGlobal = function hasGlobal(name) { var scope = this; do { if (scope.globals[name]) return true; } while (scope = scope.parent); return false; }; /** * Description */ Scope.prototype.recrawl = function recrawl() { this.path.setData("scopeInfo", null); this.crawl(); }; /** * Description */ Scope.prototype.crawl = function crawl() { var path = this.path; // var info = this.block._scopeInfo; if (info) return extend(this, info); info = this.block._scopeInfo = { bindings: object(), globals: object(), uids: object() }; extend(this, info); // ForStatement - left, init if (path.isLoop()) { for (var i = 0; i < t.FOR_INIT_KEYS.length; i++) { var node = path.get(t.FOR_INIT_KEYS[i]); if (node.isBlockScoped()) this.registerBinding("let", node); } } // FunctionExpression - id if (path.isFunctionExpression() && path.has("id")) { if (!t.isProperty(path.parent, { method: true })) { this.registerBinding("var", path.get("id")); } } // Class if (path.isClass() && path.has("id")) { this.registerBinding("var", path.get("id")); } // Function - params, rest if (path.isFunction()) { var params = path.get("params"); for (var i = 0; i < params.length; i++) { this.registerBinding("param", params[i]); } this.traverse(path.get("body").node, blockVariableVisitor, this); } // Program, BlockStatement, Function - let variables if (path.isBlockStatement() || path.isProgram()) { this.traverse(path.node, blockVariableVisitor, this); } // CatchClause - param if (path.isCatchClause()) { this.registerBinding("let", path.get("param")); } // ComprehensionExpression - blocks if (path.isComprehensionExpression()) { this.registerBinding("let", path); } // Program, Function - var variables if (path.isProgram() || path.isFunction()) { this.traverse(path.node, functionVariableVisitor, { blockId: path.get("id").node, scope: this }); } // Program if (path.isProgram()) { this.traverse(path.node, programReferenceVisitor, this); } }; /** * Description */ Scope.prototype.push = function push(opts) { var block = this.block; if (t.isLoop(block) || t.isCatchClause(block) || t.isFunction(block)) { t.ensureBlock(block); block = block.body; } if (!t.isBlockStatement(block) && !t.isProgram(block)) { block = this.getBlockParent().block; } var _block = block; if (!_block._declarations) _block._declarations = {}; block._declarations[opts.key || opts.id.name] = { kind: opts.kind || "var", id: opts.id, init: opts.init }; }; /** * Walk up the scope tree until we hit either a Function or reach the * very top and hit Program. */ Scope.prototype.getFunctionParent = function getFunctionParent() { var scope = this; while (scope.parent && !t.isFunction(scope.block)) { scope = scope.parent; } return scope; }; /** * Walk up the scope tree until we hit either a BlockStatement/Loop or reach the * very top and hit Program. */ Scope.prototype.getBlockParent = function getBlockParent() { var scope = this; while (scope.parent && !t.isFunction(scope.block) && !t.isLoop(scope.block) && !t.isFunction(scope.block)) { scope = scope.parent; } return scope; }; /** * Walks the scope tree and gathers **all** bindings. */ Scope.prototype.getAllBindings = function getAllBindings() { var ids = object(); var scope = this; do { defaults(ids, scope.bindings); scope = scope.parent; } while (scope); return ids; }; /** * Walks the scope tree and gathers all declarations of `kind`. */ Scope.prototype.getAllBindingsOfKind = function getAllBindingsOfKind() { var ids = object(); for (var i = 0; i < arguments.length; i++) { var kind = arguments[i]; var scope = this; do { for (var name in scope.bindings) { var binding = scope.bindings[name]; if (binding.kind === kind) ids[name] = binding; } scope = scope.parent; } while (scope); } return ids; }; /** * Description */ Scope.prototype.bindingIdentifierEquals = function bindingIdentifierEquals(name, node) { return this.getBindingIdentifier(name) === node; }; /** * Description */ Scope.prototype.getBinding = function getBinding(name) { var scope = this; do { var binding = scope.getOwnBindingInfo(name); if (binding) return binding; } while (scope = scope.parent); }; /** * Description */ Scope.prototype.getOwnBindingInfo = function getOwnBindingInfo(name) { return this.bindings[name]; }; /** * Description */ Scope.prototype.getBindingIdentifier = function getBindingIdentifier(name) { var info = this.getBinding(name); return info && info.identifier; }; /** * Description */ Scope.prototype.getOwnBindingIdentifier = function getOwnBindingIdentifier(name) { var binding = this.bindings[name]; return binding && binding.identifier; }; /** * Description */ Scope.prototype.hasOwnBinding = function hasOwnBinding(name) { return !!this.getOwnBindingInfo(name); }; /** * Description */ Scope.prototype.hasBinding = function hasBinding(name) { if (!name) return false; if (this.hasOwnBinding(name)) return true; if (this.parentHasBinding(name)) return true; if (this.uids[name]) return true; if (includes(Scope.globals, name)) return true; if (includes(Scope.contextVariables, name)) return true; return false; }; /** * Description */ Scope.prototype.parentHasBinding = function parentHasBinding(name) { return this.parent && this.parent.hasBinding(name); }; /** * Description */ Scope.prototype.removeOwnBinding = function removeOwnBinding(name) { this.bindings[name] = null; }; /** * Description */ Scope.prototype.removeBinding = function removeBinding(name) { var info = this.getBinding(name); if (info) info.scope.removeOwnBinding(name); }; return Scope; })(); module.exports = Scope;