UNPKG

typescript-to-lua

Version:

A generic TypeScript to Lua transpiler. Write your code in TypeScript and publish Lua!

249 lines 9.65 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.optionDeclarations = void 0; exports.updateParsedConfigFile = updateParsedConfigFile; exports.parseCommandLine = parseCommandLine; const ts = require("typescript"); const CompilerOptions_1 = require("../CompilerOptions"); const cliDiagnostics = require("./diagnostics"); exports.optionDeclarations = [ { name: "buildMode", description: "'default' or 'library'. Compiling as library will not resolve external dependencies.", type: "enum", choices: Object.values(CompilerOptions_1.BuildMode), }, { name: "extension", description: 'File extension for the resulting Lua files. Defaults to ".lua"', type: "string", }, { name: "luaBundle", description: "The name of the lua file to bundle output lua to. Requires luaBundleEntry.", type: "string", }, { name: "luaBundleEntry", description: "The entry *.ts file that will be executed when entering the luaBundle. Requires luaBundle.", type: "string", }, { name: "luaLibImport", description: "Specifies how js standard features missing in lua are imported.", type: "enum", choices: Object.values(CompilerOptions_1.LuaLibImportKind), }, { name: "luaTarget", aliases: ["lt"], description: "Specify Lua target version.", type: "enum", choices: Object.values(CompilerOptions_1.LuaTarget), }, { name: "noImplicitGlobalVariables", description: 'Specify to prevent implicitly turning "normal" variants into global variables in the transpiled output.', type: "boolean", }, { name: "noImplicitSelf", description: 'If "this" is implicitly considered an any type, do not generate a self parameter.', type: "boolean", }, { name: "noHeader", description: "Specify if a header will be added to compiled files.", type: "boolean", }, { name: "sourceMapTraceback", description: "Applies the source map to show source TS files and lines in error tracebacks.", type: "boolean", }, { name: "luaPlugins", description: "List of TypeScriptToLua plugins.", type: "json-array-of-objects", }, { name: "tstlVerbose", description: "Provide verbose output useful for diagnosing problems.", type: "boolean", }, { name: "noResolvePaths", description: "An array of paths that tstl should not resolve and keep as-is.", type: "array", }, { name: "lua51AllowTryCatchInAsyncAwait", description: "Always allow try/catch in async/await functions for Lua 5.1.", type: "boolean", }, { name: "measurePerformance", description: "Measure performance of the tstl compiler.", type: "boolean", }, ]; function updateParsedConfigFile(parsedConfigFile) { let hasRootLevelOptions = false; for (const [name, rawValue] of Object.entries(parsedConfigFile.raw)) { const option = exports.optionDeclarations.find(option => option.name === name); if (!option) continue; if (parsedConfigFile.raw.tstl === undefined) parsedConfigFile.raw.tstl = {}; parsedConfigFile.raw.tstl[name] = rawValue; hasRootLevelOptions = true; } if (parsedConfigFile.raw.tstl) { if (hasRootLevelOptions) { parsedConfigFile.errors.push(cliDiagnostics.tstlOptionsAreMovingToTheTstlObject(parsedConfigFile.raw.tstl)); } for (const [name, rawValue] of Object.entries(parsedConfigFile.raw.tstl)) { const option = exports.optionDeclarations.find(option => option.name === name); if (!option) { parsedConfigFile.errors.push(cliDiagnostics.unknownCompilerOption(name)); continue; } const { error, value } = readValue(option, rawValue, OptionSource.TsConfig); if (error) parsedConfigFile.errors.push(error); if (parsedConfigFile.options[name] === undefined) parsedConfigFile.options[name] = value; } } return parsedConfigFile; } function parseCommandLine(args) { return updateParsedCommandLine(ts.parseCommandLine(args), args); } function updateParsedCommandLine(parsedCommandLine, args) { for (let i = 0; i < args.length; i++) { if (!args[i].startsWith("-")) continue; const isShorthand = !args[i].startsWith("--"); const argumentName = args[i].substring(isShorthand ? 1 : 2); const option = exports.optionDeclarations.find(option => { if (option.name.toLowerCase() === argumentName.toLowerCase()) return true; if (isShorthand && option.aliases) { return option.aliases.some(a => a.toLowerCase() === argumentName.toLowerCase()); } return false; }); if (option) { // Ignore errors caused by tstl specific compiler options parsedCommandLine.errors = parsedCommandLine.errors.filter( // TS5023: Unknown compiler option '{0}'. // TS5025: Unknown compiler option '{0}'. Did you mean '{1}'? e => !((e.code === 5023 || e.code === 5025) && String(e.messageText).includes(`'${args[i]}'.`))); const { error, value, consumed } = readCommandLineArgument(option, args[i + 1]); if (error) parsedCommandLine.errors.push(error); parsedCommandLine.options[option.name] = value; if (consumed) { // Values of custom options are parsed as a file name, exclude them parsedCommandLine.fileNames = parsedCommandLine.fileNames.filter(f => f !== args[i + 1]); i += 1; } } } return parsedCommandLine; } function readCommandLineArgument(option, value) { if (option.type === "boolean") { if (value === "true" || value === "false") { value = value === "true"; } else { // Set boolean arguments without supplied value to true return { value: true, consumed: false }; } } if (value === undefined) { return { error: cliDiagnostics.compilerOptionExpectsAnArgument(option.name), value: undefined, consumed: false, }; } return { ...readValue(option, value, OptionSource.CommandLine), consumed: true }; } var OptionSource; (function (OptionSource) { OptionSource[OptionSource["CommandLine"] = 0] = "CommandLine"; OptionSource[OptionSource["TsConfig"] = 1] = "TsConfig"; })(OptionSource || (OptionSource = {})); function readValue(option, value, source) { if (value === null) return { value }; switch (option.type) { case "boolean": case "string": { if (typeof value !== option.type) { return { value: undefined, error: cliDiagnostics.compilerOptionRequiresAValueOfType(option.name, option.type), }; } return { value }; } case "array": case "json-array-of-objects": { const isInvalidNonCliValue = source === OptionSource.TsConfig && !Array.isArray(value); const isInvalidCliValue = source === OptionSource.CommandLine && typeof value !== "string"; if (isInvalidNonCliValue || isInvalidCliValue) { return { value: undefined, error: cliDiagnostics.compilerOptionRequiresAValueOfType(option.name, option.type), }; } const shouldParseValue = source === OptionSource.CommandLine && typeof value === "string"; if (!shouldParseValue) return { value }; if (option.type === "array") { const array = value.split(","); return { value: array }; } try { const objects = JSON.parse(value); if (!Array.isArray(objects)) { return { value: undefined, error: cliDiagnostics.compilerOptionRequiresAValueOfType(option.name, option.type), }; } return { value: objects }; } catch (e) { if (!(e instanceof SyntaxError)) throw e; return { value: undefined, error: cliDiagnostics.compilerOptionCouldNotParseJson(option.name, e.message), }; } } case "enum": { if (typeof value !== "string") { return { value: undefined, error: cliDiagnostics.compilerOptionRequiresAValueOfType(option.name, "string"), }; } const enumValue = option.choices.find(c => c.toLowerCase() === value.toLowerCase()); if (enumValue === undefined) { const optionChoices = option.choices.join(", "); return { value: undefined, error: cliDiagnostics.argumentForOptionMustBe(`--${option.name}`, optionChoices), }; } return { value: enumValue }; } } } //# sourceMappingURL=parse.js.map