diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/exclude/utils.ts b/packages/plugin-typescript/mocks/fixtures/basic-setup/exclude/utils.ts new file mode 100644 index 000000000..20f47ca5d --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/exclude/utils.ts @@ -0,0 +1,3 @@ +export function test() { + return 'test'; +} diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/src/0-no-diagnostics/constants.ts b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/0-no-diagnostics/constants.ts new file mode 100644 index 000000000..9cbbe8809 --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/0-no-diagnostics/constants.ts @@ -0,0 +1 @@ +export const TEST = 'test'; diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/src/1-syntax-errors/ts-1136-property-assignment-expected.ts b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/1-syntax-errors/ts-1136-property-assignment-expected.ts new file mode 100644 index 000000000..230ee36b7 --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/1-syntax-errors/ts-1136-property-assignment-expected.ts @@ -0,0 +1,2 @@ +const a = { ; // Error: TS1136: Property assignment expected + diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/module-resolution/ts-2307-module-not-fount.ts b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/module-resolution/ts-2307-module-not-fount.ts new file mode 100644 index 000000000..d56e71dda --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/module-resolution/ts-2307-module-not-fount.ts @@ -0,0 +1,2 @@ +// 2307 - Cannot find module. +import { nonExistentModule } from './non-existent'; diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/no-implicit-this/ts-2683-not-implicit-this.ts b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/no-implicit-this/ts-2683-not-implicit-this.ts new file mode 100644 index 000000000..9fd0e6698 --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/no-implicit-this/ts-2683-not-implicit-this.ts @@ -0,0 +1,4 @@ +// 2683 - NoImplicitThis: 'this' implicitly has type 'any'. +function noImplicitThisTS2683() { + console.log(this.value); // Error 2683 +} diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/strict-function-types/ts-2349-not-callable.ts b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/strict-function-types/ts-2349-not-callable.ts new file mode 100644 index 000000000..199c6ef3c --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/strict-function-types/ts-2349-not-callable.ts @@ -0,0 +1,3 @@ +// 2349 - Cannot call a value that is not callable. +const notCallable = 42; +notCallable(); diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/strict-null-checks/ts-2531-strict-null-checks.ts b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/strict-null-checks/ts-2531-strict-null-checks.ts new file mode 100644 index 000000000..150ae729d --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/strict-null-checks/ts-2531-strict-null-checks.ts @@ -0,0 +1,2 @@ +// 2531 - StrictNullChecks: Object is possibly 'null'. +const strictNullChecksTS2531: string = null; diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/ts-2307-cannot-find-module.ts b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/ts-2307-cannot-find-module.ts new file mode 100644 index 000000000..97a912ab0 --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/ts-2307-cannot-find-module.ts @@ -0,0 +1 @@ +const value: NonExistentType = 42; // Compiler fails to resolve the type reference diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/src/4-languale-service/ts-4114-incorrect-modifier.ts b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/4-languale-service/ts-4114-incorrect-modifier.ts new file mode 100644 index 000000000..444811dfb --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/4-languale-service/ts-4114-incorrect-modifier.ts @@ -0,0 +1,6 @@ +class Standalone { + override method() { // Error: TS4114 - 'override' modifier can only be used in a class derived from a base class. + console.log("Standalone method"); + } +} +const s = Standalone; diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/src/6-configuration-errors/ts-6059-file-is-not-under-root-dir.ts b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/6-configuration-errors/ts-6059-file-is-not-under-root-dir.ts new file mode 100644 index 000000000..0a7cbb441 --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/src/6-configuration-errors/ts-6059-file-is-not-under-root-dir.ts @@ -0,0 +1,3 @@ +import { test } from '../../exclude/utils'; + +// TS6059:: File 'exclude/utils.ts' is not under 'rootDir' '.../configuration-errors'. 'rootDir' is expected to contain all source files. diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig-config-errors.json b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig-config-errors.json new file mode 100644 index 000000000..c68a4d4d6 --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig-config-errors.json @@ -0,0 +1,10 @@ +{ + "compilerOptions": { + "rootDir": "./src", + "strict": true, + "verbatimModuleSyntax": false, + "target": "ES6", + "module": "CommonJS" + }, + "include": ["src/**/*.ts", "./out/**/*.ts", "nonexistent-file.ts"] +} diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.all-audits.json b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.all-audits.json new file mode 100644 index 000000000..2817c9d1c --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.all-audits.json @@ -0,0 +1,16 @@ +{ + "compilerOptions": { + "rootDir": "./src", + "strict": true, + "verbatimModuleSyntax": false, + "target": "ES6", + "module": "CommonJS" + }, + "exclude": ["exclude"], + "include": [ + "src/1-syntax-errors/**/*.ts", + "src/2-semantic-errors/**/*.ts", + "src/4-languale-service/**/*.ts", + "src/6-config-errors/**/*.ts" + ] +} diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.init.json b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.init.json new file mode 100644 index 000000000..ba648354a --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.init.json @@ -0,0 +1,103 @@ +{ + "compilerOptions": { + /* Visit https://aka.ms/tsconfig to read more about this file */ + + /* Projects */ + // "incremental": true, /* Save .tsbuildinfo files to allow for incremental compilation of projects. */ + // "composite": true, /* Enable constraints that allow a TypeScript project to be used with project references. */ + // "tsBuildInfoFile": "./.tsbuildinfo", /* Specify the path to .tsbuildinfo incremental compilation file. */ + // "disableSourceOfProjectReferenceRedirect": true, /* Disable preferring source files instead of declaration files when referencing composite projects. */ + // "disableSolutionSearching": true, /* Opt a project out of multi-project reference checking when editing. */ + // "disableReferencedProjectLoad": true, /* Reduce the number of projects loaded automatically by TypeScript. */ + + /* Language and Environment */ + "target": "es2016" /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */, + // "lib": [], /* Specify a set of bundled library declaration files that describe the target runtime environment. */ + // "jsx": "preserve", /* Specify what JSX code is generated. */ + // "experimentalDecorators": true, /* Enable experimental support for TC39 stage 2 draft decorators. */ + // "emitDecoratorMetadata": true, /* Emit design-type metadata for decorated declarations in source files. */ + // "jsxFactory": "", /* Specify the JSX factory function used when targeting React JSX emit, e.g. 'React.createElement' or 'h'. */ + // "jsxFragmentFactory": "", /* Specify the JSX Fragment reference used for fragments when targeting React JSX emit e.g. 'React.Fragment' or 'Fragment'. */ + // "jsxImportSource": "", /* Specify module specifier used to import the JSX factory functions when using 'jsx: react-jsx*'. */ + // "reactNamespace": "", /* Specify the object invoked for 'createElement'. This only applies when targeting 'react' JSX emit. */ + // "noLib": true, /* Disable including any library files, including the default lib.d.ts. */ + // "useDefineForClassFields": true, /* Emit ECMAScript-standard-compliant class fields. */ + // "moduleDetection": "auto", /* Control what method is used to detect module-format JS files. */ + + /* Modules */ + "module": "commonjs" /* Specify what module code is generated. */, + // "rootDir": "./", /* Specify the root folder within your source files. */ + // "moduleResolution": "node", /* Specify how TypeScript looks up a file from a given module specifier. */ + // "baseUrl": "./", /* Specify the base directory to resolve non-relative module names. */ + // "paths": {}, /* Specify a set of entries that re-map imports to additional lookup locations. */ + // "rootDirs": [], /* Allow multiple folders to be treated as one when resolving modules. */ + // "typeRoots": [], /* Specify multiple folders that act like './node_modules/@types'. */ + // "types": [], /* Specify type package names to be included without being referenced in a source file. */ + // "allowUmdGlobalAccess": true, /* Allow accessing UMD globals from modules. */ + // "moduleSuffixes": [], /* List of file name suffixes to search when resolving a module. */ + // "resolveJsonModule": true, /* Enable importing .json files. */ + // "noResolve": true, /* Disallow 'import's, 'require's or ''s from expanding the number of files TypeScript should add to a project. */ + + /* JavaScript Support */ + // "allowJs": true, /* Allow JavaScript files to be a part of your program. Use the 'checkJS' option to get errors from these files. */ + // "checkJs": true, /* Enable error reporting in type-checked JavaScript files. */ + // "maxNodeModuleJsDepth": 1, /* Specify the maximum folder depth used for checking JavaScript files from 'node_modules'. Only applicable with 'allowJs'. */ + + /* Emit */ + // "declaration": true, /* Generate .d.ts files from TypeScript and JavaScript files in your project. */ + // "declarationMap": true, /* Create sourcemaps for d.ts files. */ + // "emitDeclarationOnly": true, /* Only output d.ts files and not JavaScript files. */ + // "sourceMap": true, /* Create source map files for emitted JavaScript files. */ + // "outFile": "./", /* Specify a file that bundles all outputs into one JavaScript file. If 'declaration' is true, also designates a file that bundles all .d.ts output. */ + // "outDir": "./", /* Specify an output folder for all emitted files. */ + // "removeComments": true, /* Disable emitting comments. */ + // "noEmit": true, /* Disable emitting files from a compilation. */ + // "importHelpers": true, /* Allow importing helper functions from tslib once per project, instead of including them per-file. */ + // "importsNotUsedAsValues": "remove", /* Specify emit/checking behavior for imports that are only used for types. */ + // "downlevelIteration": true, /* Emit more compliant, but verbose and less performant JavaScript for iteration. */ + // "sourceRoot": "", /* Specify the root path for debuggers to find the reference source code. */ + // "mapRoot": "", /* Specify the location where debugger should locate map files instead of generated locations. */ + // "inlineSourceMap": true, /* Include sourcemap files inside the emitted JavaScript. */ + // "inlineSources": true, /* Include source code in the sourcemaps inside the emitted JavaScript. */ + // "emitBOM": true, /* Emit a UTF-8 Byte Order Mark (BOM) in the beginning of output files. */ + // "newLine": "crlf", /* Set the newline character for emitting files. */ + // "stripInternal": true, /* Disable emitting declarations that have '@internal' in their JSDoc comments. */ + // "noEmitHelpers": true, /* Disable generating custom helper functions like '__extends' in compiled output. */ + // "noEmitOnError": true, /* Disable emitting files if any type checking errors are reported. */ + // "preserveConstEnums": true, /* Disable erasing 'const enum' declarations in generated code. */ + // "declarationDir": "./", /* Specify the output directory for generated declaration files. */ + // "preserveValueImports": true, /* Preserve unused imported values in the JavaScript output that would otherwise be removed. */ + + /* Interop Constraints */ + // "isolatedModules": true, /* Ensure that each file can be safely transpiled without relying on other imports. */ + // "allowSyntheticDefaultImports": true, /* Allow 'import x from y' when a module doesn't have a default export. */ + "esModuleInterop": true /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables 'allowSyntheticDefaultImports' for type compatibility. */, + // "preserveSymlinks": true, /* Disable resolving symlinks to their realpath. This correlates to the same flag in node. */ + "forceConsistentCasingInFileNames": true /* Ensure that casing is correct in imports. */, + + /* Type Checking */ + "strict": true /* Enable all strict type-checking options. */, + // "noImplicitAny": true, /* Enable error reporting for expressions and declarations with an implied 'any' type. */ + // "strictNullChecks": true, /* When type checking, take into account 'null' and 'undefined'. */ + // "strictFunctionTypes": true, /* When assigning functions, check to ensure parameters and the return values are subtype-compatible. */ + // "strictBindCallApply": true, /* Check that the arguments for 'bind', 'call', and 'apply' methods match the original function. */ + // "strictPropertyInitialization": true, /* Check for class properties that are declared but not set in the constructor. */ + // "noImplicitThis": true, /* Enable error reporting when 'this' is given the type 'any'. */ + // "useUnknownInCatchVariables": true, /* Default catch clause variables as 'unknown' instead of 'any'. */ + // "alwaysStrict": true, /* Ensure 'use strict' is always emitted. */ + // "noUnusedLocals": true, /* Enable error reporting when local variables aren't read. */ + // "noUnusedParameters": true, /* Raise an error when a function parameter isn't read. */ + // "exactOptionalPropertyTypes": true, /* Interpret optional property types as written, rather than adding 'undefined'. */ + // "noImplicitReturns": true, /* Enable error reporting for codepaths that do not explicitly return in a function. */ + // "noFallthroughCasesInSwitch": true, /* Enable error reporting for fallthrough cases in switch statements. */ + // "noUncheckedIndexedAccess": true, /* Add 'undefined' to a type when accessed using an index. */ + // "noImplicitOverride": true, /* Ensure overriding members in derived classes are marked with an override modifier. */ + // "noPropertyAccessFromIndexSignature": true, /* Enforces using indexed accessors for keys declared using an indexed type. */ + // "allowUnusedLabels": true, /* Disable error reporting for unused labels. */ + // "allowUnreachableCode": true, /* Disable error reporting for unreachable code. */ + + /* Completeness */ + // "skipDefaultLibCheck": true, /* Skip type checking .d.ts files that are included with TypeScript. */ + "skipLibCheck": true /* Skip type checking all .d.ts files. */ + } +} diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.internal-errors.json b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.internal-errors.json new file mode 100644 index 000000000..1ee4ec6b5 --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.internal-errors.json @@ -0,0 +1,8 @@ +{ + "compilerOptions": { + "rootDir": "./", + "module": "CommonJS", + "out": "./dist/bundle.js" + }, + "include": ["src/0-no-diagnostics/**/*.ts"] +} diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.json b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.json new file mode 100644 index 000000000..474196ef5 --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.json @@ -0,0 +1,7 @@ +{ + "compilerOptions": { + "rootDir": "./", + "module": "CommonJS" + }, + "include": ["src/**/*.ts"] +} diff --git a/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.no-files-match.json b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.no-files-match.json new file mode 100644 index 000000000..983a18853 --- /dev/null +++ b/packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.no-files-match.json @@ -0,0 +1,8 @@ +{ + "compilerOptions": { + "rootDir": "./src", + "strict": true, + "target": "ES6", + "module": "CommonJS" + } +} diff --git a/packages/plugin-typescript/package.json b/packages/plugin-typescript/package.json index aaf672e9f..d7d9831b7 100644 --- a/packages/plugin-typescript/package.json +++ b/packages/plugin-typescript/package.json @@ -22,6 +22,12 @@ "access": "public" }, "type": "module", - "dependencies": {}, + "dependencies": { + "@code-pushup/models": "0.59.0", + "@code-pushup/utils": "0.59.0" + }, + "peerDependencies": { + "typescript": ">=4.0.0" + }, "scripts": {} } diff --git a/packages/plugin-typescript/src/lib/constants.ts b/packages/plugin-typescript/src/lib/constants.ts index b5675c81c..25edb0924 100644 --- a/packages/plugin-typescript/src/lib/constants.ts +++ b/packages/plugin-typescript/src/lib/constants.ts @@ -1 +1,73 @@ +import type { Audit, Group } from '@code-pushup/models'; +import { toSentenceCase } from '@code-pushup/utils'; +import { TS_CODE_RANGE_NAMES } from './runner/ts-error-codes.js'; +import type { AuditSlug } from './types.js'; + export const TYPESCRIPT_PLUGIN_SLUG = 'typescript'; +export const DEFAULT_TS_CONFIG = 'tsconfig.json'; + +const AUDIT_DESCRIPTIONS: Record = { + 'semantic-errors': + 'Errors that occur during type checking and type inference', + 'syntax-errors': + 'Errors that occur during parsing and lexing of TypeScript source code', + 'configuration-errors': + 'Errors that occur when parsing TypeScript configuration files', + 'declaration-and-language-service-errors': + 'Errors that occur during TypeScript language service operations', + 'internal-errors': 'Errors that occur during TypeScript internal operations', + 'no-implicit-any-errors': 'Errors related to no implicit any compiler option', + 'unknown-codes': 'Errors that do not match any known TypeScript error code', +}; +export const AUDITS: (Audit & { slug: AuditSlug })[] = Object.values( + TS_CODE_RANGE_NAMES, +).map(slug => ({ + slug, + title: toSentenceCase(slug), + description: AUDIT_DESCRIPTIONS[slug], +})); + +export const GROUPS: Group[] = [ + { + slug: 'problems', + title: 'Problems', + description: + 'Syntax, semantic, and internal compiler errors are critical for identifying and preventing bugs.', + refs: ( + [ + 'syntax-errors', + 'semantic-errors', + 'no-implicit-any-errors', + ] satisfies AuditSlug[] + ).map(slug => ({ + slug, + weight: 1, + })), + }, + { + slug: 'ts-configuration', + title: 'Configuration', + description: + 'TypeScript configuration and options errors ensure correct project setup, reducing risks from misconfiguration.', + refs: (['configuration-errors'] satisfies AuditSlug[]).map(slug => ({ + slug, + weight: 1, + })), + }, + { + slug: 'miscellaneous', + title: 'Miscellaneous', + description: + 'Errors that do not bring any specific value to the developer, but are still useful to know.', + refs: ( + [ + 'unknown-codes', + 'internal-errors', + 'declaration-and-language-service-errors', + ] satisfies AuditSlug[] + ).map(slug => ({ + slug, + weight: 1, + })), + }, +]; diff --git a/packages/plugin-typescript/src/lib/runner/__snapshots__/runner-function-all-audits.json b/packages/plugin-typescript/src/lib/runner/__snapshots__/runner-function-all-audits.json new file mode 100644 index 000000000..d5deaabe4 --- /dev/null +++ b/packages/plugin-typescript/src/lib/runner/__snapshots__/runner-function-all-audits.json @@ -0,0 +1,120 @@ +[ + { + "details": { + "issues": [ + { + "message": "TS1136: Property assignment expected.", + "severity": "error", + "source": { + "file": "packages/plugin-typescript/mocks/fixtures/basic-setup/src/1-syntax-errors/ts-1136-property-assignment-expected.ts", + "position": { + "startLine": 1, + }, + }, + }, + ], + }, + "score": 0, + "slug": "syntax-errors", + "value": 1, + }, + { + "details": { + "issues": [ + { + "message": "TS2307: Cannot find module './non-existent' or its corresponding type declarations.", + "severity": "error", + "source": { + "file": "packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/module-resolution/ts-2307-module-not-fount.ts", + "position": { + "startLine": 2, + }, + }, + }, + { + "message": "TS2683: 'this' implicitly has type 'any' because it does not have a type annotation.", + "severity": "error", + "source": { + "file": "packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/no-implicit-this/ts-2683-not-implicit-this.ts", + "position": { + "startLine": 3, + }, + }, + }, + { + "message": "TS2349: This expression is not callable. + Type 'Number' has no call signatures.", + "severity": "error", + "source": { + "file": "packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/strict-function-types/ts-2349-not-callable.ts", + "position": { + "startLine": 3, + }, + }, + }, + { + "message": "TS2322: Type 'null' is not assignable to type 'string'.", + "severity": "error", + "source": { + "file": "packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/strict/strict-null-checks/ts-2531-strict-null-checks.ts", + "position": { + "startLine": 2, + }, + }, + }, + { + "message": "TS2304: Cannot find name 'NonExistentType'.", + "severity": "error", + "source": { + "file": "packages/plugin-typescript/mocks/fixtures/basic-setup/src/2-semantic-errors/ts-2307-cannot-find-module.ts", + "position": { + "startLine": 1, + }, + }, + }, + ], + }, + "score": 0, + "slug": "semantic-errors", + "value": 5, + }, + { + "details": { + "issues": [ + { + "message": "TS4112: This member cannot have an 'override' modifier because its containing class 'Standalone' does not extend another class.", + "severity": "error", + "source": { + "file": "packages/plugin-typescript/mocks/fixtures/basic-setup/src/4-languale-service/ts-4114-incorrect-modifier.ts", + "position": { + "startLine": 2, + }, + }, + }, + ], + }, + "score": 0, + "slug": "declaration-and-language-service-errors", + "value": 1, + }, + { + "score": 1, + "slug": "internal-errors", + "value": 0, + }, + { + "score": 1, + "slug": "configuration-errors", + "value": 0, + }, + { + "score": 1, + "slug": "no-implicit-any-errors", + "value": 0, + }, + { + "score": 1, + "slug": "unknown-codes", + "value": 0, + }, +] \ No newline at end of file diff --git a/packages/plugin-typescript/src/lib/runner/runner.integration.test.ts b/packages/plugin-typescript/src/lib/runner/runner.integration.test.ts new file mode 100644 index 000000000..ea2ae2da4 --- /dev/null +++ b/packages/plugin-typescript/src/lib/runner/runner.integration.test.ts @@ -0,0 +1,17 @@ +import { describe, expect } from 'vitest'; +import { getAudits } from '../utils.js'; +import { createRunnerFunction } from './runner.js'; + +describe('createRunnerFunction', () => { + it('should create valid audit outputs when called', async () => { + await expect( + createRunnerFunction({ + tsconfig: + 'packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.all-audits.json', + expectedAudits: getAudits(), + })(() => void 0), + ).resolves.toMatchFileSnapshot( + '__snapshots__/runner-function-all-audits.json', + ); + }, 35_000); +}); diff --git a/packages/plugin-typescript/src/lib/runner/runner.ts b/packages/plugin-typescript/src/lib/runner/runner.ts new file mode 100644 index 000000000..0c12a393d --- /dev/null +++ b/packages/plugin-typescript/src/lib/runner/runner.ts @@ -0,0 +1,51 @@ +import type { + AuditOutput, + AuditOutputs, + Issue, + RunnerFunction, +} from '@code-pushup/models'; +import type { AuditSlug } from '../types.js'; +import { + type DiagnosticsOptions, + getTypeScriptDiagnostics, +} from './ts-runner.js'; +import type { CodeRangeName } from './types.js'; +import { getIssueFromDiagnostic, tsCodeToAuditSlug } from './utils.js'; + +export type RunnerOptions = DiagnosticsOptions & { + expectedAudits: { slug: AuditSlug }[]; +}; + +export function createRunnerFunction(options: RunnerOptions): RunnerFunction { + const { tsconfig, expectedAudits } = options; + return async (): Promise => { + const diagnostics = await getTypeScriptDiagnostics({ tsconfig }); + const result = diagnostics.reduce< + Partial>> + >((acc, diag) => { + const slug = tsCodeToAuditSlug(diag.code); + const existingIssues: Issue[] = acc[slug]?.details?.issues ?? []; + return { + ...acc, + [slug]: { + slug, + details: { + issues: [...existingIssues, getIssueFromDiagnostic(diag)], + }, + }, + }; + }, {}); + + return expectedAudits.map(({ slug }) => { + const { details } = result[slug] ?? {}; + + const issues = details?.issues ?? []; + return { + slug, + score: issues.length === 0 ? 1 : 0, + value: issues.length, + ...(issues.length > 0 ? { details } : {}), + } satisfies AuditOutput; + }); + }; +} diff --git a/packages/plugin-typescript/src/lib/runner/runner.unit.test.ts b/packages/plugin-typescript/src/lib/runner/runner.unit.test.ts new file mode 100644 index 000000000..804260e38 --- /dev/null +++ b/packages/plugin-typescript/src/lib/runner/runner.unit.test.ts @@ -0,0 +1,183 @@ +import { + type Diagnostic, + DiagnosticCategory, + type SourceFile, +} from 'typescript'; +import { beforeEach, describe, expect } from 'vitest'; +import { auditOutputsSchema } from '@code-pushup/models'; +import { createRunnerFunction } from './runner.js'; +import * as runnerModule from './ts-runner.js'; +import * as utilsModule from './utils.js'; + +describe('createRunnerFunction', () => { + const getTypeScriptDiagnosticsSpy = vi.spyOn( + runnerModule, + 'getTypeScriptDiagnostics', + ); + const tSCodeToAuditSlugSpy = vi.spyOn(utilsModule, 'tsCodeToAuditSlug'); + const getIssueFromDiagnosticSpy = vi.spyOn( + utilsModule, + 'getIssueFromDiagnostic', + ); + + const semanticTsCode = 2322; + const mockSemanticDiagnostic = { + code: semanticTsCode, // "Type 'string' is not assignable to type 'number'" + start: 10, // Mocked character position + messageText: "Type 'string' is not assignable to type 'number'.", + category: DiagnosticCategory.Error, + file: { + getLineAndCharacterOfPosition: () => ({ line: 5, character: 10 }), + fileName: 'example.ts', + } as unknown as SourceFile, + } as unknown as Diagnostic; + const syntacticTsCode = 1005; + const mockSyntacticDiagnostic = { + code: syntacticTsCode, // "';' expected." + start: 25, // Mocked character position + messageText: "';' expected.", + category: DiagnosticCategory.Error, + file: { + getLineAndCharacterOfPosition: () => ({ line: 10, character: 20 }), + fileName: 'example.ts', + } as unknown as SourceFile, + } as unknown as Diagnostic; + + beforeEach(() => { + getTypeScriptDiagnosticsSpy.mockReset(); + }); + + it('should return empty array if no diagnostics are found', async () => { + getTypeScriptDiagnosticsSpy.mockResolvedValue([]); + const runner = createRunnerFunction({ + tsconfig: 'tsconfig.json', + expectedAudits: [], + }); + await expect(runner(() => void 0)).resolves.toStrictEqual([]); + }); + + it('should return empty array if no supported diagnostics are found', async () => { + getTypeScriptDiagnosticsSpy.mockResolvedValue([mockSemanticDiagnostic]); + const runner = createRunnerFunction({ + tsconfig: 'tsconfig.json', + expectedAudits: [], + }); + await expect(runner(() => void 0)).resolves.toStrictEqual([]); + }); + + it('should pass the diagnostic code to tsCodeToSlug', async () => { + getTypeScriptDiagnosticsSpy.mockResolvedValue([mockSemanticDiagnostic]); + const runner = createRunnerFunction({ + tsconfig: 'tsconfig.json', + expectedAudits: [], + }); + await expect(runner(() => void 0)).resolves.toStrictEqual([]); + expect(tSCodeToAuditSlugSpy).toHaveBeenCalledTimes(1); + expect(tSCodeToAuditSlugSpy).toHaveBeenCalledWith(semanticTsCode); + }); + + it('should pass the diagnostic to getIssueFromDiagnostic', async () => { + getTypeScriptDiagnosticsSpy.mockResolvedValue([mockSemanticDiagnostic]); + const runner = createRunnerFunction({ + tsconfig: 'tsconfig.json', + expectedAudits: [], + }); + await expect(runner(() => void 0)).resolves.toStrictEqual([]); + expect(getIssueFromDiagnosticSpy).toHaveBeenCalledTimes(1); + expect(getIssueFromDiagnosticSpy).toHaveBeenCalledWith( + mockSemanticDiagnostic, + ); + }); + + it('should return multiple issues per audit', async () => { + const code = 2222; + getTypeScriptDiagnosticsSpy.mockResolvedValue([ + mockSemanticDiagnostic, + { + ...mockSemanticDiagnostic, + code, + messageText: `error text [${code}]`, + }, + ]); + const runner = createRunnerFunction({ + tsconfig: 'tsconfig.json', + expectedAudits: [{ slug: 'semantic-errors' }], + }); + + const auditOutputs = await runner(() => void 0); + expect(auditOutputs).toStrictEqual([ + { + slug: 'semantic-errors', + score: 0, + value: 2, + details: { + issues: [ + expect.objectContaining({ + message: `TS${mockSemanticDiagnostic.code}: ${mockSemanticDiagnostic.messageText}`, + }), + expect.objectContaining({ + message: `TS${code}: error text [${code}]`, + }), + ], + }, + }, + ]); + expect(() => auditOutputsSchema.parse(auditOutputs)).not.toThrow(); + }); + + it('should return multiple audits', async () => { + getTypeScriptDiagnosticsSpy.mockResolvedValue([ + mockSyntacticDiagnostic, + mockSemanticDiagnostic, + ]); + const runner = createRunnerFunction({ + tsconfig: 'tsconfig.json', + expectedAudits: [{ slug: 'semantic-errors' }, { slug: 'syntax-errors' }], + }); + + const auditOutputs = await runner(() => void 0); + expect(auditOutputs).toStrictEqual([ + expect.objectContaining({ + slug: 'semantic-errors', + details: { + issues: [ + expect.objectContaining({ + message: `TS2322: Type 'string' is not assignable to type 'number'.`, + }), + ], + }, + }), + expect.objectContaining({ + slug: 'syntax-errors', + details: { + issues: [ + expect.objectContaining({ message: "TS1005: ';' expected." }), + ], + }, + }), + ]); + }); + + it('should return valid AuditOutput shape', async () => { + getTypeScriptDiagnosticsSpy.mockResolvedValue([ + mockSyntacticDiagnostic, + { + ...mockSyntacticDiagnostic, + code: 2222, + messageText: `error text [2222]`, + }, + mockSemanticDiagnostic, + { + ...mockSemanticDiagnostic, + code: 1111, + messageText: `error text [1111]`, + }, + ]); + const runner = createRunnerFunction({ + tsconfig: 'tsconfig.json', + expectedAudits: [{ slug: 'semantic-errors' }, { slug: 'syntax-errors' }], + }); + const auditOutputs = await runner(() => void 0); + expect(() => auditOutputsSchema.parse(auditOutputs)).not.toThrow(); + }); +}); diff --git a/packages/plugin-typescript/src/lib/runner/ts-error-codes.ts b/packages/plugin-typescript/src/lib/runner/ts-error-codes.ts new file mode 100644 index 000000000..bf5eefc70 --- /dev/null +++ b/packages/plugin-typescript/src/lib/runner/ts-error-codes.ts @@ -0,0 +1,29 @@ +/** + * # Diagnostic Code Ranges and Their Grouping + * + * TypeScript diagnostic codes are grouped into ranges based on their source and purpose. Here's how they are categorized: + * + * | Code Range | Type | Description | + * |------------|---------------------------------|--------------------------------------------------| + * | 1XXX | Syntax Errors | Structural issues detected during parsing. | + * | 2XXX | Semantic Errors | Type-checking and type-system violations. | + * | 3XXX | Suggestions | Optional improvements (e.g., unused variables). | + * | 4XXX | Declaration & Language Service | Used by editors (e.g., VSCode) for IntelliSense. | + * | 5XXX | Internal Compiler Errors | Rare, unexpected failures in the compiler. | + * | 6XXX | Configuration/Options Errors | Issues with `tsconfig.json` or compiler options. | + * | 7XXX | noImplicitAny Errors | Issues with commandline compiler options. | + * + * The diagnostic messages are exposed over a undocumented and undiscoverable const names `Diagnostics`. + * Additional information is derived from [TypeScript's own guidelines on diagnostic code ranges](https://github.com/microsoft/TypeScript/wiki/Coding-guidelines#diagnostic-message-codes) + * + */ +export const TS_CODE_RANGE_NAMES = { + '1': 'syntax-errors', + '2': 'semantic-errors', + // '3': 'suggestions', + '4': 'declaration-and-language-service-errors', + '5': 'internal-errors', + '6': 'configuration-errors', + '7': 'no-implicit-any-errors', + '9': 'unknown-codes', +} as const; diff --git a/packages/plugin-typescript/src/lib/runner/ts-runner.integration.test.ts b/packages/plugin-typescript/src/lib/runner/ts-runner.integration.test.ts new file mode 100644 index 000000000..bb7402c73 --- /dev/null +++ b/packages/plugin-typescript/src/lib/runner/ts-runner.integration.test.ts @@ -0,0 +1,13 @@ +import { describe, expect } from 'vitest'; +import { getTypeScriptDiagnostics } from './ts-runner.js'; + +describe('getTypeScriptDiagnostics', () => { + it('should return valid diagnostics', async () => { + await expect( + getTypeScriptDiagnostics({ + tsconfig: + 'packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.json', + }), + ).resolves.toHaveLength(5); + }); +}); diff --git a/packages/plugin-typescript/src/lib/runner/ts-runner.ts b/packages/plugin-typescript/src/lib/runner/ts-runner.ts new file mode 100644 index 000000000..147d15e72 --- /dev/null +++ b/packages/plugin-typescript/src/lib/runner/ts-runner.ts @@ -0,0 +1,25 @@ +import { + type Diagnostic, + createProgram, + getPreEmitDiagnostics, +} from 'typescript'; +import { stringifyError } from '@code-pushup/utils'; +import { loadTargetConfig } from './utils.js'; + +export type DiagnosticsOptions = { + tsconfig: string; +}; + +export async function getTypeScriptDiagnostics({ + tsconfig, +}: DiagnosticsOptions): Promise { + const { fileNames, options } = await loadTargetConfig(tsconfig); + try { + const program = createProgram(fileNames, options); + return getPreEmitDiagnostics(program); + } catch (error) { + throw new Error( + `Can't create TS program in getDiagnostics. \n ${stringifyError(error)}`, + ); + } +} diff --git a/packages/plugin-typescript/src/lib/runner/types.ts b/packages/plugin-typescript/src/lib/runner/types.ts new file mode 100644 index 000000000..b9a599b15 --- /dev/null +++ b/packages/plugin-typescript/src/lib/runner/types.ts @@ -0,0 +1,6 @@ +import { TS_CODE_RANGE_NAMES } from './ts-error-codes.js'; + +type TsCodeRanges = typeof TS_CODE_RANGE_NAMES; +export type CodeRangeName = TsCodeRanges[keyof TsCodeRanges]; + +export type SemVerString = `${number}.${number}.${number}`; diff --git a/packages/plugin-typescript/src/lib/runner/utils.integration.test.ts b/packages/plugin-typescript/src/lib/runner/utils.integration.test.ts new file mode 100644 index 000000000..63b1f0a2f --- /dev/null +++ b/packages/plugin-typescript/src/lib/runner/utils.integration.test.ts @@ -0,0 +1,55 @@ +import * as tsModule from 'typescript'; +import { describe, expect } from 'vitest'; +import { loadTargetConfig } from './utils.js'; + +describe('loadTargetConfig', () => { + const parseConfigFileTextToJsonSpy = vi.spyOn( + tsModule, + 'parseConfigFileTextToJson', + ); + const parseJsonConfigFileContentSpy = vi.spyOn( + tsModule, + 'parseJsonConfigFileContent', + ); + + it('should return the parsed content of a tsconfig file and ist TypeScript helper to parse it', async () => { + await expect( + loadTargetConfig( + 'packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.init.json', + ), + ).resolves.toStrictEqual( + expect.objectContaining({ + fileNames: expect.any(Array), + options: { + module: 1, + configFilePath: undefined, + esModuleInterop: true, + forceConsistentCasingInFileNames: true, + skipLibCheck: true, + strict: true, + target: 3, + }, + }), + ); + expect(parseConfigFileTextToJsonSpy).toHaveBeenCalledTimes(1); + expect(parseConfigFileTextToJsonSpy).toHaveBeenCalledWith( + 'packages/plugin-typescript/mocks/fixtures/basic-setup/tsconfig.init.json', + expect.stringContaining('/* Projects */'), + ); + expect(parseJsonConfigFileContentSpy).toHaveBeenCalledTimes(1); + expect(parseJsonConfigFileContentSpy).toHaveBeenCalledWith( + expect.objectContaining({ + compilerOptions: expect.objectContaining({ + esModuleInterop: true, + forceConsistentCasingInFileNames: true, + module: 'commonjs', + skipLibCheck: true, + strict: true, + target: 'es2016', + }), + }), + expect.any(Object), + expect.any(String), + ); + }); +}); diff --git a/packages/plugin-typescript/src/lib/runner/utils.ts b/packages/plugin-typescript/src/lib/runner/utils.ts new file mode 100644 index 000000000..ab22e6d1a --- /dev/null +++ b/packages/plugin-typescript/src/lib/runner/utils.ts @@ -0,0 +1,107 @@ +// eslint-disable-next-line unicorn/import-style +import { dirname } from 'node:path'; +import { + type Diagnostic, + DiagnosticCategory, + flattenDiagnosticMessageText, + parseConfigFileTextToJson, + parseJsonConfigFileContent, + sys, +} from 'typescript'; +import type { Issue } from '@code-pushup/models'; +import { readTextFile, truncateIssueMessage } from '@code-pushup/utils'; +import { TS_CODE_RANGE_NAMES } from './ts-error-codes.js'; +import type { CodeRangeName } from './types.js'; + +/** + * Transform the TypeScript error code to the audit slug. + * @param code - The TypeScript error code. + * @returns The audit slug. + * @throws Error if the code is not supported. + */ +export function tsCodeToAuditSlug(code: number): CodeRangeName { + const rangeNumber = code + .toString() + .slice(0, 1) as keyof typeof TS_CODE_RANGE_NAMES; + return TS_CODE_RANGE_NAMES[rangeNumber] ?? 'unknown-code'; +} + +/** + * Get the severity of the issue based on the TypeScript diagnostic category. + * - ts.DiagnosticCategory.Warning (1) + * - ts.DiagnosticCategory.Error (2) + * - ts.DiagnosticCategory.Suggestion (3) + * - ts.DiagnosticCategory.Message (4) + * @param category - The TypeScript diagnostic category. + * @returns The severity of the issue. + */ +export function getSeverity(category: DiagnosticCategory): Issue['severity'] { + switch (category) { + case DiagnosticCategory.Error: + return 'error'; + case DiagnosticCategory.Warning: + return 'warning'; + default: + return 'info'; + } +} + +/** + * Get the issue from the TypeScript diagnostic. + * @param diag - The TypeScript diagnostic. + * @returns The issue. + * @throws Error if the diagnostic is global (e.g., invalid compiler option). + */ +export function getIssueFromDiagnostic(diag: Diagnostic) { + const message = `${flattenDiagnosticMessageText(diag.messageText, '\n')}`; + + const issue: Issue = { + severity: getSeverity(diag.category), + message: truncateIssueMessage(`TS${diag.code}: ${message}`), + }; + + // If undefined, the error might be global (e.g., invalid compiler option). + if (diag.file === undefined) { + return issue; + } + + const startLine = + diag.start === undefined + ? undefined + : diag.file.getLineAndCharacterOfPosition(diag.start).line + 1; + + return { + ...issue, + source: { + file: diag.file.fileName, + ...(startLine + ? { + position: { + startLine, + }, + } + : {}), + }, + } satisfies Issue; +} + +export async function loadTargetConfig(tsConfigPath: string) { + const { config } = parseConfigFileTextToJson( + tsConfigPath, + await readTextFile(tsConfigPath), + ); + + const parsedConfig = parseJsonConfigFileContent( + config, + sys, + dirname(tsConfigPath), + ); + + if (parsedConfig.fileNames.length === 0) { + throw new Error( + 'No files matched by the TypeScript configuration. Check your "include", "exclude" or "files" settings.', + ); + } + + return parsedConfig; +} diff --git a/packages/plugin-typescript/src/lib/runner/utils.unit.test.ts b/packages/plugin-typescript/src/lib/runner/utils.unit.test.ts new file mode 100644 index 000000000..2bb4a8eb9 --- /dev/null +++ b/packages/plugin-typescript/src/lib/runner/utils.unit.test.ts @@ -0,0 +1,113 @@ +import { type Diagnostic, DiagnosticCategory } from 'typescript'; +import { beforeEach, describe, expect } from 'vitest'; +import { + getIssueFromDiagnostic, + getSeverity, + tsCodeToAuditSlug, +} from './utils.js'; + +describe('tSCodeToAuditSlug', () => { + it('should transform supported code to readable audit', () => { + expect(tsCodeToAuditSlug(Number.parseInt('2345', 10))).toBe( + 'semantic-errors', + ); + }); + + it('should return unknown slug for unknown code', () => { + expect(tsCodeToAuditSlug(999)).toBe('unknown-codes'); + }); +}); + +describe('getSeverity', () => { + it.each([ + [DiagnosticCategory.Error, 'error' as const], + [DiagnosticCategory.Warning, 'warning' as const], + [DiagnosticCategory.Message, 'info' as const], + [DiagnosticCategory.Suggestion, 'info' as const], + ])('should return "error" for DiagnosticCategory.Error', (cat, severity) => { + expect(getSeverity(cat)).toBe(severity); + }); + + it('should return "info" for unknown category', () => { + expect(getSeverity(999 as DiagnosticCategory)).toBe('info'); + }); +}); + +describe('getIssueFromDiagnostic', () => { + let diagnosticMock: Diagnostic; + + beforeEach(() => { + diagnosticMock = { + code: 222, + category: DiagnosticCategory.Error, + messageText: "Type 'number' is not assignable to type 'string'.", + file: { + fileName: 'file.ts', + getLineAndCharacterOfPosition: () => ({ line: 99 }), + }, + start: 4, + } as any; + }); + + it('should return valid issue', () => { + expect(getIssueFromDiagnostic(diagnosticMock)).toStrictEqual({ + message: "TS222: Type 'number' is not assignable to type 'string'.", + severity: 'error', + source: { + file: 'file.ts', + position: { + startLine: 100, + }, + }, + }); + }); + + it('should extract messageText and provide it under message', () => { + expect(getIssueFromDiagnostic(diagnosticMock)).toStrictEqual( + expect.objectContaining({ + message: "TS222: Type 'number' is not assignable to type 'string'.", + }), + ); + }); + + it('should extract category and provide it under severity', () => { + expect(getIssueFromDiagnostic(diagnosticMock)).toStrictEqual( + expect.objectContaining({ + severity: 'error', + }), + ); + }); + + it('should extract file path and provide it under source.file', () => { + expect(getIssueFromDiagnostic(diagnosticMock)).toStrictEqual( + expect.objectContaining({ + source: expect.objectContaining({ file: 'file.ts' }), + }), + ); + }); + + it('should extract line and provide it under source.position', () => { + expect(getIssueFromDiagnostic(diagnosticMock)).toStrictEqual( + expect.objectContaining({ + source: expect.objectContaining({ position: { startLine: 100 } }), + }), + ); + }); + + it('should return issue without position if file is undefined', () => { + expect( + getIssueFromDiagnostic({ ...diagnosticMock, file: undefined }), + ).toStrictEqual({ + message: "TS222: Type 'number' is not assignable to type 'string'.", + severity: 'error', + }); + }); + + it('position.startLine should be 1 if start is undefined', () => { + const result = getIssueFromDiagnostic({ + ...diagnosticMock, + start: undefined, + }); + expect(result.source?.position).toBeUndefined(); + }); +}); diff --git a/packages/plugin-typescript/src/lib/types.ts b/packages/plugin-typescript/src/lib/types.ts new file mode 100644 index 000000000..7345bb3f0 --- /dev/null +++ b/packages/plugin-typescript/src/lib/types.ts @@ -0,0 +1,8 @@ +import type { DiagnosticsOptions } from './runner/ts-runner.js'; +import type { CodeRangeName } from './runner/types.js'; + +export type AuditSlug = CodeRangeName; + +export type FilterOptions = { onlyAudits?: AuditSlug[] | undefined }; +export type TypescriptPluginOptions = Partial & + FilterOptions; diff --git a/packages/plugin-typescript/src/lib/utils.ts b/packages/plugin-typescript/src/lib/utils.ts new file mode 100644 index 000000000..a678fa5ac --- /dev/null +++ b/packages/plugin-typescript/src/lib/utils.ts @@ -0,0 +1,141 @@ +import type { CompilerOptions } from 'typescript'; +import type { Audit, CategoryConfig, CategoryRef } from '@code-pushup/models'; +import { kebabCaseToCamelCase } from '@code-pushup/utils'; +import { AUDITS, GROUPS, TYPESCRIPT_PLUGIN_SLUG } from './constants.js'; +import type { FilterOptions, TypescriptPluginOptions } from './types.js'; + +export function filterAuditsBySlug(slugs?: string[]) { + return ({ slug }: { slug: string }) => { + if (slugs && slugs.length > 0) { + return slugs.includes(slug); + } + return true; + }; +} + +/** + * It transforms a slug code to a compiler option format + * By default, kebabCaseToCamelCase. + * It will handle also cases like emit-bom that it should be emit-BOM + * @param slug Slug to be transformed + * @returns The slug as compilerOption key + */ +function auditSlugToCompilerOption(slug: string): string { + // eslint-disable-next-line sonarjs/no-small-switch + switch (slug) { + case 'emit-bom': + return 'emitBOM'; + default: + return kebabCaseToCamelCase(slug); + } +} + +/** + * From a list of audits, it will filter out the ones that might have been disabled from the compiler options + * plus from the parameter onlyAudits + * @param compilerOptions Compiler options + * @param onlyAudits OnlyAudits + * @returns Filtered Audits + */ +export function filterAuditsByCompilerOptions( + compilerOptions: CompilerOptions, + onlyAudits?: string[], +) { + return ({ slug }: { slug: string }) => { + const option = compilerOptions[auditSlugToCompilerOption(slug)]; + return ( + option !== false && + option !== undefined && + filterAuditsBySlug(onlyAudits)({ slug }) + ); + }; +} + +export function getGroups(options?: TypescriptPluginOptions) { + return GROUPS.map(group => ({ + ...group, + refs: group.refs.filter(filterAuditsBySlug(options?.onlyAudits)), + })).filter(group => group.refs.length > 0); +} + +export function getAudits(options?: FilterOptions) { + return AUDITS.filter(filterAuditsBySlug(options?.onlyAudits)); +} + +/** + * Retrieve the category references from the groups (already processed from the audits). + * Used in the code-pushup preset + * @param opt TSPluginOptions + * @returns The array of category references + */ +export async function getCategoryRefsFromGroups( + opt?: TypescriptPluginOptions, +): Promise { + return getGroups(opt).map(({ slug }) => ({ + plugin: TYPESCRIPT_PLUGIN_SLUG, + slug, + weight: 1, + type: 'group', + })); +} + +/** + * Retrieve the category references from the audits. + * @param opt TSPluginOptions + * @returns The array of category references + */ +export async function getCategoryRefsFromAudits( + opt?: TypescriptPluginOptions, +): Promise { + return AUDITS.filter(filterAuditsBySlug(opt?.onlyAudits)).map(({ slug }) => ({ + plugin: TYPESCRIPT_PLUGIN_SLUG, + slug, + weight: 1, + type: 'audit', + })); +} + +export const CATEGORY_MAP: Record = { + typescript: { + slug: 'type-safety', + title: 'Type Safety', + description: 'TypeScript diagnostics and type-checking errors', + refs: await getCategoryRefsFromGroups(), + }, + 'bug-prevention': { + slug: 'bug-prevention', + title: 'Bug prevention', + description: 'Type checks that find **potential bugs** in your code.', + refs: await getCategoryRefsFromGroups({ + onlyAudits: [ + 'syntax-errors', + 'semantic-errors', + 'internal-errors', + 'configuration-errors', + 'no-implicit-any-errors', + ], + }), + }, + miscellaneous: { + slug: 'miscellaneous', + title: 'Miscellaneous', + description: + 'Errors that do not bring any specific value to the developer, but are still useful to know.', + refs: await getCategoryRefsFromGroups({ + onlyAudits: ['unknown-codes', 'declaration-and-language-service-errors'], + }), + }, +}; + +export function getCategories() { + return Object.values(CATEGORY_MAP); +} + +export function logSkippedAudits(audits: Audit[]) { + const skippedAudits = AUDITS.filter( + audit => !audits.some(filtered => filtered.slug === audit.slug), + ).map(audit => kebabCaseToCamelCase(audit.slug)); + if (skippedAudits.length > 0) { + console.warn(`Skipped audits: [${skippedAudits.join(', ')}]`); + } +} diff --git a/packages/plugin-typescript/src/lib/utils.unit.test.ts b/packages/plugin-typescript/src/lib/utils.unit.test.ts new file mode 100644 index 000000000..21f4b96eb --- /dev/null +++ b/packages/plugin-typescript/src/lib/utils.unit.test.ts @@ -0,0 +1,150 @@ +import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest'; +import { type Audit, categoryRefSchema } from '@code-pushup/models'; +import { AUDITS } from './constants.js'; +import { + filterAuditsByCompilerOptions, + filterAuditsBySlug, + getCategoryRefsFromGroups, + logSkippedAudits, +} from './utils.js'; + +describe('filterAuditsBySlug', () => { + const mockAudit = { slug: 'strict-function-types' } as Audit; + + it('should return true if slugs are undefined', () => { + expect(filterAuditsBySlug(undefined)(mockAudit)).toBe(true); + }); + + it('should return true if slugs are empty', () => { + expect(filterAuditsBySlug([])(mockAudit)).toBe(true); + }); + + it('should return true if slugs are including the current audit slug', () => { + expect(filterAuditsBySlug(['strict-function-types'])(mockAudit)).toBe(true); + }); + + it('should return false if slugs are not including the current audit slug', () => { + expect(filterAuditsBySlug(['verbatim-module-syntax'])(mockAudit)).toBe( + false, + ); + }); +}); + +describe('filterAuditsByCompilerOptions', () => { + it('should return false if the audit is false in compiler options', () => { + expect( + filterAuditsByCompilerOptions( + { + strictFunctionTypes: false, + }, + ['strict-function-types'], + )({ slug: 'strict-function-types' }), + ).toBe(false); + }); + + it('should return false if the audit is undefined in compiler options', () => { + expect( + filterAuditsByCompilerOptions( + { + strictFunctionTypes: undefined, + }, + ['strict-function-types'], + )({ slug: 'strict-function-types' }), + ).toBe(false); + }); + + it('should return false if the audit is enabled in compiler options but not in onlyAudits', () => { + const onlyAudits = ['strict-null-checks']; + expect( + filterAuditsByCompilerOptions( + { + strictFunctionTypes: true, + }, + onlyAudits, + )({ slug: 'strict-function-types' }), + ).toBe(false); + }); + + it('should return true if the audit is enabled in compiler options and onlyAudits is empty', () => { + expect( + filterAuditsByCompilerOptions( + { + strictFunctionTypes: true, + }, + [], + )({ slug: 'strict-function-types' }), + ).toBe(true); + }); + + it('should return true if the audit is enabled in compiler options and in onlyAudits', () => { + expect( + filterAuditsByCompilerOptions( + { + strictFunctionTypes: true, + }, + ['strict-function-types'], + )({ slug: 'strict-function-types' }), + ).toBe(true); + }); +}); + +describe('getCategoryRefsFromGroups', () => { + it('should return all groups as categoryRefs if no compiler options are given', async () => { + const categoryRefs = await getCategoryRefsFromGroups(); + expect(categoryRefs).toHaveLength(3); + expect(() => + categoryRefs.map(categoryRefSchema.parse as () => unknown), + ).not.toThrow(); + }); + + it('should return all groups as categoryRefs if compiler options are given', async () => { + const categoryRefs = await getCategoryRefsFromGroups({ + tsConfigPath: 'tsconfig.json', + }); + expect(categoryRefs).toHaveLength(3); + }); + + it('should return a subset of all groups as categoryRefs if compiler options contain onlyAudits filter', async () => { + const categoryRefs = await getCategoryRefsFromGroups({ + tsConfigPath: 'tsconfig.json', + onlyAudits: ['semantic-errors'], + }); + expect(categoryRefs).toHaveLength(1); + }); +}); + +describe('logSkippedAudits', () => { + beforeEach(() => { + vi.mock('console', () => ({ + warn: vi.fn(), + })); + }); + + afterEach(() => { + vi.restoreAllMocks(); + }); + + it('should not warn when all audits are included', () => { + logSkippedAudits(AUDITS); + + expect(console.warn).not.toHaveBeenCalled(); + }); + + it('should warn about skipped audits', () => { + logSkippedAudits(AUDITS.slice(0, -1)); + + expect(console.warn).toHaveBeenCalledTimes(1); + expect(console.warn).toHaveBeenCalledWith( + expect.stringContaining(`Skipped audits: [`), + ); + }); + + it('should camel case the slugs in the audit message', () => { + logSkippedAudits(AUDITS.slice(0, -1)); + + expect(console.warn).toHaveBeenCalledTimes(1); + expect(console.warn).toHaveBeenCalledWith( + expect.stringContaining(`unknownCodes`), + ); + }); +});