Skip to content

Commit

Permalink
feat(effectResolver): returns either all errors or only the first one…
Browse files Browse the repository at this point in the history
… based on criteriaMode (#737)
  • Loading branch information
jorisre authored Jan 30, 2025
1 parent 9a94555 commit 12d7d8e
Show file tree
Hide file tree
Showing 3 changed files with 127 additions and 7 deletions.
36 changes: 35 additions & 1 deletion effect-ts/src/__tests__/__snapshots__/effect-ts.ts.snap
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ exports[`effectTsResolver > should return a single error from effectTsResolver w
{
"errors": {
"animal": {
"message": "Expected "snake", actual ["dog"]",
"message": "Expected string, actual ["dog"]",
"ref": undefined,
"type": "Type",
},
Expand Down Expand Up @@ -38,3 +38,37 @@ exports[`effectTsResolver > should return a single error from effectTsResolver w
"values": {},
}
`;

exports[`effectTsResolver > should return all the errors from effectTsResolver when validation fails with \`validateAllFieldCriteria\` set to true 1`] = `
{
"errors": {
"phoneNumber": {
"message": "Please enter a valid phone number in international format.",
"ref": {
"name": "phoneNumber",
},
"type": "Refinement",
"types": {
"Refinement": "Please enter a valid phone number in international format.",
"Type": "Expected undefined, actual "123"",
},
},
},
"values": {},
}
`;

exports[`effectTsResolver > should return the first error from effectTsResolver when validation fails with \`validateAllFieldCriteria\` set to firstError 1`] = `
{
"errors": {
"phoneNumber": {
"message": "Please enter a valid phone number in international format.",
"ref": {
"name": "phoneNumber",
},
"type": "Refinement",
},
},
"values": {},
}
`;
61 changes: 61 additions & 0 deletions effect-ts/src/__tests__/effect-ts.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import { Schema } from 'effect';
import { effectTsResolver } from '..';
import { fields, invalidData, schema, validData } from './__fixtures__/data';

Expand All @@ -21,4 +22,64 @@ describe('effectTsResolver', () => {

expect(result).toMatchSnapshot();
});

it('should return the first error from effectTsResolver when validation fails with `validateAllFieldCriteria` set to firstError', async () => {
const SignupSchema = Schema.Struct({
phoneNumber: Schema.optional(
Schema.String.pipe(
Schema.pattern(/^\+\d{7,15}$/, {
message: () =>
'Please enter a valid phone number in international format.',
}),
),
),
});

const result = await effectTsResolver(SignupSchema)(
{ phoneNumber: '123' },
undefined,
{
fields: {
phoneNumber: {
ref: { name: 'phoneNumber' },
name: 'phoneNumber',
},
},
criteriaMode: 'firstError',
shouldUseNativeValidation,
},
);

expect(result).toMatchSnapshot();
});

it('should return all the errors from effectTsResolver when validation fails with `validateAllFieldCriteria` set to true', async () => {
const SignupSchema = Schema.Struct({
phoneNumber: Schema.optional(
Schema.String.pipe(
Schema.pattern(/^\+\d{7,15}$/, {
message: () =>
'Please enter a valid phone number in international format.',
}),
),
),
});

const result = await effectTsResolver(SignupSchema)(
{ phoneNumber: '123' },
undefined,
{
fields: {
phoneNumber: {
ref: { name: 'phoneNumber' },
name: 'phoneNumber',
},
},
criteriaMode: 'all',
shouldUseNativeValidation,
},
);

expect(result).toMatchSnapshot();
});
});
37 changes: 31 additions & 6 deletions effect-ts/src/effect-ts.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { toNestErrors, validateFieldsNatively } from '@hookform/resolvers';
import { Effect } from 'effect';

import { ArrayFormatter, decodeUnknown } from 'effect/ParseResult';
import type { FieldErrors } from 'react-hook-form';
import { appendErrors, type FieldError } from 'react-hook-form';
import type { Resolver } from './types';

export const effectTsResolver: Resolver =
Expand All @@ -16,11 +16,36 @@ export const effectTsResolver: Resolver =
Effect.flip(ArrayFormatter.formatIssue(parseIssue)),
),
Effect.mapError((issues) => {
const errors = issues.reduce((acc, current) => {
const key = current.path.join('.');
acc[key] = { message: current.message, type: current._tag };
return acc;
}, {} as FieldErrors);
const validateAllFieldCriteria =
!options.shouldUseNativeValidation && options.criteriaMode === 'all';

const errors = issues.reduce(
(acc, error) => {
const key = error.path.join('.');

if (!acc[key]) {
acc[key] = { message: error.message, type: error._tag };
}

if (validateAllFieldCriteria) {
const types = acc[key].types;
const messages = types && types[String(error._tag)];

acc[key] = appendErrors(
key,
validateAllFieldCriteria,
acc,
error._tag,
messages
? ([] as string[]).concat(messages as string[], error.message)
: error.message,
) as FieldError;
}

return acc;
},
{} as Record<string, FieldError>,
);

return toNestErrors(errors, options);
}),
Expand Down

0 comments on commit 12d7d8e

Please sign in to comment.