UNPKG

json-schema-to-typescript

Version:
260 lines (230 loc) 8.37 kB
import {JSONSchemaTypeName, LinkedJSONSchema, NormalizedJSONSchema, Parent} from './types/JSONSchema' import {appendToDescription, escapeBlockComment, isSchemaLike, justName, toSafeString, traverse} from './utils' import {Options} from './' import {applySchemaTyping} from './applySchemaTyping' import {DereferencedPaths} from './resolver' import {isDeepStrictEqual} from 'util' type Rule = ( schema: LinkedJSONSchema, fileName: string, options: Options, key: string | null, dereferencedPaths: DereferencedPaths, ) => void const rules = new Map<string, Rule>() function hasType(schema: LinkedJSONSchema, type: JSONSchemaTypeName) { return schema.type === type || (Array.isArray(schema.type) && schema.type.includes(type)) } function isObjectType(schema: LinkedJSONSchema) { return schema.properties !== undefined || hasType(schema, 'object') || hasType(schema, 'any') } function isArrayType(schema: LinkedJSONSchema) { return schema.items !== undefined || hasType(schema, 'array') || hasType(schema, 'any') } function isEnumTypeWithoutTsEnumNames(schema: LinkedJSONSchema) { return schema.type === 'string' && schema.enum !== undefined && schema.tsEnumNames === undefined } rules.set('Remove `type=["null"]` if `enum=[null]`', schema => { if ( Array.isArray(schema.enum) && schema.enum.some(e => e === null) && Array.isArray(schema.type) && schema.type.includes('null') ) { schema.type = schema.type.filter(type => type !== 'null') } }) rules.set('Destructure unary types', schema => { if (schema.type && Array.isArray(schema.type) && schema.type.length === 1) { schema.type = schema.type[0] } }) rules.set('Add empty `required` property if none is defined', schema => { if (isObjectType(schema) && !('required' in schema)) { schema.required = [] } }) rules.set('Transform `required`=false to `required`=[]', schema => { if (schema.required === false) { schema.required = [] } }) rules.set('Default additionalProperties', (schema, _, options) => { if (isObjectType(schema) && !('additionalProperties' in schema) && schema.patternProperties === undefined) { schema.additionalProperties = options.additionalProperties } }) rules.set('Transform id to $id', (schema, fileName) => { if (!isSchemaLike(schema)) { return } if (schema.id && schema.$id && schema.id !== schema.$id) { throw ReferenceError( `Schema must define either id or $id, not both. Given id=${schema.id}, $id=${schema.$id} in ${fileName}`, ) } if (schema.id) { schema.$id = schema.id delete schema.id } }) rules.set('Add an $id to anything that needs it', (schema, fileName, _options, _key, dereferencedPaths) => { if (!isSchemaLike(schema)) { return } // Top-level schema if (!schema.$id && !schema[Parent]) { schema.$id = toSafeString(justName(fileName)) return } // Sub-schemas with references if (!isArrayType(schema) && !isObjectType(schema)) { return } // We'll infer from $id and title downstream // TODO: Normalize upstream const dereferencedName = dereferencedPaths.get(schema) if (!schema.$id && !schema.title && dereferencedName) { schema.$id = toSafeString(justName(dereferencedName)) } if (dereferencedName) { dereferencedPaths.delete(schema) } }) rules.set('Escape closing JSDoc comment', schema => { escapeBlockComment(schema) }) rules.set('Add JSDoc comments for minItems and maxItems', schema => { if (!isArrayType(schema)) { return } const commentsToAppend = [ 'minItems' in schema ? `@minItems ${schema.minItems}` : '', 'maxItems' in schema ? `@maxItems ${schema.maxItems}` : '', ].filter(Boolean) if (commentsToAppend.length) { schema.description = appendToDescription(schema.description, ...commentsToAppend) } }) rules.set('Optionally remove maxItems and minItems', (schema, _fileName, options) => { if (!isArrayType(schema)) { return } if ('minItems' in schema && options.ignoreMinAndMaxItems) { delete schema.minItems } if ('maxItems' in schema && (options.ignoreMinAndMaxItems || options.maxItems === -1)) { delete schema.maxItems } }) rules.set('Normalize schema.minItems', (schema, _fileName, options) => { if (options.ignoreMinAndMaxItems) { return } // make sure we only add the props onto array types if (!isArrayType(schema)) { return } const {minItems} = schema schema.minItems = typeof minItems === 'number' ? minItems : 0 // cannot normalize maxItems because maxItems = 0 has an actual meaning }) rules.set('Remove maxItems if it is big enough to likely cause OOMs', (schema, _fileName, options) => { if (options.ignoreMinAndMaxItems || options.maxItems === -1) { return } if (!isArrayType(schema)) { return } const {maxItems, minItems} = schema // minItems is guaranteed to be a number after the previous rule runs if (maxItems !== undefined && maxItems - (minItems as number) > options.maxItems) { delete schema.maxItems } }) rules.set('Normalize schema.items', (schema, _fileName, options) => { if (options.ignoreMinAndMaxItems) { return } const {maxItems, minItems} = schema const hasMaxItems = typeof maxItems === 'number' && maxItems >= 0 const hasMinItems = typeof minItems === 'number' && minItems > 0 if (schema.items && !Array.isArray(schema.items) && (hasMaxItems || hasMinItems)) { const items = schema.items // create a tuple of length N const newItems = Array(maxItems || minItems || 0).fill(items) if (!hasMaxItems) { // if there is no maximum, then add a spread item to collect the rest schema.additionalItems = items } schema.items = newItems } if (Array.isArray(schema.items) && hasMaxItems && maxItems! < schema.items.length) { // it's perfectly valid to provide 5 item defs but require maxItems 1 // obviously we shouldn't emit a type for items that aren't expected schema.items = schema.items.slice(0, maxItems) } return schema }) rules.set('Remove extends, if it is empty', schema => { if (!schema.hasOwnProperty('extends')) { return } if (schema.extends == null || (Array.isArray(schema.extends) && schema.extends.length === 0)) { delete schema.extends } }) rules.set('Make extends always an array, if it is defined', schema => { if (schema.extends == null) { return } if (!Array.isArray(schema.extends)) { schema.extends = [schema.extends] } }) rules.set('Transform definitions to $defs', (schema, fileName) => { if (schema.definitions && schema.$defs && !isDeepStrictEqual(schema.definitions, schema.$defs)) { throw ReferenceError( `Schema must define either definitions or $defs, not both. Given id=${schema.id} in ${fileName}`, ) } if (schema.definitions) { schema.$defs = schema.definitions delete schema.definitions } }) rules.set('Transform const to singleton enum', schema => { if (schema.const !== undefined) { schema.enum = [schema.const] delete schema.const } }) rules.set('Add tsEnumNames to enum types', (schema, _, options) => { if (isEnumTypeWithoutTsEnumNames(schema) && options.inferStringEnumKeysFromValues) { schema.tsEnumNames = schema.enum?.map(String) } }) // Precalculation of the schema types is necessary because the ALL_OF type // is implemented in a way that mutates the schema object. Detection of the // NAMED_SCHEMA type relies on the presence of the $id property, which is // hoisted to a parent schema object during the ALL_OF type implementation, // and becomes unavailable if the same schema is used in multiple places. // // Precalculation of the `ALL_OF` intersection schema is necessary because // the intersection schema needs to participate in the schema cache during // the parsing step, so it cannot be re-calculated every time the schema // is encountered. rules.set('Pre-calculate schema types and intersections', schema => { if (schema !== null && typeof schema === 'object') { applySchemaTyping(schema) } }) export function normalize( rootSchema: LinkedJSONSchema, dereferencedPaths: DereferencedPaths, filename: string, options: Options, ): NormalizedJSONSchema { rules.forEach(rule => traverse(rootSchema, (schema, key) => rule(schema, filename, options, key, dereferencedPaths))) return rootSchema as NormalizedJSONSchema }