teableio/teable

View on GitHub
packages/openapi/src/generate.schema.ts

Summary

Maintainability
A
1 hr
Test Coverage
import type { RouteConfig } from '@asteasolutions/zod-to-openapi';
import { OpenAPIRegistry, OpenApiGeneratorV3 } from '@asteasolutions/zod-to-openapi';
import type { OpenAPIObject } from 'openapi3-ts/oas30';
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
import OpenAPISnippet from './openapi-snippet';
import { getRoutes } from './utils';

function registerAllRoute() {
  const registry = new OpenAPIRegistry();
  const routeObjList: RouteConfig[] = getRoutes();
  for (const routeObj of routeObjList) {
    const bearerAuth = registry.registerComponent('securitySchemes', 'bearerAuth', {
      type: 'http',
      scheme: 'bearer',
    });
    registry.registerPath({ ...routeObj, security: [{ [bearerAuth.name]: [] }] });
  }
  return registry;
}

async function generateCodeSamples(document: OpenAPIObject) {
  const routes = getRoutes();
  const langs = ['shell', 'javascript_fetch', 'node', 'python'];
  const targetTitle: Record<string, string> = {
    shell: 'Shell',
    javascript_fetch: 'JavaScript',
    node: 'Node.js',
    python: 'Python',
  };
  for (const route of routes) {
    const generated = OpenAPISnippet.getEndpointSnippets(document, route.path, route.method, langs);
    const path = document.paths?.[route.path][route.method];
    if (path) {
      path['x-codeSamples'] = [];
      for (const [index, snippet] of generated.snippets.entries()) {
        const id = snippet.id as string;
        if (targetTitle[id]) {
          path['x-codeSamples'][index] = {
            lang: targetTitle[id],
            source: await snippet.content,
          };
        }
      }
    }
  }
}

export async function getOpenApiDocumentation(config: {
  origin?: string;
  snippet?: boolean;
}): Promise<OpenAPIObject> {
  const { origin, snippet } = config;
  if (!origin && snippet) {
    throw new Error('origin is required when snippets is true, generateCodeSamples need origin');
  }
  const registry = registerAllRoute();
  const generator = new OpenApiGeneratorV3(registry.definitions);

  const document = generator.generateDocument({
    openapi: '3.0.0',
    info: {
      version: '1.0.0',
      title: 'Teable App',
      description: `Manage Data as easy as drink a cup of tea`,
    },
    servers: [{ url: origin + '/api' }],
  });
  if (snippet) {
    await generateCodeSamples(document);
  }
  return document;
}