api
Version:
Magical SDK generation from an OpenAPI definition 🪄
204 lines (173 loc) • 7.5 kB
text/typescript
import type { ConfigOptions } from './core';
import type { Operation } from 'oas';
import type { OASDocument } from 'oas/dist/rmoas.types';
import Oas from 'oas';
import Cache from './cache';
import APICore from './core';
import { PACKAGE_NAME, PACKAGE_VERSION } from './packageInfo';
interface SDKOptions {
cacheDir?: string;
}
class Sdk {
uri: string | OASDocument;
userAgent: string;
cacheDir: string | false;
constructor(uri: string | OASDocument, opts: SDKOptions = {}) {
this.uri = uri;
this.userAgent = `${PACKAGE_NAME} (node)/${PACKAGE_VERSION}`;
this.cacheDir = opts.cacheDir ? opts.cacheDir : false;
}
load() {
const cache = new Cache(this.uri, this.cacheDir);
const userAgent = this.userAgent;
const core = new APICore();
core.setUserAgent(userAgent);
let isLoaded = false;
let isCached = cache.isCached();
let sdk = {};
/**
* Create dynamic accessors for every operation with a defined operation ID. If an operation
* does not have an operation ID it can be accessed by its `.method('/path')` accessor instead.
*
*/
function loadOperations(spec: Oas) {
return Object.entries(spec.getPaths())
.map(([, operations]) => Object.values(operations))
.reduce((prev, next) => prev.concat(next), [])
.reduce((prev, next) => {
// `getOperationId()` creates dynamic operation IDs when one isn't available but we need
// to know here if we actually have one present or not. The `camelCase` option here also
// cleans up any `operationId` that we might have into something that can be used as a
// valid JS method.
const originalOperationId = next.getOperationId();
const operationId = next.getOperationId({ camelCase: true });
const op = {
[operationId]: ((operation: Operation, ...args: unknown[]) => {
return core.fetchOperation(operation, ...args);
}).bind(null, next),
};
if (operationId !== originalOperationId) {
// If we cleaned up their operation ID into a friendly method accessor (`findPetById`
// versus `find pet by id`) we should still let them use the non-friendly version if
// they want.
//
// This work is to maintain backwards compatibility with `api@4` and does not exist
// within our code generated SDKs -- those only allow the cleaner camelCase
// `operationId` to be used.
op[originalOperationId] = ((operation: Operation, ...args: unknown[]) => {
return core.fetchOperation(operation, ...args);
}).bind(null, next);
}
return Object.assign(prev, op);
}, {});
}
async function loadFromCache() {
let cachedSpec;
if (isCached) {
cachedSpec = await cache.get();
} else {
cachedSpec = await cache.load();
isCached = true;
}
const spec = new Oas(cachedSpec);
core.setSpec(spec);
sdk = Object.assign(sdk, loadOperations(spec));
isLoaded = true;
}
const sdkProxy = {
// @give this a better type than any
get(target: any, method: string) {
// Since auth returns a self-proxy, we **do not** want it to fall through into the async
// function below as when that'll happen, instead of returning a self-proxy, it'll end up
// returning a Promise. When that happens, chaining `sdk.auth().operationId()` will fail.
if (['auth', 'config'].includes(method)) {
// @todo split this up so we have better types for `auth` and `config`
return function authAndConfigHandler(...args: any) {
return target[method].apply(this, args);
};
}
return async function accessorHandler(...args: unknown[]) {
if (!(method in target)) {
// If this method doesn't exist on the proxy, have we loaded the SDK? If we have, then
// this method isn't valid.
if (isLoaded) {
throw new Error(`Sorry, \`${method}\` does not appear to be a valid operation on this API.`);
}
await loadFromCache();
// If after loading the SDK and this method still doesn't exist, then it's not real!
if (!(method in sdk)) {
throw new Error(`Sorry, \`${method}\` does not appear to be a valid operation on this API.`);
}
// @todo give sdk a better type
return (sdk as any)[method].apply(this, args);
}
return target[method].apply(this, args);
};
},
};
sdk = {
/**
* If the API you're using requires authentication you can supply the required credentials
* through this method and the library will magically determine how they should be used
* within your API request.
*
* With the exception of OpenID and MutualTLS, it supports all forms of authentication
* supported by the OpenAPI specification.
*
* @example <caption>HTTP Basic auth</caption>
* sdk.auth('username', 'password');
*
* @example <caption>Bearer tokens (HTTP or OAuth 2)</caption>
* sdk.auth('myBearerToken');
*
* @example <caption>API Keys</caption>
* sdk.auth('myApiKey');
*
* @see {@link https://spec.openapis.org/oas/v3.0.3#fixed-fields-22}
* @see {@link https://spec.openapis.org/oas/v3.1.0#fixed-fields-22}
* @param values Your auth credentials for the API. Can specify up to two strings or numbers.
*/
auth: (...values: string[] | number[]) => {
core.setAuth(...values);
},
/**
* Optionally configure various options that the SDK allows.
*
* @param config Object of supported SDK options and toggles.
* @param config.timeout Override the default `fetch` request timeout of 30 seconds (30000ms).
*/
config: (config: ConfigOptions) => {
core.setConfig(config);
},
/**
* If the API you're using offers alternate server URLs, and server variables, you can tell
* the SDK which one to use with this method. To use it you can supply either one of the
* server URLs that are contained within the OpenAPI definition (along with any server
* variables), or you can pass it a fully qualified URL to use (that may or may not exist
* within the OpenAPI definition).
*
* @example <caption>Server URL with server variables</caption>
* sdk.server('https://{region}.api.example.com/{basePath}', {
* name: 'eu',
* basePath: 'v14',
* });
*
* @example <caption>Fully qualified server URL</caption>
* sdk.server('https://eu.api.example.com/v14');
*
* @param url Server URL
* @param variables An object of variables to replace into the server URL.
*/
server: (url: string, variables = {}) => {
core.setServer(url, variables);
},
};
return new Proxy(sdk, sdkProxy);
}
}
// Why `export` vs `export default`? If we leave this as `export` then TS will transpile it into
// a `module.exports` export so that when folks load this they don't need to load it as
// `require('api').default`.
export = (uri: string | OASDocument, opts: SDKOptions = {}) => {
return new Sdk(uri, opts).load();
};