espree.js 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. /**
  2. * @fileoverview Main Espree file that converts Acorn into Esprima output.
  3. *
  4. * This file contains code from the following MIT-licensed projects:
  5. * 1. Acorn
  6. * 2. Babylon
  7. * 3. Babel-ESLint
  8. *
  9. * This file also contains code from Esprima, which is BSD licensed.
  10. *
  11. * Acorn is Copyright 2012-2015 Acorn Contributors (https://github.com/marijnh/acorn/blob/master/AUTHORS)
  12. * Babylon is Copyright 2014-2015 various contributors (https://github.com/babel/babel/blob/master/packages/babylon/AUTHORS)
  13. * Babel-ESLint is Copyright 2014-2015 Sebastian McKenzie <sebmck@gmail.com>
  14. *
  15. * Redistribution and use in source and binary forms, with or without
  16. * modification, are permitted provided that the following conditions are met:
  17. *
  18. * * Redistributions of source code must retain the above copyright
  19. * notice, this list of conditions and the following disclaimer.
  20. * * Redistributions in binary form must reproduce the above copyright
  21. * notice, this list of conditions and the following disclaimer in the
  22. * documentation and/or other materials provided with the distribution.
  23. *
  24. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  25. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  26. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  27. * ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  28. * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  29. * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  30. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  31. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  32. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  33. * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  34. *
  35. * Esprima is Copyright (c) jQuery Foundation, Inc. and Contributors, All Rights Reserved.
  36. *
  37. * Redistribution and use in source and binary forms, with or without
  38. * modification, are permitted provided that the following conditions are met:
  39. *
  40. * * Redistributions of source code must retain the above copyright
  41. * notice, this list of conditions and the following disclaimer.
  42. * * Redistributions in binary form must reproduce the above copyright
  43. * notice, this list of conditions and the following disclaimer in the
  44. * documentation and/or other materials provided with the distribution.
  45. *
  46. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  47. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  48. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  49. * ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
  50. * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  51. * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  52. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  53. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  54. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  55. * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  56. */
  57. /* eslint no-undefined:0, no-use-before-define: 0 */
  58. import * as acorn from "acorn";
  59. import jsx from "acorn-jsx";
  60. import astNodeTypes from "./lib/ast-node-types.js";
  61. import espree from "./lib/espree.js";
  62. import espreeVersion from "./lib/version.js";
  63. import * as visitorKeys from "eslint-visitor-keys";
  64. import { getLatestEcmaVersion, getSupportedEcmaVersions } from "./lib/options.js";
  65. // To initialize lazily.
  66. const parsers = {
  67. _regular: null,
  68. _jsx: null,
  69. get regular() {
  70. if (this._regular === null) {
  71. this._regular = acorn.Parser.extend(espree());
  72. }
  73. return this._regular;
  74. },
  75. get jsx() {
  76. if (this._jsx === null) {
  77. this._jsx = acorn.Parser.extend(jsx(), espree());
  78. }
  79. return this._jsx;
  80. },
  81. get(options) {
  82. const useJsx = Boolean(
  83. options &&
  84. options.ecmaFeatures &&
  85. options.ecmaFeatures.jsx
  86. );
  87. return useJsx ? this.jsx : this.regular;
  88. }
  89. };
  90. //------------------------------------------------------------------------------
  91. // Tokenizer
  92. //------------------------------------------------------------------------------
  93. /**
  94. * Tokenizes the given code.
  95. * @param {string} code The code to tokenize.
  96. * @param {Object} options Options defining how to tokenize.
  97. * @returns {Token[]} An array of tokens.
  98. * @throws {SyntaxError} If the input code is invalid.
  99. * @private
  100. */
  101. export function tokenize(code, options) {
  102. const Parser = parsers.get(options);
  103. // Ensure to collect tokens.
  104. if (!options || options.tokens !== true) {
  105. options = Object.assign({}, options, { tokens: true }); // eslint-disable-line no-param-reassign
  106. }
  107. return new Parser(options, code).tokenize();
  108. }
  109. //------------------------------------------------------------------------------
  110. // Parser
  111. //------------------------------------------------------------------------------
  112. /**
  113. * Parses the given code.
  114. * @param {string} code The code to tokenize.
  115. * @param {Object} options Options defining how to tokenize.
  116. * @returns {ASTNode} The "Program" AST node.
  117. * @throws {SyntaxError} If the input code is invalid.
  118. */
  119. export function parse(code, options) {
  120. const Parser = parsers.get(options);
  121. return new Parser(options, code).parse();
  122. }
  123. //------------------------------------------------------------------------------
  124. // Public
  125. //------------------------------------------------------------------------------
  126. export const version = espreeVersion;
  127. // Deep copy.
  128. /* istanbul ignore next */
  129. export const Syntax = (function() {
  130. let name,
  131. types = {};
  132. if (typeof Object.create === "function") {
  133. types = Object.create(null);
  134. }
  135. for (name in astNodeTypes) {
  136. if (Object.hasOwnProperty.call(astNodeTypes, name)) {
  137. types[name] = astNodeTypes[name];
  138. }
  139. }
  140. if (typeof Object.freeze === "function") {
  141. Object.freeze(types);
  142. }
  143. return types;
  144. }());
  145. /* istanbul ignore next */
  146. export const VisitorKeys = (function() {
  147. return visitorKeys.KEYS;
  148. }());
  149. export const latestEcmaVersion = getLatestEcmaVersion();
  150. export const supportedEcmaVersions = getSupportedEcmaVersions();