π Revolutionize your TS type testing with a fast, smooth, and flexible WYSIWYG experience!
demo.mp4
npm install --save-dev typroof
# Or
yarn add -D typroof
# Or
pnpm add -D typroof
This tool lets you write tests for your TypeScript type definitions just like what you would do for your runtime code like Jest, Vitest, etc. It provides similar APIs like test
, expect
, describe
in Jest, but for your type definitions.
By default, these test files should end with .proof.ts
or be placed in a proof/
directory. They will not be really executed, but instead be parsed and statically analyzed against your type definitions. Special constructs such as expect<Append<'foo', 'bar'>>().to(equal<'foobar'>)
or expect<Append<'foo', 42>>().to(error)
will be parsed and analyzed to see if they are valid.
The typroof
CLI will search for these test files and run static analysis on them. It will then report the results to the console.
[npx] typroof [path]
Use typroof --help
for usage information.
Assume that you write a string-utils.ts
file with the following type definitions:
export type Append<S extends string, Ext extends string> = `${S}${Ext}`;
export type Prepend<S extends string, Start extends string> = `${Start}${S}`;
export const append = <S extends string, Ext extends string>(s: S, ext: Ext): Append<S, Ext> =>
`${s}${ext}`;
Then you can write a string-utils.proof.ts
file to test them:
import { describe, equal, error, expect, extend, it, test } from 'typroof';
import { append } from './string-utils';
import type { Append, Prepend } from './string-utils';
test('Append', () => {
expect<Append<'foo', 'bar'>>().to(equal<'foobar'>);
// If you write an assertion that fails, you can immediately see the error message by TypeScript:
// expect<Append<'foo', 'bar'>>().to(equal<'foo'>);
// // ~~~~~~~~~~~~
// // Argument of type '...' is not assignable to parameter
// // of type '"Expect `'foobar'` to equal `'foo'`, but does not"'
expect<Append<'foo', 'bar'>>().to(extend<string>);
expect<Append<'foo', 'bar'>>().not.to(extend<number>);
expect(append('foo', 'bar')).to(equal('foobar' as const));
});
describe('Prepend', () => {
it('should prepend a string to another', () => {
expect<Prepend<'foo', 'bar'>>().to(equal<'foobar'>);
});
it('should accept only strings', () => {
// @ts-expect-error - should trigger error
expect<Prepend<42, 43>>().to(error);
});
});
You can see that the APIs are very similar to Jest. The test
function and its alias it
to create a test case. The describe
function to create a test suite (group of test cases). The expect
function to create an assertion. The toEqual
function to assert that the type is equal to the expected type.
You may notice that we use two different syntaxes for the equal
matcher. You can either pass the expected type as a type argument, or pass it as a value argument. The former is useful when testing generic types, while the latter will be useful when testing the return type of a function. You can choose whichever you like. Even expect
itself can use a value argument instead of a type argument, such as expect('foobar').to(equal<'foobar'>)
.
Then run typroof
to test your type definitions:
The equal
matcher strictly checks if the type is equal to the expected type. If you want to check if the type is assignable to the expected type, you can use the extend
matcher:
expect<Append<'foo', 'bar'>>().to(extend<string>);
Matchers are indicators that can be used via expect<T>(x?: T).to(matcher<...>)
or expect<T>(x?: T).not.to(matcher<...>)
to assert the type. Typroof provides some built-in matchers, and you can also create your own matchers (see Plugin API).
For example:
expect<Append<'foo', 'bar'>>().to(equal<'foobar'>);
// @ts-expect-error - The `error` matcher will handle `@ts-expect-error` comments correctly
expect<Append<1, 2>>().to(error);
expect<Append<'foo', 'bar'>>().not.to(beAny);
expect(append('foo', 'bar')).to(extend<string>);
expect('baz' as const).to(equal('baz' as const));
The actual type to test can either be passed as a type argument to expect
, or passed as a value argument to expect
.The former is useful when testing generic types, while the latter will be useful when testing the return type of a function. You can choose whichever you like.
The following is a list of built-in matchers. Remember that you can also use expect.not
to negate them.
Expect a pre emitted diagnostic between the start and end of the given type.
Expect the type to be equal to the given type.
Expect the type to be any
.
Expect the type to be never
.
Expect the type to be null
.
Expect the type to be undefined
.
Expect the type to be null
, undefined
or null | undefined
.
Expect the type to be true
, false
or boolean
.
Expect the type to be true
.
Expect the type to be false
.
Expect the type to be assignable to the given type (i.e. the given type should be a supertype of the type).
β Warning: In TypeScript, any
is both a subtype and a supertype of all other types. Therefore, expect<string>().to(extend<any>)
and expect<any>().to(extend<string>)
will both pass. The exception is never
, which is not assignable to any type (thus expect<any>().to(extend<never>)
fails). Keep this in mind, as it may lead to unexpected results when working with any
or never
. Use strictExtend
for a stricter version that fails if either the type or the given type is never
or any
.
Like extend
, but fails if either the type or the given type is never
or any
.
Expect the given type to be assignable to the type (i.e. the given type should be a subtype of the type).
β Warning: In TypeScript, any
is both a subtype and a supertype of all other types. Therefore, expect<string>().to(cover<any>)
and expect<any>().to(cover<string>)
will both pass. The exception is never
, which is not assignable to any type (thus expect<never>().to(cover<any>)
fails). Keep this in mind, as it may lead to unexpected results when working with any
or never
. Use strictCover
for a stricter version that fails if either the type or the given type is never
or any
.
Like cover
, but fails if either the type or the given type is never
or any
.
You can create a typroof.config.ts
file in the root directory of your project to configure Typroof. It should export the configuration object. Typroof exports a defineConfig
function to help you define the configuration object.
// typroof.config.ts
import { defineConfig } from 'typroof/config';
export default defineConfig({
testFiles: '**/*.proof.ts',
});
Don't forget to add your config file to tsconfig.json
.
β Warning: Make sure not to import value directly from typroof
in your config file (use typroof/config
, typroof/plugin
, etc. instead), otherwise you may get circular dependencies error when Typroof is loading config.
Typroof provides type definitions for the configuration object, so you can get type hints in your editor.
You can use either .ts
, .mts
, .cts
, .js
, .mjs
or .cjs
as the extension of the config file. The priority is .ts
> .mts
> .cts
> .js
> .mjs
> .cjs
.
You can invoke the typroof
CLI directly, or use the typroof
function in your code.
import typroof, { formatGroupResult, formatSummary } from 'typroof';
const startedAt = new Date();
const results = await typroof();
const finishedAt = new Date();
for (const result of results) {
console.log(formatGroupResult(result.rootGroupResult));
console.log();
}
console.log(
formatSummary({ groups: results.map((r) => r.rootGroupResult), startedAt, finishedAt }),
);
Typroof supports plugins to extend its matchers.
Matchers are just indicators to tell Typroof how to assert the type. The process actually involves two steps: The type level validation and the code analysis using ts-morph.
Take a look at how the equal
matcher is implemented:
// `registerAnalyzer` is not actually exported,
// it is just an example to show how to create custom matchers.
import { match, registerAnalyzer } from 'typroof/plugin';
// `equal` is a matcher that takes a type argument.
// If no argument is needed, you can simply use `match<'matcherName'>()`
// instead of a function.
export const equal = <U>(y?: U) => match<'equal', U>();
/**
* Check whether `T` is equal to `U`.
* It is a utility type used in the type level validation step.
*/
type Equals<T, U> =
(<G>() => G extends T ? 1 : 2) extends <G>() => G extends U ? 1 : 2 ? true : false;
// Define how the type level validation step works.
// If type-level validation is the only thing you need to do (e.g., `equal`),
// it should return a boolean type.
// Otherwise, it should return a `ToAnalyze<SomeType>`, e.g. `error` returns
// `ToAnalyze<never>`, the `ToAnalyze` means to determine whether the assertion
// passed or not needs further code analysis. You can pass any type to
// `ToAnalyze` for the code analysis step to use, but here `error` does not need it.
declare module 'typroof' {
interface Validator<T, U> {
// Here `equal` is the name of the matcher,
// it must be the same as that in `match<'equal'>()`.
equal: Equals<T, U>;
}
}
// The `registerToEqual` helper function should be called somewhere before code analysis is
// executed to register the corresponding analyzer for the `equal` matcher.
export const registerToEqual = () => {
// The registered analyzer is called when the type-level validator does not evaluate to
// `true` (i.e, `false` or `ToAnalyze<SomeType>`).
// If the validator evaluates to `ToAnalyze<SomeType>`, the `SomeType` object will be
// accessible as the `validationResult` property in the third argument of the analyzer
// as a `Type<ts.Type>` object.
// Here, since `equal` is a type-level only matcher, so the analyzer is only used to
// report the error message, and `validationResult` is not used, but it can be useful
// if you are implementing a matcher like `haveJSDoc` that needs to analyze the type
// using ts-morph.
registerAnalyzer('equal', (actual, expected, { not, validationResult }) => {
// Here `equal` is a type-level only assertion, so we just need to report the error.
// But you can do anything you want here, e.g., `error` checks if the type emits an
// error. The fourth argument provides necessary metadata for you to achieve almost
// anything you can via ts-morph.
const actualText = chalk.bold(actual.text);
const expectedType = chalk.bold(expected.getText());
const actualType = chalk.bold(actual.type.getText());
// Throw a string to report the error.
throw (
`Expect ${actualText} ${not ? 'not ' : ''}to equal ${expectedType}, ` +
`but got ${actualType}.`
);
});
};
And that is all. As you see here, it is really easy to create custom matchers, and highly customizable powered by ts-morph.
Typroof automatically generates a compile-time error message if the assertion fails:
expect<Append<'foo', 'bar'>>().to(equal<'foo'>);
// ~~~~~~~~~~~~
// Argument of type '...' is not assignable to parameter
// of type "Validation failed: equal<'foobar', 'foo'>"
But if you want a more readable error message, you can change the validator to return a string instead of a boolean if the assertion fails:
import type { Stringify } from 'typroof';
declare module 'typroof' {
interface Validator<T, U, Not extends boolean> {
// `Not` is `true` if `.not` is used, otherwise `false`.
equal: Not extends false ?
// If `.not` is not used, return a string as the error message if `T` is not equal to `U`
Equals<T, U> extends true ?
true
: `Expect \`${Stringify<T>}\` to equal \`${Stringify<U>}\`, but does not`
: // If `.not` is used, return a string as the error message if `T` is equal to `U`
Equals<T, U> extends false ? false
: `Expect the type not to equal \`${Stringify<U>}\`, but does`;
}
}
Now letβs take a look at how error
is implemented:
import { match, registerAnalyzer } from 'typroof/plugin';
import type { ToAnalyze } from 'typroof/plugin';
export const error = match<'error'>();
declare module 'typroof' {
interface Validator<T, U> {
error: ToAnalyze<never>;
}
}
export const registerToError = () => {
registerAnalyzer('error', (actual, _expected, { diagnostics, not, statement }) => {
const diagnostic = diagnostics.find((diagnostic) => {
const start = diagnostic.getStart();
if (!start) return false;
const length = diagnostic.getLength();
if (!length) return false;
const end = start + length;
return start >= actual.node.getStart() && end <= actual.node.getEnd();
});
const triggeredError =
!!diagnostic ||
// Check if the error is suppressed by `@ts-expect-error`
statement.getLeadingCommentRanges().some(
(range) =>
range
.getText()
.replace(/^\/\*+\s*/, '')
.replace(/^\/\/\s*/, '')
.startsWith('@ts-expect-error') &&
!diagnostics.find((diagnostic) => diagnostic.getStart() === range.getPos()),
);
if (not ? triggeredError : !triggeredError) {
const actualText = chalk.bold(actual.text);
throw (
`Expect ${actualText} ${not ? 'not ' : ''}to trigger error, ` +
`but ${not ? 'did' : 'did not'}.`
);
}
});
};
So how to extend Typroof with your own matchers? The code below shows how to do it:
// In your entry file, e.g., `index.ts` or `main.ts`.
declare module 'typroof' {
interface Validator<T, U, Not> {
// Define your validator here
beFoo: Not extends false ?
T extends 'foo' ?
true
: `Expect \`${Stringify<T>}\` to be \`'foo'\`, but does not`
: T extends 'foo' ? `Expect the type not to be \`'foo'\`, but does`
: false;
}
}
// In your `typroof.config.ts`
import chalk from 'chalk';
import { defineConfig } from 'typroof/config';
import type { Plugin } from 'typroof/plugin';
// It is recommended to define the plugin as a factory function
const foo = (): Plugin => ({
// `name` is required, and recommended to be named as `typroof-plugin-*`
name: 'typroof-plugin-example',
analyzers: {
// Just like what you have seen in `registerAnalyzer`
beFoo: (actual, _expected, { not }) => {
const actualText = chalk.bold(actual.text);
const expectedType = chalk.bold('"foo"');
const actualType = chalk.bold(actual.type.getText());
throw (
`Expect ${actualText} ${not ? 'not ' : ''}to be ${expectedType}, ` +
`but got ${actualType}.`
);
},
},
});
export default defineConfig({
plugins: [foo()],
});
β Warning: Donβt forget to add chalk
to your devDependencies
(or dependencies
if youβre developing a plugin) in package.json
to avoid circular dependencies error when Typroof is loading config.
If you want to publish your plugin as a library, it is recommended to export the factory function to create the plugin object as the default export, and export the matchers as named exports:
// In your `index.ts`
import { match } from 'typroof/plugin';
import type { Plugin } from 'typroof/plugin';
declare module 'typroof' {
interface Validator<T, U> {
beFoo: Not extends false ?
T extends 'foo' ?
true
: `Expect \`${Stringify<T>}\` to be \`'foo'\`, but does not`
: T extends 'foo' ? `Expect the type not to be \`'foo'\`, but does`
: false;
}
}
const foo = (): Plugin => ({
/* ... */
});
export default foo;
/**
* [Matcher] Expect the type to be `"foo"`.
*/
export const beFoo = match<'beFoo'>();
Then your users can use your plugin like this:
// In their `typroof.config.ts`
import foo from 'typroof-plugin-example';
export default defineConfig({
plugins: [foo()],
});
// And somewhere in their test files
import { beFoo } from 'typroof-plugin-example';
test('foo', () => {
expect<'foo'>().to(beFoo);
});