Skip to content

Commit

Permalink
Add contract helper
Browse files Browse the repository at this point in the history
  • Loading branch information
igorkamyshev committed Jul 30, 2024
1 parent 8c9b7c8 commit 8eb2922
Show file tree
Hide file tree
Showing 4 changed files with 151 additions and 11 deletions.
18 changes: 8 additions & 10 deletions apps/website/docs/contracts/cookbook/custom_matchers.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,24 +5,22 @@ Since `@withease/contracts` is built on top of [_Contract_](/protocols/contract)
Let us write a custom matcher that checks if an age of a user is within a certain range:

```ts
import { type Contract } from '@withease/contracts';
import { type Contract, contract, nul } from '@withease/contracts';

function age(min, max): Contract<number, number> {
return {
isData: (data) => data >= min && data <= max,
getErrorMessages: (data) =>
`Expected a number between ${min} and ${max}, but got ${data}`,
};
}
const age = contract(num, ({ min, max }: { min: number; max: number }) => ({
isData: (data) => data >= min && data <= max,
getErrorMessages: (data) =>
`Expected a number between ${min} and ${max}, but got ${data}`,
}));
```

Now you can use this matcher in your schema:

```ts
import { obj, str, and, num } from '@withease/contracts';
import { obj, str } from '@withease/contracts';

const User = obj({
name: str,
age: and(num, age(18, 100)),
age: age(18, 100),
});
```
2 changes: 1 addition & 1 deletion packages/contracts/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@
"size-limit": [
{
"path": "./dist/contracts.js",
"limit": "797 B"
"limit": "885 B"
}
]
}
79 changes: 79 additions & 0 deletions packages/contracts/src/contracts.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import {
arr,
and,
tuple,
contract,
Contract,
} from './index';

Expand Down Expand Up @@ -547,3 +548,81 @@ describe('tuple', () => {
`);
});
});

describe('contract, no base', () => {
const onlySome = contract((x: number) => ({
isData: (t): t is number => t === x,
getErrorMessages: (t) => ['expected ' + x + ', got ' + t],
}));

const onlyOne = onlySome(1);
const onlyTwo = onlySome(2);

it('valid', () => {
expect(onlyOne.isData(1)).toBeTruthy();
expect(onlyOne.getErrorMessages(1)).toEqual([]);

expect(onlyTwo.isData(2)).toBeTruthy();
expect(onlyTwo.getErrorMessages(2)).toEqual([]);
});

it('invalid', () => {
expect(onlyOne.isData(2)).toBeFalsy();
expect(onlyOne.getErrorMessages(2)).toMatchInlineSnapshot(`
[
"expected 1, got 2",
]
`);

expect(onlyTwo.isData(1)).toBeFalsy();
expect(onlyTwo.getErrorMessages(1)).toMatchInlineSnapshot(`
[
"expected 2, got 1",
]
`);
});
});

describe('contract, with base', () => {
const age = contract(num, ({ min, max }: { min: number; max: number }) => ({
isData: (t): t is number => t >= min && t <= max,
getErrorMessages: (t) => [
'expected ' + min + ' <= x <= ' + max + ', got ' + t,
],
}));

it('valid', () => {
const cntrct = age({ min: 18, max: 100 });

expect(cntrct.isData(18)).toBeTruthy();
expect(cntrct.getErrorMessages(18)).toEqual([]);

expect(cntrct.isData(100)).toBeTruthy();
expect(cntrct.getErrorMessages(100)).toEqual([]);
});

it('invalid', () => {
const cntrct = age({ min: 18, max: 100 });

expect(cntrct.isData('KEK')).toBeFalsy();
expect(cntrct.getErrorMessages('KEK')).toMatchInlineSnapshot(`
[
"expected number, got string",
]
`);

expect(cntrct.isData(17)).toBeFalsy();
expect(cntrct.getErrorMessages(17)).toMatchInlineSnapshot(`
[
"expected 18 <= x <= 100, got 17",
]
`);

expect(cntrct.isData(101)).toBeFalsy();
expect(cntrct.getErrorMessages(101)).toMatchInlineSnapshot(`
[
"expected 18 <= x <= 100, got 101",
]
`);
});
});
63 changes: 63 additions & 0 deletions packages/contracts/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -352,6 +352,69 @@ export function tuple(...contracts: Array<Contract<unknown, any>>): any {
};
}

/**
* Creates a _Contract_ based on a passed function.
*
* @overload "contract(fn)"
*
* @example
*
* const age = contract((min, max) => ({
* isData: (data) => typeof data === 'number' && data >= min && data <= max,
* getErrorMessages: (data) =>
* `Expected a number between ${min} and ${max}, but got ${data}`,
* }));
*/
export function contract<P, O>(
fn: (config: P) => Contract<unknown, O>
): (config: P) => Contract<unknown, O>;

/**
* Creates a _Contract_ based on a passed function and a base _Contract_.
* Base _Contract_ is checked first, then the _Contract_ created by the function.
*
* @overload "contract(base, fn)"
*
* @example
*
* const age = contract(num, (min, max) => ({
* isData: (data) => data >= min && data <= max,
* getErrorMessages: (data) =>
* `Expected a number between ${min} and ${max}, but got ${data}`,
* }));
*/
export function contract<P, I, O extends I>(
base: Contract<unknown, I>,
fn: (config: P) => Contract<I, O>
): (config: P) => Contract<unknown, O>;

export function contract<P, I, O extends I>(
base: Contract<unknown, I> | ((config: P) => Contract<I, O>),
fn?: (config: P) => Contract<I, O>
): (config: P) => Contract<I, O> {
const creator: any = fn ? fn : (base as (config: P) => Contract<I, O>);
const realBase = fn ? (base as Contract<unknown, I>) : null;
return (params: P) => {
const check = (data: unknown): data is O => {
if (realBase) {
return realBase.isData(data) && creator(params).isData(data);
}
return creator(params).isData(data);
};

return {
isData: check,
getErrorMessages: createGetErrorMessages(check, (data) => {
if (realBase && !realBase.isData(data)) {
return realBase.getErrorMessages(data);
}

return creator(params).getErrorMessages(data);
}),
};
};
}

// -- utils

function createSimpleContract<T>(
Expand Down

0 comments on commit 8eb2922

Please sign in to comment.