UNPKG

ts-json-schema-generator

Version:

Generate JSON schema from your Typescript sources

93 lines (76 loc) 3.99 kB
import ts from "typescript"; import { LogicError } from "../Error/Errors.js"; import type { Context, NodeParser } from "../NodeParser.js"; import type { SubNodeParser } from "../SubNodeParser.js"; import type { BaseType } from "../Type/BaseType.js"; import { LiteralType } from "../Type/LiteralType.js"; import { NeverType } from "../Type/NeverType.js"; import { NumberType } from "../Type/NumberType.js"; import { ReferenceType } from "../Type/ReferenceType.js"; import { StringType } from "../Type/StringType.js"; import { TupleType } from "../Type/TupleType.js"; import { UnionType } from "../Type/UnionType.js"; import { isErroredUnknownType } from "../Type/UnknownType.js"; import { derefType } from "../Utils/derefType.js"; import { getTypeByKey } from "../Utils/typeKeys.js"; export class IndexedAccessTypeNodeParser implements SubNodeParser { public constructor( protected typeChecker: ts.TypeChecker, protected childNodeParser: NodeParser, ) {} public supportsNode(node: ts.TypeNode): boolean { return node.kind === ts.SyntaxKind.IndexedAccessType; } private createIndexedType(objectType: ts.TypeNode, context: Context, indexType: BaseType): BaseType | undefined { if (ts.isTypeReferenceNode(objectType) && indexType instanceof LiteralType) { const declaration = this.typeChecker.getSymbolAtLocation(objectType.typeName)?.declarations?.[0]; if (!declaration || !ts.isTypeAliasDeclaration(declaration) || !ts.isTypeLiteralNode(declaration.type)) { return undefined; } const member = declaration.type.members.find( (m): m is ts.PropertySignature & { type: ts.TypeNode } => ts.isPropertySignature(m) && Boolean(m.type) && ts.isIdentifier(m.name) && m.name.text === indexType.getValue(), ); return member && this.childNodeParser.createType(member.type, context); } return undefined; } public createType(node: ts.IndexedAccessTypeNode, context: Context): BaseType { const indexType = derefType(this.childNodeParser.createType(node.indexType, context)); const indexedType = this.createIndexedType(node.objectType, context, indexType); if (indexedType && !isErroredUnknownType(indexedType)) { return indexedType; } const objectType = derefType(this.childNodeParser.createType(node.objectType, context)); if (objectType instanceof NeverType || indexType instanceof NeverType) { return new NeverType(); } const indexTypes = indexType instanceof UnionType ? indexType.getTypes() : [indexType]; const propertyTypes = indexTypes.map((type) => { if (!(type instanceof LiteralType || type instanceof StringType || type instanceof NumberType)) { throw new LogicError( node, `Unexpected type "${type.getId()}" (expected "LiteralType.js" or "StringType.js" or "NumberType.js")`, ); } const propertyType = getTypeByKey(objectType, type); if (!propertyType) { if (type instanceof NumberType && objectType instanceof TupleType) { return new UnionType(objectType.getTypes()); } if (type instanceof LiteralType) { if (objectType instanceof ReferenceType) { return objectType; } throw new LogicError(node, `Invalid index "${type.getValue()}" in type "${objectType.getId()}"`); } throw new LogicError(node, `No additional properties in type "${objectType.getId()}"`); } return propertyType; }); return propertyTypes.length === 1 ? propertyTypes[0] : new UnionType(propertyTypes); } }