@loopback/docs
Version:
Documentation files rendered at [https://loopback.io](https://loopback.io)
264 lines (217 loc) • 7.87 kB
Markdown
---
lang: en
title: 'Validation in REST Layer'
keywords: LoopBack 4.0, LoopBack 4, Node.js, TypeScript, OpenAPI
sidebar: lb4_sidebar
permalink: /doc/en/lb4/Validation-REST-layer.html
---
At the REST layer, request body is being validated against the OpenAPI schema
specification.
## Type Validation
Type validation comes out-of-the-box in LoopBack.
> Validation is applied on the parameters and the request body data. It also
> uses OpenAPI specification as the reference to infer the validation rules.
Take the `capacity` property in the [`CoffeeShop` model](Validation.md) as an
example; it is a number. When creating a `CoffeeShop` by calling `/POST`, if a
string is specified for the `capacity` property as below:
```json
{
"city": "Toronto",
"phoneNum": "416-111-1111",
"capacity": "100"
}
```
a "request body is invalid" error is expected:
```json
{
"error": {
"statusCode": 422,
"name": "UnprocessableEntityError",
"message": "The request body is invalid. See error object `details` property for more info.",
"code": "VALIDATION_FAILED",
"details": [
{
"path": ".capacity",
"code": "type",
"message": "should be number",
"info": {
"type": "number"
}
}
]
}
}
```
## Validation against OpenAPI Schema Specification
For validation against an OpenAPI schema specification, the
[AJV module](https://github.com/epoberezkin/ajv) is used to validate data with a
JSON schema generated from the OpenAPI schema specification. More details can be
found about
[validation keywords](https://github.com/epoberezkin/ajv#validation-keywords)
and
[annotation keywords](https://github.com/epoberezkin/ajv#annotation-keywords)
available in AJV. AJV can also be extended with custom keywords and formats, see
[AJV defining custom keywords page](https://ajv.js.org/custom.html).
Besides AJV, other third-party validation libraries, such as
[/joi](https://github.com/hapijs/joi) and
[class-validator](https://github.com/typestack/class-validator), can be used.
Below are a few examples of using AJV for validation. The source code of the
snippets can be found in the
[coffee-shop.model.ts in the example app](https://github.com/strongloop/loopback-next/blob/master/examples/validation-app/src/models/coffee-shop.model.ts).
{% include note.html content="The `jsonSchema` property expects [JSON Schema Draft-07](http://json-schema.org/draft/2019-09/json-schema-validation.html), which is then transformed into the [OAS 3](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md) variant." %}
### Example 1: Length limit
A typical validation example is to have a length limit on a string using the
keywords `maxLength` and `minLength`. For example:
{% include code-caption.html content="/src/models/coffee-shop.model.ts" %}
```ts
({
type: 'string',
required: true,
// --- add jsonSchema -----
jsonSchema: {
maxLength: 10,
minLength: 1,
},
// ------------------------
})
city: string;
```
If the `city` property in the request body does not satisfy the requirement as
follows:
```json
{
"city": "a long city name 123123123",
"phoneNum": "416-111-1111",
"capacity": 10
}
```
an error will occur with details on what has been violated:
```json
{
"error": {
"statusCode": 422,
"name": "UnprocessableEntityError",
"message": "The request body is invalid. See error object `details` property for more info.",
"code": "VALIDATION_FAILED",
"details": [
{
"path": ".city",
"code": "maxLength",
"message": "should NOT be longer than 10 characters",
"info": {
"limit": 10
}
}
]
}
}
```
### Example 2: Value range for a number
For numbers, the validation rules are used to specify the range of the value.
For example, any coffee shop would not be able to have more than 100 people, it
can be specified as follows:
{% include code-caption.html content="/src/models/coffee-shop.model.ts" %}
```ts
({
type: 'number',
required: true,
// --- add jsonSchema -----
jsonSchema: {
maximum: 100,
minimum: 1,
},
// ------------------------
})
capacity: number;
```
### Example 3: Pattern in a string
Model properties, such as phone number and postal/zip code, usually have certain
patterns. In this case, the `pattern` keyword is used to specify the
restrictions.
Below shows an example of the expected pattern of phone numbers, i.e. a sequence
of 10 digits separated by `-` after the 3rd and 6th digits.
{% include code-caption.html content="/src/models/coffee-shop.model.ts" %}
```ts
({
type: 'string',
required: true,
// --- add jsonSchema -----
jsonSchema: {
pattern: '\\d{3}-\\d{3}-\\d{4}',
},
// ------------------------
})
phoneNum: string;
```
{% include tip.html content="RegExp can be converted into a string with `.source` to avoid escaping backslashes" %}
## Customize validation errors
{% include note.html content="This section describes customization for legacy
[Action-based sequences](./REST-Action-sequence.md). Most LoopBack 4
applications utilize the new
[Middleware-based sequences](./REST-Middleware-sequence.md)." %}
Since the error is being caught at the REST layer, the simplest way to customize
the errors is to customize the
[sequence](https://loopback.io/doc/en/lb4/Sequence.html). It exists in all
LoopBack applications scaffolded by using the `lb4` command and can be found in
`src/sequence.ts`.
Let's take a closer look at how to customize the error. A few things to note in
the below code snippet:
1. inject RestBindings.SequenceActions.LOG_ERROR for logging error and
RestBindings.ERROR_WRITER_OPTIONS for options
2. customize error for particular endpoints
3. create a new error with customized properties
4. log the error using RestBindings.SequenceActions.LOG_ERROR
{% include code-caption.html content="/src/sequence.ts" %}
```ts
export class MySequence implements SequenceHandler {
// 1. inject RestBindings.SequenceActions.LOG_ERROR for logging error
// and RestBindings.ERROR_WRITER_OPTIONS for options
constructor(
/*..*/
(RestBindings.SequenceActions.LOG_ERROR)
protected logError: LogError,
(RestBindings.ERROR_WRITER_OPTIONS, {optional: true})
protected errorWriterOptions?: ErrorWriterOptions,
) {}
async handle(context: RequestContext) {
try {
// ...
} catch (err) {
this.handleError(context, err as HttpErrors.HttpError);
}
}
/**
* Handle errors
* If the request url is `/coffee-shops`, customize the error message.
*/
handleError(context: RequestContext, err: HttpErrors.HttpError) {
// 2. customize error for particular endpoint
if (context.request.url === '/coffee-shops') {
// if this is a validation error
if (err.statusCode === 422) {
const customizedMessage = 'My customized validation error message';
let customizedProps = {};
if (this.errorWriterOptions?.debug) {
customizedProps = {stack: err.stack};
}
// 3. Create a new error with customized properties
// you can change the status code here too
const errorData = {
statusCode: 422,
message: customizedMessage,
resolution: 'Contact your admin for troubleshooting.',
code: 'VALIDATION_FAILED',
...customizedProps,
};
context.response.status(422).send(errorData);
// 4. log the error using RestBindings.SequenceActions.LOG_ERROR
this.logError(err, err.statusCode, context.request);
// The error was handled
return;
}
}
// Otherwise fall back to the default error handler
this.reject(context, err);
}
}
```