@rnm/tscx
Version:
A tsc wrapper with many convenient features.
135 lines • 18.2 kB
JavaScript
// 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"]}