Validation messages
Specification
Validation messages reside in the separate schema, decoupled from the validation logic. The reasons for the decoupling are the following:
Establish a clear separation of concerns.
Prevent from garbaging complex validation rules with the messages.
Have a single validation schema associated with any amount of localized messages schema.
Resolver
Any message can be a plain string, or a resolver function that returns a string.
Resolvers are useful when a validation message derives not only from the field's validity, but also from its state, fields or a form. This way you can include the current value of a field in the error message, for example.
Definition
Parameters
Parameter name | Type | Description |
|
| The value of a field. |
|
| Props of a field. |
|
| State of all fields. |
|
| Form component reference. |
|
| Extra parameters passed in the async validation payload. |
Message types
Validation messages schema supports the following message types:
Type | Description |
| A message for an empty required field. |
| A message for an invalid field. |
| A message corresponding to the invalid asynchronous validation. |
Selectors
Selectors are used to associate message resolvers to the relevant field(s).
Selector | Description |
| Name-specific messages. |
| Type-specific messages. |
| General messages. Those are used as fallback values in case more specific messages are not present. |
Each selector expects the map of the selector values and its resolvers:
Named resolvers
It is possible to associate validation rules with the specific validation messages. To do that, provide the { ruleName: message }
map as the value of the rule
key in the respective selector:
Note: Named resolver must have the corresponding validation rule with the same name in order to resolve. Otherwise, the closest validation message will be returned by the resolver. Considering the example above, if
minLength
rule doesn't exist and thetype=["password"]
field is invalid, the closestinvalid
resolver will be used (which istype.password.invalid
in this case).
Fallbacks
Whenever the resolver cannot resolve it would attempt to grab the closest resolver recursively and resolve it instead. Fallback resolvers are iterated by their specificity, which can be presented as follows:
Fallback sequence
Here is the resolver paths (listed by priority, from highest to lowest):
messages.name[fieldName][ruleName]
messages.name[fieldName].invalid
/messages.name[fieldName].missing
messages.type[fieldType][ruleName]
messages.type[fieldType]invalid
/messages.type[fieldType].missing
messages.general.invalid
/messages.general.missing
Example
Consider the following validation messages:
And the following component layout:
With the given scenario the includesAt
validation rule would reject, marking the field as unexpected. This must be reflected in the UI using the validation messages schema.
This is the priority sequence in which resolvers will attempt to resolve the validation message:
name.userEmail.rule.includesAt
name.userEmail.invalid
type.email.invalid
general.invalid
The validation message resolves as soon as the resolver returns the value. The same sequence applies for the type-specific named resolvers.
Last updated