UNPKG

openapi-typescript

Version:

Convert OpenAPI 3.0 & 3.1 schemas to TypeScript

128 lines (114 loc) 4.91 kB
import ts from "typescript"; import * as changeCase from "change-case"; import { performance } from "node:perf_hooks"; import { NEVER, QUESTION_TOKEN, addJSDocComment, tsModifiers, tsPropertyIndex } from "../lib/ts.js"; import { createRef, debug, getEntries } from "../lib/utils.js"; import type { ComponentsObject, GlobalContext, SchemaObject, TransformNodeOptions } from "../types.js"; import transformHeaderObject from "./header-object.js"; import transformParameterObject from "./parameter-object.js"; import transformPathItemObject from "./path-item-object.js"; import transformRequestBodyObject from "./request-body-object.js"; import transformResponseObject from "./response-object.js"; import transformSchemaObject from "./schema-object.js"; type ComponentTransforms = keyof Omit<ComponentsObject, "examples" | "securitySchemes" | "links" | "callbacks">; const transformers: Record<ComponentTransforms, (node: any, options: TransformNodeOptions) => ts.TypeNode> = { schemas: transformSchemaObject, responses: transformResponseObject, parameters: transformParameterObject, requestBodies: transformRequestBodyObject, headers: transformHeaderObject, pathItems: transformPathItemObject, }; /** * Transform the ComponentsObject (4.8.7) * @see https://spec.openapis.org/oas/latest.html#components-object */ export default function transformComponentsObject(componentsObject: ComponentsObject, ctx: GlobalContext): ts.Node[] { const type: ts.TypeElement[] = []; const rootTypeAliases: { [key: string]: ts.TypeAliasDeclaration } = {}; for (const key of Object.keys(transformers) as ComponentTransforms[]) { const componentT = performance.now(); const items: ts.TypeElement[] = []; if (componentsObject[key]) { for (const [name, item] of getEntries<SchemaObject>(componentsObject[key], ctx)) { let subType = transformers[key](item, { path: createRef(["components", key, name]), schema: item, ctx, }); let hasQuestionToken = false; if (ctx.transform) { const result = ctx.transform(item, { path: createRef(["components", key, name]), schema: item, ctx, }); if (result) { if ("schema" in result) { subType = result.schema; hasQuestionToken = result.questionToken; } else { subType = result; } } } const property = ts.factory.createPropertySignature( /* modifiers */ tsModifiers({ readonly: ctx.immutable }), /* name */ tsPropertyIndex(name), /* questionToken */ hasQuestionToken ? QUESTION_TOKEN : undefined, /* type */ subType, ); addJSDocComment(item as unknown as any, property); items.push(property); if (ctx.rootTypes) { const componentKey = changeCase.pascalCase(singularizeComponentKey(key)); let aliasName = `${componentKey}${changeCase.pascalCase(name)}`; // Add counter suffix (e.g. "_2") if conflict in name let conflictCounter = 1; while (rootTypeAliases[aliasName] !== undefined) { conflictCounter++; aliasName = `${componentKey}${changeCase.pascalCase(name)}_${conflictCounter}`; } const ref = ts.factory.createTypeReferenceNode(`components['${key}']['${name}']`); if (ctx.rootTypesNoSchemaPrefix && key === "schemas") { aliasName = aliasName.replace(componentKey, ""); } const typeAlias = ts.factory.createTypeAliasDeclaration( /* modifiers */ tsModifiers({ export: true }), /* name */ aliasName, /* typeParameters */ undefined, /* type */ ref, ); rootTypeAliases[aliasName] = typeAlias; } } } type.push( ts.factory.createPropertySignature( /* modifiers */ undefined, /* name */ tsPropertyIndex(key), /* questionToken */ undefined, /* type */ items.length ? ts.factory.createTypeLiteralNode(items) : NEVER, ), ); debug(`Transformed components → ${key}`, "ts", performance.now() - componentT); } // Extract root types let rootTypes: ts.TypeAliasDeclaration[] = []; if (ctx.rootTypes) { rootTypes = Object.keys(rootTypeAliases).map((k) => rootTypeAliases[k]); } return [ts.factory.createTypeLiteralNode(type), ...rootTypes]; } export function singularizeComponentKey( key: `x-${string}` | "schemas" | "responses" | "parameters" | "requestBodies" | "headers" | "pathItems", ): string { switch (key) { // Handle special singular case case "requestBodies": return "requestBody"; // Default to removing the "s" default: return key.slice(0, -1); } }