espree.cjs 33 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013
  1. 'use strict';
  2. Object.defineProperty(exports, '__esModule', { value: true });
  3. var acorn = require('acorn');
  4. var jsx = require('acorn-jsx');
  5. var visitorKeys = require('eslint-visitor-keys');
  6. function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
  7. function _interopNamespace(e) {
  8. if (e && e.__esModule) return e;
  9. var n = Object.create(null);
  10. if (e) {
  11. Object.keys(e).forEach(function (k) {
  12. if (k !== 'default') {
  13. var d = Object.getOwnPropertyDescriptor(e, k);
  14. Object.defineProperty(n, k, d.get ? d : {
  15. enumerable: true,
  16. get: function () { return e[k]; }
  17. });
  18. }
  19. });
  20. }
  21. n["default"] = e;
  22. return Object.freeze(n);
  23. }
  24. var acorn__namespace = /*#__PURE__*/_interopNamespace(acorn);
  25. var jsx__default = /*#__PURE__*/_interopDefaultLegacy(jsx);
  26. var visitorKeys__namespace = /*#__PURE__*/_interopNamespace(visitorKeys);
  27. /**
  28. * @fileoverview The AST node types produced by the parser.
  29. * @author Nicholas C. Zakas
  30. */
  31. //------------------------------------------------------------------------------
  32. // Requirements
  33. //------------------------------------------------------------------------------
  34. // None!
  35. //------------------------------------------------------------------------------
  36. // Public
  37. //------------------------------------------------------------------------------
  38. var astNodeTypes = {
  39. AssignmentExpression: "AssignmentExpression",
  40. AssignmentPattern: "AssignmentPattern",
  41. ArrayExpression: "ArrayExpression",
  42. ArrayPattern: "ArrayPattern",
  43. ArrowFunctionExpression: "ArrowFunctionExpression",
  44. AwaitExpression: "AwaitExpression",
  45. BlockStatement: "BlockStatement",
  46. BinaryExpression: "BinaryExpression",
  47. BreakStatement: "BreakStatement",
  48. CallExpression: "CallExpression",
  49. CatchClause: "CatchClause",
  50. ClassBody: "ClassBody",
  51. ClassDeclaration: "ClassDeclaration",
  52. ClassExpression: "ClassExpression",
  53. ConditionalExpression: "ConditionalExpression",
  54. ContinueStatement: "ContinueStatement",
  55. DoWhileStatement: "DoWhileStatement",
  56. DebuggerStatement: "DebuggerStatement",
  57. EmptyStatement: "EmptyStatement",
  58. ExpressionStatement: "ExpressionStatement",
  59. ForStatement: "ForStatement",
  60. ForInStatement: "ForInStatement",
  61. ForOfStatement: "ForOfStatement",
  62. FunctionDeclaration: "FunctionDeclaration",
  63. FunctionExpression: "FunctionExpression",
  64. Identifier: "Identifier",
  65. IfStatement: "IfStatement",
  66. Literal: "Literal",
  67. LabeledStatement: "LabeledStatement",
  68. LogicalExpression: "LogicalExpression",
  69. MemberExpression: "MemberExpression",
  70. MetaProperty: "MetaProperty",
  71. MethodDefinition: "MethodDefinition",
  72. NewExpression: "NewExpression",
  73. ObjectExpression: "ObjectExpression",
  74. ObjectPattern: "ObjectPattern",
  75. Program: "Program",
  76. Property: "Property",
  77. RestElement: "RestElement",
  78. ReturnStatement: "ReturnStatement",
  79. SequenceExpression: "SequenceExpression",
  80. SpreadElement: "SpreadElement",
  81. Super: "Super",
  82. SwitchCase: "SwitchCase",
  83. SwitchStatement: "SwitchStatement",
  84. TaggedTemplateExpression: "TaggedTemplateExpression",
  85. TemplateElement: "TemplateElement",
  86. TemplateLiteral: "TemplateLiteral",
  87. ThisExpression: "ThisExpression",
  88. ThrowStatement: "ThrowStatement",
  89. TryStatement: "TryStatement",
  90. UnaryExpression: "UnaryExpression",
  91. UpdateExpression: "UpdateExpression",
  92. VariableDeclaration: "VariableDeclaration",
  93. VariableDeclarator: "VariableDeclarator",
  94. WhileStatement: "WhileStatement",
  95. WithStatement: "WithStatement",
  96. YieldExpression: "YieldExpression",
  97. JSXIdentifier: "JSXIdentifier",
  98. JSXNamespacedName: "JSXNamespacedName",
  99. JSXMemberExpression: "JSXMemberExpression",
  100. JSXEmptyExpression: "JSXEmptyExpression",
  101. JSXExpressionContainer: "JSXExpressionContainer",
  102. JSXElement: "JSXElement",
  103. JSXClosingElement: "JSXClosingElement",
  104. JSXOpeningElement: "JSXOpeningElement",
  105. JSXAttribute: "JSXAttribute",
  106. JSXSpreadAttribute: "JSXSpreadAttribute",
  107. JSXText: "JSXText",
  108. ExportDefaultDeclaration: "ExportDefaultDeclaration",
  109. ExportNamedDeclaration: "ExportNamedDeclaration",
  110. ExportAllDeclaration: "ExportAllDeclaration",
  111. ExportSpecifier: "ExportSpecifier",
  112. ImportDeclaration: "ImportDeclaration",
  113. ImportSpecifier: "ImportSpecifier",
  114. ImportDefaultSpecifier: "ImportDefaultSpecifier",
  115. ImportNamespaceSpecifier: "ImportNamespaceSpecifier"
  116. };
  117. /**
  118. * @fileoverview Translates tokens between Acorn format and Esprima format.
  119. * @author Nicholas C. Zakas
  120. */
  121. /* eslint no-underscore-dangle: 0 */
  122. //------------------------------------------------------------------------------
  123. // Requirements
  124. //------------------------------------------------------------------------------
  125. // none!
  126. //------------------------------------------------------------------------------
  127. // Private
  128. //------------------------------------------------------------------------------
  129. // Esprima Token Types
  130. const Token = {
  131. Boolean: "Boolean",
  132. EOF: "<end>",
  133. Identifier: "Identifier",
  134. PrivateIdentifier: "PrivateIdentifier",
  135. Keyword: "Keyword",
  136. Null: "Null",
  137. Numeric: "Numeric",
  138. Punctuator: "Punctuator",
  139. String: "String",
  140. RegularExpression: "RegularExpression",
  141. Template: "Template",
  142. JSXIdentifier: "JSXIdentifier",
  143. JSXText: "JSXText"
  144. };
  145. /**
  146. * Converts part of a template into an Esprima token.
  147. * @param {AcornToken[]} tokens The Acorn tokens representing the template.
  148. * @param {string} code The source code.
  149. * @returns {EsprimaToken} The Esprima equivalent of the template token.
  150. * @private
  151. */
  152. function convertTemplatePart(tokens, code) {
  153. const firstToken = tokens[0],
  154. lastTemplateToken = tokens[tokens.length - 1];
  155. const token = {
  156. type: Token.Template,
  157. value: code.slice(firstToken.start, lastTemplateToken.end)
  158. };
  159. if (firstToken.loc) {
  160. token.loc = {
  161. start: firstToken.loc.start,
  162. end: lastTemplateToken.loc.end
  163. };
  164. }
  165. if (firstToken.range) {
  166. token.start = firstToken.range[0];
  167. token.end = lastTemplateToken.range[1];
  168. token.range = [token.start, token.end];
  169. }
  170. return token;
  171. }
  172. /**
  173. * Contains logic to translate Acorn tokens into Esprima tokens.
  174. * @param {Object} acornTokTypes The Acorn token types.
  175. * @param {string} code The source code Acorn is parsing. This is necessary
  176. * to correct the "value" property of some tokens.
  177. * @constructor
  178. */
  179. function TokenTranslator(acornTokTypes, code) {
  180. // token types
  181. this._acornTokTypes = acornTokTypes;
  182. // token buffer for templates
  183. this._tokens = [];
  184. // track the last curly brace
  185. this._curlyBrace = null;
  186. // the source code
  187. this._code = code;
  188. }
  189. TokenTranslator.prototype = {
  190. constructor: TokenTranslator,
  191. /**
  192. * Translates a single Esprima token to a single Acorn token. This may be
  193. * inaccurate due to how templates are handled differently in Esprima and
  194. * Acorn, but should be accurate for all other tokens.
  195. * @param {AcornToken} token The Acorn token to translate.
  196. * @param {Object} extra Espree extra object.
  197. * @returns {EsprimaToken} The Esprima version of the token.
  198. */
  199. translate(token, extra) {
  200. const type = token.type,
  201. tt = this._acornTokTypes;
  202. if (type === tt.name) {
  203. token.type = Token.Identifier;
  204. // TODO: See if this is an Acorn bug
  205. if (token.value === "static") {
  206. token.type = Token.Keyword;
  207. }
  208. if (extra.ecmaVersion > 5 && (token.value === "yield" || token.value === "let")) {
  209. token.type = Token.Keyword;
  210. }
  211. } else if (type === tt.privateId) {
  212. token.type = Token.PrivateIdentifier;
  213. } else if (type === tt.semi || type === tt.comma ||
  214. type === tt.parenL || type === tt.parenR ||
  215. type === tt.braceL || type === tt.braceR ||
  216. type === tt.dot || type === tt.bracketL ||
  217. type === tt.colon || type === tt.question ||
  218. type === tt.bracketR || type === tt.ellipsis ||
  219. type === tt.arrow || type === tt.jsxTagStart ||
  220. type === tt.incDec || type === tt.starstar ||
  221. type === tt.jsxTagEnd || type === tt.prefix ||
  222. type === tt.questionDot ||
  223. (type.binop && !type.keyword) ||
  224. type.isAssign) {
  225. token.type = Token.Punctuator;
  226. token.value = this._code.slice(token.start, token.end);
  227. } else if (type === tt.jsxName) {
  228. token.type = Token.JSXIdentifier;
  229. } else if (type.label === "jsxText" || type === tt.jsxAttrValueToken) {
  230. token.type = Token.JSXText;
  231. } else if (type.keyword) {
  232. if (type.keyword === "true" || type.keyword === "false") {
  233. token.type = Token.Boolean;
  234. } else if (type.keyword === "null") {
  235. token.type = Token.Null;
  236. } else {
  237. token.type = Token.Keyword;
  238. }
  239. } else if (type === tt.num) {
  240. token.type = Token.Numeric;
  241. token.value = this._code.slice(token.start, token.end);
  242. } else if (type === tt.string) {
  243. if (extra.jsxAttrValueToken) {
  244. extra.jsxAttrValueToken = false;
  245. token.type = Token.JSXText;
  246. } else {
  247. token.type = Token.String;
  248. }
  249. token.value = this._code.slice(token.start, token.end);
  250. } else if (type === tt.regexp) {
  251. token.type = Token.RegularExpression;
  252. const value = token.value;
  253. token.regex = {
  254. flags: value.flags,
  255. pattern: value.pattern
  256. };
  257. token.value = `/${value.pattern}/${value.flags}`;
  258. }
  259. return token;
  260. },
  261. /**
  262. * Function to call during Acorn's onToken handler.
  263. * @param {AcornToken} token The Acorn token.
  264. * @param {Object} extra The Espree extra object.
  265. * @returns {void}
  266. */
  267. onToken(token, extra) {
  268. const that = this,
  269. tt = this._acornTokTypes,
  270. tokens = extra.tokens,
  271. templateTokens = this._tokens;
  272. /**
  273. * Flushes the buffered template tokens and resets the template
  274. * tracking.
  275. * @returns {void}
  276. * @private
  277. */
  278. function translateTemplateTokens() {
  279. tokens.push(convertTemplatePart(that._tokens, that._code));
  280. that._tokens = [];
  281. }
  282. if (token.type === tt.eof) {
  283. // might be one last curlyBrace
  284. if (this._curlyBrace) {
  285. tokens.push(this.translate(this._curlyBrace, extra));
  286. }
  287. return;
  288. }
  289. if (token.type === tt.backQuote) {
  290. // if there's already a curly, it's not part of the template
  291. if (this._curlyBrace) {
  292. tokens.push(this.translate(this._curlyBrace, extra));
  293. this._curlyBrace = null;
  294. }
  295. templateTokens.push(token);
  296. // it's the end
  297. if (templateTokens.length > 1) {
  298. translateTemplateTokens();
  299. }
  300. return;
  301. }
  302. if (token.type === tt.dollarBraceL) {
  303. templateTokens.push(token);
  304. translateTemplateTokens();
  305. return;
  306. }
  307. if (token.type === tt.braceR) {
  308. // if there's already a curly, it's not part of the template
  309. if (this._curlyBrace) {
  310. tokens.push(this.translate(this._curlyBrace, extra));
  311. }
  312. // store new curly for later
  313. this._curlyBrace = token;
  314. return;
  315. }
  316. if (token.type === tt.template || token.type === tt.invalidTemplate) {
  317. if (this._curlyBrace) {
  318. templateTokens.push(this._curlyBrace);
  319. this._curlyBrace = null;
  320. }
  321. templateTokens.push(token);
  322. return;
  323. }
  324. if (this._curlyBrace) {
  325. tokens.push(this.translate(this._curlyBrace, extra));
  326. this._curlyBrace = null;
  327. }
  328. tokens.push(this.translate(token, extra));
  329. }
  330. };
  331. /**
  332. * @fileoverview A collection of methods for processing Espree's options.
  333. * @author Kai Cataldo
  334. */
  335. //------------------------------------------------------------------------------
  336. // Helpers
  337. //------------------------------------------------------------------------------
  338. const SUPPORTED_VERSIONS = [
  339. 3,
  340. 5,
  341. 6,
  342. 7,
  343. 8,
  344. 9,
  345. 10,
  346. 11,
  347. 12,
  348. 13
  349. ];
  350. /**
  351. * Get the latest ECMAScript version supported by Espree.
  352. * @returns {number} The latest ECMAScript version.
  353. */
  354. function getLatestEcmaVersion() {
  355. return SUPPORTED_VERSIONS[SUPPORTED_VERSIONS.length - 1];
  356. }
  357. /**
  358. * Get the list of ECMAScript versions supported by Espree.
  359. * @returns {number[]} An array containing the supported ECMAScript versions.
  360. */
  361. function getSupportedEcmaVersions() {
  362. return [...SUPPORTED_VERSIONS];
  363. }
  364. /**
  365. * Normalize ECMAScript version from the initial config
  366. * @param {(number|"latest")} ecmaVersion ECMAScript version from the initial config
  367. * @throws {Error} throws an error if the ecmaVersion is invalid.
  368. * @returns {number} normalized ECMAScript version
  369. */
  370. function normalizeEcmaVersion(ecmaVersion = 5) {
  371. let version = ecmaVersion === "latest" ? getLatestEcmaVersion() : ecmaVersion;
  372. if (typeof version !== "number") {
  373. throw new Error(`ecmaVersion must be a number or "latest". Received value of type ${typeof ecmaVersion} instead.`);
  374. }
  375. // Calculate ECMAScript edition number from official year version starting with
  376. // ES2015, which corresponds with ES6 (or a difference of 2009).
  377. if (version >= 2015) {
  378. version -= 2009;
  379. }
  380. if (!SUPPORTED_VERSIONS.includes(version)) {
  381. throw new Error("Invalid ecmaVersion.");
  382. }
  383. return version;
  384. }
  385. /**
  386. * Normalize sourceType from the initial config
  387. * @param {string} sourceType to normalize
  388. * @throws {Error} throw an error if sourceType is invalid
  389. * @returns {string} normalized sourceType
  390. */
  391. function normalizeSourceType(sourceType = "script") {
  392. if (sourceType === "script" || sourceType === "module") {
  393. return sourceType;
  394. }
  395. if (sourceType === "commonjs") {
  396. return "script";
  397. }
  398. throw new Error("Invalid sourceType.");
  399. }
  400. /**
  401. * Normalize parserOptions
  402. * @param {Object} options the parser options to normalize
  403. * @throws {Error} throw an error if found invalid option.
  404. * @returns {Object} normalized options
  405. */
  406. function normalizeOptions(options) {
  407. const ecmaVersion = normalizeEcmaVersion(options.ecmaVersion);
  408. const sourceType = normalizeSourceType(options.sourceType);
  409. const ranges = options.range === true;
  410. const locations = options.loc === true;
  411. if (ecmaVersion !== 3 && options.allowReserved) {
  412. // a value of `false` is intentionally allowed here, so a shared config can overwrite it when needed
  413. throw new Error("`allowReserved` is only supported when ecmaVersion is 3");
  414. }
  415. if (typeof options.allowReserved !== "undefined" && typeof options.allowReserved !== "boolean") {
  416. throw new Error("`allowReserved`, when present, must be `true` or `false`");
  417. }
  418. const allowReserved = ecmaVersion === 3 ? (options.allowReserved || "never") : false;
  419. const ecmaFeatures = options.ecmaFeatures || {};
  420. const allowReturnOutsideFunction = options.sourceType === "commonjs" ||
  421. Boolean(ecmaFeatures.globalReturn);
  422. if (sourceType === "module" && ecmaVersion < 6) {
  423. throw new Error("sourceType 'module' is not supported when ecmaVersion < 2015. Consider adding `{ ecmaVersion: 2015 }` to the parser options.");
  424. }
  425. return Object.assign({}, options, {
  426. ecmaVersion,
  427. sourceType,
  428. ranges,
  429. locations,
  430. allowReserved,
  431. allowReturnOutsideFunction
  432. });
  433. }
  434. /* eslint-disable no-param-reassign*/
  435. const STATE = Symbol("espree's internal state");
  436. const ESPRIMA_FINISH_NODE = Symbol("espree's esprimaFinishNode");
  437. /**
  438. * Converts an Acorn comment to a Esprima comment.
  439. * @param {boolean} block True if it's a block comment, false if not.
  440. * @param {string} text The text of the comment.
  441. * @param {int} start The index at which the comment starts.
  442. * @param {int} end The index at which the comment ends.
  443. * @param {Location} startLoc The location at which the comment starts.
  444. * @param {Location} endLoc The location at which the comment ends.
  445. * @returns {Object} The comment object.
  446. * @private
  447. */
  448. function convertAcornCommentToEsprimaComment(block, text, start, end, startLoc, endLoc) {
  449. const comment = {
  450. type: block ? "Block" : "Line",
  451. value: text
  452. };
  453. if (typeof start === "number") {
  454. comment.start = start;
  455. comment.end = end;
  456. comment.range = [start, end];
  457. }
  458. if (typeof startLoc === "object") {
  459. comment.loc = {
  460. start: startLoc,
  461. end: endLoc
  462. };
  463. }
  464. return comment;
  465. }
  466. var espree = () => Parser => {
  467. const tokTypes = Object.assign({}, Parser.acorn.tokTypes);
  468. if (Parser.acornJsx) {
  469. Object.assign(tokTypes, Parser.acornJsx.tokTypes);
  470. }
  471. return class Espree extends Parser {
  472. constructor(opts, code) {
  473. if (typeof opts !== "object" || opts === null) {
  474. opts = {};
  475. }
  476. if (typeof code !== "string" && !(code instanceof String)) {
  477. code = String(code);
  478. }
  479. // save original source type in case of commonjs
  480. const originalSourceType = opts.sourceType;
  481. const options = normalizeOptions(opts);
  482. const ecmaFeatures = options.ecmaFeatures || {};
  483. const tokenTranslator =
  484. options.tokens === true
  485. ? new TokenTranslator(tokTypes, code)
  486. : null;
  487. // Initialize acorn parser.
  488. super({
  489. // do not use spread, because we don't want to pass any unknown options to acorn
  490. ecmaVersion: options.ecmaVersion,
  491. sourceType: options.sourceType,
  492. ranges: options.ranges,
  493. locations: options.locations,
  494. allowReserved: options.allowReserved,
  495. // Truthy value is true for backward compatibility.
  496. allowReturnOutsideFunction: options.allowReturnOutsideFunction,
  497. // Collect tokens
  498. onToken: token => {
  499. if (tokenTranslator) {
  500. // Use `tokens`, `ecmaVersion`, and `jsxAttrValueToken` in the state.
  501. tokenTranslator.onToken(token, this[STATE]);
  502. }
  503. if (token.type !== tokTypes.eof) {
  504. this[STATE].lastToken = token;
  505. }
  506. },
  507. // Collect comments
  508. onComment: (block, text, start, end, startLoc, endLoc) => {
  509. if (this[STATE].comments) {
  510. const comment = convertAcornCommentToEsprimaComment(block, text, start, end, startLoc, endLoc);
  511. this[STATE].comments.push(comment);
  512. }
  513. }
  514. }, code);
  515. /*
  516. * Data that is unique to Espree and is not represented internally in
  517. * Acorn. We put all of this data into a symbol property as a way to
  518. * avoid potential naming conflicts with future versions of Acorn.
  519. */
  520. this[STATE] = {
  521. originalSourceType: originalSourceType || options.sourceType,
  522. tokens: tokenTranslator ? [] : null,
  523. comments: options.comment === true ? [] : null,
  524. impliedStrict: ecmaFeatures.impliedStrict === true && this.options.ecmaVersion >= 5,
  525. ecmaVersion: this.options.ecmaVersion,
  526. jsxAttrValueToken: false,
  527. lastToken: null,
  528. templateElements: []
  529. };
  530. }
  531. tokenize() {
  532. do {
  533. this.next();
  534. } while (this.type !== tokTypes.eof);
  535. // Consume the final eof token
  536. this.next();
  537. const extra = this[STATE];
  538. const tokens = extra.tokens;
  539. if (extra.comments) {
  540. tokens.comments = extra.comments;
  541. }
  542. return tokens;
  543. }
  544. finishNode(...args) {
  545. const result = super.finishNode(...args);
  546. return this[ESPRIMA_FINISH_NODE](result);
  547. }
  548. finishNodeAt(...args) {
  549. const result = super.finishNodeAt(...args);
  550. return this[ESPRIMA_FINISH_NODE](result);
  551. }
  552. parse() {
  553. const extra = this[STATE];
  554. const program = super.parse();
  555. program.sourceType = extra.originalSourceType;
  556. if (extra.comments) {
  557. program.comments = extra.comments;
  558. }
  559. if (extra.tokens) {
  560. program.tokens = extra.tokens;
  561. }
  562. /*
  563. * Adjust opening and closing position of program to match Esprima.
  564. * Acorn always starts programs at range 0 whereas Esprima starts at the
  565. * first AST node's start (the only real difference is when there's leading
  566. * whitespace or leading comments). Acorn also counts trailing whitespace
  567. * as part of the program whereas Esprima only counts up to the last token.
  568. */
  569. if (program.body.length) {
  570. const [firstNode] = program.body;
  571. if (program.range) {
  572. program.range[0] = firstNode.range[0];
  573. }
  574. if (program.loc) {
  575. program.loc.start = firstNode.loc.start;
  576. }
  577. program.start = firstNode.start;
  578. }
  579. if (extra.lastToken) {
  580. if (program.range) {
  581. program.range[1] = extra.lastToken.range[1];
  582. }
  583. if (program.loc) {
  584. program.loc.end = extra.lastToken.loc.end;
  585. }
  586. program.end = extra.lastToken.end;
  587. }
  588. /*
  589. * https://github.com/eslint/espree/issues/349
  590. * Ensure that template elements have correct range information.
  591. * This is one location where Acorn produces a different value
  592. * for its start and end properties vs. the values present in the
  593. * range property. In order to avoid confusion, we set the start
  594. * and end properties to the values that are present in range.
  595. * This is done here, instead of in finishNode(), because Acorn
  596. * uses the values of start and end internally while parsing, making
  597. * it dangerous to change those values while parsing is ongoing.
  598. * By waiting until the end of parsing, we can safely change these
  599. * values without affect any other part of the process.
  600. */
  601. this[STATE].templateElements.forEach(templateElement => {
  602. const startOffset = -1;
  603. const endOffset = templateElement.tail ? 1 : 2;
  604. templateElement.start += startOffset;
  605. templateElement.end += endOffset;
  606. if (templateElement.range) {
  607. templateElement.range[0] += startOffset;
  608. templateElement.range[1] += endOffset;
  609. }
  610. if (templateElement.loc) {
  611. templateElement.loc.start.column += startOffset;
  612. templateElement.loc.end.column += endOffset;
  613. }
  614. });
  615. return program;
  616. }
  617. parseTopLevel(node) {
  618. if (this[STATE].impliedStrict) {
  619. this.strict = true;
  620. }
  621. return super.parseTopLevel(node);
  622. }
  623. /**
  624. * Overwrites the default raise method to throw Esprima-style errors.
  625. * @param {int} pos The position of the error.
  626. * @param {string} message The error message.
  627. * @throws {SyntaxError} A syntax error.
  628. * @returns {void}
  629. */
  630. raise(pos, message) {
  631. const loc = Parser.acorn.getLineInfo(this.input, pos);
  632. const err = new SyntaxError(message);
  633. err.index = pos;
  634. err.lineNumber = loc.line;
  635. err.column = loc.column + 1; // acorn uses 0-based columns
  636. throw err;
  637. }
  638. /**
  639. * Overwrites the default raise method to throw Esprima-style errors.
  640. * @param {int} pos The position of the error.
  641. * @param {string} message The error message.
  642. * @throws {SyntaxError} A syntax error.
  643. * @returns {void}
  644. */
  645. raiseRecoverable(pos, message) {
  646. this.raise(pos, message);
  647. }
  648. /**
  649. * Overwrites the default unexpected method to throw Esprima-style errors.
  650. * @param {int} pos The position of the error.
  651. * @throws {SyntaxError} A syntax error.
  652. * @returns {void}
  653. */
  654. unexpected(pos) {
  655. let message = "Unexpected token";
  656. if (pos !== null && pos !== void 0) {
  657. this.pos = pos;
  658. if (this.options.locations) {
  659. while (this.pos < this.lineStart) {
  660. this.lineStart = this.input.lastIndexOf("\n", this.lineStart - 2) + 1;
  661. --this.curLine;
  662. }
  663. }
  664. this.nextToken();
  665. }
  666. if (this.end > this.start) {
  667. message += ` ${this.input.slice(this.start, this.end)}`;
  668. }
  669. this.raise(this.start, message);
  670. }
  671. /*
  672. * Esprima-FB represents JSX strings as tokens called "JSXText", but Acorn-JSX
  673. * uses regular tt.string without any distinction between this and regular JS
  674. * strings. As such, we intercept an attempt to read a JSX string and set a flag
  675. * on extra so that when tokens are converted, the next token will be switched
  676. * to JSXText via onToken.
  677. */
  678. jsx_readString(quote) { // eslint-disable-line camelcase
  679. const result = super.jsx_readString(quote);
  680. if (this.type === tokTypes.string) {
  681. this[STATE].jsxAttrValueToken = true;
  682. }
  683. return result;
  684. }
  685. /**
  686. * Performs last-minute Esprima-specific compatibility checks and fixes.
  687. * @param {ASTNode} result The node to check.
  688. * @returns {ASTNode} The finished node.
  689. */
  690. [ESPRIMA_FINISH_NODE](result) {
  691. // Acorn doesn't count the opening and closing backticks as part of templates
  692. // so we have to adjust ranges/locations appropriately.
  693. if (result.type === "TemplateElement") {
  694. // save template element references to fix start/end later
  695. this[STATE].templateElements.push(result);
  696. }
  697. if (result.type.includes("Function") && !result.generator) {
  698. result.generator = false;
  699. }
  700. return result;
  701. }
  702. };
  703. };
  704. const version$1 = "9.3.0";
  705. /**
  706. * @fileoverview Main Espree file that converts Acorn into Esprima output.
  707. *
  708. * This file contains code from the following MIT-licensed projects:
  709. * 1. Acorn
  710. * 2. Babylon
  711. * 3. Babel-ESLint
  712. *
  713. * This file also contains code from Esprima, which is BSD licensed.
  714. *
  715. * Acorn is Copyright 2012-2015 Acorn Contributors (https://github.com/marijnh/acorn/blob/master/AUTHORS)
  716. * Babylon is Copyright 2014-2015 various contributors (https://github.com/babel/babel/blob/master/packages/babylon/AUTHORS)
  717. * Babel-ESLint is Copyright 2014-2015 Sebastian McKenzie <sebmck@gmail.com>
  718. *
  719. * Redistribution and use in source and binary forms, with or without
  720. * modification, are permitted provided that the following conditions are met:
  721. *
  722. * * Redistributions of source code must retain the above copyright
  723. * notice, this list of conditions and the following disclaimer.
  724. * * Redistributions in binary form must reproduce the above copyright
  725. * notice, this list of conditions and the following disclaimer in the
  726. * documentation and/or other materials provided with the distribution.
  727. *
  728. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  729. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  730. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  731. * ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  732. * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  733. * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  734. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  735. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  736. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  737. * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  738. *
  739. * Esprima is Copyright (c) jQuery Foundation, Inc. and Contributors, All Rights Reserved.
  740. *
  741. * Redistribution and use in source and binary forms, with or without
  742. * modification, are permitted provided that the following conditions are met:
  743. *
  744. * * Redistributions of source code must retain the above copyright
  745. * notice, this list of conditions and the following disclaimer.
  746. * * Redistributions in binary form must reproduce the above copyright
  747. * notice, this list of conditions and the following disclaimer in the
  748. * documentation and/or other materials provided with the distribution.
  749. *
  750. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  751. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  752. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  753. * ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  754. * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  755. * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  756. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  757. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  758. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  759. * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  760. */
  761. // To initialize lazily.
  762. const parsers = {
  763. _regular: null,
  764. _jsx: null,
  765. get regular() {
  766. if (this._regular === null) {
  767. this._regular = acorn__namespace.Parser.extend(espree());
  768. }
  769. return this._regular;
  770. },
  771. get jsx() {
  772. if (this._jsx === null) {
  773. this._jsx = acorn__namespace.Parser.extend(jsx__default["default"](), espree());
  774. }
  775. return this._jsx;
  776. },
  777. get(options) {
  778. const useJsx = Boolean(
  779. options &&
  780. options.ecmaFeatures &&
  781. options.ecmaFeatures.jsx
  782. );
  783. return useJsx ? this.jsx : this.regular;
  784. }
  785. };
  786. //------------------------------------------------------------------------------
  787. // Tokenizer
  788. //------------------------------------------------------------------------------
  789. /**
  790. * Tokenizes the given code.
  791. * @param {string} code The code to tokenize.
  792. * @param {Object} options Options defining how to tokenize.
  793. * @returns {Token[]} An array of tokens.
  794. * @throws {SyntaxError} If the input code is invalid.
  795. * @private
  796. */
  797. function tokenize(code, options) {
  798. const Parser = parsers.get(options);
  799. // Ensure to collect tokens.
  800. if (!options || options.tokens !== true) {
  801. options = Object.assign({}, options, { tokens: true }); // eslint-disable-line no-param-reassign
  802. }
  803. return new Parser(options, code).tokenize();
  804. }
  805. //------------------------------------------------------------------------------
  806. // Parser
  807. //------------------------------------------------------------------------------
  808. /**
  809. * Parses the given code.
  810. * @param {string} code The code to tokenize.
  811. * @param {Object} options Options defining how to tokenize.
  812. * @returns {ASTNode} The "Program" AST node.
  813. * @throws {SyntaxError} If the input code is invalid.
  814. */
  815. function parse(code, options) {
  816. const Parser = parsers.get(options);
  817. return new Parser(options, code).parse();
  818. }
  819. //------------------------------------------------------------------------------
  820. // Public
  821. //------------------------------------------------------------------------------
  822. const version = version$1;
  823. // Deep copy.
  824. /* istanbul ignore next */
  825. const Syntax = (function() {
  826. let name,
  827. types = {};
  828. if (typeof Object.create === "function") {
  829. types = Object.create(null);
  830. }
  831. for (name in astNodeTypes) {
  832. if (Object.hasOwnProperty.call(astNodeTypes, name)) {
  833. types[name] = astNodeTypes[name];
  834. }
  835. }
  836. if (typeof Object.freeze === "function") {
  837. Object.freeze(types);
  838. }
  839. return types;
  840. }());
  841. /* istanbul ignore next */
  842. const VisitorKeys = (function() {
  843. return visitorKeys__namespace.KEYS;
  844. }());
  845. const latestEcmaVersion = getLatestEcmaVersion();
  846. const supportedEcmaVersions = getSupportedEcmaVersions();
  847. exports.Syntax = Syntax;
  848. exports.VisitorKeys = VisitorKeys;
  849. exports.latestEcmaVersion = latestEcmaVersion;
  850. exports.parse = parse;
  851. exports.supportedEcmaVersions = supportedEcmaVersions;
  852. exports.tokenize = tokenize;
  853. exports.version = version;
  854. //# sourceMappingURL=espree.cjs.map