// @ts-nocheck /** * Usage: * tsx ReferenceGenerator.ts -o {output_dir} {input}.yml * * Example: * tsx ReferenceGenerator.ts -o docs/client spec/supabase.yml */ import * as fs from 'fs' import * as yaml from 'js-yaml' import { uniqBy } from 'lodash-es' import { flattenSections } from '../lib/helpers' import Example from './legacy/components/Example' import Page from './legacy/components/Page' import Tab from './legacy/components/Tab' import { OpenRef, TsDoc } from './legacy/definitions' import { slugify, tsDocCommentToMdComment, writeToDisk } from './legacy/lib/helpers' const commonDocSpecJson = JSON.parse( fs.readFileSync('spec/common-client-libs-sections.json', 'utf8') ) const flattenedCommonDocSpecJson = flattenSections(commonDocSpecJson) export default async function gen(inputFileName: string, outputDir: string) { const docSpec = yaml.load(fs.readFileSync(inputFileName, 'utf8')) const defRef = docSpec.info.definition ? fs.readFileSync(docSpec.info.definition, 'utf8') : '{}' const definition = JSON.parse(defRef) const id = docSpec.info.id const allLanguages = docSpec.info.libraries const pages = Object.entries(docSpec.functions).map(([name, x]: [string, OpenRef.Page]) => ({ ...x, pageName: name, })) // Index Page const indexFilename = outputDir + `/index.mdx` const index = generateDocsIndexPage(docSpec, inputFileName) await writeToDisk(indexFilename, index) console.log('The index was saved: ', indexFilename) // Generate Pages pages.forEach(async (pageSpec: OpenRef.Page) => { try { // get the slug from common-client-libs.yml const slug = flattenedCommonDocSpecJson.find((item) => item.id === pageSpec.id).slug const hasTsRef = pageSpec['$ref'] || null const tsDefinition = hasTsRef && extractTsDocNode(hasTsRef, definition) if (hasTsRef && !tsDefinition) throw new Error('Definition not found: ' + hasTsRef) const description = pageSpec.description || tsDocCommentToMdComment(getDescriptionFromDefinition(tsDefinition)) // Create page const content = Page({ slug: slug, id: pageSpec.id, specFileName: docSpec.info.specUrl || inputFileName, title: pageSpec.title || pageSpec.pageName, description, parameters: hasTsRef ? generateParameters(tsDefinition) : '', spotlight: generateSpotlight(id, pageSpec['examples'] || [], allLanguages), examples: generateExamples(id, pageSpec['examples'] || [], allLanguages), notes: pageSpec.notes, }) //console.log({ slug }) // Write to disk const dest = outputDir + `/${slug}.mdx` await writeToDisk(dest, content) console.log('Saved: ', dest) } catch (error) { console.error(error) } }) } function generateParameters(tsDefinition: any) { let functionDeclaration = null if (tsDefinition.kindString == 'Method') { functionDeclaration = tsDefinition } else if (tsDefinition.kindString == 'Constructor') { functionDeclaration = tsDefinition } else functionDeclaration = tsDefinition?.type?.declaration if (!functionDeclaration) return '' const paramDefinitions: TsDoc.TypeDefinition[] = functionDeclaration.signatures[0].parameters // PMC: seems flaky.. why the [0]? if (!paramDefinitions) return '' // const paramsComments: TsDoc.CommentTag = tsDefinition.comment?.tags?.filter(x => x.tag == 'param') let parameters = paramDefinitions.map((x) => recurseThroughParams(x)).join(`\n`) return methodListGroup(parameters) } function getDescriptionFromDefinition(tsDefinition) { if (!tsDefinition) return null if (['Method', 'Constructor', 'Constructor signature'].includes(tsDefinition.kindString)) return tsDefinition?.signatures[0].comment else return tsDefinition?.comment || '' } function recurseThroughParams(paramDefinition: TsDoc.TypeDefinition) { // If this is a reference to another Param, let's use the reference instead let param = isDereferenced(paramDefinition) ? paramDefinition.type?.dereferenced : paramDefinition const labelParams = generateLabelParam(param) let subContent = '' let children = param?.children if (param.type?.declaration?.children) { children = param.type?.declaration?.children } else if (isUnion(param)) { // We don't want to show the union types if it's a literal const nonLiteralVariants = param.type.types.filter(({ type }) => type !== 'literal') if (nonLiteralVariants.length === 0) { children = null } else { children = nonLiteralVariants } } else if (param.type === 'reflection') { children = param.declaration.children } if (!!children) { let properties = children .sort((a, b) => a.name?.localeCompare(b.name)) // first alphabetical .sort((a, b) => (a.flags?.isOptional ? 1 : -1)) // required params first .map((x) => recurseThroughParams(x)) let heading = `
${paramDefinition.type.name}`
} else if (paramDefinition.type?.type == 'union') {
return paramDefinition.type.types
.map((x) =>
x.value
? `${x.value}`
: x.name
? `${x.name}`
: x.type
? `${x.type}`
: ''
)
.join(' | ')
}
return 'object'
}
/**
* Iterates through the definition to find the correct definition.
* You can pass it a deeply nested node using dot notation. eg: 'LoggedInUser.data.email'
*/
export function extractTsDocNode(nodeToFind: string, definition: any) {
const nodePath = nodeToFind.split('.')
let i = 0
let previousNode = definition
let currentNode = definition
while (i < nodePath.length) {
previousNode = currentNode
currentNode = previousNode.children.find((x) => x.name == nodePath[i]) || null
if (currentNode == null) {
console.log(`Cant find ${nodePath[i]} in ${previousNode.children.map((x) => '\n' + x.name)}`)
break
}
i++
}
return currentNode
}
function generateDocsIndexPage(docSpec: any, inputFileName: string) {
return Page({
slug: (docSpec.info.slugPrefix || '') + slugify(docSpec.info.title),
id: 'index',
title: docSpec.info.title,
specFileName: inputFileName,
description: docSpec.info.description,
})
}