typescript-estree
Version:
A parser that converts TypeScript source code into an ESTree compatible form
275 lines (274 loc) • 11.1 kB
JavaScript
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
/**
* @fileoverview Parser that converts TypeScript into ESTree format.
* @author Nicholas C. Zakas
* @author James Henry <https://github.com/JamesHenry>
* @copyright jQuery Foundation and other contributors, https://jquery.org/
* MIT License
*/
const tsconfig_parser_1 = require("./tsconfig-parser");
const semver_1 = __importDefault(require("semver"));
const typescript_1 = __importDefault(require("typescript"));
const ast_converter_1 = __importDefault(require("./ast-converter"));
const convert_1 = require("./convert");
const node_utils_1 = require("./node-utils");
const semantic_errors_1 = require("./semantic-errors");
const packageJSON = require('../package.json');
const SUPPORTED_TYPESCRIPT_VERSIONS = packageJSON.devDependencies.typescript;
const ACTIVE_TYPESCRIPT_VERSION = typescript_1.default.version;
const isRunningSupportedTypeScriptVersion = semver_1.default.satisfies(ACTIVE_TYPESCRIPT_VERSION, SUPPORTED_TYPESCRIPT_VERSIONS);
let extra;
let warnedAboutTSVersion = false;
/**
* Compute the filename based on the parser options
*
* @param options Parser options
*/
function getFileName({ jsx }) {
return jsx ? 'estree.tsx' : 'estree.ts';
}
/**
* Resets the extra config object
* @returns {void}
*/
function resetExtra() {
extra = {
tokens: null,
range: false,
loc: false,
comment: false,
comments: [],
strict: false,
jsx: false,
useJSXTextNode: false,
log: console.log,
projects: [],
errorOnUnknownASTType: false,
errorOnTypeScriptSyntacticAndSemanticIssues: false,
code: '',
tsconfigRootDir: process.cwd(),
extraFileExtensions: []
};
}
/**
* @param {string} code The code of the file being linted
* @param {Object} options The config object
* @returns {{ast: ts.SourceFile, program: ts.Program} | undefined} If found, returns the source file corresponding to the code and the containing program
*/
function getASTFromProject(code, options) {
return node_utils_1.firstDefined(tsconfig_parser_1.calculateProjectParserOptions(code, options.filePath || getFileName(options), extra), currentProgram => {
const ast = currentProgram.getSourceFile(options.filePath || getFileName(options));
return ast && { ast, program: currentProgram };
});
}
/**
* @param {string} code The code of the file being linted
* @param {Object} options The config object
* @returns {{ast: ts.SourceFile, program: ts.Program} | undefined} If found, returns the source file corresponding to the code and the containing program
*/
function getASTAndDefaultProject(code, options) {
const fileName = options.filePath || getFileName(options);
const program = tsconfig_parser_1.createProgram(code, fileName, extra);
const ast = program && program.getSourceFile(fileName);
return ast && { ast, program };
}
/**
* @param {string} code The code of the file being linted
* @returns {{ast: ts.SourceFile, program: ts.Program}} Returns a new source file and program corresponding to the linted code
*/
function createNewProgram(code) {
// Even if jsx option is set in typescript compiler, filename still has to
// contain .tsx file extension
const FILENAME = getFileName(extra);
const compilerHost = {
fileExists() {
return true;
},
getCanonicalFileName() {
return FILENAME;
},
getCurrentDirectory() {
return '';
},
getDirectories() {
return [];
},
getDefaultLibFileName() {
return 'lib.d.ts';
},
// TODO: Support Windows CRLF
getNewLine() {
return '\n';
},
getSourceFile(filename) {
return typescript_1.default.createSourceFile(filename, code, typescript_1.default.ScriptTarget.Latest, true);
},
readFile() {
return undefined;
},
useCaseSensitiveFileNames() {
return true;
},
writeFile() {
return null;
}
};
const program = typescript_1.default.createProgram([FILENAME], {
noResolve: true,
target: typescript_1.default.ScriptTarget.Latest,
jsx: extra.jsx ? typescript_1.default.JsxEmit.Preserve : undefined
}, compilerHost);
const ast = program.getSourceFile(FILENAME);
return { ast, program };
}
/**
* @param {string} code The code of the file being linted
* @param {Object} options The config object
* @param {boolean} shouldProvideParserServices True iff the program should be attempted to be calculated from provided tsconfig files
* @returns {{ast: ts.SourceFile, program: ts.Program}} Returns a source file and program corresponding to the linted code
*/
function getProgramAndAST(code, options, shouldProvideParserServices) {
return ((shouldProvideParserServices && getASTFromProject(code, options)) ||
(shouldProvideParserServices && getASTAndDefaultProject(code, options)) ||
createNewProgram(code));
}
/**
* Parses the given source code to produce a valid AST
* @param {string} code TypeScript code
* @param {boolean} shouldGenerateServices Flag determining whether to generate ast maps and program or not
* @param {ParserOptions} options configuration object for the parser
* @returns {Object} the AST
*/
function generateAST(code, options = {}, shouldGenerateServices = false) {
const toString = String;
if (typeof code !== 'string' && !(code instanceof String)) {
code = toString(code);
}
resetExtra();
if (typeof options !== 'undefined') {
extra.range = typeof options.range === 'boolean' && options.range;
extra.loc = typeof options.loc === 'boolean' && options.loc;
if (typeof options.tokens === 'boolean' && options.tokens) {
extra.tokens = [];
}
if (typeof options.comment === 'boolean' && options.comment) {
extra.comment = true;
extra.comments = [];
}
if (typeof options.jsx === 'boolean' && options.jsx) {
extra.jsx = true;
}
/**
* Allow the user to cause the parser to error if it encounters an unknown AST Node Type
* (used in testing).
*/
if (typeof options.errorOnUnknownASTType === 'boolean' &&
options.errorOnUnknownASTType) {
extra.errorOnUnknownASTType = true;
}
/**
* Retrieve semantic and syntactic diagnostics from the underlying TypeScript Program
* and turn them into parse errors
*/
if (shouldGenerateServices &&
typeof options.errorOnTypeScriptSyntacticAndSemanticIssues ===
'boolean' &&
options.errorOnTypeScriptSyntacticAndSemanticIssues) {
extra.errorOnTypeScriptSyntacticAndSemanticIssues = true;
}
if (typeof options.useJSXTextNode === 'boolean' && options.useJSXTextNode) {
extra.useJSXTextNode = true;
}
/**
* Allow the user to override the function used for logging
*/
if (typeof options.loggerFn === 'function') {
extra.log = options.loggerFn;
}
else if (options.loggerFn === false) {
extra.log = Function.prototype;
}
if (typeof options.project === 'string') {
extra.projects = [options.project];
}
else if (Array.isArray(options.project) &&
options.project.every(projectPath => typeof projectPath === 'string')) {
extra.projects = options.project;
}
if (typeof options.tsconfigRootDir === 'string') {
extra.tsconfigRootDir = options.tsconfigRootDir;
}
if (Array.isArray(options.extraFileExtensions) &&
options.extraFileExtensions.every(ext => typeof ext === 'string')) {
extra.extraFileExtensions = options.extraFileExtensions;
}
}
if (!isRunningSupportedTypeScriptVersion && !warnedAboutTSVersion) {
const border = '=============';
const versionWarning = [
border,
'WARNING: You are currently running a version of TypeScript which is not officially supported by typescript-estree.',
'You may find that it works just fine, or you may not.',
`SUPPORTED TYPESCRIPT VERSIONS: ${SUPPORTED_TYPESCRIPT_VERSIONS}`,
`YOUR TYPESCRIPT VERSION: ${ACTIVE_TYPESCRIPT_VERSION}`,
'Please only submit bug reports when using the officially supported version.',
border
];
extra.log(versionWarning.join('\n\n'));
warnedAboutTSVersion = true;
}
const shouldProvideParserServices = shouldGenerateServices && extra.projects && extra.projects.length > 0;
const { ast, program } = getProgramAndAST(code, options, shouldProvideParserServices);
extra.code = code;
/**
* Convert the AST
*/
const { estree, astMaps } = ast_converter_1.default(ast, extra, shouldProvideParserServices);
/**
* Even if TypeScript parsed the source code ok, and we had no problems converting the AST,
* there may be other syntactic or semantic issues in the code that we can optionally report on.
*/
if (program && extra.errorOnTypeScriptSyntacticAndSemanticIssues) {
const error = semantic_errors_1.getFirstSemanticOrSyntacticError(program, ast);
if (error) {
throw convert_1.convertError(error);
}
}
return {
estree,
program: shouldProvideParserServices ? program : undefined,
astMaps: shouldProvideParserServices
? astMaps
: { esTreeNodeToTSNodeMap: undefined, tsNodeToESTreeNodeMap: undefined }
};
}
//------------------------------------------------------------------------------
// Public
//------------------------------------------------------------------------------
var ast_node_types_1 = require("./ast-node-types");
exports.AST_NODE_TYPES = ast_node_types_1.AST_NODE_TYPES;
const version = packageJSON.version;
exports.version = version;
function parse(code, options) {
if (options && options.errorOnTypeScriptSyntacticAndSemanticIssues) {
throw new Error(`"errorOnTypeScriptSyntacticAndSemanticIssues" is only supported for parseAndGenerateServices()`);
}
return generateAST(code, options).estree;
}
exports.parse = parse;
function parseAndGenerateServices(code, options) {
const result = generateAST(code, options, /*shouldGenerateServices*/ true);
return {
ast: result.estree,
services: {
program: result.program,
esTreeNodeToTSNodeMap: result.astMaps.esTreeNodeToTSNodeMap,
tsNodeToESTreeNodeMap: result.astMaps.tsNodeToESTreeNodeMap
}
};
}
exports.parseAndGenerateServices = parseAndGenerateServices;
;