UNPKG

json-schema-library

Version:

Customizable and hackable json-validator and json-schema utilities for traversal, data generation and validation

78 lines (70 loc) 2.62 kB
import { Keyword, JsonSchemaResolverParams, JsonSchemaValidatorParams, ValidationResult } from "../../Keyword"; import { SchemaNode } from "../../types"; import { isObject } from "../../utils/isObject"; import { validateNode } from "../../validateNode"; export const itemsKeyword: Keyword = { id: "items", keyword: "items", parse: parseItems, addResolve: (node) => (node.prefixItems || node.items) != null, resolve: itemsResolver, addValidate: ({ schema }) => schema.items != null, validate: validateItems }; function itemsResolver({ node, key }: JsonSchemaResolverParams) { if (node.prefixItems?.[key as number]) { return node.prefixItems[key as number]; } if (node.items) { return node.items; } } export function parseItems(node: SchemaNode) { const { schema, evaluationPath } = node; if (isObject(schema.items)) { const propertyNode = node.compileSchema( schema.items, `${evaluationPath}/items`, `${node.schemaLocation}/items` ); node.items = propertyNode; } else if (Array.isArray(schema.items)) { node.prefixItems = schema.items.map((itemSchema, index) => node.compileSchema(itemSchema, `${evaluationPath}/items/${index}`, `${node.schemaLocation}/items/${index}`) ); } } function validateItems({ node, data, pointer = "#", path }: JsonSchemaValidatorParams) { const { schema } = node; if (!Array.isArray(data) || data.length === 0) { return; } // @draft >= 7 bool schema if (schema.items === false) { if (Array.isArray(data) && data.length === 0) { return undefined; } return node.createError("invalid-data-error", { pointer, value: data, schema }); } const errors: ValidationResult[] = []; if (node.prefixItems) { // note: schema is valid when data does not have enough elements as defined by array-list for (let i = 0; i < Math.min(node.prefixItems.length, data.length); i += 1) { const itemData = data[i]; const itemNode = node.prefixItems[i]; const result = validateNode(itemNode, itemData, `${pointer}/${i}`, path); errors.push(...result); } return errors; } if (node.items) { for (let i = 0; i < data.length; i += 1) { const itemData = data[i]; const result = validateNode(node.items, itemData, `${pointer}/${i}`, path); if (result) { errors.push(...result); } } return errors; } }