UNPKG

@snups/rjsf-utils

Version:
139 lines (125 loc) 5.94 kB
import get from 'lodash/get'; import { ID_KEY } from '../constants'; import hashForSchema from '../hashForSchema'; import { CustomValidator, ErrorSchema, ErrorTransformer, FormContextType, RJSFSchema, RJSFValidationError, StrictRJSFSchema, UiSchema, ValidationData, ValidatorType, } from '../types'; import deepEquals from '../deepEquals'; /** The type of the map of schema hash to schema */ export type SchemaMap<S extends StrictRJSFSchema = RJSFSchema> = { [hash: string]: S; }; /** An implementation of the `ValidatorType` interface that is designed for use in capturing schemas used by the * `isValid()` function. The rest of the implementation of the interface throws errors when it is attempted to be used. * An instance of the object allows the caller to capture the schemas used in calls to the `isValid()` function. These * captured schema, along with the root schema used to construct the object are stored in the map of schemas keyed by * the hashed value of the schema. NOTE: After hashing the schema, an $id with the hash value is added to the * schema IF that schema doesn't already have an $id, prior to putting the schema into the map. */ export default class ParserValidator<T = any, S extends StrictRJSFSchema = RJSFSchema, F extends FormContextType = any> implements ValidatorType<T, S, F> { /** The rootSchema provided during construction of the class */ readonly rootSchema: S; /** The map of schemas encountered by the ParserValidator */ schemaMap: SchemaMap<S> = {}; /** Construct the ParserValidator for the given `rootSchema`. This `rootSchema` will be stashed in the `schemaMap` * first. * * @param rootSchema - The root schema against which this validator will be executed */ constructor(rootSchema: S) { this.rootSchema = rootSchema; this.addSchema(rootSchema, hashForSchema<S>(rootSchema)); } /** Resets the internal AJV validator to clear schemas from it. Can be helpful for resetting the validator for tests. */ reset() { this.schemaMap = {}; } /** Adds the given `schema` to the `schemaMap` keyed by the `hash` or `ID_KEY` if present on the `schema`. If the * schema does not have an `ID_KEY`, then the `hash` will be added as the `ID_KEY` to allow the schema to be * associated with it's `hash` for future use (by a schema compiler). * * @param schema - The schema which is to be added to the map * @param hash - The hash value at which to map the schema */ addSchema(schema: S, hash: string) { const key = get(schema, ID_KEY, hash); const identifiedSchema = { ...schema, [ID_KEY]: key }; const existing = this.schemaMap[key]; if (!existing) { this.schemaMap[key] = identifiedSchema; } else if (!deepEquals(existing, identifiedSchema)) { console.error('existing schema:', JSON.stringify(existing, null, 2)); console.error('new schema:', JSON.stringify(identifiedSchema, null, 2)); throw new Error( `Two different schemas exist with the same key ${key}! What a bad coincidence. If possible, try adding an $id to one of the schemas`, ); } } /** Returns the current `schemaMap` to the caller */ getSchemaMap() { return this.schemaMap; } /** Implements the `ValidatorType` `isValid()` method to capture the `schema` in the `schemaMap`. Throws an error when * the `rootSchema` is not the same as the root schema provided during construction. * * @param schema - The schema to record in the `schemaMap` * @param _formData - The formData parameter that is ignored * @param rootSchema - The root schema associated with the schema * @throws - Error when the given `rootSchema` differs from the root schema provided during construction */ isValid(schema: S, _formData: T, rootSchema: S): boolean { if (!deepEquals(rootSchema, this.rootSchema)) { throw new Error('Unexpectedly calling isValid() with a rootSchema that differs from the construction rootSchema'); } this.addSchema(schema, hashForSchema<S>(schema)); return false; } /** Implements the `ValidatorType` `rawValidation()` method to throw an error since it is never supposed to be called * * @param _schema - The schema parameter that is ignored * @param _formData - The formData parameter that is ignored */ rawValidation<Result = any>(_schema: S, _formData?: T): { errors?: Result[]; validationError?: Error } { throw new Error('Unexpectedly calling the `rawValidation()` method during schema parsing'); } /** Implements the `ValidatorType` `toErrorList()` method to throw an error since it is never supposed to be called * * @param _errorSchema - The error schema parameter that is ignored * @param _fieldPath - The field path parameter that is ignored */ toErrorList(_errorSchema?: ErrorSchema<T>, _fieldPath?: string[]): RJSFValidationError[] { throw new Error('Unexpectedly calling the `toErrorList()` method during schema parsing'); } /** Implements the `ValidatorType` `validateFormData()` method to throw an error since it is never supposed to be * called * * @param _formData - The formData parameter that is ignored * @param _schema - The schema parameter that is ignored * @param _customValidate - The customValidate parameter that is ignored * @param _transformErrors - The transformErrors parameter that is ignored * @param _uiSchema - The uiSchema parameter that is ignored */ validateFormData( _formData: T, _schema: S, _customValidate?: CustomValidator<T, S, F>, _transformErrors?: ErrorTransformer<T, S, F>, _uiSchema?: UiSchema<T, S, F>, ): ValidationData<T> { throw new Error('Unexpectedly calling the `validateFormData()` method during schema parsing'); } }