api
Version:
Magical SDK generation from an OpenAPI definition 🪄
416 lines (366 loc) • 16.5 kB
text/typescript
import type { ReadStream } from 'fs';
import type { Operation } from 'oas';
import type { ParameterObject, SchemaObject } from 'oas/dist/rmoas.types';
import fs from 'fs';
import path from 'path';
import stream from 'stream';
import caseless from 'caseless';
import DatauriParser from 'datauri/parser';
import datauri from 'datauri/sync';
import getStream from 'get-stream';
import lodashMerge from 'lodash/merge';
import removeUndefinedObjects from 'remove-undefined-objects';
import getJSONSchemaDefaults from './getJSONSchemaDefaults';
// These headers are normally only defined by the OpenAPI definition but we allow the user to
// manually supply them in their `metadata` parameter if they wish.
const specialHeaders = ['accept', 'authorization'];
/**
* Extract all available parameters from an operations Parameter Object into a digestable array
* that we can use to apply to the request.
*
* @see {@link https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.3.md#parameterObject}
* @see {@link https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#parameterObject}
*/
function digestParameters(parameters: ParameterObject[]): Record<string, ParameterObject> {
return parameters.reduce((prev, param) => {
if ('$ref' in param || 'allOf' in param || 'anyOf' in param || 'oneOf' in param) {
throw new Error("The OpenAPI document for this operation wasn't dereferenced before processing.");
} else if (param.name in prev) {
throw new Error(
`The operation you are using has the same parameter, ${param.name}, spread across multiple entry points. We unfortunately can't handle this right now.`,
);
}
return Object.assign(prev, { [param.name]: param });
}, {});
}
// https://github.com/you-dont-need/You-Dont-Need-Lodash-Underscore#_isempty
function isEmpty(obj: any) {
return [Object, Array].includes((obj || {}).constructor) && !Object.entries(obj || {}).length;
}
function isObject(thing: any) {
if (thing instanceof stream.Readable) {
return false;
}
return typeof thing === 'object' && thing !== null && !Array.isArray(thing);
}
function isPrimitive(obj: any) {
return obj === null || typeof obj === 'number' || typeof obj === 'string';
}
function merge(src: any, target: any) {
if (Array.isArray(target)) {
// @todo we need to add support for merging array defaults with array body/metadata arguments
return target;
} else if (!isObject(target)) {
return target;
}
return lodashMerge(src, target);
}
/**
* Ingest a file path or readable stream into a common object that we can later use to process it
* into a parameters object for making an API request.
*
*/
function processFile(
paramName: string,
file: string | ReadStream,
): Promise<{ base64: string; buffer: Buffer; filename: string; paramName: string }> {
if (typeof file === 'string') {
// In order to support relative pathed files, we need to attempt to resolve them.
const resolvedFile = path.resolve(file);
return new Promise((resolve, reject) => {
fs.stat(resolvedFile, async err => {
if (err) {
if (err.code === 'ENOENT') {
// It's less than ideal for us to handle files that don't exist like this but because
// `file` is a string it might actually be the full text contents of the file and not
// actually a path.
//
// We also can't really regex to see if `file` *looks*` like a path because one should be
// able to pass in a relative `owlbert.png` (instead of `./owlbert.png`) and though that
// doesn't *look* like a path, it is one that should still work.
return resolve(undefined);
}
return reject(err);
}
const fileMetadata = await datauri(resolvedFile);
const payloadFilename = encodeURIComponent(path.basename(resolvedFile));
return resolve({
paramName,
base64: fileMetadata.content.replace(';base64', `;name=${payloadFilename};base64`),
filename: payloadFilename,
buffer: fileMetadata.buffer,
});
});
});
} else if (file instanceof stream.Readable) {
return getStream.buffer(file).then(buffer => {
const filePath = file.path as string;
const parser = new DatauriParser();
const base64 = parser.format(filePath, buffer).content;
const payloadFilename = encodeURIComponent(path.basename(filePath));
return {
paramName,
base64: base64.replace(';base64', `;name=${payloadFilename};base64`),
filename: payloadFilename,
buffer,
};
});
}
return Promise.reject(
new TypeError(
paramName
? `The data supplied for the \`${paramName}\` request body parameter is not a file handler that we support.`
: 'The data supplied for the request body payload is not a file handler that we support.',
),
);
}
/**
* With potentially supplied body and/or metadata we need to run through them against a given API
* operation to see what's what and prepare any available parameters to be used in an API request
* with `@readme/oas-to-har`.
*
*/
export default async function prepareParams(operation: Operation, body?: unknown, metadata?: Record<string, unknown>) {
let metadataIntersected = false;
const digestedParameters = digestParameters(operation.getParameters());
const jsonSchema = operation.getParametersAsJSONSchema();
/**
* It might be common for somebody to run `sdk.findPetsByStatus({ status: 'available' }, {})`, in
* which case we want to filter out the second (metadata) parameter and treat the first parameter
* as the metadata instead. If we don't do this, their supplied `status` metadata will be treated
* as a body parameter, and because there's no `status` body parameter, and no supplied metadata
* (because it's an empty object), the request won't send a payload.
*
* @see {@link https://github.com/readmeio/api/issues/449}
*/
// eslint-disable-next-line no-param-reassign
metadata = removeUndefinedObjects(metadata);
if (!jsonSchema && (body !== undefined || metadata !== undefined)) {
let throwNoParamsError = true;
// If this operation doesn't have any parameters for us to transform to JSON Schema but they've
// sent us either an `Accept` or `Authorization` header (or both) we should let them do that.
// We should, however, only do this check for the `body` parameter as if they've sent this
// request both `body` and `metadata` we can reject it outright as the operation won't have any
// body data.
if (body !== undefined) {
if (typeof body === 'object' && body !== null && !Array.isArray(body)) {
if (Object.keys(body).length <= 2) {
const bodyParams = caseless(body);
if (specialHeaders.some(header => bodyParams.has(header))) {
throwNoParamsError = false;
}
}
}
}
if (throwNoParamsError) {
throw new Error(
"You supplied metadata and/or body data for this operation but it doesn't have any documented parameters or request payloads. If you think this is an error please contact support for the API you're using.",
);
}
}
const jsonSchemaDefaults = jsonSchema ? getJSONSchemaDefaults(jsonSchema) : {};
const params: {
body?: any;
cookie?: Record<string, string | number | boolean>;
files?: Record<string, Buffer>;
formData?: any;
header?: Record<string, string | number | boolean>;
path?: Record<string, string | number | boolean>;
query?: Record<string, string | number | boolean>;
server?: {
selected: number;
variables: Record<string, string | number>;
};
} = jsonSchemaDefaults;
// If a body argument was supplied we need to do a bit of work to see if it's actually a body
// argument or metadata because the library lets you supply either a body, metadata, or body with
// metadata.
if (typeof body !== 'undefined') {
if (Array.isArray(body) || isPrimitive(body)) {
// If the body param is an array or a primitive then we know it's absolutely a body because
// metadata can only ever be undefined or an object.
params.body = merge(params.body, body);
} else if (typeof metadata === 'undefined') {
// No metadata was explicitly provided so we need to analyze the body to determine if it's a
// body or should be actually be treated as metadata.
const headerParams = caseless({});
Object.entries(digestedParameters).forEach(([paramName, param]) => {
// Headers are sent case-insensitive so we need to make sure that we're properly
// matching them when detecting what our incoming payload looks like.
if (param.in === 'header') {
headerParams.set(paramName, '');
}
});
// `Accept` and `Authorization` headers can't be defined as normal parameters but we should
// always allow the user to supply them if they wish.
specialHeaders.forEach(header => {
if (!headerParams.has(header)) {
headerParams.set(header, '');
}
});
const intersection = Object.keys(body).filter(value => {
if (Object.keys(digestedParameters).includes(value)) {
return true;
} else if (headerParams.has(value)) {
return true;
}
return false;
}).length;
if (intersection && intersection / Object.keys(body).length > 0.25) {
/* eslint-disable no-param-reassign */
// If more than 25% of the body intersects with the parameters that we've got on hand,
// then we should treat it as a metadata object and organize into parameters.
metadataIntersected = true;
metadata = merge(params.body, body) as Record<string, unknown>;
body = undefined;
/* eslint-enable no-param-reassign */
} else {
// For all other cases, we should just treat the supplied body as a body.
params.body = merge(params.body, body);
}
} else {
// Body and metadata were both supplied.
params.body = merge(params.body, body);
}
}
if (!operation.hasRequestBody()) {
// If this operation doesn't have any documented request body then we shouldn't be sending
// anything.
delete params.body;
} else {
if (!('body' in params)) params.body = {};
// We need to retrieve the request body for this operation to search for any `binary` format
// data that the user wants to send so we know what we need to prepare for the final API
// request.
const payloadJsonSchema = jsonSchema.find(js => js.type === 'body');
if (payloadJsonSchema) {
if (!params.files) params.files = {};
const conversions = [];
// @todo add support for `type: array`, `oneOf` and `anyOf`
if (payloadJsonSchema.schema?.properties) {
Object.entries(payloadJsonSchema.schema?.properties)
.filter(([, schema]: [string, SchemaObject]) => schema?.format === 'binary')
.filter(([prop]) => Object.keys(params.body).includes(prop))
.forEach(([prop]) => {
conversions.push(processFile(prop, params.body[prop]));
});
} else if (payloadJsonSchema.schema?.type === 'string') {
if (payloadJsonSchema.schema?.format === 'binary') {
conversions.push(processFile(undefined, params.body));
}
}
await Promise.all(conversions)
.then(fileMetadata => fileMetadata.filter(Boolean))
.then(fm => {
fm.forEach(fileMetadata => {
if (!fileMetadata) {
// If we don't have any metadata here it's because the file we have is likely
// the full string content of the file so since we don't have any filenames to
// work with we shouldn't do any additional handling to the `body` or `files`
// parameters.
return;
}
if (fileMetadata.paramName) {
params.body[fileMetadata.paramName] = fileMetadata.base64;
} else {
params.body = fileMetadata.base64;
}
params.files[fileMetadata.filename] = fileMetadata.buffer;
});
});
}
}
// Form data should be placed within `formData` instead of `body` for it to properly get picked
// up by `fetch-har`.
if (operation.isFormUrlEncoded()) {
params.formData = merge(params.formData, params.body);
delete params.body;
}
// Only spend time trying to organize metadata into parameters if we were able to digest
// parameters out of the operation schema. If we couldn't digest anything, but metadata was
// supplied then we wouldn't know how to send it in the request!
if (typeof metadata !== 'undefined') {
if (!('cookie' in params)) params.cookie = {};
if (!('header' in params)) params.header = {};
if (!('path' in params)) params.path = {};
if (!('query' in params)) params.query = {};
Object.entries(digestedParameters).forEach(([paramName, param]) => {
let value: any;
let metadataHeaderParam;
if (typeof metadata === 'object' && !isEmpty(metadata)) {
if (paramName in metadata) {
value = metadata[paramName];
metadataHeaderParam = paramName;
} else if (param.in === 'header') {
// Headers are sent case-insensitive so we need to make sure that we're properly
// matching them when detecting what our incoming payload looks like.
metadataHeaderParam = Object.keys(metadata).find(k => k.toLowerCase() === paramName.toLowerCase());
value = metadata[metadataHeaderParam];
}
}
if (value === undefined) {
return;
}
/* eslint-disable no-param-reassign */
switch (param.in) {
case 'path':
params.path[paramName] = value;
delete metadata[paramName];
break;
case 'query':
params.query[paramName] = value;
delete metadata[paramName];
break;
case 'header':
params.header[paramName.toLowerCase()] = value;
delete metadata[metadataHeaderParam];
break;
case 'cookie':
params.cookie[paramName] = value;
delete metadata[paramName];
break;
default: // no-op
}
/* eslint-enable no-param-reassign */
// Because a user might have sent just a metadata object, we want to make sure that we filter
// out anything that they sent that is a parameter from also being sent as part of a form
// data payload for `x-www-form-urlencoded` requests.
if (metadataIntersected && operation.isFormUrlEncoded()) {
if (paramName in params.formData) {
delete params.formData[paramName];
}
}
});
// If there's any leftover metadata that hasn't been moved into form data for this request we
// need to move it or else it'll get tossed.
if (!isEmpty(metadata)) {
if (typeof metadata === 'object') {
// If the user supplied an `accept` or `authorization` header themselves we should allow it
// through. Normally these headers are automatically handled by `@readme/oas-to-har` but in
// the event that maybe the user wants to return XML for an API that normally returns JSON
// or specify a custom auth header (maybe we can't handle their auth case right) this is the
// only way with this library that they can do that.
specialHeaders.forEach(headerName => {
const headerParam = Object.keys(metadata).find(m => m.toLowerCase() === headerName);
if (headerParam) {
params.header[headerName] = metadata[headerParam] as string;
// eslint-disable-next-line no-param-reassign
delete metadata[headerParam];
}
});
}
if (operation.isFormUrlEncoded()) {
params.formData = merge(params.formData, metadata);
} else {
// Any other remaining unused metadata will be unused because we don't know where to place
// it in the request.
}
}
}
['body', 'cookie', 'files', 'formData', 'header', 'path', 'query'].forEach((type: keyof typeof params) => {
if (type in params && isEmpty(params[type])) {
delete params[type];
}
});
return params;
}