babel-plugin-minify-mangle-names
Advanced tools
Comparing version 0.3.0-alpha.37064b40 to 0.3.0-alpha.5ee3078e
"use strict"; | ||
module.exports = function bfsTraverseCreator(_ref) { | ||
var t = _ref.types, | ||
traverse = _ref.traverse; | ||
module.exports = function bfsTraverseCreator({ types: t, traverse }) { | ||
function getFields(path) { | ||
@@ -15,6 +12,6 @@ return t.VISITOR_KEYS[path.type]; | ||
} | ||
var visitor = traverse.explode(_visitor); | ||
const visitor = traverse.explode(_visitor); | ||
var queue = [path]; | ||
var current = void 0; | ||
const queue = [path]; | ||
let current; | ||
@@ -26,3 +23,3 @@ while (queue.length > 0) { | ||
if (visitor && visitor[current.type] && Array.isArray(visitor[current.type].enter)) { | ||
var fns = visitor[current.type].enter; | ||
const fns = visitor[current.type].enter; | ||
var _iteratorNormalCompletion = true; | ||
@@ -34,3 +31,3 @@ var _didIteratorError = false; | ||
for (var _iterator = fns[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { | ||
var fn = _step.value; | ||
const fn = _step.value; | ||
@@ -55,3 +52,3 @@ if (typeof fn === "function") fn(current); | ||
var fields = getFields(current); | ||
const fields = getFields(current); | ||
@@ -64,5 +61,5 @@ var _iteratorNormalCompletion2 = true; | ||
for (var _iterator2 = fields[Symbol.iterator](), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) { | ||
var field = _step2.value; | ||
const field = _step2.value; | ||
var child = current.get(field); | ||
const child = current.get(field); | ||
@@ -77,3 +74,3 @@ if (Array.isArray(child)) { | ||
for (var _iterator3 = child[Symbol.iterator](), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) { | ||
var c = _step3.value; | ||
const c = _step3.value; | ||
@@ -80,0 +77,0 @@ if (c.node) queue.push(c); |
"use strict"; | ||
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); | ||
const CHARSET = ("abcdefghijklmnopqrstuvwxyz" + "ABCDEFGHIJKLMNOPQRSTUVWXYZ$_").split(""); | ||
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } | ||
var CHARSET = ("abcdefghijklmnopqrstuvwxyz" + "ABCDEFGHIJKLMNOPQRSTUVWXYZ$_").split(""); | ||
module.exports = function () { | ||
function Charset(shouldConsider) { | ||
var _this = this; | ||
_classCallCheck(this, Charset); | ||
module.exports = class Charset { | ||
constructor(shouldConsider) { | ||
this.shouldConsider = shouldConsider; | ||
this.chars = CHARSET.slice(); | ||
this.frequency = {}; | ||
this.chars.forEach(function (c) { | ||
_this.frequency[c] = 0; | ||
this.chars.forEach(c => { | ||
this.frequency[c] = 0; | ||
}); | ||
@@ -24,49 +16,36 @@ this.finalized = false; | ||
_createClass(Charset, [{ | ||
key: "consider", | ||
value: function consider(str) { | ||
var _this2 = this; | ||
consider(str) { | ||
if (!this.shouldConsider) { | ||
return; | ||
} | ||
if (!this.shouldConsider) { | ||
return; | ||
str.split("").forEach(c => { | ||
if (this.frequency[c] != null) { | ||
this.frequency[c]++; | ||
} | ||
}); | ||
} | ||
str.split("").forEach(function (c) { | ||
if (_this2.frequency[c] != null) { | ||
_this2.frequency[c]++; | ||
} | ||
}); | ||
sort() { | ||
if (this.shouldConsider) { | ||
this.chars = this.chars.sort((a, b) => this.frequency[b] - this.frequency[a]); | ||
} | ||
}, { | ||
key: "sort", | ||
value: function sort() { | ||
var _this3 = this; | ||
if (this.shouldConsider) { | ||
this.chars = this.chars.sort(function (a, b) { | ||
return _this3.frequency[b] - _this3.frequency[a]; | ||
}); | ||
} | ||
this.finalized = true; | ||
} | ||
this.finalized = true; | ||
getIdentifier(num) { | ||
if (!this.finalized) { | ||
throw new Error("Should sort first"); | ||
} | ||
}, { | ||
key: "getIdentifier", | ||
value: function getIdentifier(num) { | ||
if (!this.finalized) { | ||
throw new Error("Should sort first"); | ||
} | ||
var ret = ""; | ||
num++; | ||
do { | ||
num--; | ||
ret += this.chars[num % this.chars.length]; | ||
num = Math.floor(num / this.chars.length); | ||
} while (num > 0); | ||
return ret; | ||
} | ||
}]); | ||
return Charset; | ||
}(); | ||
let ret = ""; | ||
num++; | ||
do { | ||
num--; | ||
ret += this.chars[num % this.chars.length]; | ||
num = Math.floor(num / this.chars.length); | ||
} while (num > 0); | ||
return ret; | ||
} | ||
}; |
"use strict"; | ||
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); | ||
function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } } | ||
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } | ||
// Set that counts | ||
module.exports = function () { | ||
function CountedSet() { | ||
_classCallCheck(this, CountedSet); | ||
module.exports = class CountedSet { | ||
constructor() { | ||
// because you can't simply extend Builtins yet | ||
this.map = new Map(); | ||
} | ||
_createClass(CountedSet, [{ | ||
key: "keys", | ||
value: function keys() { | ||
return [].concat(_toConsumableArray(this.map.keys())); | ||
keys() { | ||
return [...this.map.keys()]; | ||
} | ||
has(value) { | ||
return this.map.has(value); | ||
} | ||
add(value) { | ||
if (!this.has(value)) { | ||
this.map.set(value, 0); | ||
} | ||
}, { | ||
key: "has", | ||
value: function has(value) { | ||
return this.map.has(value); | ||
this.map.set(value, this.map.get(value) + 1); | ||
} | ||
delete(value) { | ||
if (!this.has(value)) return; | ||
const count = this.map.get(value); | ||
if (count <= 1) { | ||
this.map.delete(value); | ||
} else { | ||
this.map.set(value, count - 1); | ||
} | ||
}, { | ||
key: "add", | ||
value: function add(value) { | ||
if (!this.has(value)) { | ||
this.map.set(value, 0); | ||
} | ||
this.map.set(value, this.map.get(value) + 1); | ||
} | ||
}, { | ||
key: "delete", | ||
value: function _delete(value) { | ||
if (!this.has(value)) return; | ||
var count = this.map.get(value); | ||
if (count <= 1) { | ||
this.map.delete(value); | ||
} else { | ||
this.map.set(value, count - 1); | ||
} | ||
} | ||
}]); | ||
return CountedSet; | ||
}(); | ||
} | ||
}; |
@@ -12,11 +12,11 @@ "use strict"; | ||
} | ||
var fnScope = path.scope.getFunctionParent(); | ||
var bindingIds = path.getOuterBindingIdentifierPaths(); | ||
const fnScope = path.scope.getFunctionParent() || path.scope.getProgramParent(); | ||
const bindingIds = path.getOuterBindingIdentifierPaths(); | ||
for (var name in bindingIds) { | ||
var binding = path.scope.getBinding(name); | ||
for (const name in bindingIds) { | ||
const binding = path.scope.getBinding(name); | ||
// var isn't hoisted to fnScope | ||
if (binding.scope !== fnScope) { | ||
var existingBinding = fnScope.bindings[name]; | ||
const existingBinding = fnScope.bindings[name]; | ||
// make sure we are clear that the fnScope doesn't already have | ||
@@ -37,3 +37,3 @@ // an existing binding | ||
// binding.scope and fnScope | ||
var newName = fnScope.generateUid(binding.scope.generateUid(name)); | ||
const newName = fnScope.generateUid(binding.scope.generateUid(name)); | ||
@@ -40,0 +40,0 @@ // rename binding in the original scope |
851
lib/index.js
"use strict"; | ||
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); | ||
const Charset = require("./charset"); | ||
const ScopeTracker = require("./scope-tracker"); | ||
const isLabelIdentifier = require("./is-label-identifier"); | ||
const bfsTraverseCreator = require("./bfs-traverse"); | ||
const fixupVarScoping = require("./fixup-var-scoping"); | ||
function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } } | ||
var _require = require("babel-helper-mark-eval-scopes"); | ||
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } | ||
const markEvalScopes = _require.markEvalScopes, | ||
isEvalScopesMarked = _require.isMarked, | ||
hasEval = _require.hasEval; | ||
var Charset = require("./charset"); | ||
var ScopeTracker = require("./scope-tracker"); | ||
var isLabelIdentifier = require("./is-label-identifier"); | ||
var bfsTraverseCreator = require("./bfs-traverse"); | ||
var fixupVarScoping = require("./fixup-var-scoping"); | ||
var _require = require("babel-helper-mark-eval-scopes"), | ||
markEvalScopes = _require.markEvalScopes, | ||
isEvalScopesMarked = _require.isMarked, | ||
hasEval = _require.hasEval; | ||
const newIssueUrl = "https://github.com/babel/minify/issues/new"; | ||
var newIssueUrl = "https://github.com/babel/minify/issues/new"; | ||
module.exports = babel => { | ||
const t = babel.types, | ||
traverse = babel.traverse; | ||
module.exports = function (babel) { | ||
var t = babel.types, | ||
traverse = babel.traverse; | ||
const bfsTraverse = bfsTraverseCreator(babel); | ||
const hop = Object.prototype.hasOwnProperty; | ||
var bfsTraverse = bfsTraverseCreator(babel); | ||
var hop = Object.prototype.hasOwnProperty; | ||
var Mangler = function () { | ||
function Mangler(charset, program) { | ||
var _ref = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : {}, | ||
_ref$exclude = _ref.exclude, | ||
exclude = _ref$exclude === undefined ? {} : _ref$exclude, | ||
_ref$keepFnName = _ref.keepFnName, | ||
keepFnName = _ref$keepFnName === undefined ? false : _ref$keepFnName, | ||
_ref$keepClassName = _ref.keepClassName, | ||
keepClassName = _ref$keepClassName === undefined ? false : _ref$keepClassName, | ||
_ref$eval = _ref.eval, | ||
_eval = _ref$eval === undefined ? false : _ref$eval, | ||
_ref$topLevel = _ref.topLevel, | ||
topLevel = _ref$topLevel === undefined ? false : _ref$topLevel; | ||
_classCallCheck(this, Mangler); | ||
class Mangler { | ||
constructor(charset, program, { | ||
exclude = {}, | ||
keepFnName = false, | ||
keepClassName = false, | ||
eval: _eval = false, | ||
topLevel = false | ||
} = {}) { | ||
this.charset = charset; | ||
@@ -64,404 +52,368 @@ this.program = program; | ||
*/ | ||
run() { | ||
this.crawlScope(); | ||
this.collect(); | ||
this.fixup(); | ||
this.charset.sort(); | ||
this.mangle(); | ||
} | ||
/** | ||
* Tells if a variable name is excluded | ||
* @param {String} name | ||
*/ | ||
isExcluded(name) { | ||
return hop.call(this.exclude, name) && this.exclude[name]; | ||
} | ||
_createClass(Mangler, [{ | ||
key: "run", | ||
value: function run() { | ||
this.crawlScope(); | ||
this.collect(); | ||
this.fixup(); | ||
this.charset.sort(); | ||
this.mangle(); | ||
} | ||
/** | ||
* Clears traverse cache and recrawls the AST | ||
* | ||
* to recompute the bindings, references, other scope information | ||
* and paths because the other transformations in the same pipeline | ||
* (other plugins and presets) changes the AST and does NOT update | ||
* the scope objects | ||
*/ | ||
crawlScope() { | ||
(traverse.clearCache || traverse.cache.clear)(); | ||
this.program.scope.crawl(); | ||
} | ||
/** | ||
* Tells if a variable name is excluded | ||
* @param {String} name | ||
*/ | ||
/** | ||
* Re-crawling comes with a side-effect that let->var conversion | ||
* reverts the update of the binding information (block to fn scope). | ||
* This function takes care of it by updating it again. | ||
* | ||
* TODO: This is unnecessary work and needs to be fixed in babel. | ||
* https://github.com/babel/babel/issues/4818 | ||
* | ||
* When this is removed, remember to remove fixup's dependency in | ||
* ScopeTracker | ||
*/ | ||
fixup() { | ||
fixupVarScoping(this); | ||
} | ||
}, { | ||
key: "isExcluded", | ||
value: function isExcluded(name) { | ||
return hop.call(this.exclude, name) && this.exclude[name]; | ||
} | ||
/** | ||
* A single pass through the AST to collect info for | ||
* | ||
* 1. Scope Tracker | ||
* 2. Unsafe Scopes (direct eval scopes) | ||
* 3. Charset considerations for better gzip compression | ||
* | ||
* Traversed in the same fashion(BFS) the mangling is done | ||
*/ | ||
collect() { | ||
const mangler = this; | ||
const scopeTracker = mangler.scopeTracker; | ||
/** | ||
* Clears traverse cache and recrawls the AST | ||
* | ||
* to recompute the bindings, references, other scope information | ||
* and paths because the other transformations in the same pipeline | ||
* (other plugins and presets) changes the AST and does NOT update | ||
* the scope objects | ||
*/ | ||
}, { | ||
key: "crawlScope", | ||
value: function crawlScope() { | ||
(traverse.clearCache || traverse.cache.clear)(); | ||
this.program.scope.crawl(); | ||
} | ||
scopeTracker.addScope(this.program.scope); | ||
/** | ||
* Re-crawling comes with a side-effect that let->var conversion | ||
* reverts the update of the binding information (block to fn scope). | ||
* This function takes care of it by updating it again. | ||
* | ||
* TODO: This is unnecessary work and needs to be fixed in babel. | ||
* https://github.com/babel/babel/issues/4818 | ||
* | ||
* When this is removed, remember to remove fixup's dependency in | ||
* ScopeTracker | ||
* Same usage as in DCE, whichever runs first | ||
*/ | ||
}, { | ||
key: "fixup", | ||
value: function fixup() { | ||
fixupVarScoping(this); | ||
if (!isEvalScopesMarked(mangler.program)) { | ||
markEvalScopes(mangler.program); | ||
} | ||
/** | ||
* A single pass through the AST to collect info for | ||
* The visitors to be used in traversal. | ||
* | ||
* 1. Scope Tracker | ||
* 2. Unsafe Scopes (direct eval scopes) | ||
* 3. Charset considerations for better gzip compression | ||
* Note: BFS traversal supports only the `enter` handlers, `exit` | ||
* handlers are simply dropped without Errors | ||
* | ||
* Traversed in the same fashion(BFS) the mangling is done | ||
* Collects items defined in the ScopeTracker | ||
*/ | ||
const collectVisitor = { | ||
Scopable({ scope }) { | ||
scopeTracker.addScope(scope); | ||
}, { | ||
key: "collect", | ||
value: function collect() { | ||
var mangler = this; | ||
var scopeTracker = mangler.scopeTracker; | ||
scopeTracker.addScope(this.program.scope); | ||
// Collect bindings defined in the scope | ||
Object.keys(scope.bindings).forEach(name => { | ||
scopeTracker.addBinding(scope.bindings[name]); | ||
}); | ||
}, | ||
/** | ||
* Same usage as in DCE, whichever runs first | ||
* This is necessary because, in Babel, the scope.references | ||
* does NOT contain the references in that scope. Only the program | ||
* scope (top most level) contains all the references. | ||
* | ||
* We collect the references in a fashion where all the scopes between | ||
* and including the referenced scope and scope where it is declared | ||
* is considered as scope referencing that identifier | ||
*/ | ||
if (!isEvalScopesMarked(mangler.program.scope)) { | ||
markEvalScopes(mangler.program); | ||
} | ||
ReferencedIdentifier(path) { | ||
if (isLabelIdentifier(path)) return; | ||
const scope = path.scope, | ||
name = path.node.name; | ||
const binding = scope.getBinding(name); | ||
if (!binding) { | ||
// Do not collect globals as they are already available via | ||
// babel's API | ||
if (scope.hasGlobal(name)) return; | ||
// This should NOT happen ultimately. Panic if this code block is | ||
// reached | ||
throw new Error(`Binding not found for ReferencedIdentifier "${name}" ` + `present in "${path.parentPath.type}". ` + `Please report this at ${newIssueUrl}`); | ||
} else { | ||
// Add it to our scope tracker if everything is fine | ||
scopeTracker.addReference(scope, binding, name); | ||
} | ||
}, | ||
/** | ||
* The visitors to be used in traversal. | ||
* This is useful to detect binding ids and add them to the | ||
* scopeTracker's bindings | ||
* | ||
* Note: BFS traversal supports only the `enter` handlers, `exit` | ||
* handlers are simply dropped without Errors | ||
* TODO: | ||
* | ||
* Collects items defined in the ScopeTracker | ||
* This visitor is probably unnecessary. It was added to capture the | ||
* bindings that was not present in scope.bindings. But, now, it looks | ||
* like the unit and smoke tests pass without this. | ||
*/ | ||
var collectVisitor = { | ||
Scopable(_ref2) { | ||
var scope = _ref2.scope; | ||
BindingIdentifier(path) { | ||
if (isLabelIdentifier(path)) return; | ||
scopeTracker.addScope(scope); | ||
// Collect bindings defined in the scope | ||
Object.keys(scope.bindings).forEach(function (name) { | ||
scopeTracker.addBinding(scope.bindings[name]); | ||
}); | ||
}, | ||
/** | ||
* This is necessary because, in Babel, the scope.references | ||
* does NOT contain the references in that scope. Only the program | ||
* scope (top most level) contains all the references. | ||
* | ||
* We collect the references in a fashion where all the scopes between | ||
* and including the referenced scope and scope where it is declared | ||
* is considered as scope referencing that identifier | ||
*/ | ||
ReferencedIdentifier(path) { | ||
if (isLabelIdentifier(path)) return; | ||
var scope = path.scope, | ||
const scope = path.scope, | ||
name = path.node.name; | ||
var binding = scope.getBinding(name); | ||
if (!binding) { | ||
// Do not collect globals as they are already available via | ||
// babel's API | ||
if (scope.hasGlobal(name)) return; | ||
// This should NOT happen ultimately. Panic if this code block is | ||
// reached | ||
throw new Error(`Binding not found for ReferencedIdentifier "${name}" ` + `present in "${path.parentPath.type}". ` + `Please report this at ${newIssueUrl}`); | ||
} else { | ||
// Add it to our scope tracker if everything is fine | ||
scopeTracker.addReference(scope, binding, name); | ||
} | ||
}, | ||
const binding = scope.getBinding(name); | ||
/** | ||
* This is useful to detect binding ids and add them to the | ||
* scopeTracker's bindings | ||
* We have already captured the bindings when traversing through | ||
* Scopables, if a binding identifier doesn't have a binding, it | ||
* probably means that another transformation created a new binding, | ||
* refer https://github.com/babel/minify/issues/549 for example - | ||
* binding created by plugin transform-es2015-function-name | ||
* | ||
* So we just don't care about bindings that do not exist | ||
* | ||
* TODO: | ||
* | ||
* This visitor is probably unnecessary. It was added to capture the | ||
* bindings that was not present in scope.bindings. But, now, it looks | ||
* like the unit and smoke tests pass without this. | ||
* this deopts in DCE as this name can be removed for this particular | ||
* case (es2015-function-name) | ||
*/ | ||
BindingIdentifier(path) { | ||
if (isLabelIdentifier(path)) return; | ||
if (!binding) return; | ||
var scope = path.scope, | ||
name = path.node.name; | ||
/** | ||
* Detect constant violations | ||
* | ||
* If it's a constant violation, then add the Identifier Path as | ||
* a Reference instead of Binding - This is because the ScopeTracker | ||
* tracks these Re-declaration and mutation of variables as References | ||
* as it is simple to rename them | ||
*/ | ||
if (binding.identifier === path.node) { | ||
scopeTracker.addBinding(binding); | ||
} else { | ||
// constant violation | ||
scopeTracker.addReference(scope, binding, name); | ||
} | ||
} | ||
}; | ||
var binding = scope.getBinding(name); | ||
/** | ||
* These visitors are for collecting the Characters used in the program | ||
* to measure the frequency and generate variable names for mangling so | ||
* as to improve the gzip compression - as gzip likes repetition | ||
*/ | ||
if (this.charset.shouldConsider) { | ||
collectVisitor.Identifier = function Identifer(path) { | ||
const node = path.node; | ||
/** | ||
* We have already captured the bindings when traversing through | ||
* Scopables, if a binding identifier doesn't have a binding, it | ||
* probably means that another transformation created a new binding, | ||
* refer https://github.com/babel/minify/issues/549 for example - | ||
* binding created by plugin transform-es2015-function-name | ||
* | ||
* So we just don't care about bindings that do not exist | ||
* | ||
* TODO: | ||
* | ||
* this deopts in DCE as this name can be removed for this particular | ||
* case (es2015-function-name) | ||
*/ | ||
if (!binding) return; | ||
// We don't mangle properties, so we collect them as they contribute | ||
// to the frequency of characters | ||
/** | ||
* Detect constant violations | ||
* | ||
* If it's a constant violation, then add the Identifier Path as | ||
* a Reference instead of Binding - This is because the ScopeTracker | ||
* tracks these Re-declaration and mutation of variables as References | ||
* as it is simple to rename them | ||
*/ | ||
if (binding.identifier === path.node) { | ||
scopeTracker.addBinding(binding); | ||
} else { | ||
// constant violation | ||
scopeTracker.addReference(scope, binding, name); | ||
} | ||
if (path.parentPath.isMemberExpression({ property: node }) || path.parentPath.isObjectProperty({ key: node })) { | ||
mangler.charset.consider(node.name); | ||
} | ||
}; | ||
collectVisitor.Literal = function Literal({ node }) { | ||
mangler.charset.consider(String(node.value)); | ||
}; | ||
} | ||
/** | ||
* These visitors are for collecting the Characters used in the program | ||
* to measure the frequency and generate variable names for mangling so | ||
* as to improve the gzip compression - as gzip likes repetition | ||
*/ | ||
if (this.charset.shouldConsider) { | ||
collectVisitor.Identifier = function Identifer(path) { | ||
var node = path.node; | ||
// Traverse the AST | ||
bfsTraverse(mangler.program, collectVisitor); | ||
} | ||
// We don't mangle properties, so we collect them as they contribute | ||
// to the frequency of characters | ||
/** | ||
* Tells if a binding is exported as a NamedExport - so as to NOT mangle | ||
* | ||
* Babel treats NamedExports as a binding referenced by this NamedExport decl | ||
* @param {Binding} binding | ||
*/ | ||
isExportedWithName(binding) { | ||
// short circuit | ||
if (!this.topLevel) { | ||
return false; | ||
} | ||
if (path.parentPath.isMemberExpression({ property: node }) || path.parentPath.isObjectProperty({ key: node })) { | ||
mangler.charset.consider(node.name); | ||
} | ||
}; | ||
collectVisitor.Literal = function Literal(_ref3) { | ||
var node = _ref3.node; | ||
const refs = binding.referencePaths; | ||
mangler.charset.consider(String(node.value)); | ||
}; | ||
} | ||
var _iteratorNormalCompletion = true; | ||
var _didIteratorError = false; | ||
var _iteratorError = undefined; | ||
// Traverse the AST | ||
bfsTraverse(mangler.program, collectVisitor); | ||
} | ||
try { | ||
for (var _iterator = refs[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { | ||
const ref = _step.value; | ||
/** | ||
* Tells if a binding is exported as a NamedExport - so as to NOT mangle | ||
* | ||
* Babel treats NamedExports as a binding referenced by this NamedExport decl | ||
* @param {Binding} binding | ||
*/ | ||
}, { | ||
key: "isExportedWithName", | ||
value: function isExportedWithName(binding) { | ||
// short circuit | ||
if (!this.topLevel) { | ||
return false; | ||
if (ref.isExportNamedDeclaration()) { | ||
return true; | ||
} | ||
} | ||
var refs = binding.referencePaths; | ||
var _iteratorNormalCompletion = true; | ||
var _didIteratorError = false; | ||
var _iteratorError = undefined; | ||
// default | ||
} catch (err) { | ||
_didIteratorError = true; | ||
_iteratorError = err; | ||
} finally { | ||
try { | ||
for (var _iterator = refs[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { | ||
var ref = _step.value; | ||
if (ref.isExportNamedDeclaration()) { | ||
return true; | ||
} | ||
if (!_iteratorNormalCompletion && _iterator.return) { | ||
_iterator.return(); | ||
} | ||
// default | ||
} catch (err) { | ||
_didIteratorError = true; | ||
_iteratorError = err; | ||
} finally { | ||
try { | ||
if (!_iteratorNormalCompletion && _iterator.return) { | ||
_iterator.return(); | ||
} | ||
} finally { | ||
if (_didIteratorError) { | ||
throw _iteratorError; | ||
} | ||
if (_didIteratorError) { | ||
throw _iteratorError; | ||
} | ||
} | ||
return false; | ||
} | ||
/** | ||
* Mangle the scope | ||
* @param {Scope} scope | ||
*/ | ||
return false; | ||
} | ||
}, { | ||
key: "mangleScope", | ||
value: function mangleScope(scope) { | ||
var mangler = this; | ||
var scopeTracker = mangler.scopeTracker; | ||
/** | ||
* Mangle the scope | ||
* @param {Scope} scope | ||
*/ | ||
mangleScope(scope) { | ||
const mangler = this; | ||
const scopeTracker = mangler.scopeTracker; | ||
// Unsafe Scope | ||
// Unsafe Scope | ||
if (!mangler.eval && hasEval(scope)) return; | ||
if (!mangler.eval && hasEval(scope)) return; | ||
// Already visited | ||
// This is because for a function, in Babel, the function and | ||
// the function body's BlockStatement has the same scope, and will | ||
// be visited twice by the Scopable handler, and we want to mangle | ||
// it only once | ||
if (mangler.visitedScopes.has(scope)) return; | ||
mangler.visitedScopes.add(scope); | ||
// Already visited | ||
// This is because for a function, in Babel, the function and | ||
// the function body's BlockStatement has the same scope, and will | ||
// be visited twice by the Scopable handler, and we want to mangle | ||
// it only once | ||
if (mangler.visitedScopes.has(scope)) return; | ||
mangler.visitedScopes.add(scope); | ||
// Helpers to generate names | ||
var i = 0; | ||
function getNext() { | ||
return mangler.charset.getIdentifier(i++); | ||
} | ||
function resetNext() { | ||
i = 0; | ||
} | ||
// Helpers to generate names | ||
let i = 0; | ||
function getNext() { | ||
return mangler.charset.getIdentifier(i++); | ||
} | ||
function resetNext() { | ||
i = 0; | ||
} | ||
var bindings = scopeTracker.bindings.get(scope); | ||
var names = [].concat(_toConsumableArray(bindings.keys())); | ||
const bindings = scopeTracker.bindings.get(scope); | ||
const names = [...bindings.keys()]; | ||
/** | ||
* 1. Iterate through the list of BindingIdentifiers | ||
* 2. Rename each of them in-place | ||
* 3. Update the scope tree. | ||
*/ | ||
for (var _i = 0; _i < names.length; _i++) { | ||
var oldName = names[_i]; | ||
var binding = bindings.get(oldName); | ||
/** | ||
* 1. Iterate through the list of BindingIdentifiers | ||
* 2. Rename each of them in-place | ||
* 3. Update the scope tree. | ||
*/ | ||
for (let i = 0; i < names.length; i++) { | ||
const oldName = names[i]; | ||
const binding = bindings.get(oldName); | ||
// Names which should NOT be mangled | ||
if ( | ||
// arguments - for non-strict mode | ||
oldName === "arguments" || | ||
// labels | ||
binding.path.isLabeledStatement() || | ||
// ClassDeclaration has binding in two scopes | ||
// 1. The scope in which it is declared | ||
// 2. The class's own scope | ||
binding.path.isClassDeclaration() && binding.path === scope.path || | ||
// excluded | ||
mangler.isExcluded(oldName) || ( | ||
// function names | ||
mangler.keepFnName ? isFunction(binding.path) : false) || ( | ||
// class names | ||
mangler.keepClassName ? isClass(binding.path) : false) || | ||
// named export | ||
mangler.isExportedWithName(binding)) { | ||
continue; | ||
} | ||
var next = void 0; | ||
do { | ||
next = getNext(); | ||
} while (!t.isValidIdentifier(next) || scopeTracker.hasBinding(scope, next) || scope.hasGlobal(next) || scopeTracker.hasReference(scope, next) || !scopeTracker.canUseInReferencedScopes(binding, next)); | ||
// Reset so variables which are removed can be reused | ||
resetNext(); | ||
// Once we detected a valid `next` Identifier which could be used, | ||
// call the renamer | ||
mangler.rename(scope, binding, oldName, next); | ||
// Names which should NOT be mangled | ||
if ( | ||
// arguments - for non-strict mode | ||
oldName === "arguments" || | ||
// labels | ||
binding.path.isLabeledStatement() || | ||
// ClassDeclaration has binding in two scopes | ||
// 1. The scope in which it is declared | ||
// 2. The class's own scope | ||
binding.path.isClassDeclaration() && binding.path === scope.path || | ||
// excluded | ||
mangler.isExcluded(oldName) || ( | ||
// function names | ||
mangler.keepFnName ? isFunction(binding.path) : false) || ( | ||
// class names | ||
mangler.keepClassName ? isClass(binding.path) : false) || | ||
// named export | ||
mangler.isExportedWithName(binding)) { | ||
continue; | ||
} | ||
} | ||
/** | ||
* The mangle function that traverses through all the Scopes in a BFS | ||
* fashion - calls mangleScope | ||
*/ | ||
let next; | ||
do { | ||
next = getNext(); | ||
} while (!t.isValidIdentifier(next) || scopeTracker.hasBinding(scope, next) || scope.hasGlobal(next) || scopeTracker.hasReference(scope, next) || !scopeTracker.canUseInReferencedScopes(binding, next)); | ||
}, { | ||
key: "mangle", | ||
value: function mangle() { | ||
var mangler = this; | ||
// Reset so variables which are removed can be reused | ||
resetNext(); | ||
bfsTraverse(this.program, { | ||
Scopable(path) { | ||
if (!path.isProgram() || mangler.topLevel) mangler.mangleScope(path.scope); | ||
} | ||
}); | ||
// Once we detected a valid `next` Identifier which could be used, | ||
// call the renamer | ||
mangler.rename(scope, binding, oldName, next); | ||
} | ||
} | ||
/** | ||
* Given a NodePath, collects all the Identifiers which are BindingIdentifiers | ||
* and replaces them with the new name | ||
* | ||
* For example, | ||
* var a = 1, { b } = c; // a and b are BindingIdentifiers | ||
* | ||
* @param {NodePath} path | ||
* @param {String} oldName | ||
* @param {String} newName | ||
* @param {Function} predicate | ||
*/ | ||
/** | ||
* The mangle function that traverses through all the Scopes in a BFS | ||
* fashion - calls mangleScope | ||
*/ | ||
mangle() { | ||
const mangler = this; | ||
}, { | ||
key: "renameBindingIds", | ||
value: function renameBindingIds(path, oldName, newName) { | ||
var predicate = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : function () { | ||
return true; | ||
}; | ||
bfsTraverse(this.program, { | ||
Scopable(path) { | ||
if (!path.isProgram() || mangler.topLevel) mangler.mangleScope(path.scope); | ||
} | ||
}); | ||
} | ||
var bindingIds = path.getBindingIdentifierPaths(true, false); | ||
for (var name in bindingIds) { | ||
if (name !== oldName) continue; | ||
var _iteratorNormalCompletion2 = true; | ||
var _didIteratorError2 = false; | ||
var _iteratorError2 = undefined; | ||
/** | ||
* Given a NodePath, collects all the Identifiers which are BindingIdentifiers | ||
* and replaces them with the new name | ||
* | ||
* For example, | ||
* var a = 1, { b } = c; // a and b are BindingIdentifiers | ||
* | ||
* @param {NodePath} path | ||
* @param {String} oldName | ||
* @param {String} newName | ||
* @param {Function} predicate | ||
*/ | ||
renameBindingIds(path, oldName, newName, predicate = () => true) { | ||
const bindingIds = path.getBindingIdentifierPaths(true, false); | ||
for (const name in bindingIds) { | ||
if (name !== oldName) continue; | ||
var _iteratorNormalCompletion2 = true; | ||
var _didIteratorError2 = false; | ||
var _iteratorError2 = undefined; | ||
try { | ||
for (var _iterator2 = bindingIds[name][Symbol.iterator](), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) { | ||
var idPath = _step2.value; | ||
try { | ||
for (var _iterator2 = bindingIds[name][Symbol.iterator](), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) { | ||
const idPath = _step2.value; | ||
if (predicate(idPath)) { | ||
this.renamedNodes.add(idPath.node); | ||
idPath.replaceWith(t.identifier(newName)); | ||
this.renamedNodes.add(idPath.node); | ||
} | ||
if (predicate(idPath)) { | ||
this.renamedNodes.add(idPath.node); | ||
idPath.replaceWith(t.identifier(newName)); | ||
this.renamedNodes.add(idPath.node); | ||
} | ||
} catch (err) { | ||
_didIteratorError2 = true; | ||
_iteratorError2 = err; | ||
} | ||
} catch (err) { | ||
_didIteratorError2 = true; | ||
_iteratorError2 = err; | ||
} finally { | ||
try { | ||
if (!_iteratorNormalCompletion2 && _iterator2.return) { | ||
_iterator2.return(); | ||
} | ||
} finally { | ||
try { | ||
if (!_iteratorNormalCompletion2 && _iterator2.return) { | ||
_iterator2.return(); | ||
} | ||
} finally { | ||
if (_didIteratorError2) { | ||
throw _iteratorError2; | ||
} | ||
if (_didIteratorError2) { | ||
throw _iteratorError2; | ||
} | ||
@@ -471,104 +423,97 @@ } | ||
} | ||
} | ||
/** | ||
* The Renamer: | ||
* Renames the following for one Binding in a Scope | ||
* | ||
* 1. Binding in that Scope | ||
* 2. All the Binding's constant violations | ||
* 3. All its References | ||
* 4. Updates mangler.scopeTracker | ||
* 5. Updates Babel's Scope tracking | ||
* | ||
* @param {Scope} scope | ||
* @param {Binding} binding | ||
* @param {String} oldName | ||
* @param {String} newName | ||
*/ | ||
/** | ||
* The Renamer: | ||
* Renames the following for one Binding in a Scope | ||
* | ||
* 1. Binding in that Scope | ||
* 2. All the Binding's constant violations | ||
* 3. All its References | ||
* 4. Updates mangler.scopeTracker | ||
* 5. Updates Babel's Scope tracking | ||
* | ||
* @param {Scope} scope | ||
* @param {Binding} binding | ||
* @param {String} oldName | ||
* @param {String} newName | ||
*/ | ||
rename(scope, binding, oldName, newName) { | ||
const mangler = this; | ||
const scopeTracker = mangler.scopeTracker; | ||
}, { | ||
key: "rename", | ||
value: function rename(scope, binding, oldName, newName) { | ||
var mangler = this; | ||
var scopeTracker = mangler.scopeTracker; | ||
// rename at the declaration level | ||
// rename at the declaration level | ||
this.renameBindingIds(binding.path, oldName, newName, idPath => idPath.node === binding.identifier); | ||
this.renameBindingIds(binding.path, oldName, newName, function (idPath) { | ||
return idPath.node === binding.identifier; | ||
}); | ||
// update mangler's ScopeTracker | ||
scopeTracker.renameBinding(scope, oldName, newName); | ||
// update mangler's ScopeTracker | ||
scopeTracker.renameBinding(scope, oldName, newName); | ||
// update all constant violations | ||
const violations = binding.constantViolations; | ||
for (let i = 0; i < violations.length; i++) { | ||
if (violations[i].isLabeledStatement()) continue; | ||
// update all constant violations | ||
var violations = binding.constantViolations; | ||
for (var i = 0; i < violations.length; i++) { | ||
if (violations[i].isLabeledStatement()) continue; | ||
this.renameBindingIds(violations[i], oldName, newName); | ||
scopeTracker.updateReference(violations[i].scope, binding, oldName, newName); | ||
} | ||
this.renameBindingIds(violations[i], oldName, newName); | ||
scopeTracker.updateReference(violations[i].scope, binding, oldName, newName); | ||
} | ||
// update all referenced places | ||
const refs = binding.referencePaths; | ||
for (let i = 0; i < refs.length; i++) { | ||
const path = refs[i]; | ||
// update all referenced places | ||
var refs = binding.referencePaths; | ||
for (var _i2 = 0; _i2 < refs.length; _i2++) { | ||
var path = refs[_i2]; | ||
const node = path.node; | ||
var node = path.node; | ||
if (!path.isIdentifier()) { | ||
// Ideally, this should not happen | ||
// it happens in these places now - | ||
// case 1: Export Statements | ||
// This is a bug in babel | ||
// https://github.com/babel/babel/pull/3629 | ||
// case 2: Replacements in other plugins | ||
// eg: https://github.com/babel/minify/issues/122 | ||
// replacement in dce from `x` to `!x` gives referencePath as `!x` | ||
path.traverse({ | ||
ReferencedIdentifier(refPath) { | ||
if (refPath.node.name !== oldName) { | ||
return; | ||
} | ||
const actualBinding = refPath.scope.getBinding(oldName); | ||
if (actualBinding !== binding) { | ||
return; | ||
} | ||
mangler.renamedNodes.add(refPath.node); | ||
refPath.replaceWith(t.identifier(newName)); | ||
mangler.renamedNodes.add(refPath.node); | ||
if (!path.isIdentifier()) { | ||
// Ideally, this should not happen | ||
// it happens in these places now - | ||
// case 1: Export Statements | ||
// This is a bug in babel | ||
// https://github.com/babel/babel/pull/3629 | ||
// case 2: Replacements in other plugins | ||
// eg: https://github.com/babel/minify/issues/122 | ||
// replacement in dce from `x` to `!x` gives referencePath as `!x` | ||
path.traverse({ | ||
ReferencedIdentifier(refPath) { | ||
if (refPath.node.name !== oldName) { | ||
return; | ||
} | ||
var actualBinding = refPath.scope.getBinding(oldName); | ||
if (actualBinding !== binding) { | ||
return; | ||
} | ||
mangler.renamedNodes.add(refPath.node); | ||
refPath.replaceWith(t.identifier(newName)); | ||
mangler.renamedNodes.add(refPath.node); | ||
scopeTracker.updateReference(refPath.scope, binding, oldName, newName); | ||
} | ||
}); | ||
} else if (!isLabelIdentifier(path)) { | ||
if (path.node.name === oldName) { | ||
mangler.renamedNodes.add(path.node); | ||
path.replaceWith(t.identifier(newName)); | ||
mangler.renamedNodes.add(path.node); | ||
scopeTracker.updateReference(path.scope, binding, oldName, newName); | ||
} else if (mangler.renamedNodes.has(path.node)) { | ||
// already renamed, | ||
// just update the references | ||
scopeTracker.updateReference(path.scope, binding, oldName, newName); | ||
} else { | ||
throw new Error(`Unexpected Rename Error: ` + `Trying to replace "${node.name}": from "${oldName}" to "${newName}". ` + `Please report it at ${newIssueUrl}`); | ||
scopeTracker.updateReference(refPath.scope, binding, oldName, newName); | ||
} | ||
}); | ||
} else if (!isLabelIdentifier(path)) { | ||
if (path.node.name === oldName) { | ||
mangler.renamedNodes.add(path.node); | ||
path.replaceWith(t.identifier(newName)); | ||
mangler.renamedNodes.add(path.node); | ||
scopeTracker.updateReference(path.scope, binding, oldName, newName); | ||
} else if (mangler.renamedNodes.has(path.node)) { | ||
// already renamed, | ||
// just update the references | ||
scopeTracker.updateReference(path.scope, binding, oldName, newName); | ||
} else { | ||
throw new Error(`Unexpected Rename Error: ` + `Trying to replace "${node.name}": from "${oldName}" to "${newName}". ` + `Please report it at ${newIssueUrl}`); | ||
} | ||
// else label identifier - silently ignore | ||
} | ||
// else label identifier - silently ignore | ||
} | ||
// update babel's scope tracking | ||
var bindings = scope.bindings; | ||
// update babel's scope tracking | ||
const bindings = scope.bindings; | ||
bindings[newName] = binding; | ||
delete bindings[oldName]; | ||
} | ||
}]); | ||
bindings[newName] = binding; | ||
delete bindings[oldName]; | ||
} | ||
} | ||
return Mangler; | ||
}(); | ||
return { | ||
@@ -586,7 +531,7 @@ name: "minify-mangle-names", | ||
// sorting on the character set. Currently the number is pretty arbitrary. | ||
var shouldConsiderSource = path.getSource().length > 70000; | ||
const shouldConsiderSource = path.getSource().length > 70000; | ||
var charset = new Charset(shouldConsiderSource); | ||
const charset = new Charset(shouldConsiderSource); | ||
var mangler = new Mangler(charset, path, this.opts); | ||
const mangler = new Mangler(charset, path, this.opts); | ||
mangler.run(); | ||
@@ -605,4 +550,4 @@ } | ||
var map = {}; | ||
for (var i = 0; i < value.length; i++) { | ||
const map = {}; | ||
for (let i = 0; i < value.length; i++) { | ||
map[value[i]] = true; | ||
@@ -609,0 +554,0 @@ } |
@@ -6,5 +6,5 @@ "use strict"; | ||
function isLabelIdentifier(path) { | ||
var node = path.node; | ||
const node = path.node; | ||
return path.parentPath.isLabeledStatement({ label: node }) || path.parentPath.isBreakStatement({ label: node }) || path.parentPath.isContinueStatement({ label: node }); | ||
} |
"use strict"; | ||
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); | ||
const CountedSet = require("./counted-set"); | ||
const isLabelIdentifier = require("./is-label-identifier"); | ||
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } | ||
const newIssueUrl = "https://github.com/babel/minify/issues/new"; | ||
var CountedSet = require("./counted-set"); | ||
var isLabelIdentifier = require("./is-label-identifier"); | ||
var newIssueUrl = "https://github.com/babel/minify/issues/new"; | ||
/** | ||
@@ -17,6 +13,4 @@ * ScopeTracker | ||
*/ | ||
module.exports = function () { | ||
function ScopeTracker() { | ||
_classCallCheck(this, ScopeTracker); | ||
module.exports = class ScopeTracker { | ||
constructor() { | ||
this.references = new Map(); | ||
@@ -30,134 +24,114 @@ this.bindings = new Map(); | ||
*/ | ||
_createClass(ScopeTracker, [{ | ||
key: "addScope", | ||
value: function addScope(scope) { | ||
if (!this.references.has(scope)) { | ||
this.references.set(scope, new CountedSet()); | ||
} | ||
if (!this.bindings.has(scope)) { | ||
this.bindings.set(scope, new Map()); | ||
} | ||
addScope(scope) { | ||
if (!this.references.has(scope)) { | ||
this.references.set(scope, new CountedSet()); | ||
} | ||
/** | ||
* Add reference to all Scopes between and including the ReferencedScope | ||
* and Binding's Scope | ||
* @param {Scope} scope | ||
* @param {Binding} binding | ||
* @param {String} name | ||
*/ | ||
}, { | ||
key: "addReference", | ||
value: function addReference(scope, binding, name) { | ||
var parent = scope; | ||
do { | ||
this.references.get(parent).add(name); | ||
if (!binding) { | ||
throw new Error(`Binding Not Found for ${name} during scopeTracker.addRefernce. ` + `Please report at ${newIssueUrl}`); | ||
} | ||
if (binding.scope === parent) break; | ||
} while (parent = parent.parent); | ||
if (!this.bindings.has(scope)) { | ||
this.bindings.set(scope, new Map()); | ||
} | ||
} | ||
/** | ||
* has a Reference in the given {Scope} or a child Scope | ||
* | ||
* Refer {addReference} to know why the following call will be valid | ||
* for detecting references in child Scopes | ||
* | ||
* @param {Scope} scope | ||
* @param {String} name | ||
*/ | ||
/** | ||
* Add reference to all Scopes between and including the ReferencedScope | ||
* and Binding's Scope | ||
* @param {Scope} scope | ||
* @param {Binding} binding | ||
* @param {String} name | ||
*/ | ||
addReference(scope, binding, name) { | ||
let parent = scope; | ||
do { | ||
this.references.get(parent).add(name); | ||
if (!binding) { | ||
throw new Error(`Binding Not Found for ${name} during scopeTracker.addRefernce. ` + `Please report at ${newIssueUrl}`); | ||
} | ||
if (binding.scope === parent) break; | ||
} while (parent = parent.parent); | ||
} | ||
}, { | ||
key: "hasReference", | ||
value: function hasReference(scope, name) { | ||
return this.references.get(scope).has(name); | ||
} | ||
/** | ||
* has a Reference in the given {Scope} or a child Scope | ||
* | ||
* Refer {addReference} to know why the following call will be valid | ||
* for detecting references in child Scopes | ||
* | ||
* @param {Scope} scope | ||
* @param {String} name | ||
*/ | ||
hasReference(scope, name) { | ||
return this.references.get(scope).has(name); | ||
} | ||
/** | ||
* Update reference count in all scopes between and including the | ||
* Referenced Scope and the Binding's Scope | ||
* | ||
* @param {Scope} scope | ||
* @param {Binding} binding | ||
* @param {String} oldName | ||
* @param {String} newName | ||
*/ | ||
/** | ||
* Update reference count in all scopes between and including the | ||
* Referenced Scope and the Binding's Scope | ||
* | ||
* @param {Scope} scope | ||
* @param {Binding} binding | ||
* @param {String} oldName | ||
* @param {String} newName | ||
*/ | ||
updateReference(scope, binding, oldName, newName) { | ||
let parent = scope; | ||
do { | ||
const ref = this.references.get(parent); | ||
}, { | ||
key: "updateReference", | ||
value: function updateReference(scope, binding, oldName, newName) { | ||
var parent = scope; | ||
do { | ||
var ref = this.references.get(parent); | ||
ref.delete(oldName); | ||
ref.add(newName); | ||
ref.delete(oldName); | ||
ref.add(newName); | ||
if (!binding) { | ||
// Something went wrong - panic | ||
throw new Error("Binding Not Found during scopeTracker.updateRefernce " + `while updating "${oldName}" to "${newName}". ` + `Please report at ${newIssueUrl}`); | ||
} | ||
if (!binding) { | ||
// Something went wrong - panic | ||
throw new Error("Binding Not Found during scopeTracker.updateRefernce " + `while updating "${oldName}" to "${newName}". ` + `Please report at ${newIssueUrl}`); | ||
} | ||
if (binding.scope === parent) break; | ||
} while (parent = parent.parent); | ||
} | ||
if (binding.scope === parent) break; | ||
} while (parent = parent.parent); | ||
} | ||
/** | ||
* has either a Binding or a Reference | ||
* @param {Scope} scope | ||
* @param {Binding} binding | ||
* @param {String} name | ||
*/ | ||
hasBindingOrReference(scope, binding, name) { | ||
return this.hasReference(scope, name) || this.hasBinding(scope, name); | ||
} | ||
/** | ||
* has either a Binding or a Reference | ||
* @param {Scope} scope | ||
* @param {Binding} binding | ||
* @param {String} name | ||
*/ | ||
/** | ||
* For a Binding visit all places where the Binding is used and detect | ||
* if the newName {next} can be used in all these places | ||
* | ||
* 1. binding's own scope | ||
* 2. constant violations' scopes | ||
* 3. referencePaths' scopes | ||
* | ||
* @param {Binding} binding | ||
* @param {String} next | ||
*/ | ||
canUseInReferencedScopes(binding, next) { | ||
const tracker = this; | ||
}, { | ||
key: "hasBindingOrReference", | ||
value: function hasBindingOrReference(scope, binding, name) { | ||
return this.hasReference(scope, name) || this.hasBinding(scope, name); | ||
if (tracker.hasBindingOrReference(binding.scope, binding, next)) { | ||
return false; | ||
} | ||
/** | ||
* For a Binding visit all places where the Binding is used and detect | ||
* if the newName {next} can be used in all these places | ||
* | ||
* 1. binding's own scope | ||
* 2. constant violations' scopes | ||
* 3. referencePaths' scopes | ||
* | ||
* @param {Binding} binding | ||
* @param {String} next | ||
*/ | ||
}, { | ||
key: "canUseInReferencedScopes", | ||
value: function canUseInReferencedScopes(binding, next) { | ||
var tracker = this; | ||
if (tracker.hasBindingOrReference(binding.scope, binding, next)) { | ||
return false; | ||
} | ||
// Safari raises a syntax error for a `let` or `const` declaration in a | ||
// `for` loop initialization that shadows a parent function's parameter. | ||
// https://github.com/babel/minify/issues/559 | ||
// https://bugs.webkit.org/show_bug.cgi?id=171041 | ||
// https://trac.webkit.org/changeset/217200/webkit/trunk/Source | ||
var maybeDecl = binding.path.parentPath; | ||
var isBlockScoped = maybeDecl.isVariableDeclaration({ kind: "let" }) || maybeDecl.isVariableDeclaration({ kind: "const" }); | ||
if (isBlockScoped) { | ||
var maybeFor = maybeDecl.parentPath; | ||
var isForLoopBinding = maybeFor.isForStatement({ init: maybeDecl.node }) || maybeFor.isForXStatement({ left: maybeDecl.node }); | ||
if (isForLoopBinding) { | ||
var fnParent = maybeFor.getFunctionParent(); | ||
if (fnParent.isFunction({ body: maybeFor.parent })) { | ||
var parentFunctionBinding = this.bindings.get(fnParent.scope).get(next); | ||
if (parentFunctionBinding) { | ||
var parentFunctionHasParamBinding = parentFunctionBinding.kind === "param"; | ||
if (parentFunctionHasParamBinding) { | ||
return false; | ||
} | ||
// Safari raises a syntax error for a `let` or `const` declaration in a | ||
// `for` loop initialization that shadows a parent function's parameter. | ||
// https://github.com/babel/minify/issues/559 | ||
// https://bugs.webkit.org/show_bug.cgi?id=171041 | ||
// https://trac.webkit.org/changeset/217200/webkit/trunk/Source | ||
const maybeDecl = binding.path.parentPath; | ||
const isBlockScoped = maybeDecl.isVariableDeclaration({ kind: "let" }) || maybeDecl.isVariableDeclaration({ kind: "const" }); | ||
if (isBlockScoped) { | ||
const maybeFor = maybeDecl.parentPath; | ||
const isForLoopBinding = maybeFor.isForStatement({ init: maybeDecl.node }) || maybeFor.isForXStatement({ left: maybeDecl.node }); | ||
if (isForLoopBinding) { | ||
const fnParent = maybeFor.getFunctionParent(); | ||
if (fnParent.isFunction({ body: maybeFor.parent })) { | ||
const parentFunctionBinding = this.bindings.get(fnParent.scope).get(next); | ||
if (parentFunctionBinding) { | ||
const parentFunctionHasParamBinding = parentFunctionBinding.kind === "param"; | ||
if (parentFunctionHasParamBinding) { | ||
return false; | ||
} | ||
@@ -167,102 +141,88 @@ } | ||
} | ||
} | ||
for (var i = 0; i < binding.constantViolations.length; i++) { | ||
var violation = binding.constantViolations[i]; | ||
if (tracker.hasBindingOrReference(violation.scope, binding, next)) { | ||
return false; | ||
} | ||
for (let i = 0; i < binding.constantViolations.length; i++) { | ||
const violation = binding.constantViolations[i]; | ||
if (tracker.hasBindingOrReference(violation.scope, binding, next)) { | ||
return false; | ||
} | ||
} | ||
for (var _i = 0; _i < binding.referencePaths.length; _i++) { | ||
var ref = binding.referencePaths[_i]; | ||
if (!ref.isIdentifier()) { | ||
var canUse = true; | ||
ref.traverse({ | ||
ReferencedIdentifier(path) { | ||
if (path.node.name !== next) return; | ||
if (tracker.hasBindingOrReference(path.scope, binding, next)) { | ||
canUse = false; | ||
} | ||
for (let i = 0; i < binding.referencePaths.length; i++) { | ||
const ref = binding.referencePaths[i]; | ||
if (!ref.isIdentifier()) { | ||
let canUse = true; | ||
ref.traverse({ | ||
ReferencedIdentifier(path) { | ||
if (path.node.name !== next) return; | ||
if (tracker.hasBindingOrReference(path.scope, binding, next)) { | ||
canUse = false; | ||
} | ||
}); | ||
if (!canUse) { | ||
return canUse; | ||
} | ||
} else if (!isLabelIdentifier(ref)) { | ||
if (tracker.hasBindingOrReference(ref.scope, binding, next)) { | ||
return false; | ||
} | ||
}); | ||
if (!canUse) { | ||
return canUse; | ||
} | ||
} else if (!isLabelIdentifier(ref)) { | ||
if (tracker.hasBindingOrReference(ref.scope, binding, next)) { | ||
return false; | ||
} | ||
} | ||
return true; | ||
} | ||
/** | ||
* Add a binding to Tracker in binding's own Scope | ||
* @param {Binding} binding | ||
*/ | ||
return true; | ||
} | ||
}, { | ||
key: "addBinding", | ||
value: function addBinding(binding) { | ||
if (!binding) { | ||
return; | ||
} | ||
var bindings = this.bindings.get(binding.scope); | ||
var existingBinding = bindings.get(binding.identifier.name); | ||
if (existingBinding && existingBinding !== binding) { | ||
throw new Error(`scopeTracker.addBinding: ` + `Binding "${existingBinding.identifier.name}" already exists. ` + `Trying to add "${binding.identifier.name}" again.`); | ||
} | ||
bindings.set(binding.identifier.name, binding); | ||
/** | ||
* Add a binding to Tracker in binding's own Scope | ||
* @param {Binding} binding | ||
*/ | ||
addBinding(binding) { | ||
if (!binding) { | ||
return; | ||
} | ||
/** | ||
* Moves Binding from it's own Scope to {toScope} | ||
* | ||
* required for fixup-var-scope | ||
* | ||
* @param {Binding} binding | ||
* @param {Scope} toScope | ||
*/ | ||
const bindings = this.bindings.get(binding.scope); | ||
const existingBinding = bindings.get(binding.identifier.name); | ||
}, { | ||
key: "moveBinding", | ||
value: function moveBinding(binding, toScope) { | ||
this.bindings.get(binding.scope).delete(binding.identifier.name); | ||
this.bindings.get(toScope).set(binding.identifier.name, binding); | ||
if (existingBinding && existingBinding !== binding) { | ||
throw new Error(`scopeTracker.addBinding: ` + `Binding "${existingBinding.identifier.name}" already exists. ` + `Trying to add "${binding.identifier.name}" again.`); | ||
} | ||
/** | ||
* has a Binding in the current {Scope} | ||
* @param {Scope} scope | ||
* @param {String} name | ||
*/ | ||
bindings.set(binding.identifier.name, binding); | ||
} | ||
}, { | ||
key: "hasBinding", | ||
value: function hasBinding(scope, name) { | ||
return this.bindings.get(scope).has(name); | ||
} | ||
/** | ||
* Moves Binding from it's own Scope to {toScope} | ||
* | ||
* required for fixup-var-scope | ||
* | ||
* @param {Binding} binding | ||
* @param {Scope} toScope | ||
*/ | ||
moveBinding(binding, toScope) { | ||
this.bindings.get(binding.scope).delete(binding.identifier.name); | ||
this.bindings.get(toScope).set(binding.identifier.name, binding); | ||
} | ||
/** | ||
* Update the ScopeTracker on rename | ||
* @param {Scope} scope | ||
* @param {String} oldName | ||
* @param {String} newName | ||
*/ | ||
/** | ||
* has a Binding in the current {Scope} | ||
* @param {Scope} scope | ||
* @param {String} name | ||
*/ | ||
hasBinding(scope, name) { | ||
return this.bindings.get(scope).has(name); | ||
} | ||
}, { | ||
key: "renameBinding", | ||
value: function renameBinding(scope, oldName, newName) { | ||
var bindings = this.bindings.get(scope); | ||
bindings.set(newName, bindings.get(oldName)); | ||
bindings.delete(oldName); | ||
} | ||
}]); | ||
return ScopeTracker; | ||
}(); | ||
/** | ||
* Update the ScopeTracker on rename | ||
* @param {Scope} scope | ||
* @param {String} oldName | ||
* @param {String} newName | ||
*/ | ||
renameBinding(scope, oldName, newName) { | ||
const bindings = this.bindings.get(scope); | ||
bindings.set(newName, bindings.get(oldName)); | ||
bindings.delete(oldName); | ||
} | ||
}; |
{ | ||
"name": "babel-plugin-minify-mangle-names", | ||
"version": "0.3.0-alpha.37064b40", | ||
"version": "0.3.0-alpha.5ee3078e", | ||
"description": "", | ||
@@ -15,4 +15,4 @@ "keywords": [ | ||
"dependencies": { | ||
"babel-helper-mark-eval-scopes": "^0.3.0-alpha.37064b40" | ||
"babel-helper-mark-eval-scopes": "^0.3.0-alpha.5ee3078e" | ||
} | ||
} |
33461
898