Validators
Installโ
- npm
- yarn
- pnpm
npm install @basketry/typescript-validators
yarn add @basketry/typescript-validators
pnpm add @basketry/typescript-validators
Basic Usageโ
{
"source": "petstore.json",
"parser": "@basketry/openapi-3",
"generators": ["@basketry/typescript", "@basketry/typescript-validators"],
"output": "src"
}
File Structureโ
This generator will create a validators.ts
file that contains all the validation methods for the API. It will also create a sanitizers.ts
file that contians "sanitization" methods. These methods will remove all properties from an input object that are not defined by the source schema.
By default, the file will be nested within a directory named after the API major version. This behavior can be disabled by using the includeVersion
option from the Typescript Generator.
my-project/
โโโ node_modules/
โโโ src/
โ โโโ v1/ <-- generated
โ โ โโโ date-utils.ts <-- generated
โ โ โโโ sanitizers.ts <-- generated
โ โ โโโ validators.ts <-- generated
โ โ โโโ types.ts
โ โโโ index.ts
โโโ .gitignore
โโโ basketry.config.json
โโโ package.json
โโโ petstore.json
โโโ README.md
Optionsโ
This generator depends on the @basketry/typescript
generator and all of their applied options will also apply to files emitted by this generator.
See:
@basketry/typescript
options
The @basketry/typescript-validators
generator accepts the following options to customize the generated code:
typesImportPath
โ
- Type:
string
- specifies the path to the types file.
The files produced by this generator depend on the types and interfaces emitted by the @basketry/typescript
generator. In most cases, the types will exist in a file named types.ts
within the same directory as the validators file.
By default, the generated code will import the types file as follows:
import * as types from "./types.ts";
If the types file is located in a different directory, you can use this option to specify the path to the types file. This path can be any string that you would normally require
or import
. This can be useful when the types.ts
file is distributed as part of a library or package.
{
"source": "petstore.json",
"parser": "@basketry/openapi-3",
"generators": ["@basketry/typescript", "@basketry/typescript-validators"],
"output": "src",
"options": {
"typescriptValidators": {
"typesImportPath": "@petstore/sdk/lib/types"
}
}
}
This option as configured will now emit the following code at the top of the generated file:
import * as types from "@petstore/sdk/lib/types";
Examplesโ
Validator Functionsโ
A validation method is generated for each type, enum, an union defined by the source schema. These methods will return an array of ValidationError
objects if the input object does not match the schema. The ValidationError
object contains the path
to the property that failed validation and a message
describing the error.
Validation methods do not throw errors or otherwise interupt program flow. They will only return an array of ValidationError
objects. It is up to the caller to determine how to handle these errors.
import { type Pet } from "./v1/types";
import { validatePet, type ValidationError } from "./v1/validators";
app.post("/pet", (req, res) => {
const pet = req.body;
// Validate the input data
const errors: ValidationError[] = validatePet(pet);
if (errors.length) {
console.error("Validation failed:", errors);
res.status(400).send({ errors });
return;
} else {
console.log("Validation passed!");
// TODO: Handle valid data
}
});
Sanitizer Functionsโ
Sanitizer functions create a deep clone of the input object removing any properties from an input object that are undefined
or are not defined by the source schema. This can be useful when accepting data from an external source and you want to ensure that only known properties are handled by the API.
Sanitiser functions do not validate input data. Invalid properties will be copied as is to the output object. It is up to the caller to validate the input data in addition to calling the sanitiser.
import { type Pet } from "./v1/types";
import { validatePet, type ValidationError } from "./v1/validators";
app.post("/pet", (req, res) => {
const pet = req.body;
// Remove all properties not defined by the API schema
const sanitizedPet = sanitizePet(pet);
// TODO: Handle data
});
Validator Class Wrappersโ
A convience class is generated for each of the service interfaces defined by the source schema. These classes implement the service interface and wrap another instance of the service providing both validation and sanitization.
Each validator class accepts an instance of the service interface in its constructor. For each of the different reponse types defined by the service, the class will also accept a function converts ValidationErrors
and any caught exceptions into that resonse type.
Note that validator classes are best used in conjunction with services the return response envelopes. The @basketry/rules
packages provides a rule to ensure that your source schema only defines methods that return response envelopes.
import { type PetResponse, type PetService } from "./v1/types";
import { type ResponseBuilder, ValidatedPetService } from "./v1/validators";
import { MyPetService } from "./my-pet-service";
// Create an instance of the concret service implementation
const petService = new MyPetService();
// This function instructs the validator class in what do with validation errors
const buildPetResponse: ResponseBuilder<PetResponse> = (
validationErrors: ValidationError[],
caughtError: any
) => {
// TODO: Construct and return a `PetResponse` object based on the errors
};
// Wrap our service with the validator class
// Note that both our service instance and the validator class share the same `PetService` interface
const validatedPetService: PetService = new ValidatedPetService(petService, {
buildPetResponse,
});
// Now we can call the service methods. All boilerplate validation and sanitization is handled for us.
const petResponse = await validatedPetService.createPet({ name: "Fido" });
Internally, the ValidatedPetService
is implemented something like this:
import * as types from "./types";
import * as sanitizers from "./sanitizers";
export class ValidatedPetService implements types.PetService {
constructor(
private readonly service: types.PetService,
private readonly handlers: {
buildPetResponse: ResponseBuilder<types.PetResponse>;
}
) {}
async createPet(params: types.CreatePetParams) {
let validationErrors: ValidationError[] = [];
try {
validationErrors = validateCreatePetParams(params);
if (validationErrors.length) {
return sanitizers.sanitizeCreatePetResponse(
this.handlers.buildPetResponse(validationErrors, undefined)
);
}
const sanitizedParams = sanitizers.sanitizeCreatePetParams(params);
return sanitizers.sanitizeCreatePetResponse(
await this.service.CreatePet(sanitizedParams)
);
} catch (err) {
return sanitizers.sanitizeCreatePetResponse(
this.handlers.buildPetResponse(validationErrors, err)
);
}
}
}