source: imaps-frontend/node_modules/@babel/traverse/lib/scope/index.js@ 79a0317

main
Last change on this file since 79a0317 was 79a0317, checked in by stefan toskovski <stefantoska84@…>, 4 days ago

F4 Finalna Verzija

  • Property mode set to 100644
File size: 29.4 KB
Line 
1"use strict";
2
3Object.defineProperty(exports, "__esModule", {
4 value: true
5});
6exports.default = void 0;
7var _renamer = require("./lib/renamer.js");
8var _index = require("../index.js");
9var _binding = require("./binding.js");
10var _globals = require("globals");
11var _t = require("@babel/types");
12var t = _t;
13var _cache = require("../cache.js");
14var _visitors = require("../visitors.js");
15const {
16 NOT_LOCAL_BINDING,
17 assignmentExpression,
18 callExpression,
19 cloneNode,
20 getBindingIdentifiers,
21 identifier,
22 isArrayExpression,
23 isBinary,
24 isCallExpression,
25 isClass,
26 isClassBody,
27 isClassDeclaration,
28 isExportAllDeclaration,
29 isExportDefaultDeclaration,
30 isExportNamedDeclaration,
31 isFunctionDeclaration,
32 isIdentifier,
33 isImportDeclaration,
34 isLiteral,
35 isMemberExpression,
36 isMethod,
37 isModuleSpecifier,
38 isNullLiteral,
39 isObjectExpression,
40 isProperty,
41 isPureish,
42 isRegExpLiteral,
43 isSuper,
44 isTaggedTemplateExpression,
45 isTemplateLiteral,
46 isThisExpression,
47 isUnaryExpression,
48 isVariableDeclaration,
49 expressionStatement,
50 matchesPattern,
51 memberExpression,
52 numericLiteral,
53 toIdentifier,
54 variableDeclaration,
55 variableDeclarator,
56 isRecordExpression,
57 isTupleExpression,
58 isObjectProperty,
59 isTopicReference,
60 isMetaProperty,
61 isPrivateName,
62 isExportDeclaration,
63 buildUndefinedNode,
64 sequenceExpression
65} = _t;
66function gatherNodeParts(node, parts) {
67 switch (node == null ? void 0 : node.type) {
68 default:
69 if (isImportDeclaration(node) || isExportDeclaration(node)) {
70 var _node$specifiers;
71 if ((isExportAllDeclaration(node) || isExportNamedDeclaration(node) || isImportDeclaration(node)) && node.source) {
72 gatherNodeParts(node.source, parts);
73 } else if ((isExportNamedDeclaration(node) || isImportDeclaration(node)) && (_node$specifiers = node.specifiers) != null && _node$specifiers.length) {
74 for (const e of node.specifiers) gatherNodeParts(e, parts);
75 } else if ((isExportDefaultDeclaration(node) || isExportNamedDeclaration(node)) && node.declaration) {
76 gatherNodeParts(node.declaration, parts);
77 }
78 } else if (isModuleSpecifier(node)) {
79 gatherNodeParts(node.local, parts);
80 } else if (isLiteral(node) && !isNullLiteral(node) && !isRegExpLiteral(node) && !isTemplateLiteral(node)) {
81 parts.push(node.value);
82 }
83 break;
84 case "MemberExpression":
85 case "OptionalMemberExpression":
86 case "JSXMemberExpression":
87 gatherNodeParts(node.object, parts);
88 gatherNodeParts(node.property, parts);
89 break;
90 case "Identifier":
91 case "JSXIdentifier":
92 parts.push(node.name);
93 break;
94 case "CallExpression":
95 case "OptionalCallExpression":
96 case "NewExpression":
97 gatherNodeParts(node.callee, parts);
98 break;
99 case "ObjectExpression":
100 case "ObjectPattern":
101 for (const e of node.properties) {
102 gatherNodeParts(e, parts);
103 }
104 break;
105 case "SpreadElement":
106 case "RestElement":
107 gatherNodeParts(node.argument, parts);
108 break;
109 case "ObjectProperty":
110 case "ObjectMethod":
111 case "ClassProperty":
112 case "ClassMethod":
113 case "ClassPrivateProperty":
114 case "ClassPrivateMethod":
115 gatherNodeParts(node.key, parts);
116 break;
117 case "ThisExpression":
118 parts.push("this");
119 break;
120 case "Super":
121 parts.push("super");
122 break;
123 case "Import":
124 parts.push("import");
125 break;
126 case "DoExpression":
127 parts.push("do");
128 break;
129 case "YieldExpression":
130 parts.push("yield");
131 gatherNodeParts(node.argument, parts);
132 break;
133 case "AwaitExpression":
134 parts.push("await");
135 gatherNodeParts(node.argument, parts);
136 break;
137 case "AssignmentExpression":
138 gatherNodeParts(node.left, parts);
139 break;
140 case "VariableDeclarator":
141 gatherNodeParts(node.id, parts);
142 break;
143 case "FunctionExpression":
144 case "FunctionDeclaration":
145 case "ClassExpression":
146 case "ClassDeclaration":
147 gatherNodeParts(node.id, parts);
148 break;
149 case "PrivateName":
150 gatherNodeParts(node.id, parts);
151 break;
152 case "ParenthesizedExpression":
153 gatherNodeParts(node.expression, parts);
154 break;
155 case "UnaryExpression":
156 case "UpdateExpression":
157 gatherNodeParts(node.argument, parts);
158 break;
159 case "MetaProperty":
160 gatherNodeParts(node.meta, parts);
161 gatherNodeParts(node.property, parts);
162 break;
163 case "JSXElement":
164 gatherNodeParts(node.openingElement, parts);
165 break;
166 case "JSXOpeningElement":
167 gatherNodeParts(node.name, parts);
168 break;
169 case "JSXFragment":
170 gatherNodeParts(node.openingFragment, parts);
171 break;
172 case "JSXOpeningFragment":
173 parts.push("Fragment");
174 break;
175 case "JSXNamespacedName":
176 gatherNodeParts(node.namespace, parts);
177 gatherNodeParts(node.name, parts);
178 break;
179 }
180}
181const collectorVisitor = {
182 ForStatement(path) {
183 const declar = path.get("init");
184 if (declar.isVar()) {
185 const {
186 scope
187 } = path;
188 const parentScope = scope.getFunctionParent() || scope.getProgramParent();
189 parentScope.registerBinding("var", declar);
190 }
191 },
192 Declaration(path) {
193 if (path.isBlockScoped()) return;
194 if (path.isImportDeclaration()) return;
195 if (path.isExportDeclaration()) return;
196 const parent = path.scope.getFunctionParent() || path.scope.getProgramParent();
197 parent.registerDeclaration(path);
198 },
199 ImportDeclaration(path) {
200 const parent = path.scope.getBlockParent();
201 parent.registerDeclaration(path);
202 },
203 ReferencedIdentifier(path, state) {
204 state.references.push(path);
205 },
206 ForXStatement(path, state) {
207 const left = path.get("left");
208 if (left.isPattern() || left.isIdentifier()) {
209 state.constantViolations.push(path);
210 } else if (left.isVar()) {
211 const {
212 scope
213 } = path;
214 const parentScope = scope.getFunctionParent() || scope.getProgramParent();
215 parentScope.registerBinding("var", left);
216 }
217 },
218 ExportDeclaration: {
219 exit(path) {
220 const {
221 node,
222 scope
223 } = path;
224 if (isExportAllDeclaration(node)) return;
225 const declar = node.declaration;
226 if (isClassDeclaration(declar) || isFunctionDeclaration(declar)) {
227 const id = declar.id;
228 if (!id) return;
229 const binding = scope.getBinding(id.name);
230 binding == null || binding.reference(path);
231 } else if (isVariableDeclaration(declar)) {
232 for (const decl of declar.declarations) {
233 for (const name of Object.keys(getBindingIdentifiers(decl))) {
234 const binding = scope.getBinding(name);
235 binding == null || binding.reference(path);
236 }
237 }
238 }
239 }
240 },
241 LabeledStatement(path) {
242 path.scope.getBlockParent().registerDeclaration(path);
243 },
244 AssignmentExpression(path, state) {
245 state.assignments.push(path);
246 },
247 UpdateExpression(path, state) {
248 state.constantViolations.push(path);
249 },
250 UnaryExpression(path, state) {
251 if (path.node.operator === "delete") {
252 state.constantViolations.push(path);
253 }
254 },
255 BlockScoped(path) {
256 let scope = path.scope;
257 if (scope.path === path) scope = scope.parent;
258 const parent = scope.getBlockParent();
259 parent.registerDeclaration(path);
260 if (path.isClassDeclaration() && path.node.id) {
261 const id = path.node.id;
262 const name = id.name;
263 path.scope.bindings[name] = path.scope.parent.getBinding(name);
264 }
265 },
266 CatchClause(path) {
267 path.scope.registerBinding("let", path);
268 },
269 Function(path) {
270 const params = path.get("params");
271 for (const param of params) {
272 path.scope.registerBinding("param", param);
273 }
274 if (path.isFunctionExpression() && path.node.id && !path.node.id[NOT_LOCAL_BINDING]) {
275 path.scope.registerBinding("local", path.get("id"), path);
276 }
277 },
278 ClassExpression(path) {
279 if (path.node.id && !path.node.id[NOT_LOCAL_BINDING]) {
280 path.scope.registerBinding("local", path.get("id"), path);
281 }
282 },
283 TSTypeAnnotation(path) {
284 path.skip();
285 }
286};
287let uid = 0;
288class Scope {
289 constructor(path) {
290 this.uid = void 0;
291 this.path = void 0;
292 this.block = void 0;
293 this.inited = void 0;
294 this.labels = void 0;
295 this.bindings = void 0;
296 this.references = void 0;
297 this.globals = void 0;
298 this.uids = void 0;
299 this.data = void 0;
300 this.crawling = void 0;
301 const {
302 node
303 } = path;
304 const cached = _cache.scope.get(node);
305 if ((cached == null ? void 0 : cached.path) === path) {
306 return cached;
307 }
308 _cache.scope.set(node, this);
309 this.uid = uid++;
310 this.block = node;
311 this.path = path;
312 this.labels = new Map();
313 this.inited = false;
314 }
315 get parent() {
316 var _parent;
317 let parent,
318 path = this.path;
319 do {
320 var _path;
321 const shouldSkip = path.key === "key" || path.listKey === "decorators";
322 path = path.parentPath;
323 if (shouldSkip && path.isMethod()) path = path.parentPath;
324 if ((_path = path) != null && _path.isScope()) parent = path;
325 } while (path && !parent);
326 return (_parent = parent) == null ? void 0 : _parent.scope;
327 }
328 generateDeclaredUidIdentifier(name) {
329 const id = this.generateUidIdentifier(name);
330 this.push({
331 id
332 });
333 return cloneNode(id);
334 }
335 generateUidIdentifier(name) {
336 return identifier(this.generateUid(name));
337 }
338 generateUid(name = "temp") {
339 name = toIdentifier(name).replace(/^_+/, "").replace(/\d+$/g, "");
340 let uid;
341 let i = 1;
342 do {
343 uid = `_${name}`;
344 if (i > 1) uid += i;
345 i++;
346 } while (this.hasLabel(uid) || this.hasBinding(uid) || this.hasGlobal(uid) || this.hasReference(uid));
347 const program = this.getProgramParent();
348 program.references[uid] = true;
349 program.uids[uid] = true;
350 return uid;
351 }
352 generateUidBasedOnNode(node, defaultName) {
353 const parts = [];
354 gatherNodeParts(node, parts);
355 let id = parts.join("$");
356 id = id.replace(/^_/, "") || defaultName || "ref";
357 return this.generateUid(id.slice(0, 20));
358 }
359 generateUidIdentifierBasedOnNode(node, defaultName) {
360 return identifier(this.generateUidBasedOnNode(node, defaultName));
361 }
362 isStatic(node) {
363 if (isThisExpression(node) || isSuper(node) || isTopicReference(node)) {
364 return true;
365 }
366 if (isIdentifier(node)) {
367 const binding = this.getBinding(node.name);
368 if (binding) {
369 return binding.constant;
370 } else {
371 return this.hasBinding(node.name);
372 }
373 }
374 return false;
375 }
376 maybeGenerateMemoised(node, dontPush) {
377 if (this.isStatic(node)) {
378 return null;
379 } else {
380 const id = this.generateUidIdentifierBasedOnNode(node);
381 if (!dontPush) {
382 this.push({
383 id
384 });
385 return cloneNode(id);
386 }
387 return id;
388 }
389 }
390 checkBlockScopedCollisions(local, kind, name, id) {
391 if (kind === "param") return;
392 if (local.kind === "local") return;
393 const duplicate = kind === "let" || local.kind === "let" || local.kind === "const" || local.kind === "module" || local.kind === "param" && kind === "const";
394 if (duplicate) {
395 throw this.path.hub.buildError(id, `Duplicate declaration "${name}"`, TypeError);
396 }
397 }
398 rename(oldName, newName) {
399 const binding = this.getBinding(oldName);
400 if (binding) {
401 newName || (newName = this.generateUidIdentifier(oldName).name);
402 const renamer = new _renamer.default(binding, oldName, newName);
403 {
404 renamer.rename(arguments[2]);
405 }
406 }
407 }
408 dump() {
409 const sep = "-".repeat(60);
410 console.log(sep);
411 let scope = this;
412 do {
413 console.log("#", scope.block.type);
414 for (const name of Object.keys(scope.bindings)) {
415 const binding = scope.bindings[name];
416 console.log(" -", name, {
417 constant: binding.constant,
418 references: binding.references,
419 violations: binding.constantViolations.length,
420 kind: binding.kind
421 });
422 }
423 } while (scope = scope.parent);
424 console.log(sep);
425 }
426 hasLabel(name) {
427 return !!this.getLabel(name);
428 }
429 getLabel(name) {
430 return this.labels.get(name);
431 }
432 registerLabel(path) {
433 this.labels.set(path.node.label.name, path);
434 }
435 registerDeclaration(path) {
436 if (path.isLabeledStatement()) {
437 this.registerLabel(path);
438 } else if (path.isFunctionDeclaration()) {
439 this.registerBinding("hoisted", path.get("id"), path);
440 } else if (path.isVariableDeclaration()) {
441 const declarations = path.get("declarations");
442 const {
443 kind
444 } = path.node;
445 for (const declar of declarations) {
446 this.registerBinding(kind === "using" || kind === "await using" ? "const" : kind, declar);
447 }
448 } else if (path.isClassDeclaration()) {
449 if (path.node.declare) return;
450 this.registerBinding("let", path);
451 } else if (path.isImportDeclaration()) {
452 const isTypeDeclaration = path.node.importKind === "type" || path.node.importKind === "typeof";
453 const specifiers = path.get("specifiers");
454 for (const specifier of specifiers) {
455 const isTypeSpecifier = isTypeDeclaration || specifier.isImportSpecifier() && (specifier.node.importKind === "type" || specifier.node.importKind === "typeof");
456 this.registerBinding(isTypeSpecifier ? "unknown" : "module", specifier);
457 }
458 } else if (path.isExportDeclaration()) {
459 const declar = path.get("declaration");
460 if (declar.isClassDeclaration() || declar.isFunctionDeclaration() || declar.isVariableDeclaration()) {
461 this.registerDeclaration(declar);
462 }
463 } else {
464 this.registerBinding("unknown", path);
465 }
466 }
467 buildUndefinedNode() {
468 return buildUndefinedNode();
469 }
470 registerConstantViolation(path) {
471 const ids = path.getAssignmentIdentifiers();
472 for (const name of Object.keys(ids)) {
473 var _this$getBinding;
474 (_this$getBinding = this.getBinding(name)) == null || _this$getBinding.reassign(path);
475 }
476 }
477 registerBinding(kind, path, bindingPath = path) {
478 if (!kind) throw new ReferenceError("no `kind`");
479 if (path.isVariableDeclaration()) {
480 const declarators = path.get("declarations");
481 for (const declar of declarators) {
482 this.registerBinding(kind, declar);
483 }
484 return;
485 }
486 const parent = this.getProgramParent();
487 const ids = path.getOuterBindingIdentifiers(true);
488 for (const name of Object.keys(ids)) {
489 parent.references[name] = true;
490 for (const id of ids[name]) {
491 const local = this.getOwnBinding(name);
492 if (local) {
493 if (local.identifier === id) continue;
494 this.checkBlockScopedCollisions(local, kind, name, id);
495 }
496 if (local) {
497 local.reassign(bindingPath);
498 } else {
499 this.bindings[name] = new _binding.default({
500 identifier: id,
501 scope: this,
502 path: bindingPath,
503 kind: kind
504 });
505 }
506 }
507 }
508 }
509 addGlobal(node) {
510 this.globals[node.name] = node;
511 }
512 hasUid(name) {
513 let scope = this;
514 do {
515 if (scope.uids[name]) return true;
516 } while (scope = scope.parent);
517 return false;
518 }
519 hasGlobal(name) {
520 let scope = this;
521 do {
522 if (scope.globals[name]) return true;
523 } while (scope = scope.parent);
524 return false;
525 }
526 hasReference(name) {
527 return !!this.getProgramParent().references[name];
528 }
529 isPure(node, constantsOnly) {
530 if (isIdentifier(node)) {
531 const binding = this.getBinding(node.name);
532 if (!binding) return false;
533 if (constantsOnly) return binding.constant;
534 return true;
535 } else if (isThisExpression(node) || isMetaProperty(node) || isTopicReference(node) || isPrivateName(node)) {
536 return true;
537 } else if (isClass(node)) {
538 var _node$decorators;
539 if (node.superClass && !this.isPure(node.superClass, constantsOnly)) {
540 return false;
541 }
542 if (((_node$decorators = node.decorators) == null ? void 0 : _node$decorators.length) > 0) {
543 return false;
544 }
545 return this.isPure(node.body, constantsOnly);
546 } else if (isClassBody(node)) {
547 for (const method of node.body) {
548 if (!this.isPure(method, constantsOnly)) return false;
549 }
550 return true;
551 } else if (isBinary(node)) {
552 return this.isPure(node.left, constantsOnly) && this.isPure(node.right, constantsOnly);
553 } else if (isArrayExpression(node) || isTupleExpression(node)) {
554 for (const elem of node.elements) {
555 if (elem !== null && !this.isPure(elem, constantsOnly)) return false;
556 }
557 return true;
558 } else if (isObjectExpression(node) || isRecordExpression(node)) {
559 for (const prop of node.properties) {
560 if (!this.isPure(prop, constantsOnly)) return false;
561 }
562 return true;
563 } else if (isMethod(node)) {
564 var _node$decorators2;
565 if (node.computed && !this.isPure(node.key, constantsOnly)) return false;
566 if (((_node$decorators2 = node.decorators) == null ? void 0 : _node$decorators2.length) > 0) {
567 return false;
568 }
569 return true;
570 } else if (isProperty(node)) {
571 var _node$decorators3;
572 if (node.computed && !this.isPure(node.key, constantsOnly)) return false;
573 if (((_node$decorators3 = node.decorators) == null ? void 0 : _node$decorators3.length) > 0) {
574 return false;
575 }
576 if (isObjectProperty(node) || node.static) {
577 if (node.value !== null && !this.isPure(node.value, constantsOnly)) {
578 return false;
579 }
580 }
581 return true;
582 } else if (isUnaryExpression(node)) {
583 return this.isPure(node.argument, constantsOnly);
584 } else if (isTemplateLiteral(node)) {
585 for (const expression of node.expressions) {
586 if (!this.isPure(expression, constantsOnly)) return false;
587 }
588 return true;
589 } else if (isTaggedTemplateExpression(node)) {
590 return matchesPattern(node.tag, "String.raw") && !this.hasBinding("String", {
591 noGlobals: true
592 }) && this.isPure(node.quasi, constantsOnly);
593 } else if (isMemberExpression(node)) {
594 return !node.computed && isIdentifier(node.object) && node.object.name === "Symbol" && isIdentifier(node.property) && node.property.name !== "for" && !this.hasBinding("Symbol", {
595 noGlobals: true
596 });
597 } else if (isCallExpression(node)) {
598 return matchesPattern(node.callee, "Symbol.for") && !this.hasBinding("Symbol", {
599 noGlobals: true
600 }) && node.arguments.length === 1 && t.isStringLiteral(node.arguments[0]);
601 } else {
602 return isPureish(node);
603 }
604 }
605 setData(key, val) {
606 return this.data[key] = val;
607 }
608 getData(key) {
609 let scope = this;
610 do {
611 const data = scope.data[key];
612 if (data != null) return data;
613 } while (scope = scope.parent);
614 }
615 removeData(key) {
616 let scope = this;
617 do {
618 const data = scope.data[key];
619 if (data != null) scope.data[key] = null;
620 } while (scope = scope.parent);
621 }
622 init() {
623 if (!this.inited) {
624 this.inited = true;
625 this.crawl();
626 }
627 }
628 crawl() {
629 const path = this.path;
630 this.references = Object.create(null);
631 this.bindings = Object.create(null);
632 this.globals = Object.create(null);
633 this.uids = Object.create(null);
634 this.data = Object.create(null);
635 let scope = this;
636 do {
637 if (scope.crawling) return;
638 if (scope.path.isProgram()) {
639 break;
640 }
641 } while (scope = scope.parent);
642 const programParent = scope;
643 const state = {
644 references: [],
645 constantViolations: [],
646 assignments: []
647 };
648 this.crawling = true;
649 if (path.type !== "Program" && (0, _visitors.isExplodedVisitor)(collectorVisitor)) {
650 for (const visit of collectorVisitor.enter) {
651 visit.call(state, path, state);
652 }
653 const typeVisitors = collectorVisitor[path.type];
654 if (typeVisitors) {
655 for (const visit of typeVisitors.enter) {
656 visit.call(state, path, state);
657 }
658 }
659 }
660 path.traverse(collectorVisitor, state);
661 this.crawling = false;
662 for (const path of state.assignments) {
663 const ids = path.getAssignmentIdentifiers();
664 for (const name of Object.keys(ids)) {
665 if (path.scope.getBinding(name)) continue;
666 programParent.addGlobal(ids[name]);
667 }
668 path.scope.registerConstantViolation(path);
669 }
670 for (const ref of state.references) {
671 const binding = ref.scope.getBinding(ref.node.name);
672 if (binding) {
673 binding.reference(ref);
674 } else {
675 programParent.addGlobal(ref.node);
676 }
677 }
678 for (const path of state.constantViolations) {
679 path.scope.registerConstantViolation(path);
680 }
681 }
682 push(opts) {
683 let path = this.path;
684 if (path.isPattern()) {
685 path = this.getPatternParent().path;
686 } else if (!path.isBlockStatement() && !path.isProgram()) {
687 path = this.getBlockParent().path;
688 }
689 if (path.isSwitchStatement()) {
690 path = (this.getFunctionParent() || this.getProgramParent()).path;
691 }
692 const {
693 init,
694 unique,
695 kind = "var",
696 id
697 } = opts;
698 if (!init && !unique && (kind === "var" || kind === "let") && path.isFunction() && !path.node.name && isCallExpression(path.parent, {
699 callee: path.node
700 }) && path.parent.arguments.length <= path.node.params.length && isIdentifier(id)) {
701 path.pushContainer("params", id);
702 path.scope.registerBinding("param", path.get("params")[path.node.params.length - 1]);
703 return;
704 }
705 if (path.isLoop() || path.isCatchClause() || path.isFunction()) {
706 path.ensureBlock();
707 path = path.get("body");
708 }
709 const blockHoist = opts._blockHoist == null ? 2 : opts._blockHoist;
710 const dataKey = `declaration:${kind}:${blockHoist}`;
711 let declarPath = !unique && path.getData(dataKey);
712 if (!declarPath) {
713 const declar = variableDeclaration(kind, []);
714 declar._blockHoist = blockHoist;
715 [declarPath] = path.unshiftContainer("body", [declar]);
716 if (!unique) path.setData(dataKey, declarPath);
717 }
718 const declarator = variableDeclarator(id, init);
719 const len = declarPath.node.declarations.push(declarator);
720 path.scope.registerBinding(kind, declarPath.get("declarations")[len - 1]);
721 }
722 getProgramParent() {
723 let scope = this;
724 do {
725 if (scope.path.isProgram()) {
726 return scope;
727 }
728 } while (scope = scope.parent);
729 throw new Error("Couldn't find a Program");
730 }
731 getFunctionParent() {
732 let scope = this;
733 do {
734 if (scope.path.isFunctionParent()) {
735 return scope;
736 }
737 } while (scope = scope.parent);
738 return null;
739 }
740 getBlockParent() {
741 let scope = this;
742 do {
743 if (scope.path.isBlockParent()) {
744 return scope;
745 }
746 } while (scope = scope.parent);
747 throw new Error("We couldn't find a BlockStatement, For, Switch, Function, Loop or Program...");
748 }
749 getPatternParent() {
750 let scope = this;
751 do {
752 if (!scope.path.isPattern()) {
753 return scope.getBlockParent();
754 }
755 } while (scope = scope.parent.parent);
756 throw new Error("We couldn't find a BlockStatement, For, Switch, Function, Loop or Program...");
757 }
758 getAllBindings() {
759 const ids = Object.create(null);
760 let scope = this;
761 do {
762 for (const key of Object.keys(scope.bindings)) {
763 if (key in ids === false) {
764 ids[key] = scope.bindings[key];
765 }
766 }
767 scope = scope.parent;
768 } while (scope);
769 return ids;
770 }
771 bindingIdentifierEquals(name, node) {
772 return this.getBindingIdentifier(name) === node;
773 }
774 getBinding(name) {
775 let scope = this;
776 let previousPath;
777 do {
778 const binding = scope.getOwnBinding(name);
779 if (binding) {
780 var _previousPath;
781 if ((_previousPath = previousPath) != null && _previousPath.isPattern() && binding.kind !== "param" && binding.kind !== "local") {} else {
782 return binding;
783 }
784 } else if (!binding && name === "arguments" && scope.path.isFunction() && !scope.path.isArrowFunctionExpression()) {
785 break;
786 }
787 previousPath = scope.path;
788 } while (scope = scope.parent);
789 }
790 getOwnBinding(name) {
791 return this.bindings[name];
792 }
793 getBindingIdentifier(name) {
794 var _this$getBinding2;
795 return (_this$getBinding2 = this.getBinding(name)) == null ? void 0 : _this$getBinding2.identifier;
796 }
797 getOwnBindingIdentifier(name) {
798 const binding = this.bindings[name];
799 return binding == null ? void 0 : binding.identifier;
800 }
801 hasOwnBinding(name) {
802 return !!this.getOwnBinding(name);
803 }
804 hasBinding(name, opts) {
805 if (!name) return false;
806 let scope = this;
807 do {
808 if (scope.hasOwnBinding(name)) {
809 return true;
810 }
811 } while (scope = scope.parent);
812 let noGlobals;
813 let noUids;
814 if (typeof opts === "object") {
815 noGlobals = opts.noGlobals;
816 noUids = opts.noUids;
817 } else if (typeof opts === "boolean") {
818 noGlobals = opts;
819 }
820 if (!noUids && this.hasUid(name)) return true;
821 if (!noGlobals && Scope.globals.includes(name)) return true;
822 if (!noGlobals && Scope.contextVariables.includes(name)) return true;
823 return false;
824 }
825 parentHasBinding(name, opts) {
826 var _this$parent;
827 return (_this$parent = this.parent) == null ? void 0 : _this$parent.hasBinding(name, opts);
828 }
829 moveBindingTo(name, scope) {
830 const info = this.getBinding(name);
831 if (info) {
832 info.scope.removeOwnBinding(name);
833 info.scope = scope;
834 scope.bindings[name] = info;
835 }
836 }
837 removeOwnBinding(name) {
838 delete this.bindings[name];
839 }
840 removeBinding(name) {
841 var _this$getBinding3;
842 (_this$getBinding3 = this.getBinding(name)) == null || _this$getBinding3.scope.removeOwnBinding(name);
843 let scope = this;
844 do {
845 if (scope.uids[name]) {
846 scope.uids[name] = false;
847 }
848 } while (scope = scope.parent);
849 }
850 hoistVariables(emit = id => this.push({
851 id
852 })) {
853 this.crawl();
854 const seen = new Set();
855 for (const name of Object.keys(this.bindings)) {
856 const binding = this.bindings[name];
857 if (!binding) continue;
858 const {
859 path
860 } = binding;
861 if (!path.isVariableDeclarator()) continue;
862 const {
863 parent,
864 parentPath
865 } = path;
866 if (parent.kind !== "var" || seen.has(parent)) continue;
867 seen.add(path.parent);
868 let firstId;
869 const init = [];
870 for (const decl of parent.declarations) {
871 var _firstId;
872 (_firstId = firstId) != null ? _firstId : firstId = decl.id;
873 if (decl.init) {
874 init.push(assignmentExpression("=", decl.id, decl.init));
875 }
876 const ids = Object.keys(getBindingIdentifiers(decl, false, true, true));
877 for (const name of ids) {
878 emit(identifier(name), decl.init != null);
879 }
880 }
881 if (parentPath.parentPath.isFor({
882 left: parent
883 })) {
884 parentPath.replaceWith(firstId);
885 } else if (init.length === 0) {
886 parentPath.remove();
887 } else {
888 const expr = init.length === 1 ? init[0] : sequenceExpression(init);
889 if (parentPath.parentPath.isForStatement({
890 init: parent
891 })) {
892 parentPath.replaceWith(expr);
893 } else {
894 parentPath.replaceWith(expressionStatement(expr));
895 }
896 }
897 }
898 }
899}
900exports.default = Scope;
901Scope.globals = Object.keys(_globals.builtin);
902Scope.contextVariables = ["arguments", "undefined", "Infinity", "NaN"];
903{
904 Scope.prototype._renameFromMap = function _renameFromMap(map, oldName, newName, value) {
905 if (map[oldName]) {
906 map[newName] = value;
907 map[oldName] = null;
908 }
909 };
910 Scope.prototype.traverse = function (node, opts, state) {
911 (0, _index.default)(node, opts, this, state, this.path);
912 };
913 Scope.prototype._generateUid = function _generateUid(name, i) {
914 let id = name;
915 if (i > 1) id += i;
916 return `_${id}`;
917 };
918 Scope.prototype.toArray = function toArray(node, i, arrayLikeIsIterable) {
919 if (isIdentifier(node)) {
920 const binding = this.getBinding(node.name);
921 if (binding != null && binding.constant && binding.path.isGenericType("Array")) {
922 return node;
923 }
924 }
925 if (isArrayExpression(node)) {
926 return node;
927 }
928 if (isIdentifier(node, {
929 name: "arguments"
930 })) {
931 return callExpression(memberExpression(memberExpression(memberExpression(identifier("Array"), identifier("prototype")), identifier("slice")), identifier("call")), [node]);
932 }
933 let helperName;
934 const args = [node];
935 if (i === true) {
936 helperName = "toConsumableArray";
937 } else if (typeof i === "number") {
938 args.push(numericLiteral(i));
939 helperName = "slicedToArray";
940 } else {
941 helperName = "toArray";
942 }
943 if (arrayLikeIsIterable) {
944 args.unshift(this.path.hub.addHelper(helperName));
945 helperName = "maybeArrayLike";
946 }
947 return callExpression(this.path.hub.addHelper(helperName), args);
948 };
949 Scope.prototype.getAllBindingsOfKind = function getAllBindingsOfKind(...kinds) {
950 const ids = Object.create(null);
951 for (const kind of kinds) {
952 let scope = this;
953 do {
954 for (const name of Object.keys(scope.bindings)) {
955 const binding = scope.bindings[name];
956 if (binding.kind === kind) ids[name] = binding;
957 }
958 scope = scope.parent;
959 } while (scope);
960 }
961 return ids;
962 };
963 Object.defineProperties(Scope.prototype, {
964 parentBlock: {
965 configurable: true,
966 enumerable: true,
967 get() {
968 return this.path.parent;
969 }
970 },
971 hub: {
972 configurable: true,
973 enumerable: true,
974 get() {
975 return this.path.hub;
976 }
977 }
978 });
979}
980
981//# sourceMappingURL=index.js.map
Note: See TracBrowser for help on using the repository browser.