eslint-scope.cjs 68 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238
  1. 'use strict';
  2. Object.defineProperty(exports, '__esModule', { value: true });
  3. var assert = require('assert');
  4. var estraverse = require('estraverse');
  5. var esrecurse = require('esrecurse');
  6. function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
  7. var assert__default = /*#__PURE__*/_interopDefaultLegacy(assert);
  8. var estraverse__default = /*#__PURE__*/_interopDefaultLegacy(estraverse);
  9. var esrecurse__default = /*#__PURE__*/_interopDefaultLegacy(esrecurse);
  10. /*
  11. Copyright (C) 2015 Yusuke Suzuki <utatane.tea@gmail.com>
  12. Redistribution and use in source and binary forms, with or without
  13. modification, are permitted provided that the following conditions are met:
  14. * Redistributions of source code must retain the above copyright
  15. notice, this list of conditions and the following disclaimer.
  16. * Redistributions in binary form must reproduce the above copyright
  17. notice, this list of conditions and the following disclaimer in the
  18. documentation and/or other materials provided with the distribution.
  19. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  20. AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  21. IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  22. ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  23. DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  24. (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  25. LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  26. ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  27. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  28. THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  29. */
  30. const READ = 0x1;
  31. const WRITE = 0x2;
  32. const RW = READ | WRITE;
  33. /**
  34. * A Reference represents a single occurrence of an identifier in code.
  35. * @constructor Reference
  36. */
  37. class Reference {
  38. constructor(ident, scope, flag, writeExpr, maybeImplicitGlobal, partial, init) {
  39. /**
  40. * Identifier syntax node.
  41. * @member {espreeIdentifier} Reference#identifier
  42. */
  43. this.identifier = ident;
  44. /**
  45. * Reference to the enclosing Scope.
  46. * @member {Scope} Reference#from
  47. */
  48. this.from = scope;
  49. /**
  50. * Whether the reference comes from a dynamic scope (such as 'eval',
  51. * 'with', etc.), and may be trapped by dynamic scopes.
  52. * @member {boolean} Reference#tainted
  53. */
  54. this.tainted = false;
  55. /**
  56. * The variable this reference is resolved with.
  57. * @member {Variable} Reference#resolved
  58. */
  59. this.resolved = null;
  60. /**
  61. * The read-write mode of the reference. (Value is one of {@link
  62. * Reference.READ}, {@link Reference.RW}, {@link Reference.WRITE}).
  63. * @member {number} Reference#flag
  64. * @private
  65. */
  66. this.flag = flag;
  67. if (this.isWrite()) {
  68. /**
  69. * If reference is writeable, this is the tree being written to it.
  70. * @member {espreeNode} Reference#writeExpr
  71. */
  72. this.writeExpr = writeExpr;
  73. /**
  74. * Whether the Reference might refer to a partial value of writeExpr.
  75. * @member {boolean} Reference#partial
  76. */
  77. this.partial = partial;
  78. /**
  79. * Whether the Reference is to write of initialization.
  80. * @member {boolean} Reference#init
  81. */
  82. this.init = init;
  83. }
  84. this.__maybeImplicitGlobal = maybeImplicitGlobal;
  85. }
  86. /**
  87. * Whether the reference is static.
  88. * @function Reference#isStatic
  89. * @returns {boolean} static
  90. */
  91. isStatic() {
  92. return !this.tainted && this.resolved && this.resolved.scope.isStatic();
  93. }
  94. /**
  95. * Whether the reference is writeable.
  96. * @function Reference#isWrite
  97. * @returns {boolean} write
  98. */
  99. isWrite() {
  100. return !!(this.flag & Reference.WRITE);
  101. }
  102. /**
  103. * Whether the reference is readable.
  104. * @function Reference#isRead
  105. * @returns {boolean} read
  106. */
  107. isRead() {
  108. return !!(this.flag & Reference.READ);
  109. }
  110. /**
  111. * Whether the reference is read-only.
  112. * @function Reference#isReadOnly
  113. * @returns {boolean} read only
  114. */
  115. isReadOnly() {
  116. return this.flag === Reference.READ;
  117. }
  118. /**
  119. * Whether the reference is write-only.
  120. * @function Reference#isWriteOnly
  121. * @returns {boolean} write only
  122. */
  123. isWriteOnly() {
  124. return this.flag === Reference.WRITE;
  125. }
  126. /**
  127. * Whether the reference is read-write.
  128. * @function Reference#isReadWrite
  129. * @returns {boolean} read write
  130. */
  131. isReadWrite() {
  132. return this.flag === Reference.RW;
  133. }
  134. }
  135. /**
  136. * @constant Reference.READ
  137. * @private
  138. */
  139. Reference.READ = READ;
  140. /**
  141. * @constant Reference.WRITE
  142. * @private
  143. */
  144. Reference.WRITE = WRITE;
  145. /**
  146. * @constant Reference.RW
  147. * @private
  148. */
  149. Reference.RW = RW;
  150. /* vim: set sw=4 ts=4 et tw=80 : */
  151. /*
  152. Copyright (C) 2015 Yusuke Suzuki <utatane.tea@gmail.com>
  153. Redistribution and use in source and binary forms, with or without
  154. modification, are permitted provided that the following conditions are met:
  155. * Redistributions of source code must retain the above copyright
  156. notice, this list of conditions and the following disclaimer.
  157. * Redistributions in binary form must reproduce the above copyright
  158. notice, this list of conditions and the following disclaimer in the
  159. documentation and/or other materials provided with the distribution.
  160. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  161. AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  162. IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  163. ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  164. DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  165. (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  166. LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  167. ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  168. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  169. THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  170. */
  171. /**
  172. * A Variable represents a locally scoped identifier. These include arguments to
  173. * functions.
  174. * @constructor Variable
  175. */
  176. class Variable {
  177. constructor(name, scope) {
  178. /**
  179. * The variable name, as given in the source code.
  180. * @member {string} Variable#name
  181. */
  182. this.name = name;
  183. /**
  184. * List of defining occurrences of this variable (like in 'var ...'
  185. * statements or as parameter), as AST nodes.
  186. * @member {espree.Identifier[]} Variable#identifiers
  187. */
  188. this.identifiers = [];
  189. /**
  190. * List of {@link Reference|references} of this variable (excluding parameter entries)
  191. * in its defining scope and all nested scopes. For defining
  192. * occurrences only see {@link Variable#defs}.
  193. * @member {Reference[]} Variable#references
  194. */
  195. this.references = [];
  196. /**
  197. * List of defining occurrences of this variable (like in 'var ...'
  198. * statements or as parameter), as custom objects.
  199. * @member {Definition[]} Variable#defs
  200. */
  201. this.defs = [];
  202. this.tainted = false;
  203. /**
  204. * Whether this is a stack variable.
  205. * @member {boolean} Variable#stack
  206. */
  207. this.stack = true;
  208. /**
  209. * Reference to the enclosing Scope.
  210. * @member {Scope} Variable#scope
  211. */
  212. this.scope = scope;
  213. }
  214. }
  215. Variable.CatchClause = "CatchClause";
  216. Variable.Parameter = "Parameter";
  217. Variable.FunctionName = "FunctionName";
  218. Variable.ClassName = "ClassName";
  219. Variable.Variable = "Variable";
  220. Variable.ImportBinding = "ImportBinding";
  221. Variable.ImplicitGlobalVariable = "ImplicitGlobalVariable";
  222. /* vim: set sw=4 ts=4 et tw=80 : */
  223. /*
  224. Copyright (C) 2015 Yusuke Suzuki <utatane.tea@gmail.com>
  225. Redistribution and use in source and binary forms, with or without
  226. modification, are permitted provided that the following conditions are met:
  227. * Redistributions of source code must retain the above copyright
  228. notice, this list of conditions and the following disclaimer.
  229. * Redistributions in binary form must reproduce the above copyright
  230. notice, this list of conditions and the following disclaimer in the
  231. documentation and/or other materials provided with the distribution.
  232. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  233. AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  234. IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  235. ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  236. DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  237. (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  238. LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  239. ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  240. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  241. THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  242. */
  243. /**
  244. * @constructor Definition
  245. */
  246. class Definition {
  247. constructor(type, name, node, parent, index, kind) {
  248. /**
  249. * @member {string} Definition#type - type of the occurrence (e.g. "Parameter", "Variable", ...).
  250. */
  251. this.type = type;
  252. /**
  253. * @member {espree.Identifier} Definition#name - the identifier AST node of the occurrence.
  254. */
  255. this.name = name;
  256. /**
  257. * @member {espree.Node} Definition#node - the enclosing node of the identifier.
  258. */
  259. this.node = node;
  260. /**
  261. * @member {espree.Node?} Definition#parent - the enclosing statement node of the identifier.
  262. */
  263. this.parent = parent;
  264. /**
  265. * @member {number?} Definition#index - the index in the declaration statement.
  266. */
  267. this.index = index;
  268. /**
  269. * @member {string?} Definition#kind - the kind of the declaration statement.
  270. */
  271. this.kind = kind;
  272. }
  273. }
  274. /**
  275. * @constructor ParameterDefinition
  276. */
  277. class ParameterDefinition extends Definition {
  278. constructor(name, node, index, rest) {
  279. super(Variable.Parameter, name, node, null, index, null);
  280. /**
  281. * Whether the parameter definition is a part of a rest parameter.
  282. * @member {boolean} ParameterDefinition#rest
  283. */
  284. this.rest = rest;
  285. }
  286. }
  287. /* vim: set sw=4 ts=4 et tw=80 : */
  288. /*
  289. Copyright (C) 2015 Yusuke Suzuki <utatane.tea@gmail.com>
  290. Redistribution and use in source and binary forms, with or without
  291. modification, are permitted provided that the following conditions are met:
  292. * Redistributions of source code must retain the above copyright
  293. notice, this list of conditions and the following disclaimer.
  294. * Redistributions in binary form must reproduce the above copyright
  295. notice, this list of conditions and the following disclaimer in the
  296. documentation and/or other materials provided with the distribution.
  297. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  298. AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  299. IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  300. ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  301. DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  302. (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  303. LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  304. ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  305. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  306. THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  307. */
  308. const { Syntax: Syntax$2 } = estraverse__default["default"];
  309. /**
  310. * Test if scope is struct
  311. * @param {Scope} scope scope
  312. * @param {Block} block block
  313. * @param {boolean} isMethodDefinition is method definition
  314. * @param {boolean} useDirective use directive
  315. * @returns {boolean} is strict scope
  316. */
  317. function isStrictScope(scope, block, isMethodDefinition, useDirective) {
  318. let body;
  319. // When upper scope is exists and strict, inner scope is also strict.
  320. if (scope.upper && scope.upper.isStrict) {
  321. return true;
  322. }
  323. if (isMethodDefinition) {
  324. return true;
  325. }
  326. if (scope.type === "class" || scope.type === "module") {
  327. return true;
  328. }
  329. if (scope.type === "block" || scope.type === "switch") {
  330. return false;
  331. }
  332. if (scope.type === "function") {
  333. if (block.type === Syntax$2.ArrowFunctionExpression && block.body.type !== Syntax$2.BlockStatement) {
  334. return false;
  335. }
  336. if (block.type === Syntax$2.Program) {
  337. body = block;
  338. } else {
  339. body = block.body;
  340. }
  341. if (!body) {
  342. return false;
  343. }
  344. } else if (scope.type === "global") {
  345. body = block;
  346. } else {
  347. return false;
  348. }
  349. // Search 'use strict' directive.
  350. if (useDirective) {
  351. for (let i = 0, iz = body.body.length; i < iz; ++i) {
  352. const stmt = body.body[i];
  353. if (stmt.type !== Syntax$2.DirectiveStatement) {
  354. break;
  355. }
  356. if (stmt.raw === "\"use strict\"" || stmt.raw === "'use strict'") {
  357. return true;
  358. }
  359. }
  360. } else {
  361. for (let i = 0, iz = body.body.length; i < iz; ++i) {
  362. const stmt = body.body[i];
  363. if (stmt.type !== Syntax$2.ExpressionStatement) {
  364. break;
  365. }
  366. const expr = stmt.expression;
  367. if (expr.type !== Syntax$2.Literal || typeof expr.value !== "string") {
  368. break;
  369. }
  370. if (expr.raw !== null && expr.raw !== undefined) {
  371. if (expr.raw === "\"use strict\"" || expr.raw === "'use strict'") {
  372. return true;
  373. }
  374. } else {
  375. if (expr.value === "use strict") {
  376. return true;
  377. }
  378. }
  379. }
  380. }
  381. return false;
  382. }
  383. /**
  384. * Register scope
  385. * @param {ScopeManager} scopeManager scope manager
  386. * @param {Scope} scope scope
  387. * @returns {void}
  388. */
  389. function registerScope(scopeManager, scope) {
  390. scopeManager.scopes.push(scope);
  391. const scopes = scopeManager.__nodeToScope.get(scope.block);
  392. if (scopes) {
  393. scopes.push(scope);
  394. } else {
  395. scopeManager.__nodeToScope.set(scope.block, [scope]);
  396. }
  397. }
  398. /**
  399. * Should be statically
  400. * @param {Object} def def
  401. * @returns {boolean} should be statically
  402. */
  403. function shouldBeStatically(def) {
  404. return (
  405. (def.type === Variable.ClassName) ||
  406. (def.type === Variable.Variable && def.parent.kind !== "var")
  407. );
  408. }
  409. /**
  410. * @constructor Scope
  411. */
  412. class Scope {
  413. constructor(scopeManager, type, upperScope, block, isMethodDefinition) {
  414. /**
  415. * One of "global", "module", "function", "function-expression-name", "block", "switch", "catch", "with", "for",
  416. * "class", "class-field-initializer", "class-static-block".
  417. * @member {string} Scope#type
  418. */
  419. this.type = type;
  420. /**
  421. * The scoped {@link Variable}s of this scope, as <code>{ Variable.name
  422. * : Variable }</code>.
  423. * @member {Map} Scope#set
  424. */
  425. this.set = new Map();
  426. /**
  427. * The tainted variables of this scope, as <code>{ Variable.name :
  428. * boolean }</code>.
  429. * @member {Map} Scope#taints */
  430. this.taints = new Map();
  431. /**
  432. * Generally, through the lexical scoping of JS you can always know
  433. * which variable an identifier in the source code refers to. There are
  434. * a few exceptions to this rule. With 'global' and 'with' scopes you
  435. * can only decide at runtime which variable a reference refers to.
  436. * Moreover, if 'eval()' is used in a scope, it might introduce new
  437. * bindings in this or its parent scopes.
  438. * All those scopes are considered 'dynamic'.
  439. * @member {boolean} Scope#dynamic
  440. */
  441. this.dynamic = this.type === "global" || this.type === "with";
  442. /**
  443. * A reference to the scope-defining syntax node.
  444. * @member {espree.Node} Scope#block
  445. */
  446. this.block = block;
  447. /**
  448. * The {@link Reference|references} that are not resolved with this scope.
  449. * @member {Reference[]} Scope#through
  450. */
  451. this.through = [];
  452. /**
  453. * The scoped {@link Variable}s of this scope. In the case of a
  454. * 'function' scope this includes the automatic argument <em>arguments</em> as
  455. * its first element, as well as all further formal arguments.
  456. * @member {Variable[]} Scope#variables
  457. */
  458. this.variables = [];
  459. /**
  460. * Any variable {@link Reference|reference} found in this scope. This
  461. * includes occurrences of local variables as well as variables from
  462. * parent scopes (including the global scope). For local variables
  463. * this also includes defining occurrences (like in a 'var' statement).
  464. * In a 'function' scope this does not include the occurrences of the
  465. * formal parameter in the parameter list.
  466. * @member {Reference[]} Scope#references
  467. */
  468. this.references = [];
  469. /**
  470. * For 'global' and 'function' scopes, this is a self-reference. For
  471. * other scope types this is the <em>variableScope</em> value of the
  472. * parent scope.
  473. * @member {Scope} Scope#variableScope
  474. */
  475. this.variableScope =
  476. this.type === "global" ||
  477. this.type === "module" ||
  478. this.type === "function" ||
  479. this.type === "class-field-initializer" ||
  480. this.type === "class-static-block"
  481. ? this
  482. : upperScope.variableScope;
  483. /**
  484. * Whether this scope is created by a FunctionExpression.
  485. * @member {boolean} Scope#functionExpressionScope
  486. */
  487. this.functionExpressionScope = false;
  488. /**
  489. * Whether this is a scope that contains an 'eval()' invocation.
  490. * @member {boolean} Scope#directCallToEvalScope
  491. */
  492. this.directCallToEvalScope = false;
  493. /**
  494. * @member {boolean} Scope#thisFound
  495. */
  496. this.thisFound = false;
  497. this.__left = [];
  498. /**
  499. * Reference to the parent {@link Scope|scope}.
  500. * @member {Scope} Scope#upper
  501. */
  502. this.upper = upperScope;
  503. /**
  504. * Whether 'use strict' is in effect in this scope.
  505. * @member {boolean} Scope#isStrict
  506. */
  507. this.isStrict = isStrictScope(this, block, isMethodDefinition, scopeManager.__useDirective());
  508. /**
  509. * List of nested {@link Scope}s.
  510. * @member {Scope[]} Scope#childScopes
  511. */
  512. this.childScopes = [];
  513. if (this.upper) {
  514. this.upper.childScopes.push(this);
  515. }
  516. this.__declaredVariables = scopeManager.__declaredVariables;
  517. registerScope(scopeManager, this);
  518. }
  519. __shouldStaticallyClose(scopeManager) {
  520. return (!this.dynamic || scopeManager.__isOptimistic());
  521. }
  522. __shouldStaticallyCloseForGlobal(ref) {
  523. // On global scope, let/const/class declarations should be resolved statically.
  524. const name = ref.identifier.name;
  525. if (!this.set.has(name)) {
  526. return false;
  527. }
  528. const variable = this.set.get(name);
  529. const defs = variable.defs;
  530. return defs.length > 0 && defs.every(shouldBeStatically);
  531. }
  532. __staticCloseRef(ref) {
  533. if (!this.__resolve(ref)) {
  534. this.__delegateToUpperScope(ref);
  535. }
  536. }
  537. __dynamicCloseRef(ref) {
  538. // notify all names are through to global
  539. let current = this;
  540. do {
  541. current.through.push(ref);
  542. current = current.upper;
  543. } while (current);
  544. }
  545. __globalCloseRef(ref) {
  546. // let/const/class declarations should be resolved statically.
  547. // others should be resolved dynamically.
  548. if (this.__shouldStaticallyCloseForGlobal(ref)) {
  549. this.__staticCloseRef(ref);
  550. } else {
  551. this.__dynamicCloseRef(ref);
  552. }
  553. }
  554. __close(scopeManager) {
  555. let closeRef;
  556. if (this.__shouldStaticallyClose(scopeManager)) {
  557. closeRef = this.__staticCloseRef;
  558. } else if (this.type !== "global") {
  559. closeRef = this.__dynamicCloseRef;
  560. } else {
  561. closeRef = this.__globalCloseRef;
  562. }
  563. // Try Resolving all references in this scope.
  564. for (let i = 0, iz = this.__left.length; i < iz; ++i) {
  565. const ref = this.__left[i];
  566. closeRef.call(this, ref);
  567. }
  568. this.__left = null;
  569. return this.upper;
  570. }
  571. // To override by function scopes.
  572. // References in default parameters isn't resolved to variables which are in their function body.
  573. __isValidResolution(ref, variable) { // eslint-disable-line class-methods-use-this, no-unused-vars
  574. return true;
  575. }
  576. __resolve(ref) {
  577. const name = ref.identifier.name;
  578. if (!this.set.has(name)) {
  579. return false;
  580. }
  581. const variable = this.set.get(name);
  582. if (!this.__isValidResolution(ref, variable)) {
  583. return false;
  584. }
  585. variable.references.push(ref);
  586. variable.stack = variable.stack && ref.from.variableScope === this.variableScope;
  587. if (ref.tainted) {
  588. variable.tainted = true;
  589. this.taints.set(variable.name, true);
  590. }
  591. ref.resolved = variable;
  592. return true;
  593. }
  594. __delegateToUpperScope(ref) {
  595. if (this.upper) {
  596. this.upper.__left.push(ref);
  597. }
  598. this.through.push(ref);
  599. }
  600. __addDeclaredVariablesOfNode(variable, node) {
  601. if (node === null || node === undefined) {
  602. return;
  603. }
  604. let variables = this.__declaredVariables.get(node);
  605. if (variables === null || variables === undefined) {
  606. variables = [];
  607. this.__declaredVariables.set(node, variables);
  608. }
  609. if (variables.indexOf(variable) === -1) {
  610. variables.push(variable);
  611. }
  612. }
  613. __defineGeneric(name, set, variables, node, def) {
  614. let variable;
  615. variable = set.get(name);
  616. if (!variable) {
  617. variable = new Variable(name, this);
  618. set.set(name, variable);
  619. variables.push(variable);
  620. }
  621. if (def) {
  622. variable.defs.push(def);
  623. this.__addDeclaredVariablesOfNode(variable, def.node);
  624. this.__addDeclaredVariablesOfNode(variable, def.parent);
  625. }
  626. if (node) {
  627. variable.identifiers.push(node);
  628. }
  629. }
  630. __define(node, def) {
  631. if (node && node.type === Syntax$2.Identifier) {
  632. this.__defineGeneric(
  633. node.name,
  634. this.set,
  635. this.variables,
  636. node,
  637. def
  638. );
  639. }
  640. }
  641. __referencing(node, assign, writeExpr, maybeImplicitGlobal, partial, init) {
  642. // because Array element may be null
  643. if (!node || node.type !== Syntax$2.Identifier) {
  644. return;
  645. }
  646. // Specially handle like `this`.
  647. if (node.name === "super") {
  648. return;
  649. }
  650. const ref = new Reference(node, this, assign || Reference.READ, writeExpr, maybeImplicitGlobal, !!partial, !!init);
  651. this.references.push(ref);
  652. this.__left.push(ref);
  653. }
  654. __detectEval() {
  655. let current = this;
  656. this.directCallToEvalScope = true;
  657. do {
  658. current.dynamic = true;
  659. current = current.upper;
  660. } while (current);
  661. }
  662. __detectThis() {
  663. this.thisFound = true;
  664. }
  665. __isClosed() {
  666. return this.__left === null;
  667. }
  668. /**
  669. * returns resolved {Reference}
  670. * @function Scope#resolve
  671. * @param {Espree.Identifier} ident identifier to be resolved.
  672. * @returns {Reference} reference
  673. */
  674. resolve(ident) {
  675. let ref, i, iz;
  676. assert__default["default"](this.__isClosed(), "Scope should be closed.");
  677. assert__default["default"](ident.type === Syntax$2.Identifier, "Target should be identifier.");
  678. for (i = 0, iz = this.references.length; i < iz; ++i) {
  679. ref = this.references[i];
  680. if (ref.identifier === ident) {
  681. return ref;
  682. }
  683. }
  684. return null;
  685. }
  686. /**
  687. * returns this scope is static
  688. * @function Scope#isStatic
  689. * @returns {boolean} static
  690. */
  691. isStatic() {
  692. return !this.dynamic;
  693. }
  694. /**
  695. * returns this scope has materialized arguments
  696. * @function Scope#isArgumentsMaterialized
  697. * @returns {boolean} arguemnts materialized
  698. */
  699. isArgumentsMaterialized() { // eslint-disable-line class-methods-use-this
  700. return true;
  701. }
  702. /**
  703. * returns this scope has materialized `this` reference
  704. * @function Scope#isThisMaterialized
  705. * @returns {boolean} this materialized
  706. */
  707. isThisMaterialized() { // eslint-disable-line class-methods-use-this
  708. return true;
  709. }
  710. isUsedName(name) {
  711. if (this.set.has(name)) {
  712. return true;
  713. }
  714. for (let i = 0, iz = this.through.length; i < iz; ++i) {
  715. if (this.through[i].identifier.name === name) {
  716. return true;
  717. }
  718. }
  719. return false;
  720. }
  721. }
  722. class GlobalScope extends Scope {
  723. constructor(scopeManager, block) {
  724. super(scopeManager, "global", null, block, false);
  725. this.implicit = {
  726. set: new Map(),
  727. variables: [],
  728. /**
  729. * List of {@link Reference}s that are left to be resolved (i.e. which
  730. * need to be linked to the variable they refer to).
  731. * @member {Reference[]} Scope#implicit#left
  732. */
  733. left: []
  734. };
  735. }
  736. __close(scopeManager) {
  737. const implicit = [];
  738. for (let i = 0, iz = this.__left.length; i < iz; ++i) {
  739. const ref = this.__left[i];
  740. if (ref.__maybeImplicitGlobal && !this.set.has(ref.identifier.name)) {
  741. implicit.push(ref.__maybeImplicitGlobal);
  742. }
  743. }
  744. // create an implicit global variable from assignment expression
  745. for (let i = 0, iz = implicit.length; i < iz; ++i) {
  746. const info = implicit[i];
  747. this.__defineImplicit(info.pattern,
  748. new Definition(
  749. Variable.ImplicitGlobalVariable,
  750. info.pattern,
  751. info.node,
  752. null,
  753. null,
  754. null
  755. ));
  756. }
  757. this.implicit.left = this.__left;
  758. return super.__close(scopeManager);
  759. }
  760. __defineImplicit(node, def) {
  761. if (node && node.type === Syntax$2.Identifier) {
  762. this.__defineGeneric(
  763. node.name,
  764. this.implicit.set,
  765. this.implicit.variables,
  766. node,
  767. def
  768. );
  769. }
  770. }
  771. }
  772. class ModuleScope extends Scope {
  773. constructor(scopeManager, upperScope, block) {
  774. super(scopeManager, "module", upperScope, block, false);
  775. }
  776. }
  777. class FunctionExpressionNameScope extends Scope {
  778. constructor(scopeManager, upperScope, block) {
  779. super(scopeManager, "function-expression-name", upperScope, block, false);
  780. this.__define(block.id,
  781. new Definition(
  782. Variable.FunctionName,
  783. block.id,
  784. block,
  785. null,
  786. null,
  787. null
  788. ));
  789. this.functionExpressionScope = true;
  790. }
  791. }
  792. class CatchScope extends Scope {
  793. constructor(scopeManager, upperScope, block) {
  794. super(scopeManager, "catch", upperScope, block, false);
  795. }
  796. }
  797. class WithScope extends Scope {
  798. constructor(scopeManager, upperScope, block) {
  799. super(scopeManager, "with", upperScope, block, false);
  800. }
  801. __close(scopeManager) {
  802. if (this.__shouldStaticallyClose(scopeManager)) {
  803. return super.__close(scopeManager);
  804. }
  805. for (let i = 0, iz = this.__left.length; i < iz; ++i) {
  806. const ref = this.__left[i];
  807. ref.tainted = true;
  808. this.__delegateToUpperScope(ref);
  809. }
  810. this.__left = null;
  811. return this.upper;
  812. }
  813. }
  814. class BlockScope extends Scope {
  815. constructor(scopeManager, upperScope, block) {
  816. super(scopeManager, "block", upperScope, block, false);
  817. }
  818. }
  819. class SwitchScope extends Scope {
  820. constructor(scopeManager, upperScope, block) {
  821. super(scopeManager, "switch", upperScope, block, false);
  822. }
  823. }
  824. class FunctionScope extends Scope {
  825. constructor(scopeManager, upperScope, block, isMethodDefinition) {
  826. super(scopeManager, "function", upperScope, block, isMethodDefinition);
  827. // section 9.2.13, FunctionDeclarationInstantiation.
  828. // NOTE Arrow functions never have an arguments objects.
  829. if (this.block.type !== Syntax$2.ArrowFunctionExpression) {
  830. this.__defineArguments();
  831. }
  832. }
  833. isArgumentsMaterialized() {
  834. // TODO(Constellation)
  835. // We can more aggressive on this condition like this.
  836. //
  837. // function t() {
  838. // // arguments of t is always hidden.
  839. // function arguments() {
  840. // }
  841. // }
  842. if (this.block.type === Syntax$2.ArrowFunctionExpression) {
  843. return false;
  844. }
  845. if (!this.isStatic()) {
  846. return true;
  847. }
  848. const variable = this.set.get("arguments");
  849. assert__default["default"](variable, "Always have arguments variable.");
  850. return variable.tainted || variable.references.length !== 0;
  851. }
  852. isThisMaterialized() {
  853. if (!this.isStatic()) {
  854. return true;
  855. }
  856. return this.thisFound;
  857. }
  858. __defineArguments() {
  859. this.__defineGeneric(
  860. "arguments",
  861. this.set,
  862. this.variables,
  863. null,
  864. null
  865. );
  866. this.taints.set("arguments", true);
  867. }
  868. // References in default parameters isn't resolved to variables which are in their function body.
  869. // const x = 1
  870. // function f(a = x) { // This `x` is resolved to the `x` in the outer scope.
  871. // const x = 2
  872. // console.log(a)
  873. // }
  874. __isValidResolution(ref, variable) {
  875. // If `options.nodejsScope` is true, `this.block` becomes a Program node.
  876. if (this.block.type === "Program") {
  877. return true;
  878. }
  879. const bodyStart = this.block.body.range[0];
  880. // It's invalid resolution in the following case:
  881. return !(
  882. variable.scope === this &&
  883. ref.identifier.range[0] < bodyStart && // the reference is in the parameter part.
  884. variable.defs.every(d => d.name.range[0] >= bodyStart) // the variable is in the body.
  885. );
  886. }
  887. }
  888. class ForScope extends Scope {
  889. constructor(scopeManager, upperScope, block) {
  890. super(scopeManager, "for", upperScope, block, false);
  891. }
  892. }
  893. class ClassScope extends Scope {
  894. constructor(scopeManager, upperScope, block) {
  895. super(scopeManager, "class", upperScope, block, false);
  896. }
  897. }
  898. class ClassFieldInitializerScope extends Scope {
  899. constructor(scopeManager, upperScope, block) {
  900. super(scopeManager, "class-field-initializer", upperScope, block, true);
  901. }
  902. }
  903. class ClassStaticBlockScope extends Scope {
  904. constructor(scopeManager, upperScope, block) {
  905. super(scopeManager, "class-static-block", upperScope, block, true);
  906. }
  907. }
  908. /* vim: set sw=4 ts=4 et tw=80 : */
  909. /*
  910. Copyright (C) 2015 Yusuke Suzuki <utatane.tea@gmail.com>
  911. Redistribution and use in source and binary forms, with or without
  912. modification, are permitted provided that the following conditions are met:
  913. * Redistributions of source code must retain the above copyright
  914. notice, this list of conditions and the following disclaimer.
  915. * Redistributions in binary form must reproduce the above copyright
  916. notice, this list of conditions and the following disclaimer in the
  917. documentation and/or other materials provided with the distribution.
  918. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  919. AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  920. IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  921. ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  922. DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  923. (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  924. LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  925. ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  926. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  927. THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  928. */
  929. /**
  930. * @constructor ScopeManager
  931. */
  932. class ScopeManager {
  933. constructor(options) {
  934. this.scopes = [];
  935. this.globalScope = null;
  936. this.__nodeToScope = new WeakMap();
  937. this.__currentScope = null;
  938. this.__options = options;
  939. this.__declaredVariables = new WeakMap();
  940. }
  941. __useDirective() {
  942. return this.__options.directive;
  943. }
  944. __isOptimistic() {
  945. return this.__options.optimistic;
  946. }
  947. __ignoreEval() {
  948. return this.__options.ignoreEval;
  949. }
  950. __isNodejsScope() {
  951. return this.__options.nodejsScope || this.__options.sourceType === "commonjs";
  952. }
  953. isModule() {
  954. return this.__options.sourceType === "module";
  955. }
  956. isImpliedStrict() {
  957. return this.__options.impliedStrict;
  958. }
  959. isStrictModeSupported() {
  960. return this.__options.ecmaVersion >= 5;
  961. }
  962. // Returns appropriate scope for this node.
  963. __get(node) {
  964. return this.__nodeToScope.get(node);
  965. }
  966. /**
  967. * Get variables that are declared by the node.
  968. *
  969. * "are declared by the node" means the node is same as `Variable.defs[].node` or `Variable.defs[].parent`.
  970. * If the node declares nothing, this method returns an empty array.
  971. * CAUTION: This API is experimental. See https://github.com/estools/escope/pull/69 for more details.
  972. * @param {Espree.Node} node a node to get.
  973. * @returns {Variable[]} variables that declared by the node.
  974. */
  975. getDeclaredVariables(node) {
  976. return this.__declaredVariables.get(node) || [];
  977. }
  978. /**
  979. * acquire scope from node.
  980. * @function ScopeManager#acquire
  981. * @param {Espree.Node} node node for the acquired scope.
  982. * @param {?boolean} [inner=false] look up the most inner scope, default value is false.
  983. * @returns {Scope?} Scope from node
  984. */
  985. acquire(node, inner) {
  986. /**
  987. * predicate
  988. * @param {Scope} testScope scope to test
  989. * @returns {boolean} predicate
  990. */
  991. function predicate(testScope) {
  992. if (testScope.type === "function" && testScope.functionExpressionScope) {
  993. return false;
  994. }
  995. return true;
  996. }
  997. const scopes = this.__get(node);
  998. if (!scopes || scopes.length === 0) {
  999. return null;
  1000. }
  1001. // Heuristic selection from all scopes.
  1002. // If you would like to get all scopes, please use ScopeManager#acquireAll.
  1003. if (scopes.length === 1) {
  1004. return scopes[0];
  1005. }
  1006. if (inner) {
  1007. for (let i = scopes.length - 1; i >= 0; --i) {
  1008. const scope = scopes[i];
  1009. if (predicate(scope)) {
  1010. return scope;
  1011. }
  1012. }
  1013. } else {
  1014. for (let i = 0, iz = scopes.length; i < iz; ++i) {
  1015. const scope = scopes[i];
  1016. if (predicate(scope)) {
  1017. return scope;
  1018. }
  1019. }
  1020. }
  1021. return null;
  1022. }
  1023. /**
  1024. * acquire all scopes from node.
  1025. * @function ScopeManager#acquireAll
  1026. * @param {Espree.Node} node node for the acquired scope.
  1027. * @returns {Scopes?} Scope array
  1028. */
  1029. acquireAll(node) {
  1030. return this.__get(node);
  1031. }
  1032. /**
  1033. * release the node.
  1034. * @function ScopeManager#release
  1035. * @param {Espree.Node} node releasing node.
  1036. * @param {?boolean} [inner=false] look up the most inner scope, default value is false.
  1037. * @returns {Scope?} upper scope for the node.
  1038. */
  1039. release(node, inner) {
  1040. const scopes = this.__get(node);
  1041. if (scopes && scopes.length) {
  1042. const scope = scopes[0].upper;
  1043. if (!scope) {
  1044. return null;
  1045. }
  1046. return this.acquire(scope.block, inner);
  1047. }
  1048. return null;
  1049. }
  1050. attach() { } // eslint-disable-line class-methods-use-this
  1051. detach() { } // eslint-disable-line class-methods-use-this
  1052. __nestScope(scope) {
  1053. if (scope instanceof GlobalScope) {
  1054. assert__default["default"](this.__currentScope === null);
  1055. this.globalScope = scope;
  1056. }
  1057. this.__currentScope = scope;
  1058. return scope;
  1059. }
  1060. __nestGlobalScope(node) {
  1061. return this.__nestScope(new GlobalScope(this, node));
  1062. }
  1063. __nestBlockScope(node) {
  1064. return this.__nestScope(new BlockScope(this, this.__currentScope, node));
  1065. }
  1066. __nestFunctionScope(node, isMethodDefinition) {
  1067. return this.__nestScope(new FunctionScope(this, this.__currentScope, node, isMethodDefinition));
  1068. }
  1069. __nestForScope(node) {
  1070. return this.__nestScope(new ForScope(this, this.__currentScope, node));
  1071. }
  1072. __nestCatchScope(node) {
  1073. return this.__nestScope(new CatchScope(this, this.__currentScope, node));
  1074. }
  1075. __nestWithScope(node) {
  1076. return this.__nestScope(new WithScope(this, this.__currentScope, node));
  1077. }
  1078. __nestClassScope(node) {
  1079. return this.__nestScope(new ClassScope(this, this.__currentScope, node));
  1080. }
  1081. __nestClassFieldInitializerScope(node) {
  1082. return this.__nestScope(new ClassFieldInitializerScope(this, this.__currentScope, node));
  1083. }
  1084. __nestClassStaticBlockScope(node) {
  1085. return this.__nestScope(new ClassStaticBlockScope(this, this.__currentScope, node));
  1086. }
  1087. __nestSwitchScope(node) {
  1088. return this.__nestScope(new SwitchScope(this, this.__currentScope, node));
  1089. }
  1090. __nestModuleScope(node) {
  1091. return this.__nestScope(new ModuleScope(this, this.__currentScope, node));
  1092. }
  1093. __nestFunctionExpressionNameScope(node) {
  1094. return this.__nestScope(new FunctionExpressionNameScope(this, this.__currentScope, node));
  1095. }
  1096. __isES6() {
  1097. return this.__options.ecmaVersion >= 6;
  1098. }
  1099. }
  1100. /* vim: set sw=4 ts=4 et tw=80 : */
  1101. /*
  1102. Copyright (C) 2015 Yusuke Suzuki <utatane.tea@gmail.com>
  1103. Redistribution and use in source and binary forms, with or without
  1104. modification, are permitted provided that the following conditions are met:
  1105. * Redistributions of source code must retain the above copyright
  1106. notice, this list of conditions and the following disclaimer.
  1107. * Redistributions in binary form must reproduce the above copyright
  1108. notice, this list of conditions and the following disclaimer in the
  1109. documentation and/or other materials provided with the distribution.
  1110. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  1111. AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  1112. IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  1113. ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  1114. DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  1115. (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  1116. LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  1117. ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  1118. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  1119. THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  1120. */
  1121. const { Syntax: Syntax$1 } = estraverse__default["default"];
  1122. /**
  1123. * Get last array element
  1124. * @param {Array} xs array
  1125. * @returns {any} Last elment
  1126. */
  1127. function getLast(xs) {
  1128. return xs[xs.length - 1] || null;
  1129. }
  1130. class PatternVisitor extends esrecurse__default["default"].Visitor {
  1131. static isPattern(node) {
  1132. const nodeType = node.type;
  1133. return (
  1134. nodeType === Syntax$1.Identifier ||
  1135. nodeType === Syntax$1.ObjectPattern ||
  1136. nodeType === Syntax$1.ArrayPattern ||
  1137. nodeType === Syntax$1.SpreadElement ||
  1138. nodeType === Syntax$1.RestElement ||
  1139. nodeType === Syntax$1.AssignmentPattern
  1140. );
  1141. }
  1142. constructor(options, rootPattern, callback) {
  1143. super(null, options);
  1144. this.rootPattern = rootPattern;
  1145. this.callback = callback;
  1146. this.assignments = [];
  1147. this.rightHandNodes = [];
  1148. this.restElements = [];
  1149. }
  1150. Identifier(pattern) {
  1151. const lastRestElement = getLast(this.restElements);
  1152. this.callback(pattern, {
  1153. topLevel: pattern === this.rootPattern,
  1154. rest: lastRestElement !== null && lastRestElement !== undefined && lastRestElement.argument === pattern,
  1155. assignments: this.assignments
  1156. });
  1157. }
  1158. Property(property) {
  1159. // Computed property's key is a right hand node.
  1160. if (property.computed) {
  1161. this.rightHandNodes.push(property.key);
  1162. }
  1163. // If it's shorthand, its key is same as its value.
  1164. // If it's shorthand and has its default value, its key is same as its value.left (the value is AssignmentPattern).
  1165. // If it's not shorthand, the name of new variable is its value's.
  1166. this.visit(property.value);
  1167. }
  1168. ArrayPattern(pattern) {
  1169. for (let i = 0, iz = pattern.elements.length; i < iz; ++i) {
  1170. const element = pattern.elements[i];
  1171. this.visit(element);
  1172. }
  1173. }
  1174. AssignmentPattern(pattern) {
  1175. this.assignments.push(pattern);
  1176. this.visit(pattern.left);
  1177. this.rightHandNodes.push(pattern.right);
  1178. this.assignments.pop();
  1179. }
  1180. RestElement(pattern) {
  1181. this.restElements.push(pattern);
  1182. this.visit(pattern.argument);
  1183. this.restElements.pop();
  1184. }
  1185. MemberExpression(node) {
  1186. // Computed property's key is a right hand node.
  1187. if (node.computed) {
  1188. this.rightHandNodes.push(node.property);
  1189. }
  1190. // the object is only read, write to its property.
  1191. this.rightHandNodes.push(node.object);
  1192. }
  1193. //
  1194. // ForInStatement.left and AssignmentExpression.left are LeftHandSideExpression.
  1195. // By spec, LeftHandSideExpression is Pattern or MemberExpression.
  1196. // (see also: https://github.com/estree/estree/pull/20#issuecomment-74584758)
  1197. // But espree 2.0 parses to ArrayExpression, ObjectExpression, etc...
  1198. //
  1199. SpreadElement(node) {
  1200. this.visit(node.argument);
  1201. }
  1202. ArrayExpression(node) {
  1203. node.elements.forEach(this.visit, this);
  1204. }
  1205. AssignmentExpression(node) {
  1206. this.assignments.push(node);
  1207. this.visit(node.left);
  1208. this.rightHandNodes.push(node.right);
  1209. this.assignments.pop();
  1210. }
  1211. CallExpression(node) {
  1212. // arguments are right hand nodes.
  1213. node.arguments.forEach(a => {
  1214. this.rightHandNodes.push(a);
  1215. });
  1216. this.visit(node.callee);
  1217. }
  1218. }
  1219. /* vim: set sw=4 ts=4 et tw=80 : */
  1220. /*
  1221. Copyright (C) 2015 Yusuke Suzuki <utatane.tea@gmail.com>
  1222. Redistribution and use in source and binary forms, with or without
  1223. modification, are permitted provided that the following conditions are met:
  1224. * Redistributions of source code must retain the above copyright
  1225. notice, this list of conditions and the following disclaimer.
  1226. * Redistributions in binary form must reproduce the above copyright
  1227. notice, this list of conditions and the following disclaimer in the
  1228. documentation and/or other materials provided with the distribution.
  1229. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  1230. AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  1231. IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  1232. ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  1233. DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  1234. (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  1235. LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  1236. ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  1237. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  1238. THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  1239. */
  1240. const { Syntax } = estraverse__default["default"];
  1241. /**
  1242. * Traverse identifier in pattern
  1243. * @param {Object} options options
  1244. * @param {pattern} rootPattern root pattern
  1245. * @param {Refencer} referencer referencer
  1246. * @param {callback} callback callback
  1247. * @returns {void}
  1248. */
  1249. function traverseIdentifierInPattern(options, rootPattern, referencer, callback) {
  1250. // Call the callback at left hand identifier nodes, and Collect right hand nodes.
  1251. const visitor = new PatternVisitor(options, rootPattern, callback);
  1252. visitor.visit(rootPattern);
  1253. // Process the right hand nodes recursively.
  1254. if (referencer !== null && referencer !== undefined) {
  1255. visitor.rightHandNodes.forEach(referencer.visit, referencer);
  1256. }
  1257. }
  1258. // Importing ImportDeclaration.
  1259. // http://people.mozilla.org/~jorendorff/es6-draft.html#sec-moduledeclarationinstantiation
  1260. // https://github.com/estree/estree/blob/master/es6.md#importdeclaration
  1261. // FIXME: Now, we don't create module environment, because the context is
  1262. // implementation dependent.
  1263. class Importer extends esrecurse__default["default"].Visitor {
  1264. constructor(declaration, referencer) {
  1265. super(null, referencer.options);
  1266. this.declaration = declaration;
  1267. this.referencer = referencer;
  1268. }
  1269. visitImport(id, specifier) {
  1270. this.referencer.visitPattern(id, pattern => {
  1271. this.referencer.currentScope().__define(pattern,
  1272. new Definition(
  1273. Variable.ImportBinding,
  1274. pattern,
  1275. specifier,
  1276. this.declaration,
  1277. null,
  1278. null
  1279. ));
  1280. });
  1281. }
  1282. ImportNamespaceSpecifier(node) {
  1283. const local = (node.local || node.id);
  1284. if (local) {
  1285. this.visitImport(local, node);
  1286. }
  1287. }
  1288. ImportDefaultSpecifier(node) {
  1289. const local = (node.local || node.id);
  1290. this.visitImport(local, node);
  1291. }
  1292. ImportSpecifier(node) {
  1293. const local = (node.local || node.id);
  1294. if (node.name) {
  1295. this.visitImport(node.name, node);
  1296. } else {
  1297. this.visitImport(local, node);
  1298. }
  1299. }
  1300. }
  1301. // Referencing variables and creating bindings.
  1302. class Referencer extends esrecurse__default["default"].Visitor {
  1303. constructor(options, scopeManager) {
  1304. super(null, options);
  1305. this.options = options;
  1306. this.scopeManager = scopeManager;
  1307. this.parent = null;
  1308. this.isInnerMethodDefinition = false;
  1309. }
  1310. currentScope() {
  1311. return this.scopeManager.__currentScope;
  1312. }
  1313. close(node) {
  1314. while (this.currentScope() && node === this.currentScope().block) {
  1315. this.scopeManager.__currentScope = this.currentScope().__close(this.scopeManager);
  1316. }
  1317. }
  1318. pushInnerMethodDefinition(isInnerMethodDefinition) {
  1319. const previous = this.isInnerMethodDefinition;
  1320. this.isInnerMethodDefinition = isInnerMethodDefinition;
  1321. return previous;
  1322. }
  1323. popInnerMethodDefinition(isInnerMethodDefinition) {
  1324. this.isInnerMethodDefinition = isInnerMethodDefinition;
  1325. }
  1326. referencingDefaultValue(pattern, assignments, maybeImplicitGlobal, init) {
  1327. const scope = this.currentScope();
  1328. assignments.forEach(assignment => {
  1329. scope.__referencing(
  1330. pattern,
  1331. Reference.WRITE,
  1332. assignment.right,
  1333. maybeImplicitGlobal,
  1334. pattern !== assignment.left,
  1335. init
  1336. );
  1337. });
  1338. }
  1339. visitPattern(node, options, callback) {
  1340. let visitPatternOptions = options;
  1341. let visitPatternCallback = callback;
  1342. if (typeof options === "function") {
  1343. visitPatternCallback = options;
  1344. visitPatternOptions = { processRightHandNodes: false };
  1345. }
  1346. traverseIdentifierInPattern(
  1347. this.options,
  1348. node,
  1349. visitPatternOptions.processRightHandNodes ? this : null,
  1350. visitPatternCallback
  1351. );
  1352. }
  1353. visitFunction(node) {
  1354. let i, iz;
  1355. // FunctionDeclaration name is defined in upper scope
  1356. // NOTE: Not referring variableScope. It is intended.
  1357. // Since
  1358. // in ES5, FunctionDeclaration should be in FunctionBody.
  1359. // in ES6, FunctionDeclaration should be block scoped.
  1360. if (node.type === Syntax.FunctionDeclaration) {
  1361. // id is defined in upper scope
  1362. this.currentScope().__define(node.id,
  1363. new Definition(
  1364. Variable.FunctionName,
  1365. node.id,
  1366. node,
  1367. null,
  1368. null,
  1369. null
  1370. ));
  1371. }
  1372. // FunctionExpression with name creates its special scope;
  1373. // FunctionExpressionNameScope.
  1374. if (node.type === Syntax.FunctionExpression && node.id) {
  1375. this.scopeManager.__nestFunctionExpressionNameScope(node);
  1376. }
  1377. // Consider this function is in the MethodDefinition.
  1378. this.scopeManager.__nestFunctionScope(node, this.isInnerMethodDefinition);
  1379. const that = this;
  1380. /**
  1381. * Visit pattern callback
  1382. * @param {pattern} pattern pattern
  1383. * @param {Object} info info
  1384. * @returns {void}
  1385. */
  1386. function visitPatternCallback(pattern, info) {
  1387. that.currentScope().__define(pattern,
  1388. new ParameterDefinition(
  1389. pattern,
  1390. node,
  1391. i,
  1392. info.rest
  1393. ));
  1394. that.referencingDefaultValue(pattern, info.assignments, null, true);
  1395. }
  1396. // Process parameter declarations.
  1397. for (i = 0, iz = node.params.length; i < iz; ++i) {
  1398. this.visitPattern(node.params[i], { processRightHandNodes: true }, visitPatternCallback);
  1399. }
  1400. // if there's a rest argument, add that
  1401. if (node.rest) {
  1402. this.visitPattern({
  1403. type: "RestElement",
  1404. argument: node.rest
  1405. }, pattern => {
  1406. this.currentScope().__define(pattern,
  1407. new ParameterDefinition(
  1408. pattern,
  1409. node,
  1410. node.params.length,
  1411. true
  1412. ));
  1413. });
  1414. }
  1415. // In TypeScript there are a number of function-like constructs which have no body,
  1416. // so check it exists before traversing
  1417. if (node.body) {
  1418. // Skip BlockStatement to prevent creating BlockStatement scope.
  1419. if (node.body.type === Syntax.BlockStatement) {
  1420. this.visitChildren(node.body);
  1421. } else {
  1422. this.visit(node.body);
  1423. }
  1424. }
  1425. this.close(node);
  1426. }
  1427. visitClass(node) {
  1428. if (node.type === Syntax.ClassDeclaration) {
  1429. this.currentScope().__define(node.id,
  1430. new Definition(
  1431. Variable.ClassName,
  1432. node.id,
  1433. node,
  1434. null,
  1435. null,
  1436. null
  1437. ));
  1438. }
  1439. this.visit(node.superClass);
  1440. this.scopeManager.__nestClassScope(node);
  1441. if (node.id) {
  1442. this.currentScope().__define(node.id,
  1443. new Definition(
  1444. Variable.ClassName,
  1445. node.id,
  1446. node
  1447. ));
  1448. }
  1449. this.visit(node.body);
  1450. this.close(node);
  1451. }
  1452. visitProperty(node) {
  1453. let previous;
  1454. if (node.computed) {
  1455. this.visit(node.key);
  1456. }
  1457. const isMethodDefinition = node.type === Syntax.MethodDefinition;
  1458. if (isMethodDefinition) {
  1459. previous = this.pushInnerMethodDefinition(true);
  1460. }
  1461. this.visit(node.value);
  1462. if (isMethodDefinition) {
  1463. this.popInnerMethodDefinition(previous);
  1464. }
  1465. }
  1466. visitForIn(node) {
  1467. if (node.left.type === Syntax.VariableDeclaration && node.left.kind !== "var") {
  1468. this.scopeManager.__nestForScope(node);
  1469. }
  1470. if (node.left.type === Syntax.VariableDeclaration) {
  1471. this.visit(node.left);
  1472. this.visitPattern(node.left.declarations[0].id, pattern => {
  1473. this.currentScope().__referencing(pattern, Reference.WRITE, node.right, null, true, true);
  1474. });
  1475. } else {
  1476. this.visitPattern(node.left, { processRightHandNodes: true }, (pattern, info) => {
  1477. let maybeImplicitGlobal = null;
  1478. if (!this.currentScope().isStrict) {
  1479. maybeImplicitGlobal = {
  1480. pattern,
  1481. node
  1482. };
  1483. }
  1484. this.referencingDefaultValue(pattern, info.assignments, maybeImplicitGlobal, false);
  1485. this.currentScope().__referencing(pattern, Reference.WRITE, node.right, maybeImplicitGlobal, true, false);
  1486. });
  1487. }
  1488. this.visit(node.right);
  1489. this.visit(node.body);
  1490. this.close(node);
  1491. }
  1492. visitVariableDeclaration(variableTargetScope, type, node, index) {
  1493. const decl = node.declarations[index];
  1494. const init = decl.init;
  1495. this.visitPattern(decl.id, { processRightHandNodes: true }, (pattern, info) => {
  1496. variableTargetScope.__define(
  1497. pattern,
  1498. new Definition(
  1499. type,
  1500. pattern,
  1501. decl,
  1502. node,
  1503. index,
  1504. node.kind
  1505. )
  1506. );
  1507. this.referencingDefaultValue(pattern, info.assignments, null, true);
  1508. if (init) {
  1509. this.currentScope().__referencing(pattern, Reference.WRITE, init, null, !info.topLevel, true);
  1510. }
  1511. });
  1512. }
  1513. AssignmentExpression(node) {
  1514. if (PatternVisitor.isPattern(node.left)) {
  1515. if (node.operator === "=") {
  1516. this.visitPattern(node.left, { processRightHandNodes: true }, (pattern, info) => {
  1517. let maybeImplicitGlobal = null;
  1518. if (!this.currentScope().isStrict) {
  1519. maybeImplicitGlobal = {
  1520. pattern,
  1521. node
  1522. };
  1523. }
  1524. this.referencingDefaultValue(pattern, info.assignments, maybeImplicitGlobal, false);
  1525. this.currentScope().__referencing(pattern, Reference.WRITE, node.right, maybeImplicitGlobal, !info.topLevel, false);
  1526. });
  1527. } else {
  1528. this.currentScope().__referencing(node.left, Reference.RW, node.right);
  1529. }
  1530. } else {
  1531. this.visit(node.left);
  1532. }
  1533. this.visit(node.right);
  1534. }
  1535. CatchClause(node) {
  1536. this.scopeManager.__nestCatchScope(node);
  1537. this.visitPattern(node.param, { processRightHandNodes: true }, (pattern, info) => {
  1538. this.currentScope().__define(pattern,
  1539. new Definition(
  1540. Variable.CatchClause,
  1541. node.param,
  1542. node,
  1543. null,
  1544. null,
  1545. null
  1546. ));
  1547. this.referencingDefaultValue(pattern, info.assignments, null, true);
  1548. });
  1549. this.visit(node.body);
  1550. this.close(node);
  1551. }
  1552. Program(node) {
  1553. this.scopeManager.__nestGlobalScope(node);
  1554. if (this.scopeManager.__isNodejsScope()) {
  1555. // Force strictness of GlobalScope to false when using node.js scope.
  1556. this.currentScope().isStrict = false;
  1557. this.scopeManager.__nestFunctionScope(node, false);
  1558. }
  1559. if (this.scopeManager.__isES6() && this.scopeManager.isModule()) {
  1560. this.scopeManager.__nestModuleScope(node);
  1561. }
  1562. if (this.scopeManager.isStrictModeSupported() && this.scopeManager.isImpliedStrict()) {
  1563. this.currentScope().isStrict = true;
  1564. }
  1565. this.visitChildren(node);
  1566. this.close(node);
  1567. }
  1568. Identifier(node) {
  1569. this.currentScope().__referencing(node);
  1570. }
  1571. // eslint-disable-next-line class-methods-use-this
  1572. PrivateIdentifier() {
  1573. // Do nothing.
  1574. }
  1575. UpdateExpression(node) {
  1576. if (PatternVisitor.isPattern(node.argument)) {
  1577. this.currentScope().__referencing(node.argument, Reference.RW, null);
  1578. } else {
  1579. this.visitChildren(node);
  1580. }
  1581. }
  1582. MemberExpression(node) {
  1583. this.visit(node.object);
  1584. if (node.computed) {
  1585. this.visit(node.property);
  1586. }
  1587. }
  1588. Property(node) {
  1589. this.visitProperty(node);
  1590. }
  1591. PropertyDefinition(node) {
  1592. const { computed, key, value } = node;
  1593. if (computed) {
  1594. this.visit(key);
  1595. }
  1596. if (value) {
  1597. this.scopeManager.__nestClassFieldInitializerScope(value);
  1598. this.visit(value);
  1599. this.close(value);
  1600. }
  1601. }
  1602. StaticBlock(node) {
  1603. this.scopeManager.__nestClassStaticBlockScope(node);
  1604. this.visitChildren(node);
  1605. this.close(node);
  1606. }
  1607. MethodDefinition(node) {
  1608. this.visitProperty(node);
  1609. }
  1610. BreakStatement() {} // eslint-disable-line class-methods-use-this
  1611. ContinueStatement() {} // eslint-disable-line class-methods-use-this
  1612. LabeledStatement(node) {
  1613. this.visit(node.body);
  1614. }
  1615. ForStatement(node) {
  1616. // Create ForStatement declaration.
  1617. // NOTE: In ES6, ForStatement dynamically generates
  1618. // per iteration environment. However, escope is
  1619. // a static analyzer, we only generate one scope for ForStatement.
  1620. if (node.init && node.init.type === Syntax.VariableDeclaration && node.init.kind !== "var") {
  1621. this.scopeManager.__nestForScope(node);
  1622. }
  1623. this.visitChildren(node);
  1624. this.close(node);
  1625. }
  1626. ClassExpression(node) {
  1627. this.visitClass(node);
  1628. }
  1629. ClassDeclaration(node) {
  1630. this.visitClass(node);
  1631. }
  1632. CallExpression(node) {
  1633. // Check this is direct call to eval
  1634. if (!this.scopeManager.__ignoreEval() && node.callee.type === Syntax.Identifier && node.callee.name === "eval") {
  1635. // NOTE: This should be `variableScope`. Since direct eval call always creates Lexical environment and
  1636. // let / const should be enclosed into it. Only VariableDeclaration affects on the caller's environment.
  1637. this.currentScope().variableScope.__detectEval();
  1638. }
  1639. this.visitChildren(node);
  1640. }
  1641. BlockStatement(node) {
  1642. if (this.scopeManager.__isES6()) {
  1643. this.scopeManager.__nestBlockScope(node);
  1644. }
  1645. this.visitChildren(node);
  1646. this.close(node);
  1647. }
  1648. ThisExpression() {
  1649. this.currentScope().variableScope.__detectThis();
  1650. }
  1651. WithStatement(node) {
  1652. this.visit(node.object);
  1653. // Then nest scope for WithStatement.
  1654. this.scopeManager.__nestWithScope(node);
  1655. this.visit(node.body);
  1656. this.close(node);
  1657. }
  1658. VariableDeclaration(node) {
  1659. const variableTargetScope = (node.kind === "var") ? this.currentScope().variableScope : this.currentScope();
  1660. for (let i = 0, iz = node.declarations.length; i < iz; ++i) {
  1661. const decl = node.declarations[i];
  1662. this.visitVariableDeclaration(variableTargetScope, Variable.Variable, node, i);
  1663. if (decl.init) {
  1664. this.visit(decl.init);
  1665. }
  1666. }
  1667. }
  1668. // sec 13.11.8
  1669. SwitchStatement(node) {
  1670. this.visit(node.discriminant);
  1671. if (this.scopeManager.__isES6()) {
  1672. this.scopeManager.__nestSwitchScope(node);
  1673. }
  1674. for (let i = 0, iz = node.cases.length; i < iz; ++i) {
  1675. this.visit(node.cases[i]);
  1676. }
  1677. this.close(node);
  1678. }
  1679. FunctionDeclaration(node) {
  1680. this.visitFunction(node);
  1681. }
  1682. FunctionExpression(node) {
  1683. this.visitFunction(node);
  1684. }
  1685. ForOfStatement(node) {
  1686. this.visitForIn(node);
  1687. }
  1688. ForInStatement(node) {
  1689. this.visitForIn(node);
  1690. }
  1691. ArrowFunctionExpression(node) {
  1692. this.visitFunction(node);
  1693. }
  1694. ImportDeclaration(node) {
  1695. assert__default["default"](this.scopeManager.__isES6() && this.scopeManager.isModule(), "ImportDeclaration should appear when the mode is ES6 and in the module context.");
  1696. const importer = new Importer(node, this);
  1697. importer.visit(node);
  1698. }
  1699. visitExportDeclaration(node) {
  1700. if (node.source) {
  1701. return;
  1702. }
  1703. if (node.declaration) {
  1704. this.visit(node.declaration);
  1705. return;
  1706. }
  1707. this.visitChildren(node);
  1708. }
  1709. // TODO: ExportDeclaration doesn't exist. for bc?
  1710. ExportDeclaration(node) {
  1711. this.visitExportDeclaration(node);
  1712. }
  1713. ExportAllDeclaration(node) {
  1714. this.visitExportDeclaration(node);
  1715. }
  1716. ExportDefaultDeclaration(node) {
  1717. this.visitExportDeclaration(node);
  1718. }
  1719. ExportNamedDeclaration(node) {
  1720. this.visitExportDeclaration(node);
  1721. }
  1722. ExportSpecifier(node) {
  1723. // TODO: `node.id` doesn't exist. for bc?
  1724. const local = (node.id || node.local);
  1725. this.visit(local);
  1726. }
  1727. MetaProperty() { // eslint-disable-line class-methods-use-this
  1728. // do nothing.
  1729. }
  1730. }
  1731. /* vim: set sw=4 ts=4 et tw=80 : */
  1732. const version = "7.1.0";
  1733. /*
  1734. Copyright (C) 2012-2014 Yusuke Suzuki <utatane.tea@gmail.com>
  1735. Copyright (C) 2013 Alex Seville <hi@alexanderseville.com>
  1736. Copyright (C) 2014 Thiago de Arruda <tpadilha84@gmail.com>
  1737. Redistribution and use in source and binary forms, with or without
  1738. modification, are permitted provided that the following conditions are met:
  1739. * Redistributions of source code must retain the above copyright
  1740. notice, this list of conditions and the following disclaimer.
  1741. * Redistributions in binary form must reproduce the above copyright
  1742. notice, this list of conditions and the following disclaimer in the
  1743. documentation and/or other materials provided with the distribution.
  1744. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  1745. AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  1746. IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  1747. ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  1748. DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  1749. (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  1750. LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  1751. ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  1752. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  1753. THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  1754. */
  1755. /**
  1756. * Set the default options
  1757. * @returns {Object} options
  1758. */
  1759. function defaultOptions() {
  1760. return {
  1761. optimistic: false,
  1762. directive: false,
  1763. nodejsScope: false,
  1764. impliedStrict: false,
  1765. sourceType: "script", // one of ['script', 'module', 'commonjs']
  1766. ecmaVersion: 5,
  1767. childVisitorKeys: null,
  1768. fallback: "iteration"
  1769. };
  1770. }
  1771. /**
  1772. * Preform deep update on option object
  1773. * @param {Object} target Options
  1774. * @param {Object} override Updates
  1775. * @returns {Object} Updated options
  1776. */
  1777. function updateDeeply(target, override) {
  1778. /**
  1779. * Is hash object
  1780. * @param {Object} value Test value
  1781. * @returns {boolean} Result
  1782. */
  1783. function isHashObject(value) {
  1784. return typeof value === "object" && value instanceof Object && !(value instanceof Array) && !(value instanceof RegExp);
  1785. }
  1786. for (const key in override) {
  1787. if (Object.prototype.hasOwnProperty.call(override, key)) {
  1788. const val = override[key];
  1789. if (isHashObject(val)) {
  1790. if (isHashObject(target[key])) {
  1791. updateDeeply(target[key], val);
  1792. } else {
  1793. target[key] = updateDeeply({}, val);
  1794. }
  1795. } else {
  1796. target[key] = val;
  1797. }
  1798. }
  1799. }
  1800. return target;
  1801. }
  1802. /**
  1803. * Main interface function. Takes an Espree syntax tree and returns the
  1804. * analyzed scopes.
  1805. * @function analyze
  1806. * @param {espree.Tree} tree Abstract Syntax Tree
  1807. * @param {Object} providedOptions Options that tailor the scope analysis
  1808. * @param {boolean} [providedOptions.optimistic=false] the optimistic flag
  1809. * @param {boolean} [providedOptions.directive=false] the directive flag
  1810. * @param {boolean} [providedOptions.ignoreEval=false] whether to check 'eval()' calls
  1811. * @param {boolean} [providedOptions.nodejsScope=false] whether the whole
  1812. * script is executed under node.js environment. When enabled, escope adds
  1813. * a function scope immediately following the global scope.
  1814. * @param {boolean} [providedOptions.impliedStrict=false] implied strict mode
  1815. * (if ecmaVersion >= 5).
  1816. * @param {string} [providedOptions.sourceType='script'] the source type of the script. one of 'script', 'module', and 'commonjs'
  1817. * @param {number} [providedOptions.ecmaVersion=5] which ECMAScript version is considered
  1818. * @param {Object} [providedOptions.childVisitorKeys=null] Additional known visitor keys. See [esrecurse](https://github.com/estools/esrecurse)'s the `childVisitorKeys` option.
  1819. * @param {string} [providedOptions.fallback='iteration'] A kind of the fallback in order to encounter with unknown node. See [esrecurse](https://github.com/estools/esrecurse)'s the `fallback` option.
  1820. * @returns {ScopeManager} ScopeManager
  1821. */
  1822. function analyze(tree, providedOptions) {
  1823. const options = updateDeeply(defaultOptions(), providedOptions);
  1824. const scopeManager = new ScopeManager(options);
  1825. const referencer = new Referencer(options, scopeManager);
  1826. referencer.visit(tree);
  1827. assert__default["default"](scopeManager.__currentScope === null, "currentScope should be null.");
  1828. return scopeManager;
  1829. }
  1830. /* vim: set sw=4 ts=4 et tw=80 : */
  1831. exports.Definition = Definition;
  1832. exports.PatternVisitor = PatternVisitor;
  1833. exports.Reference = Reference;
  1834. exports.Referencer = Referencer;
  1835. exports.Scope = Scope;
  1836. exports.ScopeManager = ScopeManager;
  1837. exports.Variable = Variable;
  1838. exports.analyze = analyze;
  1839. exports.version = version;
  1840. //# sourceMappingURL=eslint-scope.cjs.map