UNPKG

@loopback/docs

Version:

Documentation files rendered at [https://loopback.io](https://loopback.io)

264 lines (217 loc) 7.87 kB
--- 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 [@hapi/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 @property({ 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 @property({ 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 @property({ 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( /*..*/ @inject(RestBindings.SequenceActions.LOG_ERROR) protected logError: LogError, @inject(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); } } ```