UNPKG

@rnm/tscx

Version:

A tsc wrapper with many convenient features.

135 lines 18.2 kB
// this file should not have `async` and `await` import childProcess from "node:child_process"; import path from "node:path"; import process from "node:process"; import { copyfiles, exec, remove, script, tsc } from "./cmd/index.js"; export class Compiler { options; id = ""; currentSubprocess; tsconfig; rootDir; outDir; constructor(options) { this.options = options; // setup tsconfig this.tsconfig = this.getTsConfig(); this.rootDir = this.getRootDir(); this.outDir = this.getOutDir(); } compile() { const id = `${Date.now().toString()}_${Math.random().toString(36).slice(2)}`; this.id = id; if (!this.currentSubprocess) { this.execTasks(id); return; } if (typeof this.currentSubprocess.exitCode === "number") { this.execTasks(id); return; } if (!this.currentSubprocess.killed) { this.currentSubprocess.kill(); } this.currentSubprocess.removeAllListeners("close"); this.currentSubprocess.on("close", () => { this.execTasks(id); }); } getTasks() { const { project, remove: rm, copyfiles: cp, script: scr, exec: ex, ...others } = this.options; return [ ...(rm ? [() => remove(this.outDir)] : []), () => tsc({ project, ...others }), ...(cp ? [() => copyfiles(this.rootDir, this.outDir)] : []), ...(scr ? [() => script(scr)] : []), ...(ex ? [() => exec(ex)] : []), ]; } execTasks(id) { if (this.id !== id) { return; } const tasks = this.getTasks(); const execNextTask = (index = 0) => { const currentTask = tasks[index]; if (!currentTask || this.id !== id) { return; } this.currentSubprocess = currentTask(); this.currentSubprocess.on("close", (code, signal) => { // manually exiting or unexpected exception will not execute next task if (code || signal) { return; } execNextTask(index + 1); }); }; execNextTask(); } refreshTsConfig() { this.tsconfig = this.getTsConfig(); this.rootDir = this.getRootDir(); this.outDir = this.getOutDir(); } getTsConfig() { const tscPath = path.resolve(process.cwd(), "node_modules", "typescript", "bin", "tsc"); const cmd = `node ${tscPath} --showConfig --project ${this.options.project}`; const config = JSON.parse(childProcess.execSync(cmd).toString("utf8")); if (!config.compilerOptions || Object.keys(config.compilerOptions).length <= 0) { throw new Error("Tsconfig.compilerOptions is empty!"); } return config; } getInclude() { return this.tsconfig.include; } getOutDir() { const outDir = this.tsconfig.compilerOptions?.outDir; if (!outDir) { throw new Error('"outDir" is not found'); } const absoluteOutDir = path.resolve(process.cwd(), outDir); if (process.cwd().startsWith(absoluteOutDir)) { throw new Error('"outDir" in tsconfig.json should not be current or parent directory'); } return absoluteOutDir; } getRootDir() { const rootDir = this.tsconfig.compilerOptions?.rootDir; return rootDir ? path.resolve(process.cwd(), rootDir) : path.resolve(process.cwd(), this.getRootDirByFiles(this.tsconfig.files ?? [])); } /** * Get the longest common dir. https://www.typescriptlang.org/tsconfig#rootDir * @param files file paths like ['./src/index.ts', './index.ts'] * @returns absolute path */ getRootDirByFiles(files) { if (files.length === 0) { throw new Error("Cannot get the longest common dir when the arguments is empty"); } const folder = files .map((file) => file.split(path.sep).slice(0, -1)) .reduce((prev, item) => { if (prev.length === 0) { return item; } const result = []; for (let i = 0; i < prev.length && i < item.length; i += 1) { const sub = prev[i]; if (sub && sub === item[i]) { result[i] = sub; } else { break; } } return result; }, []); return path.join(...folder); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"compiler.js","sourceRoot":"","sources":["../src/compiler.ts"],"names":[],"mappings":"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","sourcesContent":["// this file should not have `async` and `await`\nimport childProcess from \"node:child_process\";\nimport path from \"node:path\";\nimport process from \"node:process\";\nimport type ts from \"typescript\";\nimport { copyfiles, exec, remove, script, tsc } from \"./cmd/index.ts\";\n\nexport interface CompilerOptions extends Record<string, string | boolean> {\n  project: string;\n  remove: boolean;\n  copyfiles: boolean;\n  script?: string;\n  exec?: string;\n}\n\nexport interface TsConfig {\n  compilerOptions?: ts.CompilerOptions;\n  include?: string[];\n  exclude?: string[];\n  files?: string[];\n}\n\nexport class Compiler {\n  private id = \"\";\n  private currentSubprocess?: childProcess.ChildProcess;\n  private tsconfig: TsConfig;\n  private rootDir: string;\n  private outDir: string;\n\n  constructor(private readonly options: CompilerOptions) {\n    // setup tsconfig\n    this.tsconfig = this.getTsConfig();\n    this.rootDir = this.getRootDir();\n    this.outDir = this.getOutDir();\n  }\n\n  compile() {\n    const id = `${Date.now().toString()}_${Math.random().toString(36).slice(2)}`;\n    this.id = id;\n\n    if (!this.currentSubprocess) {\n      this.execTasks(id);\n      return;\n    }\n    if (typeof this.currentSubprocess.exitCode === \"number\") {\n      this.execTasks(id);\n      return;\n    }\n    if (!this.currentSubprocess.killed) {\n      this.currentSubprocess.kill();\n    }\n    this.currentSubprocess.removeAllListeners(\"close\");\n    this.currentSubprocess.on(\"close\", () => {\n      this.execTasks(id);\n    });\n  }\n\n  private getTasks(): Array<() => childProcess.ChildProcess> {\n    const {\n      project,\n      remove: rm,\n      copyfiles: cp,\n      script: scr,\n      exec: ex,\n      ...others\n    } = this.options;\n    return [\n      ...(rm ? [() => remove(this.outDir)] : []),\n      () => tsc({ project, ...others }),\n      ...(cp ? [() => copyfiles(this.rootDir, this.outDir)] : []),\n      ...(scr ? [() => script(scr)] : []),\n      ...(ex ? [() => exec(ex)] : []),\n    ];\n  }\n\n  private execTasks(id: string) {\n    if (this.id !== id) {\n      return;\n    }\n\n    const tasks = this.getTasks();\n    const execNextTask = (index = 0) => {\n      const currentTask = tasks[index];\n      if (!currentTask || this.id !== id) {\n        return;\n      }\n      this.currentSubprocess = currentTask();\n      this.currentSubprocess.on(\"close\", (code, signal) => {\n        // manually exiting or unexpected exception will not execute next task\n        if (code || signal) {\n          return;\n        }\n        execNextTask(index + 1);\n      });\n    };\n    execNextTask();\n  }\n\n  refreshTsConfig() {\n    this.tsconfig = this.getTsConfig();\n    this.rootDir = this.getRootDir();\n    this.outDir = this.getOutDir();\n  }\n\n  private getTsConfig(): TsConfig {\n    const tscPath = path.resolve(\n      process.cwd(),\n      \"node_modules\",\n      \"typescript\",\n      \"bin\",\n      \"tsc\",\n    );\n    const cmd = `node ${tscPath} --showConfig --project ${this.options.project}`;\n    const config: TsConfig = JSON.parse(\n      childProcess.execSync(cmd).toString(\"utf8\"),\n    );\n    if (\n      !config.compilerOptions ||\n      Object.keys(config.compilerOptions).length <= 0\n    ) {\n      throw new Error(\"Tsconfig.compilerOptions is empty!\");\n    }\n    return config;\n  }\n\n  getInclude() {\n    return this.tsconfig.include;\n  }\n\n  getOutDir() {\n    const outDir = this.tsconfig.compilerOptions?.outDir;\n    if (!outDir) {\n      throw new Error('\"outDir\" is not found');\n    }\n    const absoluteOutDir = path.resolve(process.cwd(), outDir);\n    if (process.cwd().startsWith(absoluteOutDir)) {\n      throw new Error(\n        '\"outDir\" in tsconfig.json should not be current or parent directory',\n      );\n    }\n    return absoluteOutDir;\n  }\n\n  private getRootDir() {\n    const rootDir = this.tsconfig.compilerOptions?.rootDir;\n    return rootDir\n      ? path.resolve(process.cwd(), rootDir)\n      : path.resolve(\n          process.cwd(),\n          this.getRootDirByFiles(this.tsconfig.files ?? []),\n        );\n  }\n\n  /**\n   * Get the longest common dir. https://www.typescriptlang.org/tsconfig#rootDir\n   * @param files file paths like ['./src/index.ts', './index.ts']\n   * @returns absolute path\n   */\n  private getRootDirByFiles(files: string[]) {\n    if (files.length === 0) {\n      throw new Error(\n        \"Cannot get the longest common dir when the arguments is empty\",\n      );\n    }\n\n    const folder = files\n      .map((file) => file.split(path.sep).slice(0, -1))\n      .reduce<string[]>((prev, item) => {\n        if (prev.length === 0) {\n          return item;\n        }\n        const result: string[] = [];\n        for (let i = 0; i < prev.length && i < item.length; i += 1) {\n          const sub = prev[i];\n          if (sub && sub === item[i]) {\n            result[i] = sub;\n          } else {\n            break;\n          }\n        }\n        return result;\n      }, []);\n\n    return path.join(...folder);\n  }\n}\n"]}