@snups/rjsf-utils
Version:
Utility functions for @snups/rjsf-core
139 lines (125 loc) • 5.94 kB
text/typescript
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');
}
}