/* * Copyright (c) 2022 Huawei Device Co., Ltd. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import * as path from "path"; import * as ts from "typescript"; import * as fs from "fs"; import { CmdOptions } from "./cmdOptions"; import { CompilerDriver } from "./compilerDriver"; import * as diag from "./diagnostic"; import * as jshelpers from "./jshelpers"; import { LOGE } from "./log"; import { setGlobalDeclare, setGlobalStrict } from "./strictMode"; import { TypeChecker } from "./typeChecker"; import { setPos, isBase64Str } from "./base/util"; function checkIsGlobalDeclaration(sourceFile: ts.SourceFile) { for (let statement of sourceFile.statements) { if (statement.modifiers) { for (let modifier of statement.modifiers) { if (modifier.kind === ts.SyntaxKind.ExportKeyword) { return false; } } } else if (statement.kind === ts.SyntaxKind.ExportAssignment) { return false; } else if (statement.kind === ts.SyntaxKind.ImportKeyword || statement.kind === ts.SyntaxKind.ImportDeclaration) { return false; } } return true; } function generateDTs(node: ts.SourceFile, options: ts.CompilerOptions) { let outputBinName = getOutputBinName(node); let compilerDriver = new CompilerDriver(outputBinName); setGlobalStrict(jshelpers.isEffectiveStrictModeSourceFile(node, options)); compilerDriver.compile(node); compilerDriver.showStatistics(); } function main(fileNames: string[], options: ts.CompilerOptions) { const host = ts.createCompilerHost(options); if (!CmdOptions.needGenerateTmpFile()) { host.writeFile = () => {}; } let program = ts.createProgram(fileNames, options, host); let typeChecker = TypeChecker.getInstance(); typeChecker.setTypeChecker(program.getTypeChecker()); if (CmdOptions.needRecordDtsType()) { for (let sourceFile of program.getSourceFiles()) { if (sourceFile.isDeclarationFile && !program.isSourceFileDefaultLibrary(sourceFile)) { setGlobalDeclare(checkIsGlobalDeclaration(sourceFile)); generateDTs(sourceFile, options); } } } let emitResult = program.emit( undefined, undefined, undefined, undefined, { before: [ // @ts-ignore (ctx: ts.TransformationContext) => { return (node: ts.SourceFile) => { let outputBinName = getOutputBinName(node); let compilerDriver = new CompilerDriver(outputBinName); compilerDriver.compileForSyntaxCheck(node); return node; } } ], after: [ // @ts-ignore (ctx: ts.TransformationContext) => { return (node: ts.SourceFile) => { if (ts.getEmitHelpers(node)) { let newStatements = []; ts.getEmitHelpers(node)?.forEach( item => { let emitHelperSourceFile = ts.createSourceFile(node.fileName, item.text, options.target!, true, ts.ScriptKind.JS); emitHelperSourceFile.statements.forEach(emitStatement => { let emitNode = setPos(emitStatement); newStatements.push(emitNode); }); } ) newStatements.push(...node.statements); node = ts.factory.updateSourceFile(node, newStatements); } let outputBinName = getOutputBinName(node); let compilerDriver = new CompilerDriver(outputBinName); setGlobalStrict(jshelpers.isEffectiveStrictModeSourceFile(node, options)); compilerDriver.compile(node); compilerDriver.showStatistics(); return node; } } ] } ); let allDiagnostics = ts .getPreEmitDiagnostics(program) .concat(emitResult.diagnostics); allDiagnostics.forEach(diagnostic => { diag.printDiagnostic(diagnostic); }); } function getOutputBinName(node: ts.SourceFile) { let outputBinName = CmdOptions.getOutputBinName(); let fileName = node.fileName.substring(0, node.fileName.lastIndexOf('.')); let inputFileName = CmdOptions.getInputFileName(); if (/^win/.test(require('os').platform())) { var inputFileTmps = inputFileName.split(path.sep); inputFileName = path.posix.join(...inputFileTmps); } if (fileName != inputFileName) { outputBinName = fileName + ".abc"; } return outputBinName; } function getDtsFiles(libDir: string): string[] { let dtsFiles:string[] = []; function finDtsFile(dir){ let files = fs.readdirSync(dir); files.forEach(function (item, _) { let fPath = path.join(dir,item); let stat = fs.statSync(fPath); if(stat.isDirectory() === true) { finDtsFile(fPath); } if (stat.isFile() === true && item.endsWith(".d.ts") === true) { dtsFiles.push(fPath); } }); } finDtsFile(libDir); return dtsFiles; } const stopWatchingStr = "####"; const watchAbcFileDefaultTimeOut = 10; const watchFileName = "watch_expressions"; function updateWatchJsFile() { let ideIputStr = CmdOptions.getEvaluateExpression(); if (!isBase64Str(ideIputStr)) { throw new Error("Passed expression string for evaluating is not base64 style."); } let watchAbcFileTimeOut = watchAbcFileDefaultTimeOut; if (CmdOptions.getWatchTimeOutValue() != 0) { watchAbcFileTimeOut = CmdOptions.getWatchTimeOutValue(); } let watchFilePrefix = CmdOptions.getWatchJsPath() + path.sep + watchFileName; let originExpre = Buffer.from(ideIputStr, 'base64').toString(); let jsFileName = watchFilePrefix + ".js"; let abcFileName = watchFilePrefix + ".abc"; let errorMsgFileName = watchFilePrefix + ".err"; fs.watchFile(errorMsgFileName, { persistent: true, interval: 50 }, (curr, prev) => { if (+curr.mtime <= +prev.mtime) { fs.unwatchFile(jsFileName); fs.unwatchFile(abcFileName); throw new Error("watched errMsg file has not been initialized"); } console.log("error in genarate abc file for this expression."); fs.unwatchFile(abcFileName); fs.unwatchFile(errorMsgFileName); process.exit(); }); fs.watchFile(abcFileName, { persistent: true, interval: 50 }, (curr, prev) => { if (+curr.mtime <= +prev.mtime) { fs.unwatchFile(jsFileName); fs.unwatchFile(errorMsgFileName); throw new Error("watched abc file has not been initialized"); } let base64data = fs.readFileSync(abcFileName); let watchResStr = Buffer.from(base64data).toString('base64'); console.log(watchResStr); fs.unwatchFile(abcFileName); fs.unwatchFile(errorMsgFileName); process.exit(); }); fs.writeFileSync(jsFileName, originExpre); setTimeout(() => { fs.unwatchFile(jsFileName); fs.unwatchFile(abcFileName); fs.unwatchFile(errorMsgFileName); fs.unlinkSync(jsFileName); fs.unlinkSync(abcFileName); fs.unlinkSync(errorMsgFileName); throw new Error("watchFileServer has not been initialized"); }, watchAbcFileTimeOut*1000); } function compileWatchExpression(jsFileName: string, errorMsgFileName: string, options: ts.CompilerOptions, watchedProgram: ts.Program) { CmdOptions.setWatchEvaluateExpressionArgs(['','']); let fileName = watchFileName + ".js"; let errorMsgRecordFlag = false; let sourceFile = ts.createSourceFile(fileName, fs.readFileSync(jsFileName).toString(), ts.ScriptTarget.ES2017); let jsFileDiagnostics = watchedProgram.getSyntacticDiagnostics(sourceFile); jsFileDiagnostics.forEach(diagnostic => { if (!errorMsgRecordFlag) { fs.writeFileSync(errorMsgFileName, "There are syntax errors in input expression.\n"); errorMsgRecordFlag = true; } diag.printDiagnostic(diagnostic); return; }); if (errorMsgRecordFlag) { return; } watchedProgram.emit( undefined, undefined, undefined, undefined, { before: [ // @ts-ignore (ctx: ts.TransformationContext) => { return (node: ts.SourceFile) => { if (path.basename(node.fileName) == fileName) { node = sourceFile; } let outputBinName = getOutputBinName(node); let compilerDriver = new CompilerDriver(outputBinName); compilerDriver.compileForSyntaxCheck(node); return node; } } ], after: [ // @ts-ignore (ctx: ts.TransformationContext) => { return (node: ts.SourceFile) => { if (ts.getEmitHelpers(node)) { let newStatements = []; ts.getEmitHelpers(node)?.forEach( item => { let emitHelperSourceFile = ts.createSourceFile(node.fileName, item.text, options.target!, true, ts.ScriptKind.JS); emitHelperSourceFile.statements.forEach(emitStatement => { let emitNode = setPos(emitStatement); newStatements.push(emitNode); }); } ) newStatements.push(...node.statements); node = ts.factory.updateSourceFile(node, newStatements); } let outputBinName = getOutputBinName(node); let compilerDriver = new CompilerDriver(outputBinName); setGlobalStrict(jshelpers.isEffectiveStrictModeSourceFile(node, options)); compilerDriver.compile(node); return node; } } ] } ); } function launchWatchEvaluateDeamon(parsed: ts.ParsedCommandLine | undefined) { let deamonFilePrefix = CmdOptions.getEvaluateDeamonPath() + path.sep + watchFileName; let jsFileName = deamonFilePrefix + ".js"; let abcFileName = deamonFilePrefix + ".abc"; let errorMsgFileName = deamonFilePrefix + ".err"; if (fs.existsSync(jsFileName)) { console.log("watchFileServer has been initialized supportTimeout"); return; } let files: string[] = parsed.fileNames; fs.writeFileSync(jsFileName, "initJsFile\n"); fs.writeFileSync(errorMsgFileName, "initErrMsgFile\n"); files.unshift(jsFileName); let watchedProgram = ts.createProgram(files, parsed.options); compileWatchExpression(jsFileName, errorMsgFileName, parsed.options, watchedProgram); fs.watchFile(jsFileName, { persistent: true, interval: 50 }, (curr, prev) => { if (+curr.mtime <= +prev.mtime) { throw new Error("watched js file has not been initialized"); } if (fs.readFileSync(jsFileName).toString() == stopWatchingStr) { fs.unwatchFile(jsFileName); console.log("stopWatchingSuccess"); return; } compileWatchExpression(jsFileName, errorMsgFileName, parsed.options, watchedProgram); }); console.log("startWatchingSuccess supportTimeout"); process.on("exit", () => { fs.unlinkSync(jsFileName); fs.unlinkSync(abcFileName); fs.unlinkSync(errorMsgFileName); }); } namespace Compiler { export namespace Options { export let Default: ts.CompilerOptions = { outDir: "../tmp/build", allowJs: true, noEmitOnError: true, noImplicitAny: true, target: ts.ScriptTarget.ES2017, module: ts.ModuleKind.ES2015, strictNullChecks: true, skipLibCheck: true, alwaysStrict: true }; } } function run(args: string[], options?: ts.CompilerOptions): void { let parsed = CmdOptions.parseUserCmd(args); if (!parsed) { return; } if (options) { if (!((parsed.options.project) || (parsed.options.build))) { parsed.options = options; } } try { if (CmdOptions.isWatchEvaluateDeamonMode()) { launchWatchEvaluateDeamon(parsed); return; } if (CmdOptions.isStopEvaluateDeamonMode()) { fs.writeFileSync(CmdOptions.getEvaluateDeamonPath() + path.sep + watchFileName + ".js", stopWatchingStr); return; } if (CmdOptions.isWatchEvaluateExpressionMode()) { updateWatchJsFile(); return; } main(parsed.fileNames.concat(CmdOptions.getIncludedFiles()), parsed.options); } catch (err) { if (err instanceof diag.DiagnosticError) { let diagnostic = diag.getDiagnostic(err.code); if (diagnostic != undefined) { let diagnosticLog = diag.createDiagnostic(err.file, err.irnode, diagnostic, ...err.args); diag.printDiagnostic(diagnosticLog); } } else if (err instanceof SyntaxError) { LOGE(err.name, err.message); } else { throw err; } } } let dtsFiles = getDtsFiles(path["join"](__dirname, "../node_modules/typescript/lib")); process.argv.push(...dtsFiles); run(process.argv.slice(2), Compiler.Options.Default); global.gc();