-
Notifications
You must be signed in to change notification settings - Fork 67.1k
Expand file tree
/
Copy pathoperation.ts
More file actions
235 lines (215 loc) · 8.94 KB
/
operation.ts
File metadata and controls
235 lines (215 loc) · 8.94 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
import httpStatusCodes from 'http-status-code'
import { get, isPlainObject } from 'lodash-es'
import { parseTemplate } from 'url-template'
import mergeAllOf from 'json-schema-merge-allof'
import { renderContent } from './render-content'
import getCodeSamples from './create-rest-examples'
import operationSchema from './operation-schema'
import { validateJson } from '@/tests/lib/validate-json-schema'
import { getBodyParams } from './get-body-params'
export default class Operation {
// OpenAPI operation object - schema is dynamic and varies by endpoint
#operation: any
serverUrl: string
verb: string
requestPath: string
title: string
category: string
subcategory: string
// OpenAPI parameters vary by endpoint, no fixed schema available
parameters: any[]
// Body parameters are dynamically generated from OpenAPI schema
bodyParameters: any[]
descriptionHTML?: string
// Code examples structure varies by language and endpoint
codeExamples?: any[]
// Status codes are dynamically generated from OpenAPI responses
statusCodes?: any[]
previews?: any[]
// Programmatic access data structure varies by operation
progAccess?: any
// OpenAPI operation and globalServers objects have dynamic schema
constructor(verb: string, requestPath: string, operation: any, globalServers?: any[]) {
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) {
// Template variables structure comes from OpenAPI server variables
const templateVariables: Record<string, any> = {}
for (const key of Object.keys(serverVariables)) {
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.category = operation['x-github'].category
this.subcategory = operation['x-github'].subcategory
// Shallow-clone each parameter so that renderParameterDescriptions() can
// safely delete fields (e.g. deprecated, example, examples) without
// mutating this.#operation.parameters, which renderCodeExamples() reads
// concurrently via getParameterExamples().
this.parameters = (operation.parameters || []).map((p: any) => ({ ...p }))
this.bodyParameters = []
return this
}
// Programmatic access data structure varies by operation and is not strongly typed
async process(progAccessData: any): Promise<void> {
await Promise.all([
this.renderCodeExamples(),
this.renderDescription(),
this.renderStatusCodes(),
this.renderParameterDescriptions(),
this.renderBodyParameterDescriptions(),
this.renderPreviewNotes(),
this.programmaticAccess(progAccessData),
])
const { isValid, errors } = validateJson(operationSchema, this)
if (!isValid) {
console.error(JSON.stringify(errors, null, 2))
throw new Error('Invalid OpenAPI operation found')
}
}
async renderDescription(): Promise<this> {
try {
this.descriptionHTML = await renderContent(this.#operation.description)
return this
} catch (error) {
console.error(error)
throw new Error(`Error rendering description for ${this.verb} ${this.requestPath}`)
}
}
async renderCodeExamples(): Promise<any[]> {
const codeExamples = await getCodeSamples(this.#operation)
try {
this.codeExamples = await Promise.all(
// Code example structure varies by endpoint and language
codeExamples.map(async (codeExample: any) => {
codeExample.response.description = await renderContent(codeExample.response.description)
return codeExample
}),
)
return this.codeExamples
} catch (error) {
console.error(error)
throw new Error(`Error generating code examples for ${this.verb} ${this.requestPath}`)
}
}
async renderStatusCodes(): Promise<void> {
const responses = this.#operation.responses
const responseKeys = Object.keys(responses)
if (responseKeys.length === 0) return
try {
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 || response.description?.toLowerCase() === 'response'
? await renderContent(httpStatusMessage)
: await renderContent(response.description)
return {
httpStatusCode,
description: responseDescription,
}
}),
)
} catch (error) {
console.error(error)
throw new Error(`Error rendering status codes for ${this.verb} ${this.requestPath}`)
}
}
async renderParameterDescriptions(): Promise<any[]> {
try {
return Promise.all(
this.parameters.map(async (param) => {
param.description = await renderContent(param.description)
// Remove fields that are not used at runtime to keep schema.json lean
delete param.deprecated
delete param.example
delete param.examples
delete param['x-multi-segment']
// Strip unused parameter schema sub-fields; only type, default, and
// enum are consumed by renderers
if (param.schema && typeof param.schema === 'object') {
const { type, default: defaultVal, enum: enumVal } = param.schema
param.schema = { type }
if (defaultVal !== undefined) param.schema.default = defaultVal
if (enumVal !== undefined) param.schema.enum = enumVal
}
return param
}),
)
} catch (error) {
console.error(error)
throw new Error(`Error rendering parameter descriptions for ${this.verb} ${this.requestPath}`)
}
}
async renderBodyParameterDescriptions(): Promise<void> {
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`, {})
// Merges any instances of allOf in the schema using a deep merge
const mergedAllofSchema = mergeAllOf(schema)
try {
this.bodyParameters = isPlainObject(schema)
? await getBodyParams(mergedAllofSchema as any, true)
: []
} catch (error) {
console.error(error)
throw new Error(
`Error rendering body parameter descriptions for ${this.verb} ${this.requestPath}`,
)
}
}
async renderPreviewNotes(): Promise<void> {
const previews = get(this.#operation, 'x-github.previews', [])
try {
this.previews = await Promise.all(
// Preview note structure from OpenAPI x-github extension is dynamic
previews.map(async (preview: any) => {
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)
}),
)
} catch (error) {
console.error(error)
throw new Error(`Error rendering preview notes for ${this.verb} ${this.requestPath}`)
}
}
// Programmatic access data structure varies by operation and is not strongly typed
programmaticAccess(progAccessData: any): void {
this.progAccess = progAccessData[this.#operation.operationId]
if (this.progAccess) {
delete this.progAccess.disabledForPatV2
}
}
}