2019-10-19 14:39:53 -07:00
|
|
|
const changeCase = require('change-case');
|
|
|
|
|
2019-10-19 02:25:09 -07:00
|
|
|
// flatMap: https://gist.github.com/samgiles/762ee337dff48623e729
|
|
|
|
// [B](f: (A) ⇒ [B]): [B] ; Although the types in the arrays aren't strict (:
|
|
|
|
Array.prototype.flatMap = function(lambda) {
|
|
|
|
return Array.prototype.concat.apply([], this.map(lambda));
|
|
|
|
};
|
|
|
|
Array.prototype.groupBy = function(lambda) {
|
|
|
|
return Object.entries(this.reduce((agg, x) => {
|
|
|
|
const k = lambda(x);
|
|
|
|
(agg[k] = agg[k] || []).push(x);
|
|
|
|
return agg;
|
|
|
|
}, {}));
|
2019-10-31 01:09:20 -07:00
|
|
|
};
|
|
|
|
Array.prototype.sortBy = function(lambda) {
|
|
|
|
return this.sort((a, b) => {
|
|
|
|
const va = lambda(a);
|
|
|
|
const vb = lambda(b);
|
|
|
|
if ((typeof va) !== (typeof vb))
|
|
|
|
throw Error(`Mismatched sort types: ${typeof va}, ${typeof vb}.`);
|
|
|
|
if (typeof va === 'number')
|
|
|
|
return va - vb;
|
|
|
|
if (typeof va === 'string')
|
|
|
|
return va.localeCompare(vb);
|
|
|
|
throw Error(`Unknown sort type: ${typeof va}.`);
|
|
|
|
});
|
|
|
|
};
|
2019-10-19 02:25:09 -07:00
|
|
|
|
2019-10-23 00:39:40 -07:00
|
|
|
function preamble() {
|
2019-10-25 22:19:00 -07:00
|
|
|
return `\
|
|
|
|
///////////////////////////////////////////////
|
|
|
|
// //
|
|
|
|
// ! //
|
|
|
|
// This file is automatically generated! //
|
|
|
|
// Do not directly edit! //
|
|
|
|
// //
|
|
|
|
///////////////////////////////////////////////`;
|
2019-10-23 00:39:40 -07:00
|
|
|
}
|
|
|
|
|
2019-10-19 02:25:09 -07:00
|
|
|
function capitalize(input) {
|
|
|
|
return input[0].toUpperCase() + input.slice(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
function decapitalize(input) {
|
|
|
|
return input[0].toLowerCase() + input.slice(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
function normalizeSchemaName(name) {
|
|
|
|
return name.replace(/DTO/ig, '');
|
|
|
|
}
|
|
|
|
|
|
|
|
function normalizeArgName(name) {
|
2019-10-19 14:39:53 -07:00
|
|
|
const tokens = name.split('_');
|
|
|
|
const argName = decapitalize(tokens.map(capitalize).join(''));
|
2019-10-19 02:25:09 -07:00
|
|
|
return 'base' === argName ? 'Base' : argName;
|
|
|
|
}
|
|
|
|
|
2019-10-31 01:09:20 -07:00
|
|
|
function normalizePropName(propName) {
|
2021-05-30 22:27:06 -07:00
|
|
|
let out = changeCase.snakeCase(propName);
|
|
|
|
if (/^\d/.test(out)) // No leading digits.
|
|
|
|
out = 'x' + out;
|
2019-10-19 14:39:53 -07:00
|
|
|
if ('type' === out)
|
|
|
|
return 'r#' + out;
|
|
|
|
return out;
|
2019-10-19 02:25:09 -07:00
|
|
|
}
|
|
|
|
|
2019-10-31 01:09:20 -07:00
|
|
|
function stringifyType(prop, { endpoint = null, optional = false, fullpath = true, owned = true }) {
|
2019-10-19 02:25:09 -07:00
|
|
|
if (prop.anyOf) {
|
|
|
|
prop = prop.anyOf[0];
|
|
|
|
}
|
2019-10-31 01:09:20 -07:00
|
|
|
if (optional) {
|
2020-07-10 01:56:11 -07:00
|
|
|
return `Option<${stringifyType(prop, { endpoint, fullpath, owned })}>`;
|
2019-10-31 01:09:20 -07:00
|
|
|
}
|
2019-10-19 02:25:09 -07:00
|
|
|
|
2019-10-25 22:19:00 -07:00
|
|
|
let enumType = prop['x-enum'];
|
|
|
|
if (enumType && 'locale' !== enumType)
|
|
|
|
return 'crate::consts::' + changeCase.pascalCase(enumType);
|
|
|
|
|
2019-10-19 02:25:09 -07:00
|
|
|
let refType = prop['$ref'];
|
|
|
|
if (refType) {
|
2019-10-22 14:28:23 -07:00
|
|
|
return (!endpoint ? '' : changeCase.snakeCase(endpoint) + '::') +
|
2019-10-19 02:25:09 -07:00
|
|
|
normalizeSchemaName(refType.slice(refType.indexOf('.') + 1));
|
|
|
|
}
|
|
|
|
switch (prop.type) {
|
2019-10-19 14:39:53 -07:00
|
|
|
case 'boolean': return 'bool';
|
|
|
|
case 'integer': return ('int32' === prop.format ? 'i32' : 'i64');
|
|
|
|
case 'number': return ('float' === prop.format ? 'f32' : 'f64');
|
|
|
|
case 'array':
|
2019-10-31 01:09:20 -07:00
|
|
|
const subprop = stringifyType(prop.items, { endpoint, optional, fullpath, owned });
|
2019-10-19 14:39:53 -07:00
|
|
|
return (owned ? (fullpath ? 'std::vec::' : '') + `Vec<${subprop}>` : `&[${subprop}]`);
|
|
|
|
case 'string': return (owned ? 'String' : '&str');
|
2019-10-19 02:25:09 -07:00
|
|
|
case 'object':
|
2019-10-31 01:09:20 -07:00
|
|
|
return 'std::collections::HashMap<' + stringifyType(prop['x-key'], { endpoint, optional, fullpath, owned }) + ', ' +
|
|
|
|
stringifyType(prop.additionalProperties, { endpoint, optional, fullpath, owned }) + '>';
|
2019-10-19 14:39:53 -07:00
|
|
|
default: return prop.type;
|
2019-10-19 02:25:09 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function formatJsonProperty(name) {
|
2019-10-19 14:39:53 -07:00
|
|
|
return `#[serde(rename = "${name}")]`;
|
2019-10-19 02:25:09 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
function formatAddQueryParam(param) {
|
2021-05-21 19:04:04 -07:00
|
|
|
const k = `"${param.name}"`;
|
|
|
|
const name = changeCase.snakeCase(param.name);
|
|
|
|
const condStart = param.required ? '' : `mut request = request; if let Some(${name}) = ${name} { `;
|
|
|
|
const condEnd = param.required ? '' : ' }'
|
|
|
|
const prop = param.schema;
|
2019-10-29 16:51:41 -07:00
|
|
|
switch (prop.type) {
|
2021-05-21 19:04:04 -07:00
|
|
|
case 'array': return `let ${condStart}request = request.query(&*${name}.iter()`
|
|
|
|
+ `.map(|w| ( ${k}, w )).collect::<Vec<_>>());${condEnd}`;
|
|
|
|
case 'object':
|
|
|
|
throw 'unsupported';
|
|
|
|
default:
|
|
|
|
return `let ${condStart}request = request.query(&[ (${k}, ${name}) ]);${condEnd}`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function formatAddHeaderParam(param) {
|
|
|
|
const k = `"${param.name}"`;
|
|
|
|
const name = changeCase.snakeCase(param.name);
|
|
|
|
const condStart = param.required ? '' : `mut request = request; if let Some(${name}) = ${name} { `;
|
|
|
|
const condEnd = param.required ? '' : ' }'
|
|
|
|
const prop = param.schema;
|
|
|
|
switch (prop.type) {
|
|
|
|
case 'string':
|
|
|
|
return `let ${condStart}request = request.header(${k}, ${name});${condEnd}`;
|
|
|
|
case 'object':
|
|
|
|
throw 'unsupported';
|
2019-10-29 16:51:41 -07:00
|
|
|
default:
|
2021-05-21 19:04:04 -07:00
|
|
|
return `let ${condStart}request = request.header(${k}, ${name}.to_string());${condEnd}`;
|
2019-10-29 16:51:41 -07:00
|
|
|
}
|
2019-10-19 02:25:09 -07:00
|
|
|
}
|
|
|
|
|
2019-10-19 14:39:53 -07:00
|
|
|
function formatRouteArgument(route, pathParams = []) {
|
|
|
|
if (!pathParams.length)
|
2021-05-21 17:31:52 -07:00
|
|
|
return `"${route}"`;
|
2019-10-19 14:39:53 -07:00
|
|
|
|
|
|
|
route = route.replace(/\{\S+?\}/g, '{}');
|
|
|
|
const args = pathParams
|
|
|
|
.map(({name}) => name)
|
|
|
|
.map(changeCase.snakeCase)
|
|
|
|
.join(', ');
|
2021-05-21 17:31:52 -07:00
|
|
|
return `&format!("${route}", ${args})`;
|
2019-10-19 14:39:53 -07:00
|
|
|
}
|
|
|
|
|
2019-10-19 02:25:09 -07:00
|
|
|
module.exports = {
|
2019-10-19 14:39:53 -07:00
|
|
|
changeCase,
|
2019-10-23 00:39:40 -07:00
|
|
|
preamble,
|
2019-10-19 02:25:09 -07:00
|
|
|
capitalize,
|
|
|
|
decapitalize,
|
|
|
|
normalizeSchemaName,
|
|
|
|
normalizeArgName,
|
|
|
|
normalizePropName,
|
|
|
|
stringifyType,
|
|
|
|
formatJsonProperty,
|
2019-10-19 14:39:53 -07:00
|
|
|
formatAddQueryParam,
|
2021-05-21 19:04:04 -07:00
|
|
|
formatAddHeaderParam,
|
2019-10-19 14:39:53 -07:00
|
|
|
formatRouteArgument,
|
2019-10-19 02:25:09 -07:00
|
|
|
};
|