188 lines
6.5 KiB
JavaScript
188 lines
6.5 KiB
JavaScript
#!/usr/bin/env node
|
|
import Ajv from 'ajv'
|
|
import httpStatusCodes from 'http-status-code'
|
|
import { readFile } from 'fs/promises'
|
|
import { get, isPlainObject } from 'lodash-es'
|
|
import { parseTemplate } from 'url-template'
|
|
|
|
import renderContent from '../../../lib/render-content/index.js'
|
|
import getCodeSamples from './create-rest-examples.js'
|
|
import operationSchema from './operation-schema.js'
|
|
import { getBodyParams } from './get-body-params.js'
|
|
|
|
const { operationUrls } = JSON.parse(
|
|
await readFile('script/rest/utils/rest-api-overrides.json', 'utf8')
|
|
)
|
|
|
|
export default class Operation {
|
|
#operation
|
|
constructor(verb, requestPath, operation, globalServers) {
|
|
this.#operation = operation
|
|
// The global server object sets metadata including the base url for
|
|
// all operations in a version. Individual operations can override
|
|
// the global server url at the operation level.
|
|
this.serverUrl = operation.servers ? operation.servers[0].url : globalServers[0].url
|
|
|
|
const serverVariables = operation.servers
|
|
? operation.servers[0].variables
|
|
: globalServers[0].variables
|
|
if (serverVariables) {
|
|
const templateVariables = {}
|
|
Object.keys(serverVariables).forEach(
|
|
(key) => (templateVariables[key] = serverVariables[key].default)
|
|
)
|
|
this.serverUrl = parseTemplate(this.serverUrl).expand(templateVariables)
|
|
}
|
|
|
|
this.serverUrl = this.serverUrl.replace('http:', 'http(s):')
|
|
|
|
// Attach some global properties to the operation object to use
|
|
// during processing
|
|
this.#operation.serverUrl = this.serverUrl
|
|
this.#operation.requestPath = requestPath
|
|
this.#operation.verb = verb
|
|
|
|
this.verb = verb
|
|
this.requestPath = requestPath
|
|
this.title = operation.summary
|
|
this.setCategories()
|
|
this.parameters = operation.parameters || []
|
|
this.bodyParameters = []
|
|
this.enabledForGitHubApps = operation['x-github'].enabledForGitHubApps
|
|
this.codeExamples = getCodeSamples(this.#operation)
|
|
return this
|
|
}
|
|
|
|
setCategories() {
|
|
const operationId = this.#operation.operationId
|
|
const xGithub = this.#operation['x-github']
|
|
// Set category
|
|
// A temporary override file allows us to override the category defined in
|
|
// the openapi schema. Without it, we'd have to update several
|
|
// @documentation_urls in the api code every time we move
|
|
// an endpoint to a new page.
|
|
this.category = operationUrls[operationId]
|
|
? operationUrls[operationId].category
|
|
: xGithub.category
|
|
|
|
// Set subcategory
|
|
// A temporary override file allows us to override the subcategory
|
|
// defined in the openapi schema. Without it, we'd have to update several
|
|
// @documentation_urls in the api code every time we move
|
|
// an endpoint to a new page.
|
|
if (operationUrls[operationId]) {
|
|
if (operationUrls[operationId].subcategory) {
|
|
this.subcategory = operationUrls[operationId].subcategory
|
|
}
|
|
} else if (xGithub.subcategory) {
|
|
this.subcategory = xGithub.subcategory
|
|
}
|
|
}
|
|
|
|
async process() {
|
|
await Promise.all([
|
|
this.renderDescription(),
|
|
this.renderStatusCodes(),
|
|
this.renderParameterDescriptions(),
|
|
this.renderBodyParameterDescriptions(),
|
|
this.renderExampleResponseDescriptions(),
|
|
this.renderPreviewNotes(),
|
|
])
|
|
|
|
const ajv = new Ajv()
|
|
const valid = ajv.validate(operationSchema, this)
|
|
if (!valid) {
|
|
console.error(JSON.stringify(ajv.errors, null, 2))
|
|
throw new Error('Invalid OpenAPI operation found')
|
|
}
|
|
}
|
|
|
|
getExternalDocs() {
|
|
return this.#operation.externalDocs
|
|
}
|
|
|
|
async renderDescription() {
|
|
this.descriptionHTML = await renderContent(this.#operation.description)
|
|
return this
|
|
}
|
|
|
|
async renderExampleResponseDescriptions() {
|
|
return Promise.all(
|
|
this.codeExamples.map(async (codeExample) => {
|
|
codeExample.response.description = await renderContent(codeExample.response.description)
|
|
return codeExample
|
|
})
|
|
)
|
|
}
|
|
|
|
async renderStatusCodes() {
|
|
const responses = this.#operation.responses
|
|
const responseKeys = Object.keys(responses)
|
|
if (responseKeys.length === 0) return []
|
|
|
|
this.statusCodes = await Promise.all(
|
|
responseKeys.map(async (responseCode) => {
|
|
const response = responses[responseCode]
|
|
const httpStatusCode = responseCode
|
|
const httpStatusMessage = httpStatusCodes.getMessage(Number(responseCode), 'HTTP/2')
|
|
// The OpenAPI should be updated to provide better descriptions, but
|
|
// until then, we can catch some known generic descriptions and replace
|
|
// them with the default http status message.
|
|
const responseDescription =
|
|
response.description.toLowerCase() === 'response'
|
|
? await renderContent(httpStatusMessage)
|
|
: await renderContent(response.description)
|
|
|
|
return {
|
|
httpStatusCode,
|
|
description: responseDescription,
|
|
}
|
|
})
|
|
)
|
|
}
|
|
|
|
async renderParameterDescriptions() {
|
|
return Promise.all(
|
|
this.parameters.map(async (param) => {
|
|
param.description = await renderContent(param.description)
|
|
return param
|
|
})
|
|
)
|
|
}
|
|
|
|
async renderBodyParameterDescriptions() {
|
|
if (!this.#operation.requestBody) return []
|
|
// There is currently only one operation with more than one content type
|
|
// and the request body parameter types are the same for both.
|
|
// Operation Id: markdown/render-raw
|
|
const contentType = Object.keys(this.#operation.requestBody.content)[0]
|
|
const schema = get(this.#operation, `requestBody.content.${contentType}.schema`, {})
|
|
// TODO: Remove this check
|
|
if (this.#operation.operationId === 'checks/create') {
|
|
delete schema.oneOf
|
|
}
|
|
|
|
this.bodyParameters = isPlainObject(schema) ? await getBodyParams(schema, true) : []
|
|
}
|
|
|
|
async renderPreviewNotes() {
|
|
const previews = get(this.#operation, 'x-github.previews', [])
|
|
this.previews = await Promise.all(
|
|
previews.map(async (preview) => {
|
|
const note = preview.note
|
|
// remove extra leading and trailing newlines
|
|
.replace(/```\n\n\n/gm, '```\n')
|
|
.replace(/```\n\n/gm, '```\n')
|
|
.replace(/\n\n\n```/gm, '\n```')
|
|
.replace(/\n\n```/gm, '\n```')
|
|
|
|
// convert single-backtick code snippets to fully fenced triple-backtick blocks
|
|
// example: This is the description.\n\n`application/vnd.github.machine-man-preview+json`
|
|
.replace(/\n`application/, '\n```\napplication')
|
|
.replace(/json`$/, 'json\n```')
|
|
return await renderContent(note)
|
|
})
|
|
)
|
|
}
|
|
}
|