2020-04-05 00:57:34 +09:00
|
|
|
import {StrictTypeParser as STP} from './utils/StrictTypeParser';
|
|
|
|
const warn = (x: any) => console.warn('\x1b[1;33mWarning: '+x+'\x1b[0m');
|
2020-05-24 15:15:53 +09:00
|
|
|
type Dict<T> = {[_: string]: T};
|
2020-04-05 00:57:34 +09:00
|
|
|
|
|
|
|
/* ==== type declaration ==== */
|
|
|
|
export interface OpenAPI {
|
|
|
|
paths: Paths;
|
|
|
|
components?: Components;
|
|
|
|
}
|
|
|
|
|
|
|
|
// path
|
|
|
|
interface Paths {
|
|
|
|
[path: string]: PathItem
|
|
|
|
}
|
|
|
|
interface PathItem {
|
|
|
|
get?: Operation;
|
|
|
|
put?: Operation;
|
|
|
|
post?: Operation;
|
|
|
|
delete?: Operation;
|
|
|
|
patch?: Operation;
|
|
|
|
[_: string]: any;
|
|
|
|
}
|
|
|
|
type EMethod = 'get' | 'put' | 'post' | 'delete' | 'patch';
|
|
|
|
const ELMethod: Array<EMethod> = ['get', 'put', 'post', 'delete', 'patch'];
|
|
|
|
interface Operation {
|
2020-05-24 15:15:53 +09:00
|
|
|
responses: Dict<Response | Reference>;
|
|
|
|
parameters?: Array<Parameter | Reference>;
|
|
|
|
requestBody?: RequestBody | Reference;
|
2020-04-05 00:57:34 +09:00
|
|
|
operationId?: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
// response
|
|
|
|
interface Response {
|
|
|
|
// headers?: Header;
|
|
|
|
content?: TMediaTypes;
|
|
|
|
}
|
|
|
|
type TMediaTypes = {[contentType: string]: MediaType};
|
|
|
|
interface MediaType {
|
|
|
|
schema?: Schema | Reference;
|
|
|
|
example?: any;
|
|
|
|
examples?: {[_: string]: object};
|
|
|
|
}
|
|
|
|
|
|
|
|
// parameter
|
|
|
|
interface Parameter {
|
|
|
|
name: string;
|
|
|
|
in: EParameterIn;
|
|
|
|
description?: string;
|
|
|
|
required?: boolean;
|
|
|
|
deprecated?: boolean;
|
|
|
|
style?: string;
|
|
|
|
schema?: Schema | Reference;
|
|
|
|
}
|
|
|
|
type EParameterIn = 'query' | 'header' | 'path' | 'cookie';
|
|
|
|
export const ELParameterIn: Array<EParameterIn> = [
|
|
|
|
'path', 'query', 'header', 'cookie'];
|
|
|
|
|
|
|
|
// request body
|
|
|
|
interface RequestBody {
|
|
|
|
description: string;
|
2020-05-24 15:15:53 +09:00
|
|
|
content: Dict<MediaType>;
|
2020-04-05 00:57:34 +09:00
|
|
|
required?: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
// components
|
|
|
|
interface Components {
|
2020-05-24 15:15:53 +09:00
|
|
|
schemas: Dict<Schema | Reference>;
|
|
|
|
responses: Dict<Response | Reference>;
|
|
|
|
parameters: Dict<Parameter | Reference>;
|
|
|
|
requestBodies: Dict<RequestBody | Reference>;
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
// schemeType
|
2020-05-24 15:15:53 +09:00
|
|
|
export interface Schema {
|
2020-04-05 00:57:34 +09:00
|
|
|
type: string;
|
|
|
|
format?: string;
|
|
|
|
nullable?: boolean;
|
|
|
|
readOnly?: boolean;
|
|
|
|
maxSize?: number;
|
2020-05-20 05:17:08 +09:00
|
|
|
required?: string[];
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
interface ArraySchema extends Schema {
|
|
|
|
items: Schema | Reference;
|
|
|
|
}
|
|
|
|
export function isArraySchema(x: any): x is ArraySchema {
|
|
|
|
return x.type === 'array';
|
|
|
|
}
|
|
|
|
interface ObjectSchema extends Schema {
|
|
|
|
properties: {[name: string]: Schema | Reference};
|
|
|
|
}
|
|
|
|
export function isObjectSchema(x: any): x is ObjectSchema {
|
|
|
|
return x.type === 'object';
|
|
|
|
}
|
2020-05-24 15:15:53 +09:00
|
|
|
export interface Reference {
|
2020-04-05 00:57:34 +09:00
|
|
|
$ref: string;
|
|
|
|
maxSize?: string | number;
|
|
|
|
}
|
|
|
|
function isReference(x: any): x is Reference {
|
|
|
|
return typeof x.$ref === 'string';
|
|
|
|
}
|
|
|
|
|
|
|
|
// api
|
|
|
|
class APIFunction {
|
|
|
|
constructor(
|
|
|
|
public method: string,
|
|
|
|
public url: string,
|
|
|
|
public reqTypes: TReqTypes,
|
|
|
|
public resTypes: TResTypes,
|
|
|
|
) {}
|
|
|
|
}
|
|
|
|
type TReqTypes = {
|
|
|
|
query?: {[name: string]: SchemaType};
|
|
|
|
header?: {[name: string]: SchemaType};
|
|
|
|
path?: {[name: string]: SchemaType};
|
|
|
|
cookie?: {[name: string]: SchemaType};
|
|
|
|
body?: SchemaType;
|
|
|
|
};
|
|
|
|
type TResTypes = {[status: string]: SchemaType};
|
|
|
|
/* ==== ==== */
|
|
|
|
|
2020-05-24 15:15:53 +09:00
|
|
|
// Reference
|
|
|
|
export function resolveRef<T>(
|
|
|
|
obj: T|Reference, dict: Dict<T|Reference>|undefined, prefix: string,
|
|
|
|
): T | undefined {
|
|
|
|
do {
|
|
|
|
if (!isReference(obj)) return obj;
|
|
|
|
const ref = obj.$ref;
|
|
|
|
if (ref.startsWith(prefix)) {
|
|
|
|
const name = ref.substring(prefix.length+1); // $prefix/
|
|
|
|
const obj0 = dict?.[name];
|
|
|
|
if (obj0 === undefined) {
|
|
|
|
console.error(`ref not found: ${ref}`);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
obj = obj0;
|
|
|
|
} else {
|
|
|
|
console.error(`Invalid ref: ${ref}, expect prefix ${prefix}`);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} while (true);
|
|
|
|
}
|
|
|
|
|
|
|
|
function mediaTypes2type(
|
|
|
|
content?: TMediaTypes, required?: boolean,
|
|
|
|
): SchemaType {
|
2020-04-05 00:57:34 +09:00
|
|
|
const media = content?.['application/json']; // TODO
|
|
|
|
if (media == null) {
|
|
|
|
if (Object.keys(content ?? {}).length > 0) {
|
|
|
|
warn('only support application/json now');
|
|
|
|
}
|
2020-08-17 13:12:34 +08:00
|
|
|
return new SchemaType('any', false, false);
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
// schema
|
|
|
|
const {schema} = media;
|
2020-08-17 13:12:34 +08:00
|
|
|
return new SchemaType(schema ?? 'any', required ?? false, false);
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
export class SchemaType {
|
|
|
|
private _typeName?: string;
|
|
|
|
get typeName(): string {
|
|
|
|
return this._typeName ??
|
2020-08-17 13:12:34 +08:00
|
|
|
(this._typeName = SchemaType.typeNameOf(this.schema, this._sameFile));
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
get required(): boolean {
|
|
|
|
return this._required;
|
|
|
|
}
|
|
|
|
get maxSize(): string | number | undefined {
|
|
|
|
return this.schema.maxSize;
|
|
|
|
}
|
|
|
|
forProp(prop: string): string {
|
|
|
|
return `${prop}${this.required ? '' : '?'}: ${this.typeName}`;
|
|
|
|
}
|
2020-08-17 13:12:34 +08:00
|
|
|
stp(
|
|
|
|
prop: string, label: string,
|
|
|
|
partial: boolean=false, sameFile: boolean=false,
|
|
|
|
): string {
|
|
|
|
const stp = SchemaType.gcStp(prop, this.schema, label, partial, sameFile);
|
2020-04-15 07:45:51 +09:00
|
|
|
return (this.required ? '' : `${prop}===void 0 ? void 0 : `)+stp;
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
private schema: Schema | Reference;
|
2020-08-17 13:12:34 +08:00
|
|
|
constructor(
|
|
|
|
schema: Schema | Reference | string,
|
|
|
|
private _required: boolean,
|
|
|
|
private _sameFile: boolean,
|
|
|
|
) {
|
2020-04-05 00:57:34 +09:00
|
|
|
this.schema = typeof schema === 'string' ? {type: schema} : schema;
|
|
|
|
}
|
|
|
|
|
2020-08-17 13:12:34 +08:00
|
|
|
static typeNameOf(schema: Schema | Reference, sameFile: boolean): string {
|
2020-04-05 00:57:34 +09:00
|
|
|
if (isReference(schema)) {
|
|
|
|
const {$ref} = schema;
|
|
|
|
const typeName = /^#\/components\/schemas\/(\w+)$/g.exec($ref)?.[1];
|
|
|
|
if (typeName == null) {
|
|
|
|
warn(`Invalid $ref, use any instead: ${$ref}`);
|
|
|
|
return 'any';
|
|
|
|
}
|
2020-08-17 13:12:34 +08:00
|
|
|
return sameFile ? typeName : `Schemas.${typeName}`;
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
const {
|
|
|
|
type, format, nullable, readOnly,
|
|
|
|
} = schema;
|
|
|
|
let sType = type;
|
|
|
|
if (isArraySchema(schema)) {
|
2020-08-17 13:12:34 +08:00
|
|
|
sType = `Array<${SchemaType.typeNameOf(schema.items, sameFile)}>`;
|
2020-04-05 00:57:34 +09:00
|
|
|
} else if (isObjectSchema(schema)) {
|
|
|
|
sType = '{';
|
|
|
|
for (const [name, sub] of Object.entries(schema.properties)) {
|
2020-08-17 13:12:34 +08:00
|
|
|
sType += `${name}: ${SchemaType.typeNameOf(sub, sameFile)}, `;
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
sType += '}';
|
|
|
|
} else if (type === 'string') {
|
|
|
|
if (format === 'date-time') sType = 'Date';
|
|
|
|
else if (format === 'date') sType = 'FullDate';
|
|
|
|
else if (format === 'byte') sType = 'string'; // TODO Buffer
|
|
|
|
else if (format === 'binary') sType = 'string'; // TODO Buffer
|
|
|
|
else if (format) warn(`Unknown format ${format}, use string instead`);
|
|
|
|
} else if (type === 'integer') sType = 'number'; // TODO integer
|
|
|
|
if (nullable) sType = `${sType} | null`;
|
|
|
|
if (readOnly) sType = `Readonly<${sType}>`;
|
|
|
|
return sType;
|
|
|
|
}
|
2020-04-08 20:17:57 +09:00
|
|
|
static gcStp(para: string, schema: Schema | Reference,
|
2020-08-17 13:12:34 +08:00
|
|
|
label: string, partial: boolean, sameFile: boolean): string {
|
2020-04-08 20:17:57 +09:00
|
|
|
// partial: Object only, 1 layer only
|
2020-04-05 00:57:34 +09:00
|
|
|
// object
|
|
|
|
if (isReference(schema)) {
|
2020-08-17 13:12:34 +08:00
|
|
|
const typeName = new SchemaType(schema, true, sameFile).typeName;
|
2020-04-15 07:45:51 +09:00
|
|
|
return `${typeName}.${partial ? 'Partial': 'from'}(${para})`;
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
// any
|
|
|
|
const {type, nullable, format} = schema;
|
2020-04-08 20:17:57 +09:00
|
|
|
let sStp;
|
2020-04-05 00:57:34 +09:00
|
|
|
if (type === 'any') return para;
|
|
|
|
if (isArraySchema(schema)) {
|
2020-08-17 13:12:34 +08:00
|
|
|
sStp = `(v, l)=>STP._Array(v, l, elm=>${SchemaType.gcStp(
|
|
|
|
'elm', schema.items, `${label}[]`, false, sameFile)})`;
|
2020-04-05 00:57:34 +09:00
|
|
|
} else if (isObjectSchema(schema)) {
|
2020-04-08 20:17:57 +09:00
|
|
|
sStp = '()=>({';
|
2020-04-05 00:57:34 +09:00
|
|
|
for (const [name, sub] of Object.entries(schema.properties)) {
|
2020-08-17 13:12:34 +08:00
|
|
|
sStp += `${name}: ${SchemaType.gcStp(
|
|
|
|
para+'.'+name, sub, label+'.'+name, false, sameFile)}, `;
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
2020-04-08 20:17:57 +09:00
|
|
|
sStp += '})';
|
2020-04-05 00:57:34 +09:00
|
|
|
} else {
|
|
|
|
let t;
|
|
|
|
if (type === 'string') {
|
|
|
|
if (format === 'date-time') t = 'Date';
|
|
|
|
else if (format === 'date') t = 'FullDate';
|
2020-04-08 20:17:57 +09:00
|
|
|
else if (format === 'byte') t = 'byte';
|
|
|
|
else if (format === 'binary') t = 'binary';
|
2020-04-05 00:57:34 +09:00
|
|
|
else {
|
2020-04-08 20:17:57 +09:00
|
|
|
if (format) {
|
|
|
|
warn(`Unknown string format ${format}, use string instead`);
|
|
|
|
}
|
2020-04-05 00:57:34 +09:00
|
|
|
t = 'string';
|
|
|
|
}
|
2020-04-08 20:17:57 +09:00
|
|
|
} else if (type === 'integer') {
|
|
|
|
if (format === 'int32') t = 'int32';
|
|
|
|
else {
|
2020-05-20 05:17:08 +09:00
|
|
|
if (format && format != 'int64') {
|
|
|
|
warn(`Unsupport integer format ${format}, use number instead`);
|
|
|
|
}
|
2020-04-08 20:17:57 +09:00
|
|
|
t = 'number'; // TODO int64
|
|
|
|
}
|
|
|
|
} else t = type;
|
2020-04-05 00:57:34 +09:00
|
|
|
if (!STP.supportTypes.includes(t)) {
|
2020-04-08 20:17:57 +09:00
|
|
|
warn(`Unsupport type ${type} ${format}, use any instead`);
|
2020-04-05 00:57:34 +09:00
|
|
|
return para;
|
2020-04-08 20:17:57 +09:00
|
|
|
}
|
|
|
|
sStp = `STP._${t}`;
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
// nullable
|
2020-04-08 20:17:57 +09:00
|
|
|
const funcName = nullable ? 'nullableParse' : 'parse';
|
|
|
|
// result
|
|
|
|
const sLabel = `'${label.replace(/'/g, '\\\'')}'`; // escape
|
|
|
|
return `STP.${funcName}(${sStp}, ${para}, ${sLabel})`;
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export type APIFunctions = {[_: string]: APIFunction};
|
|
|
|
export function apiFunctionsOf(openAPI: OpenAPI): APIFunctions {
|
2020-05-24 15:15:53 +09:00
|
|
|
const {paths, components: comps} = openAPI;
|
|
|
|
const compPrefix = '#/components/';
|
2020-04-05 00:57:34 +09:00
|
|
|
const functions: APIFunctions = {};
|
|
|
|
for (const [url, pathItem] of Object.entries(paths)) {
|
|
|
|
for (const method of ELMethod) {
|
|
|
|
const op = pathItem[method];
|
|
|
|
if (op == null) continue;
|
|
|
|
// operationId
|
|
|
|
const {
|
|
|
|
operationId, parameters, requestBody, responses,
|
|
|
|
} = op;
|
|
|
|
if (operationId == null) {
|
|
|
|
warn(`ignore operation in ${method} ${url}: ` +
|
|
|
|
'operationId should be given');
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const name = operationId;
|
|
|
|
const reqTypes: TReqTypes = {};
|
|
|
|
const resTypes: TResTypes = {};
|
|
|
|
// reqParas
|
|
|
|
if (parameters != null) {
|
2020-05-24 15:15:53 +09:00
|
|
|
for (const rPara of parameters) {
|
|
|
|
const para = resolveRef(
|
|
|
|
rPara, comps?.parameters, compPrefix+'parameters');
|
|
|
|
if (para == null) continue;
|
2020-04-05 00:57:34 +09:00
|
|
|
const {
|
|
|
|
name, in: _in, required, schema,
|
|
|
|
} = para;
|
|
|
|
// add
|
|
|
|
if (reqTypes[_in] == null) reqTypes[_in] = {};
|
|
|
|
reqTypes[_in]![name] = new SchemaType(
|
2020-08-17 13:12:34 +08:00
|
|
|
schema ?? 'any', required ?? false, false);
|
2020-04-05 00:57:34 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// requestBody
|
|
|
|
if (requestBody != null) {
|
2020-05-24 15:15:53 +09:00
|
|
|
const requestBodyO = resolveRef(
|
|
|
|
requestBody, comps?.requestBodies, compPrefix+'requestBodies');
|
|
|
|
if (requestBodyO == null) continue;
|
2020-04-05 00:57:34 +09:00
|
|
|
reqTypes.body = mediaTypes2type(
|
2020-05-24 15:15:53 +09:00
|
|
|
requestBodyO.content,
|
|
|
|
requestBodyO.required,
|
2020-04-05 00:57:34 +09:00
|
|
|
);
|
|
|
|
}
|
|
|
|
// responses
|
2020-05-24 15:15:53 +09:00
|
|
|
for (const [status, rRes] of Object.entries(responses)) {
|
|
|
|
const res = resolveRef(rRes, comps?.responses, compPrefix+'responses');
|
|
|
|
if (res == null) continue;
|
2020-04-05 00:57:34 +09:00
|
|
|
resTypes[status] = mediaTypes2type(res.content, true);
|
|
|
|
}
|
|
|
|
// add to group
|
|
|
|
const saf = new APIFunction(method, url, reqTypes, resTypes);
|
|
|
|
functions[name] = saf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return functions;
|
|
|
|
}
|