From 783555d3da1c8a0089634cd7062c16ecd371ba64 Mon Sep 17 00:00:00 2001 From: Andrew Branch Date: Wed, 7 Feb 2024 16:46:53 -0800 Subject: [PATCH 1/4] Expose renderer in CLI package --- packages/cli/package.json | 7 ++- packages/cli/src/index.ts | 36 +++++++------ packages/cli/src/problemUtils.ts | 4 +- packages/cli/src/render/index.ts | 11 ++++ packages/cli/src/render/typed.ts | 54 ++++++++++---------- packages/cli/src/render/untyped.ts | 2 +- packages/cli/test/snapshots/ejs@3.1.9.tgz.md | 6 ++- packages/cli/tsconfig.json | 1 - 8 files changed, 72 insertions(+), 49 deletions(-) diff --git a/packages/cli/package.json b/packages/cli/package.json index 933d188..2e34acf 100644 --- a/packages/cli/package.json +++ b/packages/cli/package.json @@ -21,11 +21,16 @@ }, "files": [ "LICENSE", - "dist/**/*.js" + "dist/**/*.js", + "dist/**/*.js.map", + "dist/**/*.d.ts" ], "bin": { "attw": "./dist/index.js" }, + "exports": { + "./internal/render": "./dist/render/index.js" + }, "publishConfig": { "access": "public" }, diff --git a/packages/cli/src/index.ts b/packages/cli/src/index.ts index 779e64b..f78f003 100644 --- a/packages/cli/src/index.ts +++ b/packages/cli/src/index.ts @@ -18,21 +18,20 @@ import { major, minor } from "semver"; const packageJson = createRequire(import.meta.url)("../package.json"); const version = packageJson.version; -const formats = ["auto", "table", "table-flipped", "ascii", "json"] as const; - -type Format = (typeof formats)[number]; - -export interface Opts { +const formats = Object.keys({ + auto: true, + json: true, + ascii: true, + table: true, + "table-flipped": true, +} satisfies Record) as render.Format[]; + +interface Opts extends render.RenderOptions { pack?: boolean; fromNpm?: boolean; definitelyTyped?: boolean | string; - summary?: boolean; - emoji?: boolean; - color?: boolean; quiet?: boolean; configPath?: string; - ignoreRules?: string[]; - format: Format; entrypoints?: string[]; includeEntrypoints?: string[]; @@ -81,9 +80,6 @@ particularly ESM-related module resolution issues.`, .action(async (fileOrDirectory = ".") => { const opts = program.opts(); await readConfig(program, opts.configPath); - opts.ignoreRules = opts.ignoreRules?.map( - (value) => Object.keys(problemFlags).find((key) => problemFlags[key as core.ProblemKind] === value) as string, - ); if (opts.quiet) { console.log = () => {}; @@ -210,7 +206,10 @@ particularly ESM-related module resolution issues.`, console.log(JSON.stringify(result)); - if (analysis.types && analysis.problems.some((problem) => !opts.ignoreRules?.includes(problem.kind))) + if ( + analysis.types && + analysis.problems.some((problem) => !opts.ignoreRules?.includes(problemFlags[problem.kind])) + ) process.exit(1); return; @@ -218,13 +217,16 @@ particularly ESM-related module resolution issues.`, console.log(); if (analysis.types) { - await render.typed(analysis, opts); + console.log(await render.typed(analysis, opts)); - if (analysis.types && analysis.problems.some((problem) => !opts.ignoreRules?.includes(problem.kind))) { + if ( + analysis.types && + analysis.problems.some((problem) => !opts.ignoreRules?.includes(problemFlags[problem.kind])) + ) { process.exitCode = 1; } } else { - render.untyped(analysis as core.UntypedResult); + console.log(render.untyped(analysis as core.UntypedResult)); } if (deleteTgz) { diff --git a/packages/cli/src/problemUtils.ts b/packages/cli/src/problemUtils.ts index 1c76153..f9eefe8 100644 --- a/packages/cli/src/problemUtils.ts +++ b/packages/cli/src/problemUtils.ts @@ -1,7 +1,7 @@ import * as core from "@arethetypeswrong/core"; import type { ProblemKind } from "@arethetypeswrong/core"; -export const problemFlags: Record = { +export const problemFlags = { NoResolution: "no-resolution", UntypedResolution: "untyped-resolution", FalseCJS: "false-cjs", @@ -13,7 +13,7 @@ export const problemFlags: Record = { MissingExportEquals: "missing-export-equals", UnexpectedModuleSyntax: "unexpected-module-syntax", InternalResolutionError: "internal-resolution-error", -}; +} as const satisfies Record; export const resolutionKinds: Record = { node10: "node10", diff --git a/packages/cli/src/render/index.ts b/packages/cli/src/render/index.ts index 28db131..955c11b 100644 --- a/packages/cli/src/render/index.ts +++ b/packages/cli/src/render/index.ts @@ -1,2 +1,13 @@ +import type { problemFlags } from "../problemUtils.js"; + +export type Format = "auto" | "table" | "table-flipped" | "ascii" | "json"; +export interface RenderOptions { + ignoreRules?: (typeof problemFlags)[keyof typeof problemFlags][]; + format?: Format; + color?: boolean; + summary?: boolean; + emoji?: boolean; +} + export * from "./typed.js"; export * from "./untyped.js"; diff --git a/packages/cli/src/render/typed.ts b/packages/cli/src/render/typed.ts index 607a3c3..9c2f171 100644 --- a/packages/cli/src/render/typed.ts +++ b/packages/cli/src/render/typed.ts @@ -1,48 +1,44 @@ import * as core from "@arethetypeswrong/core"; +import { filterProblems, problemAffectsEntrypoint, problemKindInfo } from "@arethetypeswrong/core/problems"; import { allResolutionKinds, getResolutionOption, groupProblemsByKind } from "@arethetypeswrong/core/utils"; import chalk from "chalk"; import Table, { type GenericTable, type HorizontalTableRow } from "cli-table3"; import { marked } from "marked"; - -import { filterProblems, problemAffectsEntrypoint, problemKindInfo } from "@arethetypeswrong/core/problems"; -import type { Opts } from "../index.js"; +import TerminalRenderer from "marked-terminal"; import { moduleKinds, problemFlags, resolutionKinds } from "../problemUtils.js"; import { asciiTable } from "./asciiTable.js"; -import TerminalRenderer from "marked-terminal"; +import type { RenderOptions } from "./index.js"; -export async function typed(analysis: core.Analysis, opts: Opts) { - const problems = analysis.problems.filter((problem) => !opts.ignoreRules || !opts.ignoreRules.includes(problem.kind)); +export async function typed(analysis: core.Analysis, opts: RenderOptions): Promise { + let output = ""; + const problems = analysis.problems.filter( + (problem) => !opts.ignoreRules || !opts.ignoreRules.includes(problemFlags[problem.kind]), + ); const grouped = groupProblemsByKind(problems); const entrypoints = Object.keys(analysis.entrypoints); marked.setOptions({ renderer: new TerminalRenderer(), }); - console.log(`${analysis.packageName} v${analysis.packageVersion}`); + out(`${analysis.packageName} v${analysis.packageVersion}`); if (analysis.types.kind === "@types") { - console.log(`${analysis.types.packageName} v${analysis.types.packageVersion}`); + out(`${analysis.types.packageName} v${analysis.types.packageVersion}`); } - console.log(); + out(); if (Object.keys(analysis.buildTools).length) { - console.log("Build tools:"); - console.log( + out("Build tools:"); + out( Object.entries(analysis.buildTools) .map(([tool, version]) => { return `- ${tool}@${version}`; }) .join("\n"), ); - console.log(); + out(); } if (opts.ignoreRules && opts.ignoreRules.length) { - console.log( - chalk.gray( - ` (ignoring rules: ${opts.ignoreRules - .map((rule) => `'${problemFlags[rule as core.ProblemKind]}'`) - .join(", ")})\n`, - ), - ); + out(chalk.gray(` (ignoring rules: ${opts.ignoreRules.map((rule) => `'${rule}'`).join(", ")})\n`)); } if (opts.summary) { @@ -54,7 +50,7 @@ export async function typed(analysis: core.Analysis, opts: Opts) { return `${emoji}${description}`; }); - console.log(summaryTexts.join("") || defaultSummary); + out(summaryTexts.join("") || defaultSummary); } const entrypointNames = entrypoints.map( @@ -119,25 +115,31 @@ export async function typed(analysis: core.Analysis, opts: Opts) { switch (opts.format) { case "table": - console.log(table!.toString()); + out(table!.toString()); break; case "table-flipped": - console.log(flippedTable!.toString()); + out(flippedTable!.toString()); break; case "ascii": - console.log(asciiTable(table!)); + out(asciiTable(table!)); break; case "auto": const terminalWidth = process.stdout.columns || 133; // This looks like GitHub Actions' width if (table!.width <= terminalWidth) { - console.log(table!.toString()); + out(table!.toString()); } else if (flippedTable!.width <= terminalWidth) { - console.log(flippedTable!.toString()); + out(flippedTable!.toString()); } else { - console.log(asciiTable(table!)); + out(asciiTable(table!)); } break; } + + return output.trimEnd(); + + function out(s: string = "") { + output += s + "\n"; + } } function memo(fn: (...args: Args) => Result): (...args: Args) => Result { diff --git a/packages/cli/src/render/untyped.ts b/packages/cli/src/render/untyped.ts index 819c1b6..89254d2 100644 --- a/packages/cli/src/render/untyped.ts +++ b/packages/cli/src/render/untyped.ts @@ -1,5 +1,5 @@ import * as core from "@arethetypeswrong/core"; export function untyped(analysis: core.UntypedResult) { - console.log("This package does not contain types.\nDetails: ", analysis); + return "This package does not contain types.\nDetails: " + JSON.stringify(analysis, null, 2); } diff --git a/packages/cli/test/snapshots/ejs@3.1.9.tgz.md b/packages/cli/test/snapshots/ejs@3.1.9.tgz.md index 2ef0ca0..7bdeed1 100644 --- a/packages/cli/test/snapshots/ejs@3.1.9.tgz.md +++ b/packages/cli/test/snapshots/ejs@3.1.9.tgz.md @@ -5,7 +5,11 @@ $ attw ejs@3.1.9.tgz -f table-flipped This package does not contain types. -Details: { packageName: 'ejs', packageVersion: '3.1.9', types: false } +Details: { + "packageName": "ejs", + "packageVersion": "3.1.9", + "types": false +} ``` diff --git a/packages/cli/tsconfig.json b/packages/cli/tsconfig.json index 1a5b1f4..2affa8e 100644 --- a/packages/cli/tsconfig.json +++ b/packages/cli/tsconfig.json @@ -5,7 +5,6 @@ "rootDir": "src", "types": ["ts-expose-internals", "node"], "outDir": "./dist", - "declarationDir": "./lib", "sourceMap": true }, "include": ["src"], From 856e8b6a53a8ccdc63e7a35df65c7dddef44d771 Mon Sep 17 00:00:00 2001 From: Andrew Branch Date: Wed, 7 Feb 2024 16:48:38 -0800 Subject: [PATCH 2/4] Add changeset --- .changeset/nervous-carrots-yell.md | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 .changeset/nervous-carrots-yell.md diff --git a/.changeset/nervous-carrots-yell.md b/.changeset/nervous-carrots-yell.md new file mode 100644 index 0000000..ca5516e --- /dev/null +++ b/.changeset/nervous-carrots-yell.md @@ -0,0 +1,5 @@ +--- +"@arethetypeswrong/cli": patch +--- + +Expose internal renderer API From 2984c12fefa2e1be67dc19b50bea24889906c16f Mon Sep 17 00:00:00 2001 From: Andrew Branch Date: Thu, 8 Feb 2024 10:10:43 -0800 Subject: [PATCH 3/4] Add exit code util --- .changeset/nervous-carrots-yell.md | 2 +- package.json | 4 ++-- packages/cli/package.json | 5 +++-- packages/cli/src/getExitCode.ts | 13 +++++++++++++ packages/cli/src/index.ts | 22 ++++++++++----------- packages/cli/tsconfig.json | 2 +- packages/core/package.json | 2 +- pnpm-lock.yaml | 31 +++++++++++++----------------- 8 files changed, 44 insertions(+), 37 deletions(-) create mode 100644 packages/cli/src/getExitCode.ts diff --git a/.changeset/nervous-carrots-yell.md b/.changeset/nervous-carrots-yell.md index ca5516e..75cc323 100644 --- a/.changeset/nervous-carrots-yell.md +++ b/.changeset/nervous-carrots-yell.md @@ -2,4 +2,4 @@ "@arethetypeswrong/cli": patch --- -Expose internal renderer API +Expose internal renderer and exit code API diff --git a/package.json b/package.json index dc170e2..90df885 100644 --- a/package.json +++ b/package.json @@ -18,7 +18,7 @@ "devDependencies": { "@changesets/cli": "^2.27.1", "prettier": "^3.0.3", - "typescript": "5.3.3" + "typescript": "5.4.0-dev.20240207" }, "engines": { "node": ">=18", @@ -26,7 +26,7 @@ }, "pnpm": { "overrides": { - "ts-expose-internals-conditionally": "$typescript" + "ts-expose-internals-conditionally": "5.3.3" } } } diff --git a/packages/cli/package.json b/packages/cli/package.json index 2e34acf..c6c4ebc 100644 --- a/packages/cli/package.json +++ b/packages/cli/package.json @@ -29,6 +29,7 @@ "attw": "./dist/index.js" }, "exports": { + "./internal/getExitCode": "./dist/getExitCode.js", "./internal/render": "./dist/render/index.js" }, "publishConfig": { @@ -47,8 +48,8 @@ "@types/marked-terminal": "^3.1.3", "@types/node": "^20.2.5", "@types/semver": "^7.5.3", - "ts-expose-internals": "5.3.2", - "typescript": "5.3.2" + "ts-expose-internals-conditionally": "1.0.0-empty.0", + "typescript": "5.4.0-dev.20240207" }, "dependencies": { "@arethetypeswrong/core": "0.13.6", diff --git a/packages/cli/src/getExitCode.ts b/packages/cli/src/getExitCode.ts new file mode 100644 index 0000000..61078c4 --- /dev/null +++ b/packages/cli/src/getExitCode.ts @@ -0,0 +1,13 @@ +import type { CheckResult } from "@arethetypeswrong/core"; +import { problemFlags } from "./problemUtils.js"; +import type { RenderOptions } from "./render/index.js"; + +export function getExitCode(analysis: CheckResult, opts?: RenderOptions): number { + if (!analysis.types) { + return 0; + } + if (!opts?.ignoreRules) { + return analysis.problems.length > 0 ? 1 : 0; + } + return analysis.problems.some((problem) => !opts.ignoreRules!.includes(problemFlags[problem.kind])) ? 1 : 0; +} diff --git a/packages/cli/src/index.ts b/packages/cli/src/index.ts index f78f003..b722758 100644 --- a/packages/cli/src/index.ts +++ b/packages/cli/src/index.ts @@ -14,6 +14,7 @@ import { problemFlags } from "./problemUtils.js"; import { readConfig } from "./readConfig.js"; import * as render from "./render/index.js"; import { major, minor } from "semver"; +import { getExitCode } from "./getExitCode.js"; const packageJson = createRequire(import.meta.url)("../package.json"); const version = packageJson.version; @@ -206,11 +207,14 @@ particularly ESM-related module resolution issues.`, console.log(JSON.stringify(result)); - if ( - analysis.types && - analysis.problems.some((problem) => !opts.ignoreRules?.includes(problemFlags[problem.kind])) - ) - process.exit(1); + if (deleteTgz) { + await unlink(deleteTgz); + } + + const exitCode = getExitCode(analysis, opts); + if (exitCode) { + process.exit(exitCode); + } return; } @@ -218,13 +222,7 @@ particularly ESM-related module resolution issues.`, console.log(); if (analysis.types) { console.log(await render.typed(analysis, opts)); - - if ( - analysis.types && - analysis.problems.some((problem) => !opts.ignoreRules?.includes(problemFlags[problem.kind])) - ) { - process.exitCode = 1; - } + process.exitCode = getExitCode(analysis, opts); } else { console.log(render.untyped(analysis as core.UntypedResult)); } diff --git a/packages/cli/tsconfig.json b/packages/cli/tsconfig.json index 2affa8e..de50a14 100644 --- a/packages/cli/tsconfig.json +++ b/packages/cli/tsconfig.json @@ -3,7 +3,7 @@ "compilerOptions": { "module": "nodenext", "rootDir": "src", - "types": ["ts-expose-internals", "node"], + "types": ["ts-expose-internals-conditionally", "node"], "outDir": "./dist", "sourceMap": true }, diff --git a/packages/core/package.json b/packages/core/package.json index c759fa5..e5a470f 100644 --- a/packages/core/package.json +++ b/packages/core/package.json @@ -51,7 +51,7 @@ "fflate": "^0.7.4", "semver": "^7.5.4", "ts-expose-internals-conditionally": "1.0.0-empty.0", - "typescript": "5.3.3", + "typescript": "5.4.0-dev.20240207", "validate-npm-package-name": "^5.0.0" }, "devDependencies": { diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 1fc95ff..3b61f11 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -18,8 +18,8 @@ importers: specifier: ^3.0.3 version: 3.0.3 typescript: - specifier: 5.3.3 - version: 5.3.3 + specifier: 5.4.0-dev.20240207 + version: 5.4.0-dev.20240207 packages/cli: dependencies: @@ -57,12 +57,12 @@ importers: '@types/semver': specifier: ^7.5.3 version: 7.5.3 - ts-expose-internals: - specifier: 5.3.2 - version: 5.3.2 + ts-expose-internals-conditionally: + specifier: 5.3.3 + version: 5.3.3 typescript: - specifier: 5.3.2 - version: 5.3.2 + specifier: 5.4.0-dev.20240207 + version: 5.4.0-dev.20240207 packages/core: dependencies: @@ -79,8 +79,8 @@ importers: specifier: 5.3.3 version: 5.3.3 typescript: - specifier: 5.3.3 - version: 5.3.3 + specifier: 5.4.0-dev.20240207 + version: 5.4.0-dev.20240207 validate-npm-package-name: specifier: ^5.0.0 version: 5.0.0 @@ -3640,11 +3640,6 @@ packages: /ts-expose-internals-conditionally@5.3.3: resolution: {integrity: sha512-k3wz3FnxrtuD8ue3Y9GmqEsph3PB1YCZs6zWy6d6T60A9/0zgy3w0wQBQOdKPoyWN0T+JQCbY5cE8Cg80VlzNw==} - dev: false - - /ts-expose-internals@5.3.2: - resolution: {integrity: sha512-lKFeJay+cwMvqfvf+Uv/+pkTwKc4OOtjEPjxKMUUcVaqKGlTEuN3yssT29VG0maDb0QbpAGb6iDzKj7/cf4WWQ==} - dev: true /ts-expose-internals@5.3.3: resolution: {integrity: sha512-0lW96u0pa08dq9QLYUBRbgGkvsD+LvPAVGe94kr2tHaRZAxgckwJ+qqc1BSRoR1rpkTZf7AWAa6DKrLGFmctwg==} @@ -3757,14 +3752,14 @@ packages: resolution: {integrity: sha512-+wpbd+gNRNaHeYdJESWL0dh0SbjitfegVUCpLavYUJ0NEc149r3+NYZz2Wv2ttB+80ZJXjLpLXKBNB/E87wyyA==} dev: true - /typescript@5.3.2: - resolution: {integrity: sha512-6l+RyNy7oAHDfxC4FzSJcz9vnjTKxrLpDG5M2Vu4SHRVNg6xzqZp6LYSR9zjqQTu8DU/f5xwxUdADOkbrIX2gQ==} + /typescript@5.3.3: + resolution: {integrity: sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw==} engines: {node: '>=14.17'} hasBin: true dev: true - /typescript@5.3.3: - resolution: {integrity: sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw==} + /typescript@5.4.0-dev.20240207: + resolution: {integrity: sha512-BD7gF4QnjiTzOPqstgkgBxANkltT6VsZS6gFjXf7haPWTgrcebvCNQr1yGazb0o18RVtNXeHs6lDbXKqhJhiQQ==} engines: {node: '>=14.17'} hasBin: true From fdb18ff83061344616ed44a51f3c0f416648e462 Mon Sep 17 00:00:00 2001 From: Andrew Branch Date: Thu, 8 Feb 2024 11:57:31 -0800 Subject: [PATCH 4/4] Revert TS upgrade --- package.json | 4 ++-- packages/cli/package.json | 2 +- packages/core/package.json | 2 +- pnpm-lock.yaml | 18 ++++++------------ 4 files changed, 10 insertions(+), 16 deletions(-) diff --git a/package.json b/package.json index 90df885..dc170e2 100644 --- a/package.json +++ b/package.json @@ -18,7 +18,7 @@ "devDependencies": { "@changesets/cli": "^2.27.1", "prettier": "^3.0.3", - "typescript": "5.4.0-dev.20240207" + "typescript": "5.3.3" }, "engines": { "node": ">=18", @@ -26,7 +26,7 @@ }, "pnpm": { "overrides": { - "ts-expose-internals-conditionally": "5.3.3" + "ts-expose-internals-conditionally": "$typescript" } } } diff --git a/packages/cli/package.json b/packages/cli/package.json index c6c4ebc..42c9412 100644 --- a/packages/cli/package.json +++ b/packages/cli/package.json @@ -49,7 +49,7 @@ "@types/node": "^20.2.5", "@types/semver": "^7.5.3", "ts-expose-internals-conditionally": "1.0.0-empty.0", - "typescript": "5.4.0-dev.20240207" + "typescript": "5.3.3" }, "dependencies": { "@arethetypeswrong/core": "0.13.6", diff --git a/packages/core/package.json b/packages/core/package.json index e5a470f..c759fa5 100644 --- a/packages/core/package.json +++ b/packages/core/package.json @@ -51,7 +51,7 @@ "fflate": "^0.7.4", "semver": "^7.5.4", "ts-expose-internals-conditionally": "1.0.0-empty.0", - "typescript": "5.4.0-dev.20240207", + "typescript": "5.3.3", "validate-npm-package-name": "^5.0.0" }, "devDependencies": { diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 3b61f11..69bde91 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -18,8 +18,8 @@ importers: specifier: ^3.0.3 version: 3.0.3 typescript: - specifier: 5.4.0-dev.20240207 - version: 5.4.0-dev.20240207 + specifier: 5.3.3 + version: 5.3.3 packages/cli: dependencies: @@ -61,8 +61,8 @@ importers: specifier: 5.3.3 version: 5.3.3 typescript: - specifier: 5.4.0-dev.20240207 - version: 5.4.0-dev.20240207 + specifier: 5.3.3 + version: 5.3.3 packages/core: dependencies: @@ -79,8 +79,8 @@ importers: specifier: 5.3.3 version: 5.3.3 typescript: - specifier: 5.4.0-dev.20240207 - version: 5.4.0-dev.20240207 + specifier: 5.3.3 + version: 5.3.3 validate-npm-package-name: specifier: ^5.0.0 version: 5.0.0 @@ -3756,12 +3756,6 @@ packages: resolution: {integrity: sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw==} engines: {node: '>=14.17'} hasBin: true - dev: true - - /typescript@5.4.0-dev.20240207: - resolution: {integrity: sha512-BD7gF4QnjiTzOPqstgkgBxANkltT6VsZS6gFjXf7haPWTgrcebvCNQr1yGazb0o18RVtNXeHs6lDbXKqhJhiQQ==} - engines: {node: '>=14.17'} - hasBin: true /unbox-primitive@1.0.2: resolution: {integrity: sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==}