@visulima/api-platform 1.0.2 → 1.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +16 -0
- package/bin/index.js +46 -0
- package/dist/{chunk-XXZ56SKG.mjs → chunk-2M45MXC2.mjs} +1 -2
- package/dist/chunk-2M45MXC2.mjs.map +1 -0
- package/dist/chunk-3Y3YBDRS.mjs +284 -0
- package/dist/chunk-3Y3YBDRS.mjs.map +1 -0
- package/dist/chunk-NY75KGAH.js +284 -0
- package/dist/chunk-NY75KGAH.js.map +1 -0
- package/dist/{chunk-AJKZCWFG.js → chunk-S46GRBOS.js} +1 -2
- package/dist/chunk-S46GRBOS.js.map +1 -0
- package/dist/index-server.d.ts +2 -0
- package/dist/index-server.js +9 -10
- package/dist/index-server.js.map +1 -1
- package/dist/index-server.mjs +7 -8
- package/dist/index-server.mjs.map +1 -1
- package/dist/next/cli/index.js +8 -8
- package/dist/next/cli/index.js.map +1 -1
- package/dist/next/cli/index.mjs +8 -8
- package/dist/next/cli/index.mjs.map +1 -1
- package/dist/next/index-browser.js +2 -2
- package/dist/next/index-browser.mjs +1 -1
- package/dist/next/index-server.d.ts +9 -5
- package/dist/next/index-server.js +4 -4
- package/dist/next/index-server.mjs +2 -2
- package/next/cli/package.json +12 -3
- package/next/package.json +9 -0
- package/package.json +20 -20
- package/recipes/api/swagger.ts +12 -0
- package/recipes/pages/redoc-ui.tsx +5 -0
- package/recipes/pages/swagger-ui.tsx +5 -0
- package/dist/chunk-2LATTLUM.mjs +0 -166
- package/dist/chunk-2LATTLUM.mjs.map +0 -1
- package/dist/chunk-AJKZCWFG.js.map +0 -1
- package/dist/chunk-S7GUPAL4.js +0 -166
- package/dist/chunk-S7GUPAL4.js.map +0 -1
- package/dist/chunk-XXZ56SKG.mjs.map +0 -1
- package/src/connect/create-node-router.ts +0 -44
- package/src/connect/handler.ts +0 -46
- package/src/connect/middleware/cors-middleware.ts +0 -10
- package/src/connect/middleware/http-header-normalizer.ts +0 -93
- package/src/connect/middleware/rate-limiter-middleware.ts +0 -43
- package/src/connect/middleware/serializers-middleware.ts +0 -121
- package/src/connect/serializers/types.d.ts +0 -1
- package/src/connect/serializers/xml.ts +0 -13
- package/src/connect/serializers/yaml.ts +0 -7
- package/src/error-handler/jsonapi-error-handler.ts +0 -46
- package/src/error-handler/problem-error-handler.ts +0 -44
- package/src/error-handler/types.d.ts +0 -14
- package/src/error-handler/utils.ts +0 -39
- package/src/index-browser.tsx +0 -1
- package/src/index-server.ts +0 -75
- package/src/next/cli/index.ts +0 -2
- package/src/next/cli/list/api-route-file-parser.ts +0 -74
- package/src/next/cli/list/collect-api-route-files.ts +0 -42
- package/src/next/cli/list/list-command.ts +0 -105
- package/src/next/cli/list/routes-render.ts +0 -62
- package/src/next/cli/list/types.d.ts +0 -1
- package/src/next/index-browser.tsx +0 -3
- package/src/next/index-server.ts +0 -6
- package/src/next/routes/api/swagger.ts +0 -23
- package/src/next/routes/pages/swagger/get-static-properties-swagger.ts +0 -32
- package/src/next/routes/pages/swagger/redoc.tsx +0 -35
- package/src/next/routes/pages/swagger/swagger.tsx +0 -44
- package/src/next/webpack/with-open-api.ts +0 -63
- package/src/swagger/extend-swagger-spec.ts +0 -167
- package/src/swagger/swagger-handler.ts +0 -83
- package/src/utils.ts +0 -37
- package/src/zod/date-in-schema.ts +0 -57
- package/src/zod/date-out-schema.ts +0 -41
- package/src/zod/index.ts +0 -9
|
@@ -0,0 +1,284 @@
|
|
|
1
|
+
"use strict";Object.defineProperty(exports, "__esModule", {value: true}); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }// src/swagger/swagger-handler.ts
|
|
2
|
+
var _crud = require('@visulima/crud');
|
|
3
|
+
var _debug = require('debug'); var _debug2 = _interopRequireDefault(_debug);
|
|
4
|
+
var _lodashmerge = require('lodash.merge'); var _lodashmerge2 = _interopRequireDefault(_lodashmerge);
|
|
5
|
+
var _fs = require('fs');
|
|
6
|
+
var _path = require('path'); var _path2 = _interopRequireDefault(_path);
|
|
7
|
+
|
|
8
|
+
// src/connect/serializers/yaml.ts
|
|
9
|
+
var _yaml = require('yaml');
|
|
10
|
+
var yamlTransformer = (data) => _yaml.stringify.call(void 0, data, { indent: 2 });
|
|
11
|
+
var yaml_default = yamlTransformer;
|
|
12
|
+
|
|
13
|
+
// src/swagger/extend-swagger-spec.ts
|
|
14
|
+
var _case = require('case');
|
|
15
|
+
var _jstoxml = require('jstoxml');
|
|
16
|
+
|
|
17
|
+
var jsonMediaType = "application/json";
|
|
18
|
+
var prepareStatusContent = (methodSpec, status, mediaType) => {
|
|
19
|
+
var _a, _b, _c, _d, _e;
|
|
20
|
+
if (((_b = (_a = methodSpec == null ? void 0 : methodSpec.responses) == null ? void 0 : _a[status]) == null ? void 0 : _b.content) === void 0) {
|
|
21
|
+
methodSpec.responses[status].content = {};
|
|
22
|
+
}
|
|
23
|
+
if (((_e = (_d = (_c = methodSpec == null ? void 0 : methodSpec.responses) == null ? void 0 : _c[status]) == null ? void 0 : _d.content) == null ? void 0 : _e[mediaType]) === void 0) {
|
|
24
|
+
methodSpec.responses[status].content[mediaType] = {};
|
|
25
|
+
}
|
|
26
|
+
};
|
|
27
|
+
var extendComponentSchemas = (spec, schemaName, schema) => {
|
|
28
|
+
if (typeof spec.components !== "object") {
|
|
29
|
+
spec.components = {};
|
|
30
|
+
}
|
|
31
|
+
if (typeof spec.components.schemas !== "object") {
|
|
32
|
+
spec.components.schemas = {};
|
|
33
|
+
}
|
|
34
|
+
if (spec.components.schemas[schemaName] === void 0) {
|
|
35
|
+
spec.components.schemas[schemaName] = schema;
|
|
36
|
+
}
|
|
37
|
+
};
|
|
38
|
+
var extendResponseSchema = (methodSpec, status, mediaType, schemaName, schemaIsArray) => {
|
|
39
|
+
var _a, _b, _c, _d;
|
|
40
|
+
prepareStatusContent(methodSpec, status, mediaType);
|
|
41
|
+
if (((_d = (_c = (_b = (_a = methodSpec == null ? void 0 : methodSpec.responses) == null ? void 0 : _a[status]) == null ? void 0 : _b.content) == null ? void 0 : _c[mediaType]) == null ? void 0 : _d.schema) === void 0) {
|
|
42
|
+
methodSpec.responses[status].content[mediaType].schema = {};
|
|
43
|
+
}
|
|
44
|
+
methodSpec.responses[status].content[mediaType].schema = schemaIsArray ? {
|
|
45
|
+
type: "array",
|
|
46
|
+
items: {
|
|
47
|
+
$ref: `#/components/schemas/${schemaName}`
|
|
48
|
+
}
|
|
49
|
+
} : {
|
|
50
|
+
$ref: `#/components/schemas/${schemaName}`
|
|
51
|
+
};
|
|
52
|
+
};
|
|
53
|
+
var extendSwaggerWithMediaTypeSchema = (methodSpec, responseSpec, allowedMediaTypes, pathKey, spec, status) => {
|
|
54
|
+
let example;
|
|
55
|
+
let examples;
|
|
56
|
+
Object.entries(responseSpec.content).forEach(([mediaName, contentSpec]) => {
|
|
57
|
+
if (typeof contentSpec.schema === "object") {
|
|
58
|
+
const { schema } = contentSpec;
|
|
59
|
+
if (mediaName === jsonMediaType && contentSpec.examples !== void 0) {
|
|
60
|
+
examples = contentSpec.examples;
|
|
61
|
+
} else if (mediaName === jsonMediaType && contentSpec.example !== void 0) {
|
|
62
|
+
example = contentSpec.example;
|
|
63
|
+
}
|
|
64
|
+
const schemaIsArray = schema.type === "array";
|
|
65
|
+
Object.entries(allowedMediaTypes || {}).forEach(([mediaType, allowed]) => {
|
|
66
|
+
if (!allowed) {
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
let schemaName;
|
|
70
|
+
if (schema.$ref === void 0) {
|
|
71
|
+
schemaName = `${_case.header.call(void 0, pathKey.trim().replace("/", ""))}${mediaType === "application/ld+json" ? ".jsonld" : ""}`;
|
|
72
|
+
extendComponentSchemas(spec, schemaName, schema);
|
|
73
|
+
} else {
|
|
74
|
+
schemaName = schema.$ref.replace("#/components/schemas/", "");
|
|
75
|
+
}
|
|
76
|
+
extendResponseSchema(methodSpec, status, mediaType, schemaName, schemaIsArray);
|
|
77
|
+
});
|
|
78
|
+
}
|
|
79
|
+
});
|
|
80
|
+
return { examples, example };
|
|
81
|
+
};
|
|
82
|
+
var extendSwaggerWithMediaTypeExample = (methodSpec, responseSpec, status, allowedMediaTypes, transformers, example) => {
|
|
83
|
+
Object.keys(responseSpec.content).forEach((mediaName) => {
|
|
84
|
+
if (mediaName === jsonMediaType) {
|
|
85
|
+
return;
|
|
86
|
+
}
|
|
87
|
+
Object.entries(allowedMediaTypes || {}).forEach(([mediaType, allowed]) => {
|
|
88
|
+
var _a, _b, _c, _d;
|
|
89
|
+
if (!allowed) {
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
prepareStatusContent(methodSpec, status, mediaType);
|
|
93
|
+
if (((_d = (_c = (_b = (_a = methodSpec == null ? void 0 : methodSpec.responses) == null ? void 0 : _a[status]) == null ? void 0 : _b.content) == null ? void 0 : _c[mediaType]) == null ? void 0 : _d.example) === void 0) {
|
|
94
|
+
methodSpec.responses[status].content[mediaType].example = {};
|
|
95
|
+
}
|
|
96
|
+
let transformed = false;
|
|
97
|
+
transformers.forEach(({ regex, transformer }) => {
|
|
98
|
+
if (!transformed && regex.test(mediaType)) {
|
|
99
|
+
methodSpec.responses[status].content[mediaType].example = transformer(example);
|
|
100
|
+
transformed = true;
|
|
101
|
+
}
|
|
102
|
+
});
|
|
103
|
+
if (!transformed) {
|
|
104
|
+
methodSpec.responses[status].content[mediaType].example = example;
|
|
105
|
+
}
|
|
106
|
+
});
|
|
107
|
+
});
|
|
108
|
+
};
|
|
109
|
+
var extendComponentExamples = (spec, exampleName, examples) => {
|
|
110
|
+
if (typeof spec.components !== "object") {
|
|
111
|
+
spec.components = {};
|
|
112
|
+
}
|
|
113
|
+
if (typeof spec.components.examples !== "object") {
|
|
114
|
+
spec.components.examples = {};
|
|
115
|
+
}
|
|
116
|
+
if (spec.components.examples[exampleName] === void 0 && examples[exampleName] !== void 0) {
|
|
117
|
+
spec.components.examples[exampleName] = examples[exampleName];
|
|
118
|
+
}
|
|
119
|
+
};
|
|
120
|
+
var prepareResponseExamples = (spec, methodSpec, status, mediaType, transformers, examples) => {
|
|
121
|
+
var _a, _b, _c, _d;
|
|
122
|
+
prepareStatusContent(methodSpec, status, mediaType);
|
|
123
|
+
if (((_d = (_c = (_b = (_a = methodSpec == null ? void 0 : methodSpec.responses) == null ? void 0 : _a[status]) == null ? void 0 : _b.content) == null ? void 0 : _c[mediaType]) == null ? void 0 : _d.examples) === void 0) {
|
|
124
|
+
methodSpec.responses[status].content[mediaType].examples = {};
|
|
125
|
+
}
|
|
126
|
+
const transformedExamples = {};
|
|
127
|
+
Object.entries(examples).forEach(([exampleName, example]) => {
|
|
128
|
+
var _a2, _b2;
|
|
129
|
+
let transformed = false;
|
|
130
|
+
transformers.forEach(({ regex, transformer }) => {
|
|
131
|
+
var _a3, _b3, _c2, _d2, _e, _f;
|
|
132
|
+
if (!transformed && regex.test(mediaType)) {
|
|
133
|
+
let data = "";
|
|
134
|
+
if (((_b3 = (_a3 = spec.components) == null ? void 0 : _a3.examples) == null ? void 0 : _b3[exampleName]) !== void 0) {
|
|
135
|
+
data = ((_d2 = (_c2 = spec.components) == null ? void 0 : _c2.examples) == null ? void 0 : _d2[exampleName]).value;
|
|
136
|
+
} else if (example) {
|
|
137
|
+
data = ((_f = (_e = spec.components) == null ? void 0 : _e.examples) == null ? void 0 : _f[example.$ref.replace("#/components/examples/", "")]).value;
|
|
138
|
+
} else if (typeof (example == null ? void 0 : example.value) === "string") {
|
|
139
|
+
data = example.value;
|
|
140
|
+
}
|
|
141
|
+
transformedExamples[exampleName] = {
|
|
142
|
+
value: transformer(data)
|
|
143
|
+
};
|
|
144
|
+
transformed = true;
|
|
145
|
+
}
|
|
146
|
+
});
|
|
147
|
+
if (!transformed) {
|
|
148
|
+
transformedExamples[exampleName] = ((_b2 = (_a2 = spec.components) == null ? void 0 : _a2.examples) == null ? void 0 : _b2[exampleName]) === void 0 ? example : {
|
|
149
|
+
$ref: `#/components/examples/${exampleName}`
|
|
150
|
+
};
|
|
151
|
+
}
|
|
152
|
+
});
|
|
153
|
+
methodSpec.responses[status].content[mediaType].examples = transformedExamples;
|
|
154
|
+
};
|
|
155
|
+
var extendSwaggerWithMediaTypeExamples = (spec, methodSpec, status, responseSpec, allowedMediaTypes, pathKey, transformers, examples) => {
|
|
156
|
+
const examplesName = `${_case.header.call(void 0, pathKey.trim().replace("/", ""))}`;
|
|
157
|
+
Object.keys(responseSpec.content).forEach((mediaName) => {
|
|
158
|
+
if (mediaName === jsonMediaType) {
|
|
159
|
+
return;
|
|
160
|
+
}
|
|
161
|
+
Object.entries(allowedMediaTypes || {}).forEach(([mediaType, allowed]) => {
|
|
162
|
+
if (!allowed) {
|
|
163
|
+
return;
|
|
164
|
+
}
|
|
165
|
+
extendComponentExamples(spec, examplesName, examples);
|
|
166
|
+
prepareResponseExamples(spec, methodSpec, status, mediaType, transformers, examples);
|
|
167
|
+
});
|
|
168
|
+
});
|
|
169
|
+
};
|
|
170
|
+
function extendSwaggerSpec(spec, allowedMediaTypes, transformers = [
|
|
171
|
+
{
|
|
172
|
+
regex: /xml/,
|
|
173
|
+
transformer: (value) => _jstoxml.toXML.call(void 0, value, {
|
|
174
|
+
header: true,
|
|
175
|
+
indent: " "
|
|
176
|
+
})
|
|
177
|
+
},
|
|
178
|
+
{
|
|
179
|
+
regex: /yaml|yml/,
|
|
180
|
+
transformer: (value) => _yaml.stringify.call(void 0, value, { indent: 2 })
|
|
181
|
+
}
|
|
182
|
+
]) {
|
|
183
|
+
if (typeof spec === "object" && typeof spec.paths === "object") {
|
|
184
|
+
Object.entries(spec.paths).forEach(([pathKey, pathSpec]) => {
|
|
185
|
+
Object.values(pathSpec).forEach((methodSpec) => {
|
|
186
|
+
if (typeof methodSpec.responses === "object") {
|
|
187
|
+
Object.entries(methodSpec.responses).forEach(([status, responseSpec]) => {
|
|
188
|
+
if (typeof responseSpec.content === "object") {
|
|
189
|
+
const { examples, example } = extendSwaggerWithMediaTypeSchema(
|
|
190
|
+
methodSpec,
|
|
191
|
+
responseSpec,
|
|
192
|
+
allowedMediaTypes,
|
|
193
|
+
pathKey,
|
|
194
|
+
spec,
|
|
195
|
+
status
|
|
196
|
+
);
|
|
197
|
+
if (example !== void 0) {
|
|
198
|
+
extendSwaggerWithMediaTypeExample(
|
|
199
|
+
methodSpec,
|
|
200
|
+
responseSpec,
|
|
201
|
+
status,
|
|
202
|
+
allowedMediaTypes,
|
|
203
|
+
transformers,
|
|
204
|
+
example
|
|
205
|
+
);
|
|
206
|
+
} else if (examples !== void 0) {
|
|
207
|
+
extendSwaggerWithMediaTypeExamples(
|
|
208
|
+
spec,
|
|
209
|
+
methodSpec,
|
|
210
|
+
status,
|
|
211
|
+
responseSpec,
|
|
212
|
+
allowedMediaTypes,
|
|
213
|
+
pathKey,
|
|
214
|
+
transformers,
|
|
215
|
+
examples
|
|
216
|
+
);
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
});
|
|
220
|
+
}
|
|
221
|
+
});
|
|
222
|
+
});
|
|
223
|
+
}
|
|
224
|
+
return spec;
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
// src/swagger/swagger-handler.ts
|
|
228
|
+
var swaggerCrudDebug = _debug2.default.call(void 0, "visulima:api-platform:swagger:crud:get-static-properties-swagger");
|
|
229
|
+
var swaggerHandler = (options = {}) => {
|
|
230
|
+
const {
|
|
231
|
+
allowedMediaTypes = {
|
|
232
|
+
"application/json": true
|
|
233
|
+
},
|
|
234
|
+
swaggerFilePath,
|
|
235
|
+
crud,
|
|
236
|
+
specs
|
|
237
|
+
} = options;
|
|
238
|
+
return async (request, response) => {
|
|
239
|
+
const swaggerPath = _path2.default.join(process.cwd(), swaggerFilePath || "swagger/swagger.json");
|
|
240
|
+
if (!_fs.existsSync.call(void 0, swaggerPath)) {
|
|
241
|
+
throw new Error(`Swagger file not found at ${swaggerPath}. Did you change the output path in "withOpenApi" inside the next.config.js file?`);
|
|
242
|
+
}
|
|
243
|
+
const fileContents = _fs.readFileSync.call(void 0, swaggerPath, "utf8");
|
|
244
|
+
let spec = extendSwaggerSpec(JSON.parse(fileContents), allowedMediaTypes);
|
|
245
|
+
let crudSwagger = {};
|
|
246
|
+
if (crud !== void 0) {
|
|
247
|
+
try {
|
|
248
|
+
const modelsOpenApi = await _crud.modelsToOpenApi.call(void 0, crud);
|
|
249
|
+
crudSwagger = {
|
|
250
|
+
components: { schemas: modelsOpenApi.schemas, examples: modelsOpenApi.examples },
|
|
251
|
+
tags: modelsOpenApi.tags,
|
|
252
|
+
paths: modelsOpenApi.paths
|
|
253
|
+
};
|
|
254
|
+
crudSwagger = extendSwaggerSpec(crudSwagger, allowedMediaTypes);
|
|
255
|
+
swaggerCrudDebug(JSON.stringify(crudSwagger, null, 2));
|
|
256
|
+
spec = _lodashmerge2.default.call(void 0, spec, crudSwagger);
|
|
257
|
+
} catch (error) {
|
|
258
|
+
console.log(error);
|
|
259
|
+
throw new Error("Please install @visulima/crud to use the crud swagger generator.");
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
if (Array.isArray(specs)) {
|
|
263
|
+
specs.forEach((value) => {
|
|
264
|
+
spec = _lodashmerge2.default.call(void 0, spec, extendSwaggerSpec(value, allowedMediaTypes));
|
|
265
|
+
});
|
|
266
|
+
}
|
|
267
|
+
if (typeof request.headers.accept === "string" && /yaml|yml/.test(request.headers.accept)) {
|
|
268
|
+
response.statusCode = 200;
|
|
269
|
+
response.setHeader("Content-Type", request.headers.accept);
|
|
270
|
+
response.end(yaml_default(spec));
|
|
271
|
+
} else {
|
|
272
|
+
response.statusCode = 200;
|
|
273
|
+
response.setHeader("Content-Type", "application/json");
|
|
274
|
+
response.end(JSON.stringify(spec, null, 2));
|
|
275
|
+
}
|
|
276
|
+
};
|
|
277
|
+
};
|
|
278
|
+
var swagger_handler_default = swaggerHandler;
|
|
279
|
+
|
|
280
|
+
|
|
281
|
+
|
|
282
|
+
|
|
283
|
+
exports.yaml_default = yaml_default; exports.swagger_handler_default = swagger_handler_default;
|
|
284
|
+
//# sourceMappingURL=chunk-NY75KGAH.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/swagger/swagger-handler.ts","../src/connect/serializers/yaml.ts","../src/swagger/extend-swagger-spec.ts"],"names":["stringify","_a","_b","_c","_d"],"mappings":"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","sourcesContent":["// eslint-disable-next-line unicorn/prevent-abbreviations,import/no-extraneous-dependencies\nimport type { ModelsToOpenApiParameters, SwaggerModelsConfig } from \"@visulima/crud\";\n// eslint-disable-next-line unicorn/prevent-abbreviations,import/no-extraneous-dependencies\nimport { modelsToOpenApi } from \"@visulima/crud\";\nimport debug from \"debug\";\nimport merge from \"lodash.merge\";\nimport { existsSync, readFileSync } from \"node:fs\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport path from \"node:path\";\nimport type { OpenAPIV3 } from \"openapi-types\";\n\nimport yamlTransformer from \"../connect/serializers/yaml\";\nimport extendSwaggerSpec from \"./extend-swagger-spec\";\n\n// eslint-disable-next-line testing-library/no-debugging-utils\nconst swaggerCrudDebug = debug(\"visulima:api-platform:swagger:crud:get-static-properties-swagger\");\n\nconst swaggerHandler = (\n options: Partial<{\n allowedMediaTypes: { [key: string]: boolean };\n swaggerFilePath: string;\n crud: Exclude<ModelsToOpenApiParameters, \"swagger\"> & {\n swagger?: {\n models?: SwaggerModelsConfig<string>;\n };\n };\n specs?: Partial<OpenAPIV3.Document>[];\n }> = {},\n) => {\n const {\n allowedMediaTypes = {\n \"application/json\": true,\n },\n swaggerFilePath,\n crud,\n specs,\n } = options;\n\n return async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response) => {\n const swaggerPath = path.join(process.cwd(), swaggerFilePath || \"swagger/swagger.json\");\n\n if (!existsSync(swaggerPath)) {\n throw new Error(`Swagger file not found at ${swaggerPath}. Did you change the output path in \"withOpenApi\" inside the next.config.js file?`);\n }\n\n const fileContents = readFileSync(swaggerPath, \"utf8\");\n\n let spec = extendSwaggerSpec(JSON.parse(fileContents) as OpenAPIV3.Document, allowedMediaTypes) as OpenAPIV3.Document;\n let crudSwagger: Partial<OpenAPIV3.Document> = {};\n\n if (crud !== undefined) {\n try {\n const modelsOpenApi = await modelsToOpenApi(crud);\n\n crudSwagger = {\n components: { schemas: modelsOpenApi.schemas, examples: modelsOpenApi.examples },\n tags: modelsOpenApi.tags as OpenAPIV3.TagObject[],\n paths: modelsOpenApi.paths,\n };\n\n crudSwagger = extendSwaggerSpec(crudSwagger, allowedMediaTypes);\n\n swaggerCrudDebug(JSON.stringify(crudSwagger, null, 2));\n\n spec = merge(spec, crudSwagger);\n } catch (error) {\n // eslint-disable-next-line no-console\n console.log(error);\n\n throw new Error(\"Please install @visulima/crud to use the crud swagger generator.\");\n }\n }\n\n if (Array.isArray(specs)) {\n specs.forEach((value) => {\n spec = merge(spec, extendSwaggerSpec(value, allowedMediaTypes));\n });\n }\n\n if (typeof request.headers.accept === \"string\" && /yaml|yml/.test(request.headers.accept)) {\n response.statusCode = 200;\n response.setHeader(\"Content-Type\", request.headers.accept);\n response.end(yamlTransformer(spec));\n } else {\n response.statusCode = 200;\n response.setHeader(\"Content-Type\", \"application/json\");\n response.end(JSON.stringify(spec, null, 2));\n }\n };\n};\n\nexport default swaggerHandler;\n","import { stringify } from \"yaml\";\n\nimport type { Serializer } from \"./types\";\n\nconst yamlTransformer: Serializer = (data) => stringify(data, { indent: 2 });\n\nexport default yamlTransformer;\n","/* eslint-disable no-param-reassign */\nimport { header as headerCase } from \"case\";\nimport { toXML } from \"jstoxml\";\nimport type { OpenAPIV3 } from \"openapi-types\";\nimport { stringify } from \"yaml\";\n\ntype Transformers = { regex: RegExp; transformer: (data: any) => string }[];\n\nconst jsonMediaType = \"application/json\";\n\nconst prepareStatusContent = (methodSpec: OpenAPIV3.OperationObject<{}>, status: string, mediaType: string) => {\n if (((methodSpec?.responses as unknown as OpenAPIV3.ResponsesObject)?.[status] as OpenAPIV3.ResponseObject)?.content === undefined) {\n ((methodSpec.responses as unknown as OpenAPIV3.ResponsesObject)[status] as OpenAPIV3.ResponseObject).content = {};\n }\n\n if (((methodSpec?.responses as unknown as OpenAPIV3.ResponsesObject)?.[status] as OpenAPIV3.ResponseObject)?.content?.[mediaType] === undefined) {\n (\n ((methodSpec.responses as unknown as OpenAPIV3.ResponsesObject)[status] as OpenAPIV3.ResponseObject).content as {\n [key: string]: OpenAPIV3.MediaTypeObject;\n }\n )[mediaType] = {} as OpenAPIV3.MediaTypeObject;\n }\n};\n\nconst extendComponentSchemas = (spec: Partial<OpenAPIV3.Document>, schemaName: string, schema: OpenAPIV3.SchemaObject) => {\n if (typeof spec.components !== \"object\") {\n spec.components = {};\n }\n\n if (typeof spec.components.schemas !== \"object\") {\n spec.components.schemas = {};\n }\n\n if (spec.components.schemas[schemaName] === undefined) {\n spec.components.schemas[schemaName] = schema;\n }\n};\n\nconst extendResponseSchema = (methodSpec: OpenAPIV3.OperationObject, status: string, mediaType: string, schemaName: string, schemaIsArray: boolean) => {\n prepareStatusContent(methodSpec, status, mediaType);\n\n if (((methodSpec?.responses as unknown as OpenAPIV3.ResponsesObject)?.[status] as OpenAPIV3.ResponseObject)?.content?.[mediaType]?.schema === undefined) {\n (\n (\n ((methodSpec.responses as unknown as OpenAPIV3.ResponsesObject)[status] as OpenAPIV3.ResponseObject).content as {\n [key: string]: OpenAPIV3.MediaTypeObject;\n }\n )[mediaType] as OpenAPIV3.MediaTypeObject\n ).schema = {} as OpenAPIV3.SchemaObject;\n }\n\n (\n (\n ((methodSpec.responses as unknown as OpenAPIV3.ResponsesObject)[status] as OpenAPIV3.ResponseObject).content as {\n [key: string]: OpenAPIV3.MediaTypeObject;\n }\n )[mediaType] as OpenAPIV3.MediaTypeObject\n ).schema = schemaIsArray\n ? {\n type: \"array\",\n items: {\n $ref: `#/components/schemas/${schemaName}`,\n },\n }\n : {\n $ref: `#/components/schemas/${schemaName}`,\n };\n};\n\nconst extendSwaggerWithMediaTypeSchema = (\n methodSpec: OpenAPIV3.OperationObject,\n responseSpec: OpenAPIV3.ResponseObject,\n allowedMediaTypes: { [p: string]: boolean } | undefined,\n pathKey: string,\n spec: Partial<OpenAPIV3.Document>,\n status: string,\n): {\n example?: any;\n examples?: {\n [media: string]: OpenAPIV3.ReferenceObject | OpenAPIV3.ExampleObject;\n };\n} => {\n let example: any | undefined;\n let examples:\n | {\n [media: string]: OpenAPIV3.ReferenceObject | OpenAPIV3.ExampleObject;\n }\n | undefined;\n\n // eslint-disable-next-line radar/cognitive-complexity\n Object.entries(responseSpec.content as object).forEach(([mediaName, contentSpec]) => {\n if (typeof contentSpec.schema === \"object\") {\n const { schema } = contentSpec;\n\n if (mediaName === jsonMediaType && contentSpec.examples !== undefined) {\n examples = contentSpec.examples;\n } else if (mediaName === jsonMediaType && contentSpec.example !== undefined) {\n example = contentSpec.example;\n }\n\n const schemaIsArray = (schema as OpenAPIV3.SchemaObject).type === \"array\";\n\n Object.entries(allowedMediaTypes || {}).forEach(([mediaType, allowed]) => {\n if (!allowed) {\n return;\n }\n\n let schemaName: string;\n\n if ((schema as OpenAPIV3.ReferenceObject).$ref === undefined) {\n // eslint-disable-next-line max-len\n schemaName = `${headerCase(pathKey.trim().replace(\"/\", \"\"))}${mediaType === \"application/ld+json\" ? \".jsonld\" : \"\"}`;\n\n extendComponentSchemas(spec as OpenAPIV3.Document, schemaName, schema as OpenAPIV3.SchemaObject);\n } else {\n // eslint-disable-next-line max-len\n schemaName = (schema as OpenAPIV3.ReferenceObject).$ref.replace(\"#/components/schemas/\", \"\");\n }\n\n extendResponseSchema(methodSpec, status, mediaType, schemaName, schemaIsArray);\n });\n }\n });\n\n return { examples, example };\n};\n\nconst extendSwaggerWithMediaTypeExample = (\n methodSpec: OpenAPIV3.OperationObject,\n responseSpec: OpenAPIV3.ResponseObject,\n status: string,\n allowedMediaTypes: { [p: string]: boolean } | undefined,\n transformers: Transformers,\n example: any,\n) => {\n Object.keys(responseSpec.content as object).forEach((mediaName) => {\n if (mediaName === jsonMediaType) {\n return;\n }\n\n Object.entries(allowedMediaTypes || {}).forEach(([mediaType, allowed]) => {\n if (!allowed) {\n return;\n }\n\n prepareStatusContent(methodSpec, status, mediaType);\n\n if (\n ((methodSpec?.responses as unknown as OpenAPIV3.ResponsesObject)?.[status] as OpenAPIV3.ResponseObject)?.content?.[mediaType]?.example\n === undefined\n ) {\n (\n (\n ((methodSpec.responses as unknown as OpenAPIV3.ResponsesObject)[status] as OpenAPIV3.ResponseObject).content as {\n [key: string]: OpenAPIV3.MediaTypeObject;\n }\n )[mediaType] as OpenAPIV3.MediaTypeObject\n ).example = {};\n }\n\n let transformed = false;\n\n transformers.forEach(({ regex, transformer }) => {\n if (!transformed && regex.test(mediaType)) {\n (\n (\n ((methodSpec.responses as unknown as OpenAPIV3.ResponsesObject)[status] as OpenAPIV3.ResponseObject).content as {\n [key: string]: OpenAPIV3.MediaTypeObject;\n }\n )[mediaType] as OpenAPIV3.MediaTypeObject\n ).example = transformer(example);\n\n transformed = true;\n }\n });\n\n if (!transformed) {\n (\n (\n ((methodSpec.responses as unknown as OpenAPIV3.ResponsesObject)[status] as OpenAPIV3.ResponseObject).content as {\n [key: string]: OpenAPIV3.MediaTypeObject;\n }\n )[mediaType] as OpenAPIV3.MediaTypeObject\n ).example = example;\n }\n });\n });\n};\n\nconst extendComponentExamples = (\n spec: Partial<OpenAPIV3.Document>,\n exampleName: string,\n examples: {\n [media: string]: OpenAPIV3.ReferenceObject | OpenAPIV3.ExampleObject;\n },\n) => {\n if (typeof spec.components !== \"object\") {\n spec.components = {};\n }\n\n if (typeof spec.components.examples !== \"object\") {\n spec.components.examples = {};\n }\n\n if (spec.components.examples[exampleName] === undefined && examples[exampleName] !== undefined) {\n spec.components.examples[exampleName] = examples[exampleName] as OpenAPIV3.ExampleObject;\n }\n};\n\nconst prepareResponseExamples = (\n spec: Partial<OpenAPIV3.Document>,\n methodSpec: OpenAPIV3.OperationObject,\n status: string,\n mediaType: string,\n transformers: Transformers,\n examples: {\n [media: string]: OpenAPIV3.ReferenceObject | OpenAPIV3.ExampleObject;\n },\n // eslint-disable-next-line radar/cognitive-complexity\n) => {\n prepareStatusContent(methodSpec, status, mediaType);\n\n if (((methodSpec?.responses as unknown as OpenAPIV3.ResponsesObject)?.[status] as OpenAPIV3.ResponseObject)?.content?.[mediaType]?.examples === undefined) {\n (\n (\n ((methodSpec.responses as unknown as OpenAPIV3.ResponsesObject)[status] as OpenAPIV3.ResponseObject).content as {\n [key: string]: OpenAPIV3.MediaTypeObject;\n }\n )[mediaType] as OpenAPIV3.MediaTypeObject\n ).examples = {};\n }\n\n const transformedExamples: {\n [media: string]: OpenAPIV3.ReferenceObject | OpenAPIV3.ExampleObject;\n } = {};\n\n Object.entries(examples).forEach(([exampleName, example]) => {\n let transformed = false;\n\n transformers.forEach(({ regex, transformer }) => {\n if (!transformed && regex.test(mediaType)) {\n let data: any = \"\";\n\n if ((spec.components?.examples?.[exampleName] as OpenAPIV3.ExampleObject) !== undefined) {\n data = (spec.components?.examples?.[exampleName] as OpenAPIV3.ExampleObject).value;\n } else if (example as OpenAPIV3.ReferenceObject) {\n data = (\n spec.components?.examples?.[\n (example as OpenAPIV3.ReferenceObject).$ref.replace(\"#/components/examples/\", \"\")\n ] as OpenAPIV3.ExampleObject\n ).value;\n } else if (typeof (example as OpenAPIV3.ExampleObject)?.value === \"string\") {\n data = (example as OpenAPIV3.ExampleObject).value;\n }\n\n transformedExamples[exampleName] = {\n value: transformer(data),\n };\n\n transformed = true;\n }\n });\n\n if (!transformed) {\n transformedExamples[exampleName] = spec.components?.examples?.[exampleName] === undefined\n ? example\n : {\n $ref: `#/components/examples/${exampleName}`,\n };\n }\n });\n\n (\n (\n ((methodSpec.responses as unknown as OpenAPIV3.ResponsesObject)[status] as OpenAPIV3.ResponseObject).content as {\n [key: string]: OpenAPIV3.MediaTypeObject;\n }\n )[mediaType] as OpenAPIV3.MediaTypeObject\n ).examples = transformedExamples;\n};\n\nconst extendSwaggerWithMediaTypeExamples = (\n spec: Partial<OpenAPIV3.Document>,\n methodSpec: OpenAPIV3.OperationObject,\n status: string,\n responseSpec: OpenAPIV3.ResponseObject,\n allowedMediaTypes: { [p: string]: boolean } | undefined,\n pathKey: string,\n transformers: Transformers,\n examples: {\n [media: string]: OpenAPIV3.ReferenceObject | OpenAPIV3.ExampleObject;\n },\n) => {\n const examplesName = `${headerCase(pathKey.trim().replace(\"/\", \"\"))}`;\n\n Object.keys(responseSpec.content as object).forEach((mediaName) => {\n if (mediaName === jsonMediaType) {\n return;\n }\n\n Object.entries(allowedMediaTypes || {}).forEach(([mediaType, allowed]) => {\n if (!allowed) {\n return;\n }\n\n extendComponentExamples(spec, examplesName, examples);\n\n prepareResponseExamples(spec, methodSpec, status, mediaType, transformers, examples);\n });\n });\n};\n\n// eslint-disable-next-line radar/cognitive-complexity\nexport default function extendSwaggerSpec(\n spec: Partial<OpenAPIV3.Document>,\n allowedMediaTypes?: { [key: string]: boolean },\n transformers: Transformers = [\n {\n regex: /xml/,\n transformer: (value) => toXML(value, {\n header: true,\n indent: \" \",\n }),\n },\n {\n regex: /yaml|yml/,\n transformer: (value) => stringify(value, { indent: 2 }),\n },\n ],\n): Partial<OpenAPIV3.Document> {\n if (typeof spec === \"object\" && typeof spec.paths === \"object\") {\n Object.entries(spec.paths).forEach(([pathKey, pathSpec]) => {\n Object.values(pathSpec as OpenAPIV3.PathsObject & OpenAPIV3.OperationObject).forEach((methodSpec) => {\n if (typeof (methodSpec as OpenAPIV3.OperationObject).responses === \"object\") {\n Object.entries((methodSpec as OpenAPIV3.OperationObject).responses).forEach(([status, responseSpec]) => {\n if (typeof (responseSpec as OpenAPIV3.ResponseObject).content === \"object\") {\n const { examples, example } = extendSwaggerWithMediaTypeSchema(\n methodSpec as OpenAPIV3.OperationObject,\n responseSpec as OpenAPIV3.ResponseObject,\n allowedMediaTypes,\n pathKey,\n spec,\n status,\n );\n\n if (example !== undefined) {\n extendSwaggerWithMediaTypeExample(\n methodSpec as OpenAPIV3.OperationObject,\n responseSpec as OpenAPIV3.ResponseObject,\n status,\n allowedMediaTypes,\n transformers,\n example,\n );\n } else if (examples !== undefined) {\n extendSwaggerWithMediaTypeExamples(\n spec,\n methodSpec as OpenAPIV3.OperationObject,\n status,\n responseSpec as OpenAPIV3.ResponseObject,\n allowedMediaTypes,\n pathKey,\n transformers,\n examples,\n );\n }\n }\n });\n }\n });\n });\n }\n\n return spec;\n}\n\n/* eslint-enable no-param-reassign */\n"]}
|
|
@@ -17,7 +17,6 @@ var _dynamic = require('next/dynamic'); var _dynamic2 = _interopRequireDefault(_
|
|
|
17
17
|
|
|
18
18
|
|
|
19
19
|
var SwaggerUI = _dynamic2.default.call(void 0, Promise.resolve().then(() => require("swagger-ui-react")), { ssr: false });
|
|
20
|
-
_dynamic2.default.call(void 0, Promise.resolve().then(() => require("swagger-ui-react/swagger-ui.css")), { ssr: false });
|
|
21
20
|
var SwaggerApiDocument = (name, swagger = {}) => ({ swaggerData }) => /* @__PURE__ */ _react2.default.createElement(_react2.default.Fragment, null, /* @__PURE__ */ _react2.default.createElement(_head2.default, null, /* @__PURE__ */ _react2.default.createElement("title", null, name), /* @__PURE__ */ _react2.default.createElement("style", null, `
|
|
22
21
|
body {
|
|
23
22
|
background: #fafafa !important;
|
|
@@ -49,4 +48,4 @@ var get_static_properties_swagger_default = getStaticProps;
|
|
|
49
48
|
|
|
50
49
|
|
|
51
50
|
exports.redoc_default = redoc_default; exports.swagger_default = swagger_default; exports.get_static_properties_swagger_default = get_static_properties_swagger_default;
|
|
52
|
-
//# sourceMappingURL=chunk-
|
|
51
|
+
//# sourceMappingURL=chunk-S46GRBOS.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/next/routes/pages/swagger/redoc.tsx","../src/next/routes/pages/swagger/swagger.tsx","../src/next/routes/pages/swagger/get-static-properties-swagger.ts"],"names":["Head","React"],"mappings":";AAGA,OAAO,UAAU;AACjB,OAAO,WAAW;AAIlB,SAAS,uBAAuB;AAKhC,IAAM,mBAI0D,CAAC,MAAM,UAAU,CAAC,MAAM,CAAC,EAAE,YAAY,MAC3F,0DACI,oCAAC,YACG,oCAAC,eAAO,IAAK,GACb,oCAAC,eACI;AAAA;AAAA;AAAA;AAAA,CAKL,CACJ,GAEA,oCAAC;AAAA,EAAiB,GAAG;AAAA,EAAS,MAAM;AAAA,CAAa,CACrD;AAGZ,IAAO,gBAAQ;;;AChCf,OAAO,aAAa;AAEpB,OAAOA,WAAU;AAEjB,OAAOC,YAAW;AAKlB,IAAM,YAAY,QAAQ,OAAO,qBAAqB,EAAE,KAAK,MAAM,CAAC;AAGpE,IAAM,qBAI0D,CAAC,MAAM,UAAU,CAAC,MAAM,CAAC,EAAE,YAAY,MAC3F,gBAAAA,OAAA,cAAAA,OAAA,gBACI,gBAAAA,OAAA,cAACD,OAAA,MACG,gBAAAC,OAAA,cAAC,eAAO,IAAK,GACb,gBAAAA,OAAA,cAAC,eACI;AAAA;AAAA;AAAA;AAAA,CAKL,CACJ,GAEA,gBAAAA,OAAA,cAAC;AAAA,EAAW,GAAG;AAAA,EAAS,MAAM;AAAA,CAAa,CAC/C;AAGZ,IAAO,kBAAQ;;;AClCf,OAAO,WAAW;AAKlB,IAAM,eAAe,MAAM,6DAA6D;AAGxF,IAAM,iBAEgB,CAAC,eAAe,YAKhC;AAEF,QAAM,WAAW,MAAM,MAAM,UAAU;AACvC,QAAM,cAAc,MAAM,SAAS,KAAK;AAExC,eAAa,WAAW;AAExB,SAAO;AAAA,IACH,OAAO;AAAA,MACH;AAAA,MACA,aAAa,KAAK,MAAM,KAAK,UAAU,WAAW,CAAC;AAAA,IACvD;AAAA,EACJ;AACJ;AAEA,IAAO,wCAAQ","sourcesContent":["// eslint-disable-next-line import/no-extraneous-dependencies\nimport type { InferGetStaticPropsType, NextPage } from \"next\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport Head from \"next/head\";\nimport React from \"react\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport type { RedocStandaloneProps } from \"redoc\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport { RedocStandalone } from \"redoc\";\n\nimport getStaticProps from \"./get-static-properties-swagger\";\n\n// eslint-disable-next-line max-len\nconst RedocApiDocument: (\n name: string,\n swagger?: Exclude<RedocStandaloneProps, \"spec\">,\n // eslint-disable-next-line max-len,unicorn/no-useless-undefined\n) => NextPage<InferGetStaticPropsType<typeof getStaticProps>> = (name, swagger = {}) => ({ swaggerData }: InferGetStaticPropsType<typeof getStaticProps>) => (\n <>\n <Head>\n <title>{name}</title>\n <style>\n {`\nbody {\n background: #fafafa !important;\n}\n`}\n </style>\n </Head>\n {/* eslint-disable-next-line react/jsx-props-no-spreading */}\n <RedocStandalone {...swagger} spec={swaggerData} />\n </>\n);\n\nexport default RedocApiDocument;\n","import type { InferGetStaticPropsType, NextPage } from \"next\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport dynamic from \"next/dynamic\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport Head from \"next/head\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport React from \"react\";\nimport type { SwaggerUIProps } from \"swagger-ui-react\";\n\nimport getStaticProps from \"./get-static-properties-swagger\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nconst SwaggerUI = dynamic(import(\"swagger-ui-react\"), { ssr: false });\n\n// eslint-disable-next-line max-len\nconst SwaggerApiDocument: (\n name: string,\n swagger?: Exclude<SwaggerUIProps, \"spec\">,\n // eslint-disable-next-line max-len,unicorn/no-useless-undefined\n) => NextPage<InferGetStaticPropsType<typeof getStaticProps>> = (name, swagger = {}) => ({ swaggerData }: InferGetStaticPropsType<typeof getStaticProps>) => (\n <>\n <Head>\n <title>{name}</title>\n <style>\n {`\nbody {\n background: #fafafa !important;\n}\n`}\n </style>\n </Head>\n {/* eslint-disable-next-line react/jsx-props-no-spreading */}\n <SwaggerUI {...swagger} spec={swaggerData} />\n </>\n);\n\nexport default SwaggerApiDocument;\n","// eslint-disable-next-line unicorn/prevent-abbreviations\nimport debug from \"debug\";\nimport type { GetStaticProps } from \"next\";\nimport type { OpenAPIV3 } from \"openapi-types\";\n\n// eslint-disable-next-line testing-library/no-debugging-utils\nconst swaggerDebug = debug(\"visulima:api-platform:swagger:get-static-properties-swagger\");\n\n// eslint-disable-next-line unicorn/consistent-function-scoping\nconst getStaticProps: (\n swaggerUrl: string,\n) => GetStaticProps = (swaggerUrl) => async (): Promise<{\n props: {\n swaggerUrl: string;\n swaggerData: OpenAPIV3.Document;\n };\n}> => {\n // eslint-disable-next-line compat/compat\n const response = await fetch(swaggerUrl);\n const swaggerData = await response.json();\n\n swaggerDebug(swaggerData);\n\n return {\n props: {\n swaggerUrl,\n swaggerData: JSON.parse(JSON.stringify(swaggerData)),\n },\n };\n};\n\nexport default getStaticProps;\n"]}
|
package/dist/index-server.d.ts
CHANGED
|
@@ -10,6 +10,7 @@ import { NextApiResponse } from 'next';
|
|
|
10
10
|
import { RateLimiterAbstract, RateLimiterRes } from 'rate-limiter-flexible';
|
|
11
11
|
import { CorsOptions, CorsOptionsDelegate } from 'cors';
|
|
12
12
|
import { ModelsToOpenApiParameters, SwaggerModelsConfig } from '@visulima/crud';
|
|
13
|
+
import { OpenAPIV3 } from 'openapi-types';
|
|
13
14
|
|
|
14
15
|
type ErrorHandler = <Request extends IncomingMessage, Response extends ServerResponse>(
|
|
15
16
|
error: any,
|
|
@@ -82,6 +83,7 @@ declare const swaggerHandler: (options?: Partial<{
|
|
|
82
83
|
models?: SwaggerModelsConfig<string>;
|
|
83
84
|
};
|
|
84
85
|
};
|
|
86
|
+
specs?: Partial<OpenAPIV3.Document<{}>>[] | undefined;
|
|
85
87
|
}>) => <Request_1 extends IncomingMessage, Response_1 extends ServerResponse<IncomingMessage>>(request: Request_1, response: Response_1) => Promise<void>;
|
|
86
88
|
|
|
87
89
|
export { corsMiddleware, createNodeRouter, httpHeaderNormalizerMiddleware, onError, onNoMatch, rateLimiterMiddleware, serializersMiddleware, swaggerHandler };
|
package/dist/index-server.js
CHANGED
|
@@ -6,7 +6,7 @@ var _chunkT25VSNTFjs = require('./chunk-T25VSNTF.js');
|
|
|
6
6
|
|
|
7
7
|
|
|
8
8
|
|
|
9
|
-
var
|
|
9
|
+
var _chunkNY75KGAHjs = require('./chunk-NY75KGAH.js');
|
|
10
10
|
require('./chunk-JC4IRQUL.js');
|
|
11
11
|
|
|
12
12
|
// src/index-server.ts
|
|
@@ -79,10 +79,10 @@ var sendJson = (response, jsonBody) => {
|
|
|
79
79
|
response.end(JSON.stringify(jsonBody));
|
|
80
80
|
};
|
|
81
81
|
var addStatusCodeToResponse = (response, error) => {
|
|
82
|
-
if (
|
|
82
|
+
if (error.statusCode !== void 0) {
|
|
83
83
|
response.statusCode = error.statusCode;
|
|
84
84
|
}
|
|
85
|
-
if (
|
|
85
|
+
if (error.status !== void 0) {
|
|
86
86
|
response.statusCode = error.status;
|
|
87
87
|
}
|
|
88
88
|
if (response.statusCode < 400) {
|
|
@@ -238,7 +238,7 @@ var httpHeaderNormalizerMiddleware = (options_) => {
|
|
|
238
238
|
Object.keys(request.headers).forEach((key) => {
|
|
239
239
|
rawHeaders[key] = request.headers[key];
|
|
240
240
|
const normalizedKey = options.normalizeHeaderKey(key, options.canonical);
|
|
241
|
-
if (
|
|
241
|
+
if (normalizedKey !== void 0) {
|
|
242
242
|
headers[normalizedKey] = request.headers[key];
|
|
243
243
|
}
|
|
244
244
|
});
|
|
@@ -256,11 +256,10 @@ var _case = require('case');
|
|
|
256
256
|
|
|
257
257
|
// src/connect/serializers/xml.ts
|
|
258
258
|
var _jstoxml = require('jstoxml');
|
|
259
|
-
var
|
|
259
|
+
var xmlTransformer = (data) => _jstoxml.toXML.call(void 0, data, {
|
|
260
260
|
header: true,
|
|
261
261
|
indent: " "
|
|
262
|
-
};
|
|
263
|
-
var xmlTransformer = (data) => _jstoxml.toXML.call(void 0, data, config);
|
|
262
|
+
});
|
|
264
263
|
var xml_default = xmlTransformer;
|
|
265
264
|
|
|
266
265
|
// src/connect/middleware/serializers-middleware.ts
|
|
@@ -299,7 +298,7 @@ var serialize = (serializers, request, response, data, options) => {
|
|
|
299
298
|
if (!breakTypes) {
|
|
300
299
|
if (/yaml|yml/.test(type)) {
|
|
301
300
|
response.setHeader(contentTypeKey, type);
|
|
302
|
-
serializedData =
|
|
301
|
+
serializedData = _chunkNY75KGAHjs.yaml_default.call(void 0, hasJsonStructure(data) ? JSON.parse(data) : data);
|
|
303
302
|
} else if (/xml/.test(type)) {
|
|
304
303
|
response.setHeader(contentTypeKey, type);
|
|
305
304
|
serializedData = xml_default({
|
|
@@ -363,7 +362,7 @@ var create_node_router_default = createNodeRouter;
|
|
|
363
362
|
var getIP = (request) => (request == null ? void 0 : request.ip) || request.headers["x-forwarded-for"] || request.headers["x-real-ip"] || request.connection.remoteAddress;
|
|
364
363
|
var rateLimiterMiddleware = (rateLimiter, headers) => async (request, response, next) => {
|
|
365
364
|
const ip = getIP(request);
|
|
366
|
-
if (
|
|
365
|
+
if (ip === void 0) {
|
|
367
366
|
throw _httperrors2.default.call(void 0, 400, "Missing IP");
|
|
368
367
|
}
|
|
369
368
|
try {
|
|
@@ -450,5 +449,5 @@ var cors_middleware_default = corsMiddleware;
|
|
|
450
449
|
|
|
451
450
|
|
|
452
451
|
|
|
453
|
-
exports.BadGateway = _httperrors.BadGateway; exports.BadRequest = _httperrors.BadRequest; exports.BandwidthLimitExceeded = _httperrors.BandwidthLimitExceeded; exports.Conflict = _httperrors.Conflict; exports.EdgeRouter = _connect.EdgeRouter; exports.ExpectationFailed = _httperrors.ExpectationFailed; exports.FailedDependency = _httperrors.FailedDependency; exports.Forbidden = _httperrors.Forbidden; exports.GatewayTimeout = _httperrors.GatewayTimeout; exports.Gone = _httperrors.Gone; exports.HTTPVersionNotSupported = _httperrors.HTTPVersionNotSupported; exports.ImATeapot = _httperrors.ImATeapot; exports.InsufficientStorage = _httperrors.InsufficientStorage; exports.InternalServerError = _httperrors.InternalServerError; exports.LengthRequired = _httperrors.LengthRequired; exports.Locked = _httperrors.Locked; exports.LoopDetected = _httperrors.LoopDetected; exports.MethodNotAllowed = _httperrors.MethodNotAllowed; exports.MisdirectedRequest = _httperrors.MisdirectedRequest; exports.NetworkAuthenticationRequire = _httperrors.NetworkAuthenticationRequire; exports.NodeRouter = _connect.NodeRouter; exports.NotAcceptable = _httperrors.NotAcceptable; exports.NotExtended = _httperrors.NotExtended; exports.NotFound = _httperrors.NotFound; exports.NotImplemented = _httperrors.NotImplemented; exports.PayloadTooLarge = _httperrors.PayloadTooLarge; exports.PaymentRequired = _httperrors.PaymentRequired; exports.PreconditionFailed = _httperrors.PreconditionFailed; exports.PreconditionRequired = _httperrors.PreconditionRequired; exports.ProxyAuthenticationRequired = _httperrors.ProxyAuthenticationRequired; exports.RangeNotSatisfiable = _httperrors.RangeNotSatisfiable; exports.RequestHeaderFieldsTooLarge = _httperrors.RequestHeaderFieldsTooLarge; exports.RequestTimeout = _httperrors.RequestTimeout; exports.Router = _connect.Router; exports.ServiceUnavailable = _httperrors.ServiceUnavailable; exports.TooManyRequests = _httperrors.TooManyRequests; exports.URITooLong = _httperrors.URITooLong; exports.Unauthorized = _httperrors.Unauthorized; exports.UnavailableForLegalReasons = _httperrors.UnavailableForLegalReasons; exports.UnorderedCollection = _httperrors.UnorderedCollection; exports.UnprocessableEntity = _httperrors.UnprocessableEntity; exports.UnsupportedMediaType = _httperrors.UnsupportedMediaType; exports.UpgradeRequired = _httperrors.UpgradeRequired; exports.VariantAlsoNegotiates = _httperrors.VariantAlsoNegotiates; exports.corsMiddleware = cors_middleware_default; exports.createEdgeRouter = _connect.createEdgeRouter; exports.createHttpError = _httperrors2.default; exports.createNodeRouter = create_node_router_default; exports.dateIn = _chunkT25VSNTFjs.dateIn; exports.dateOut = _chunkT25VSNTFjs.dateOut; exports.expressWrapper = _connect.expressWrapper; exports.httpHeaderNormalizerMiddleware = http_header_normalizer_default; exports.onError = onError; exports.onNoMatch = onNoMatch; exports.rateLimiterMiddleware = rate_limiter_middleware_default; exports.sendJson = _connect.sendJson; exports.serializersMiddleware = serializers_middleware_default; exports.swaggerHandler =
|
|
452
|
+
exports.BadGateway = _httperrors.BadGateway; exports.BadRequest = _httperrors.BadRequest; exports.BandwidthLimitExceeded = _httperrors.BandwidthLimitExceeded; exports.Conflict = _httperrors.Conflict; exports.EdgeRouter = _connect.EdgeRouter; exports.ExpectationFailed = _httperrors.ExpectationFailed; exports.FailedDependency = _httperrors.FailedDependency; exports.Forbidden = _httperrors.Forbidden; exports.GatewayTimeout = _httperrors.GatewayTimeout; exports.Gone = _httperrors.Gone; exports.HTTPVersionNotSupported = _httperrors.HTTPVersionNotSupported; exports.ImATeapot = _httperrors.ImATeapot; exports.InsufficientStorage = _httperrors.InsufficientStorage; exports.InternalServerError = _httperrors.InternalServerError; exports.LengthRequired = _httperrors.LengthRequired; exports.Locked = _httperrors.Locked; exports.LoopDetected = _httperrors.LoopDetected; exports.MethodNotAllowed = _httperrors.MethodNotAllowed; exports.MisdirectedRequest = _httperrors.MisdirectedRequest; exports.NetworkAuthenticationRequire = _httperrors.NetworkAuthenticationRequire; exports.NodeRouter = _connect.NodeRouter; exports.NotAcceptable = _httperrors.NotAcceptable; exports.NotExtended = _httperrors.NotExtended; exports.NotFound = _httperrors.NotFound; exports.NotImplemented = _httperrors.NotImplemented; exports.PayloadTooLarge = _httperrors.PayloadTooLarge; exports.PaymentRequired = _httperrors.PaymentRequired; exports.PreconditionFailed = _httperrors.PreconditionFailed; exports.PreconditionRequired = _httperrors.PreconditionRequired; exports.ProxyAuthenticationRequired = _httperrors.ProxyAuthenticationRequired; exports.RangeNotSatisfiable = _httperrors.RangeNotSatisfiable; exports.RequestHeaderFieldsTooLarge = _httperrors.RequestHeaderFieldsTooLarge; exports.RequestTimeout = _httperrors.RequestTimeout; exports.Router = _connect.Router; exports.ServiceUnavailable = _httperrors.ServiceUnavailable; exports.TooManyRequests = _httperrors.TooManyRequests; exports.URITooLong = _httperrors.URITooLong; exports.Unauthorized = _httperrors.Unauthorized; exports.UnavailableForLegalReasons = _httperrors.UnavailableForLegalReasons; exports.UnorderedCollection = _httperrors.UnorderedCollection; exports.UnprocessableEntity = _httperrors.UnprocessableEntity; exports.UnsupportedMediaType = _httperrors.UnsupportedMediaType; exports.UpgradeRequired = _httperrors.UpgradeRequired; exports.VariantAlsoNegotiates = _httperrors.VariantAlsoNegotiates; exports.corsMiddleware = cors_middleware_default; exports.createEdgeRouter = _connect.createEdgeRouter; exports.createHttpError = _httperrors2.default; exports.createNodeRouter = create_node_router_default; exports.dateIn = _chunkT25VSNTFjs.dateIn; exports.dateOut = _chunkT25VSNTFjs.dateOut; exports.expressWrapper = _connect.expressWrapper; exports.httpHeaderNormalizerMiddleware = http_header_normalizer_default; exports.onError = onError; exports.onNoMatch = onNoMatch; exports.rateLimiterMiddleware = rate_limiter_middleware_default; exports.sendJson = _connect.sendJson; exports.serializersMiddleware = serializers_middleware_default; exports.swaggerHandler = _chunkNY75KGAHjs.swagger_handler_default; exports.withZod = _connect.withZod; exports.zod = _chunkT25VSNTFjs.zod_exports;
|
|
454
453
|
//# sourceMappingURL=index-server.js.map
|
package/dist/index-server.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index-server.ts","../src/connect/create-node-router.ts","../src/connect/handler.ts","../src/error-handler/jsonapi-error-handler.ts","../src/error-handler/utils.ts","../src/error-handler/problem-error-handler.ts","../src/connect/middleware/http-header-normalizer.ts","../src/connect/middleware/serializers-middleware.ts","../src/connect/serializers/xml.ts","../src/connect/middleware/rate-limiter-middleware.ts","../src/connect/middleware/cors-middleware.ts"],"names":["default","HttpError","getReasonPhrase","defaultTitle","expressWrapper","NodeRouter","sendJson","createHttpError"],"mappings":"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","sourcesContent":["export * from \"./index-browser\";\n\nexport {\n default as createHttpError,\n BadRequest,\n Forbidden,\n BadGateway,\n BandwidthLimitExceeded,\n Conflict,\n ExpectationFailed,\n FailedDependency,\n GatewayTimeout,\n Gone,\n HTTPVersionNotSupported,\n ImATeapot,\n InsufficientStorage,\n InternalServerError,\n VariantAlsoNegotiates,\n ProxyAuthenticationRequired,\n NetworkAuthenticationRequire,\n LengthRequired,\n LoopDetected,\n Locked,\n MethodNotAllowed,\n MisdirectedRequest,\n NotAcceptable,\n NotExtended,\n NotFound,\n NotImplemented,\n PayloadTooLarge,\n RequestHeaderFieldsTooLarge,\n PaymentRequired,\n PreconditionFailed,\n PreconditionRequired,\n RangeNotSatisfiable,\n RequestTimeout,\n ServiceUnavailable,\n TooManyRequests,\n Unauthorized,\n UnorderedCollection,\n UnprocessableEntity,\n UnavailableForLegalReasons,\n UnsupportedMediaType,\n UpgradeRequired,\n URITooLong,\n} from \"http-errors\";\n\nexport { default as createNodeRouter } from \"./connect/create-node-router\";\nexport { onError, onNoMatch } from \"./connect/handler\";\nexport type {\n EdgeRequestHandler,\n ExpressRequestHandler,\n NodeRequestHandler,\n Route,\n HandlerOptions,\n NextHandler,\n FunctionLike,\n Nextable,\n ValueOrPromise,\n FindResult,\n RouteShortcutMethod,\n HttpMethod,\n} from \"@visulima/connect\";\nexport {\n createEdgeRouter, EdgeRouter, expressWrapper, NodeRouter, Router, withZod, sendJson,\n} from \"@visulima/connect\";\n\nexport { default as rateLimiterMiddleware } from \"./connect/middleware/rate-limiter-middleware\";\nexport { default as corsMiddleware } from \"./connect/middleware/cors-middleware\";\nexport { default as serializersMiddleware } from \"./connect/middleware/serializers-middleware\";\nexport { default as httpHeaderNormalizerMiddleware } from \"./connect/middleware/http-header-normalizer\";\n\nexport { default as swaggerHandler } from \"./swagger/swagger-handler\";\n\nexport { dateIn, dateOut } from \"./zod\";\n","import { NodeRouter } from \"@visulima/connect\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { AnyZodObject } from \"zod\";\nimport { ZodObject } from \"zod\";\n\nimport type { ErrorHandlers } from \"../error-handler/types\";\nimport { onError, onNoMatch } from \"./handler\";\nimport httpHeaderNormalizerMiddleware from \"./middleware/http-header-normalizer\";\nimport type { Serializers } from \"./middleware/serializers-middleware\";\nimport serializersMiddleware from \"./middleware/serializers-middleware\";\n\nconst createNodeRouter = <\n Request extends IncomingMessage,\n Response extends ServerResponse,\n Schema extends AnyZodObject = ZodObject<{ body?: AnyZodObject; headers?: AnyZodObject; query?: AnyZodObject }>,\n>(\n options: {\n middlewares?: {\n \"http-header-normalizer\"?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string };\n serializers?: {\n serializers?: Serializers;\n defaultContentType?: string;\n };\n };\n errorHandlers?: ErrorHandlers;\n showTrace?: boolean;\n } = {},\n ) => {\n const router = new NodeRouter<Request, Response, Schema>({\n onNoMatch,\n onError: onError(options.errorHandlers || [], options.showTrace || false),\n });\n\n return router\n .use(httpHeaderNormalizerMiddleware(options?.middlewares?.[\"http-header-normalizer\"] || {}))\n .use(\n serializersMiddleware(\n options?.middlewares?.serializers?.serializers || [],\n options?.middlewares?.serializers?.defaultContentType || \"application/json; charset=utf-8\",\n ),\n );\n};\n\nexport default createNodeRouter;\n","import type {\n FunctionLike, Nextable, Route, ValueOrPromise,\n} from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport JsonapiErrorHandler from \"../error-handler/jsonapi-error-handler\";\nimport ProblemErrorHandler from \"../error-handler/problem-error-handler\";\nimport type { ErrorHandler, ErrorHandlers } from \"../error-handler/types\";\n\n// eslint-disable-next-line unicorn/consistent-function-scoping,max-len\nexport const onError = <Request extends IncomingMessage, Response extends ServerResponse>(errorHandlers: ErrorHandlers, showTrace: boolean) => async (error: unknown, request: Request, response: Response): Promise<void> => {\n const apiFormat: string = request.headers.accept as string;\n\n let errorHandler: ErrorHandler = ProblemErrorHandler;\n\n if (apiFormat === \"application/vnd.api+json\") {\n errorHandler = JsonapiErrorHandler;\n }\n\n // eslint-disable-next-line no-restricted-syntax\n for (const { regex, handler } of errorHandlers) {\n if (regex.test(apiFormat)) {\n errorHandler = handler;\n break;\n }\n }\n\n // eslint-disable-next-line no-param-reassign\n (error as { expose: boolean } & Error).expose = showTrace;\n\n errorHandler(error, request, response);\n};\n\nexport const onNoMatch: <Request extends IncomingMessage, Response extends ServerResponse>(\n request: Request,\n response: Response,\n routes: Route<Nextable<FunctionLike>>[],\n) => ValueOrPromise<void> = async (request, response, routes) => {\n const uniqueMethods = [...new Set(routes.map((route) => route.method))].join(\", \");\n\n response.setHeader(\"Allow\", uniqueMethods);\n response.statusCode = 405;\n\n throw createHttpError(405, `No route with [${request.method}] method found.`);\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\nimport { ErrorSerializer, JapiError } from \"ts-japi\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultTitle = \"An error occurred\";\n\nconst jsonapiErrorHandler: ErrorHandler = (error: HttpError | JapiError | Error, _request, response) => {\n addStatusCodeToResponse(response, error);\n\n setErrorHeaders(response, error);\n\n if (error instanceof JapiError || JapiError.isLikeJapiError(error)) {\n const serializer = new ErrorSerializer();\n\n sendJson(response, serializer.serialize(error));\n } else if (error instanceof HttpError) {\n const { statusCode, title, message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: statusCode,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n } else {\n const { message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: \"500\",\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n }\n};\n\nexport default jsonapiErrorHandler;\n","import { StatusCodes } from \"http-status-codes\";\nimport type { ServerResponse } from \"node:http\";\n\nexport const setErrorHeaders = (response: ServerResponse, error: any) => {\n const headers: { [key: string]: number | string | ReadonlyArray<string> } = error.headers || {};\n\n Object.keys(headers).forEach((header: string) => {\n response.setHeader(header, headers[header] as number | string | ReadonlyArray<string>);\n });\n};\n\n/**\n * Send `JSON` object\n * @param response response object\n * @param jsonBody of data\n */\nexport const sendJson = (response: ServerResponse, jsonBody: any): void => {\n // Set header to application/json\n response.setHeader(\"content-type\", \"application/json; charset=utf-8\");\n\n response.end(JSON.stringify(jsonBody));\n};\n\nexport const addStatusCodeToResponse = (response: ServerResponse, error: any): void => {\n // respect err.statusCode\n if (typeof error.statusCode !== \"undefined\") {\n response.statusCode = error.statusCode;\n }\n\n // respect err.status\n if (typeof error.status !== \"undefined\") {\n response.statusCode = error.status;\n }\n\n // default status code to 500\n if (response.statusCode < 400) {\n response.statusCode = StatusCodes.INTERNAL_SERVER_ERROR;\n }\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultType = \"https://tools.ietf.org/html/rfc2616#section-10\";\nconst defaultTitle = \"An error occurred\";\n/**\n * Normalizes errors according to the API Problem spec (RFC 7807).\n *\n * @see https://tools.ietf.org/html/rfc7807\n */\nconst problemErrorHandler: ErrorHandler = (error: HttpError | Error, _request, response) => {\n const { stack, message } = error;\n\n if (error instanceof HttpError) {\n const {\n statusCode, expose, title, type,\n } = error;\n\n response.statusCode = statusCode;\n\n setErrorHeaders(response, error);\n\n sendJson(response, {\n type: type || defaultType,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n details: message,\n ...(expose ? { trace: stack } : {}),\n });\n } else {\n addStatusCodeToResponse(response, error);\n\n sendJson(response, {\n type: defaultType,\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n details: message,\n ...((error as { expose: boolean } & Error).expose ? { trace: stack } : {}),\n });\n }\n};\n\nexport default problemErrorHandler;\n","import type { NextHandler } from \"@visulima/connect\";\nimport type { IncomingHttpHeaders, IncomingMessage } from \"node:http\";\n\nconst exceptions = {\n alpn: \"ALPN\",\n \"c-pep\": \"C-PEP\",\n \"c-pep-info\": \"C-PEP-Info\",\n \"caldav-timezones\": \"CalDAV-Timezones\",\n \"content-id\": \"Content-ID\",\n \"content-md5\": \"Content-MD5\",\n dasl: \"DASL\",\n dav: \"DAV\",\n dnt: \"DNT\",\n etag: \"ETag\",\n getprofile: \"GetProfile\",\n \"http2-settings\": \"HTTP2-Settings\",\n \"last-event-id\": \"Last-Event-ID\",\n \"mime-version\": \"MIME-Version\",\n \"optional-www-authenticate\": \"Optional-WWW-Authenticate\",\n \"sec-websocket-accept\": \"Sec-WebSocket-Accept\",\n \"sec-websocket-extensions\": \"Sec-WebSocket-Extensions\",\n \"sec-webSocket-key\": \"Sec-WebSocket-Key\",\n \"sec-webSocket-protocol\": \"Sec-WebSocket-Protocol\",\n \"sec-webSocket-version\": \"Sec-WebSocket-Version\",\n slug: \"SLUG\",\n tcn: \"TCN\",\n te: \"TE\",\n ttl: \"TTL\",\n \"www-authenticate\": \"WWW-Authenticate\",\n \"x-att-deviceid\": \"X-ATT-DeviceId\",\n \"x-dnsprefetch-control\": \"X-DNSPrefetch-Control\",\n \"x-uidh\": \"X-UIDH\",\n};\n\nconst normalizeHeaderKey = (key: string, canonical: boolean) => {\n const lowerCaseKey = key.toLowerCase();\n\n if (!canonical) {\n return lowerCaseKey;\n }\n\n if (exceptions[lowerCaseKey as keyof typeof exceptions]) {\n return exceptions[lowerCaseKey as keyof typeof exceptions];\n }\n\n return (\n lowerCaseKey\n .split(\"-\")\n // eslint-disable-next-line no-unsafe-optional-chaining\n .map((text) => text[0]?.toUpperCase() + text.slice(1))\n .join(\"-\")\n );\n};\n\nconst defaults = {\n canonical: false,\n normalizeHeaderKey,\n};\n\n/**\n * HTTP headers are case-insensitive.\n * That's why NodeJS makes them lower case by default.\n * While sensible, sometimes, for example for compatibility reasons, you might need them in their more common form.\n */\nconst httpHeaderNormalizerMiddleware = (options_?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string }) => {\n const options = { ...defaults, ...options_ };\n\n return async <Request extends IncomingMessage>(request: Request, _: any, next: NextHandler) => {\n if (request.headers) {\n const rawHeaders: IncomingHttpHeaders = {};\n const headers: IncomingHttpHeaders = {};\n\n Object.keys(request.headers).forEach((key) => {\n rawHeaders[key] = request.headers[key];\n\n const normalizedKey = options.normalizeHeaderKey(key, options.canonical);\n\n if (typeof normalizedKey !== \"undefined\") {\n headers[normalizedKey] = request.headers[key];\n }\n });\n\n request.headers = headers;\n // @TODO at type `request.rawHeaders` to global scope\n // @ts-ignore\n request.rawHeaders = rawHeaders;\n }\n\n return next();\n };\n};\n\nexport default httpHeaderNormalizerMiddleware;\n","import type { NextHandler } from \"@visulima/connect\";\nimport accepts from \"accepts\";\nimport { header as headerCase } from \"case\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport type { Serializer } from \"../serializers/types\";\nimport xmlTransformer from \"../serializers/xml\";\nimport yamlTransformer from \"../serializers/yaml\";\n\nfunction hasJsonStructure(string_: any): boolean {\n if (typeof string_ !== \"string\") {\n return false;\n }\n\n try {\n const result = JSON.parse(string_);\n const type = Object.prototype.toString.call(result);\n\n return type === \"[object Object]\" || type === \"[object Array]\";\n } catch {\n return false;\n }\n}\n\nconst contentTypeKey = \"Content-Type\";\n\n// eslint-disable-next-line max-len\nconst serialize = <Request extends IncomingMessage, Response extends ServerResponse>(\n serializers: Serializers,\n request: Request,\n response: Response | NextApiResponse,\n data: any,\n options: {\n defaultContentType: string;\n },\n // eslint-disable-next-line radar/cognitive-complexity\n) => {\n const contentType = response.getHeader(contentTypeKey) as string | undefined;\n\n // skip serialization when Content-Type is already set\n if (typeof contentType === \"string\") {\n return data;\n }\n\n const accept = accepts(request);\n const types: string[] = [...(accept.types() as string[]), options.defaultContentType];\n\n let serializedData = data;\n\n // eslint-disable-next-line no-restricted-syntax\n types.every((type) => {\n let breakTypes = false;\n\n serializers.forEach(({ regex, serializer }) => {\n if (!regex.test(type)) {\n return;\n }\n\n response.setHeader(contentTypeKey, type);\n serializedData = serializer(serializedData);\n breakTypes = true;\n });\n\n if (!breakTypes) {\n if (/yaml|yml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = yamlTransformer(hasJsonStructure(data) ? JSON.parse(data) : data);\n } else if (/xml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = xmlTransformer({\n [headerCase(`${request.url?.replace(\"/api/\", \"\")}`.trim())]: hasJsonStructure(data) ? JSON.parse(data) : data,\n });\n }\n }\n\n return breakTypes;\n });\n\n // eslint-disable-next-line no-param-reassign\n return serializedData;\n};\n\n// eslint-disable-next-line max-len\nconst serializersMiddleware = (serializers: Serializers = [], defaultContentType: string = \"application/json; charset=utf-8\") => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n if (typeof (response as NextApiResponse)?.send === \"function\") {\n const oldSend = (response as NextApiResponse).send;\n\n (response as NextApiResponse).send = (data) => {\n (response as NextApiResponse).send = oldSend;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n return (response as NextApiResponse).send(data);\n };\n } else {\n const oldEnd = response.end;\n\n // @ts-ignore\n response.end = (data, ...arguments_) => {\n response.end = oldEnd;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n // @ts-ignore\n return response.end(data, ...arguments_);\n };\n }\n\n return next();\n};\n\nexport type Serializers = {\n regex: RegExp;\n serializer: Serializer;\n}[];\nexport default serializersMiddleware;\n","import type { XmlElement } from \"jstoxml\";\nimport { toXML } from \"jstoxml\";\n\nimport type { Serializer } from \"./types\";\n\nconst config = {\n header: true,\n indent: \" \",\n};\n\nconst xmlTransformer: Serializer = (data?: XmlElement | XmlElement[]) => toXML(data, config);\n\nexport default xmlTransformer;\n","import type { NextHandler } from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { RateLimiterAbstract, RateLimiterRes } from \"rate-limiter-flexible\";\n\n// eslint-disable-next-line max-len\nconst getIP: (request: IncomingMessage & { ip?: string }) => string | undefined = (request) => request?.ip\n || (request.headers[\"x-forwarded-for\"] as string | undefined)\n || (request.headers[\"x-real-ip\"] as string | undefined)\n || request.connection.remoteAddress;\n\ntype HeaderValue = string | number | ReadonlyArray<string>;\n\n// eslint-disable-next-line max-len\nconst rateLimiterMiddleware = (rateLimiter: RateLimiterAbstract, headers?: (limiterResponse: RateLimiterRes) => { [key: string]: HeaderValue }) => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n const ip = getIP(request);\n\n if (typeof ip === \"undefined\") {\n throw createHttpError(400, \"Missing IP\");\n }\n\n try {\n const limiter = await rateLimiter.consume(ip);\n\n const mergedHeaders: { [key: string]: HeaderValue } = {\n \"Retry-After\": Math.round(limiter.msBeforeNext / 1000) || 1,\n \"X-RateLimit-Remaining\": limiter.remainingPoints,\n \"X-RateLimit-Reset\": new Date(Date.now() + limiter.msBeforeNext).toISOString(),\n ...headers,\n };\n\n Object.keys(mergedHeaders).forEach((key) => {\n response.setHeader(key, mergedHeaders[key] as HeaderValue);\n });\n\n await next();\n } catch {\n throw createHttpError(429, \"Too Many Requests\");\n }\n};\n\nexport default rateLimiterMiddleware;\n","import { expressWrapper } from \"@visulima/connect\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport type { CorsOptions, CorsOptionsDelegate } from \"cors\";\nimport cors from \"cors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\n// eslint-disable-next-line max-len\nconst corsMiddleware = <Request extends IncomingMessage, Response extends ServerResponse>(options?: CorsOptions | CorsOptionsDelegate) => expressWrapper<Request, Response>(cors(options));\n\nexport default corsMiddleware;\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/index-server.ts","../src/connect/create-node-router.ts","../src/connect/handler.ts","../src/error-handler/jsonapi-error-handler.ts","../src/error-handler/utils.ts","../src/error-handler/problem-error-handler.ts","../src/connect/middleware/http-header-normalizer.ts","../src/connect/middleware/serializers-middleware.ts","../src/connect/serializers/xml.ts","../src/connect/middleware/rate-limiter-middleware.ts","../src/connect/middleware/cors-middleware.ts"],"names":["default","HttpError","getReasonPhrase","defaultTitle","expressWrapper","NodeRouter","sendJson","createHttpError"],"mappings":"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","sourcesContent":["export * from \"./index-browser\";\n\nexport {\n default as createHttpError,\n BadRequest,\n Forbidden,\n BadGateway,\n BandwidthLimitExceeded,\n Conflict,\n ExpectationFailed,\n FailedDependency,\n GatewayTimeout,\n Gone,\n HTTPVersionNotSupported,\n ImATeapot,\n InsufficientStorage,\n InternalServerError,\n VariantAlsoNegotiates,\n ProxyAuthenticationRequired,\n NetworkAuthenticationRequire,\n LengthRequired,\n LoopDetected,\n Locked,\n MethodNotAllowed,\n MisdirectedRequest,\n NotAcceptable,\n NotExtended,\n NotFound,\n NotImplemented,\n PayloadTooLarge,\n RequestHeaderFieldsTooLarge,\n PaymentRequired,\n PreconditionFailed,\n PreconditionRequired,\n RangeNotSatisfiable,\n RequestTimeout,\n ServiceUnavailable,\n TooManyRequests,\n Unauthorized,\n UnorderedCollection,\n UnprocessableEntity,\n UnavailableForLegalReasons,\n UnsupportedMediaType,\n UpgradeRequired,\n URITooLong,\n} from \"http-errors\";\n\nexport { default as createNodeRouter } from \"./connect/create-node-router\";\nexport { onError, onNoMatch } from \"./connect/handler\";\nexport type {\n EdgeRequestHandler,\n ExpressRequestHandler,\n NodeRequestHandler,\n Route,\n HandlerOptions,\n NextHandler,\n FunctionLike,\n Nextable,\n ValueOrPromise,\n FindResult,\n RouteShortcutMethod,\n HttpMethod,\n} from \"@visulima/connect\";\nexport {\n createEdgeRouter, EdgeRouter, expressWrapper, NodeRouter, Router, withZod, sendJson,\n} from \"@visulima/connect\";\n\nexport { default as rateLimiterMiddleware } from \"./connect/middleware/rate-limiter-middleware\";\nexport { default as corsMiddleware } from \"./connect/middleware/cors-middleware\";\nexport { default as serializersMiddleware } from \"./connect/middleware/serializers-middleware\";\nexport { default as httpHeaderNormalizerMiddleware } from \"./connect/middleware/http-header-normalizer\";\n\nexport { default as swaggerHandler } from \"./swagger/swagger-handler\";\n\nexport { dateIn, dateOut } from \"./zod\";\n","import { NodeRouter } from \"@visulima/connect\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { AnyZodObject } from \"zod\";\nimport { ZodObject } from \"zod\";\n\nimport type { ErrorHandlers } from \"../error-handler/types\";\nimport { onError, onNoMatch } from \"./handler\";\nimport httpHeaderNormalizerMiddleware from \"./middleware/http-header-normalizer\";\nimport type { Serializers } from \"./middleware/serializers-middleware\";\nimport serializersMiddleware from \"./middleware/serializers-middleware\";\n\nconst createNodeRouter = <\n Request extends IncomingMessage,\n Response extends ServerResponse,\n Schema extends AnyZodObject = ZodObject<{ body?: AnyZodObject; headers?: AnyZodObject; query?: AnyZodObject }>,\n>(\n options: {\n middlewares?: {\n \"http-header-normalizer\"?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string };\n serializers?: {\n serializers?: Serializers;\n defaultContentType?: string;\n };\n };\n errorHandlers?: ErrorHandlers;\n showTrace?: boolean;\n } = {},\n ) => {\n const router = new NodeRouter<Request, Response, Schema>({\n onNoMatch,\n onError: onError(options.errorHandlers || [], options.showTrace || false),\n });\n\n return router\n .use(httpHeaderNormalizerMiddleware(options?.middlewares?.[\"http-header-normalizer\"] || {}))\n .use(\n serializersMiddleware(\n options?.middlewares?.serializers?.serializers || [],\n options?.middlewares?.serializers?.defaultContentType || \"application/json; charset=utf-8\",\n ),\n );\n};\n\nexport default createNodeRouter;\n","import type {\n FunctionLike, Nextable, Route, ValueOrPromise,\n} from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport JsonapiErrorHandler from \"../error-handler/jsonapi-error-handler\";\nimport ProblemErrorHandler from \"../error-handler/problem-error-handler\";\nimport type { ErrorHandler, ErrorHandlers } from \"../error-handler/types\";\n\n// eslint-disable-next-line unicorn/consistent-function-scoping,max-len\nexport const onError = <Request extends IncomingMessage, Response extends ServerResponse>(errorHandlers: ErrorHandlers, showTrace: boolean) => async (error: unknown, request: Request, response: Response): Promise<void> => {\n const apiFormat: string = request.headers.accept as string;\n\n let errorHandler: ErrorHandler = ProblemErrorHandler;\n\n if (apiFormat === \"application/vnd.api+json\") {\n errorHandler = JsonapiErrorHandler;\n }\n\n // eslint-disable-next-line no-restricted-syntax\n for (const { regex, handler } of errorHandlers) {\n if (regex.test(apiFormat)) {\n errorHandler = handler;\n break;\n }\n }\n\n // eslint-disable-next-line no-param-reassign\n (error as { expose: boolean } & Error).expose = showTrace;\n\n errorHandler(error, request, response);\n};\n\nexport const onNoMatch: <Request extends IncomingMessage, Response extends ServerResponse>(\n request: Request,\n response: Response,\n routes: Route<Nextable<FunctionLike>>[],\n) => ValueOrPromise<void> = async (request, response, routes) => {\n const uniqueMethods = [...new Set(routes.map((route) => route.method))].join(\", \");\n\n response.setHeader(\"Allow\", uniqueMethods);\n response.statusCode = 405;\n\n throw createHttpError(405, `No route with [${request.method}] method found.`);\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\nimport { ErrorSerializer, JapiError } from \"ts-japi\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultTitle = \"An error occurred\";\n\nconst jsonapiErrorHandler: ErrorHandler = (error: HttpError | JapiError | Error, _request, response) => {\n addStatusCodeToResponse(response, error);\n\n setErrorHeaders(response, error);\n\n if (error instanceof JapiError || JapiError.isLikeJapiError(error)) {\n const serializer = new ErrorSerializer();\n\n sendJson(response, serializer.serialize(error));\n } else if (error instanceof HttpError) {\n const { statusCode, title, message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: statusCode,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n } else {\n const { message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: \"500\",\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n }\n};\n\nexport default jsonapiErrorHandler;\n","import { StatusCodes } from \"http-status-codes\";\nimport type { ServerResponse } from \"node:http\";\n\nexport const setErrorHeaders = (response: ServerResponse, error: any) => {\n const headers: { [key: string]: number | string | ReadonlyArray<string> } = error.headers || {};\n\n Object.keys(headers).forEach((header: string) => {\n response.setHeader(header, headers[header] as number | string | ReadonlyArray<string>);\n });\n};\n\n/**\n * Send `JSON` object\n * @param response response object\n * @param jsonBody of data\n */\nexport const sendJson = (response: ServerResponse, jsonBody: any): void => {\n // Set header to application/json\n response.setHeader(\"content-type\", \"application/json; charset=utf-8\");\n\n response.end(JSON.stringify(jsonBody));\n};\n\nexport const addStatusCodeToResponse = (response: ServerResponse, error: any): void => {\n // respect err.statusCode\n if (error.statusCode !== undefined) {\n response.statusCode = error.statusCode;\n }\n\n // respect err.status\n if (error.status !== undefined) {\n response.statusCode = error.status;\n }\n\n // default status code to 500\n if (response.statusCode < 400) {\n response.statusCode = StatusCodes.INTERNAL_SERVER_ERROR;\n }\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultType = \"https://tools.ietf.org/html/rfc2616#section-10\";\nconst defaultTitle = \"An error occurred\";\n/**\n * Normalizes errors according to the API Problem spec (RFC 7807).\n *\n * @see https://tools.ietf.org/html/rfc7807\n */\nconst problemErrorHandler: ErrorHandler = (error: HttpError | Error, _request, response) => {\n const { stack, message } = error;\n\n if (error instanceof HttpError) {\n const {\n statusCode, expose, title, type,\n } = error;\n\n response.statusCode = statusCode;\n\n setErrorHeaders(response, error);\n\n sendJson(response, {\n type: type || defaultType,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n details: message,\n ...(expose ? { trace: stack } : {}),\n });\n } else {\n addStatusCodeToResponse(response, error);\n\n sendJson(response, {\n type: defaultType,\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n details: message,\n ...((error as { expose: boolean } & Error).expose ? { trace: stack } : {}),\n });\n }\n};\n\nexport default problemErrorHandler;\n","import type { NextHandler } from \"@visulima/connect\";\nimport type { IncomingHttpHeaders, IncomingMessage } from \"node:http\";\n\nconst exceptions = {\n alpn: \"ALPN\",\n \"c-pep\": \"C-PEP\",\n \"c-pep-info\": \"C-PEP-Info\",\n \"caldav-timezones\": \"CalDAV-Timezones\",\n \"content-id\": \"Content-ID\",\n \"content-md5\": \"Content-MD5\",\n dasl: \"DASL\",\n dav: \"DAV\",\n dnt: \"DNT\",\n etag: \"ETag\",\n getprofile: \"GetProfile\",\n \"http2-settings\": \"HTTP2-Settings\",\n \"last-event-id\": \"Last-Event-ID\",\n \"mime-version\": \"MIME-Version\",\n \"optional-www-authenticate\": \"Optional-WWW-Authenticate\",\n \"sec-websocket-accept\": \"Sec-WebSocket-Accept\",\n \"sec-websocket-extensions\": \"Sec-WebSocket-Extensions\",\n \"sec-webSocket-key\": \"Sec-WebSocket-Key\",\n \"sec-webSocket-protocol\": \"Sec-WebSocket-Protocol\",\n \"sec-webSocket-version\": \"Sec-WebSocket-Version\",\n slug: \"SLUG\",\n tcn: \"TCN\",\n te: \"TE\",\n ttl: \"TTL\",\n \"www-authenticate\": \"WWW-Authenticate\",\n \"x-att-deviceid\": \"X-ATT-DeviceId\",\n \"x-dnsprefetch-control\": \"X-DNSPrefetch-Control\",\n \"x-uidh\": \"X-UIDH\",\n};\n\nconst normalizeHeaderKey = (key: string, canonical: boolean) => {\n const lowerCaseKey = key.toLowerCase();\n\n if (!canonical) {\n return lowerCaseKey;\n }\n\n if (exceptions[lowerCaseKey as keyof typeof exceptions]) {\n return exceptions[lowerCaseKey as keyof typeof exceptions];\n }\n\n return (\n lowerCaseKey\n .split(\"-\")\n // eslint-disable-next-line no-unsafe-optional-chaining\n .map((text) => text[0]?.toUpperCase() + text.slice(1))\n .join(\"-\")\n );\n};\n\nconst defaults = {\n canonical: false,\n normalizeHeaderKey,\n};\n\n/**\n * HTTP headers are case-insensitive.\n * That's why NodeJS makes them lower case by default.\n * While sensible, sometimes, for example for compatibility reasons, you might need them in their more common form.\n */\nconst httpHeaderNormalizerMiddleware = (options_?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string }) => {\n const options = { ...defaults, ...options_ };\n\n return async <Request extends IncomingMessage>(request: Request, _: any, next: NextHandler) => {\n if (request.headers) {\n const rawHeaders: IncomingHttpHeaders = {};\n const headers: IncomingHttpHeaders = {};\n\n Object.keys(request.headers).forEach((key) => {\n rawHeaders[key] = request.headers[key];\n\n const normalizedKey = options.normalizeHeaderKey(key, options.canonical);\n\n if (normalizedKey !== undefined) {\n headers[normalizedKey] = request.headers[key];\n }\n });\n\n request.headers = headers;\n // @TODO at type `request.rawHeaders` to global scope\n // @ts-ignore\n request.rawHeaders = rawHeaders;\n }\n\n return next();\n };\n};\n\nexport default httpHeaderNormalizerMiddleware;\n","import type { NextHandler } from \"@visulima/connect\";\nimport accepts from \"accepts\";\nimport { header as headerCase } from \"case\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport type { Serializer } from \"../serializers/types\";\nimport xmlTransformer from \"../serializers/xml\";\nimport yamlTransformer from \"../serializers/yaml\";\n\nfunction hasJsonStructure(string_: any): boolean {\n if (typeof string_ !== \"string\") {\n return false;\n }\n\n try {\n const result = JSON.parse(string_);\n const type = Object.prototype.toString.call(result);\n\n return type === \"[object Object]\" || type === \"[object Array]\";\n } catch {\n return false;\n }\n}\n\nconst contentTypeKey = \"Content-Type\";\n\n// eslint-disable-next-line max-len\nconst serialize = <Request extends IncomingMessage, Response extends ServerResponse>(\n serializers: Serializers,\n request: Request,\n response: Response | NextApiResponse,\n data: any,\n options: {\n defaultContentType: string;\n },\n // eslint-disable-next-line radar/cognitive-complexity\n) => {\n const contentType = response.getHeader(contentTypeKey) as string | undefined;\n\n // skip serialization when Content-Type is already set\n if (typeof contentType === \"string\") {\n return data;\n }\n\n const accept = accepts(request);\n const types: string[] = [...(accept.types() as string[]), options.defaultContentType];\n\n let serializedData = data;\n\n // eslint-disable-next-line no-restricted-syntax\n types.every((type) => {\n let breakTypes = false;\n\n serializers.forEach(({ regex, serializer }) => {\n if (!regex.test(type)) {\n return;\n }\n\n response.setHeader(contentTypeKey, type);\n serializedData = serializer(serializedData);\n breakTypes = true;\n });\n\n if (!breakTypes) {\n if (/yaml|yml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = yamlTransformer(hasJsonStructure(data) ? JSON.parse(data) : data);\n } else if (/xml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = xmlTransformer({\n [headerCase(`${request.url?.replace(\"/api/\", \"\")}`.trim())]: hasJsonStructure(data) ? JSON.parse(data) : data,\n });\n }\n }\n\n return breakTypes;\n });\n\n // eslint-disable-next-line no-param-reassign\n return serializedData;\n};\n\n// eslint-disable-next-line max-len\nconst serializersMiddleware = (serializers: Serializers = [], defaultContentType: string = \"application/json; charset=utf-8\") => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n if (typeof (response as NextApiResponse)?.send === \"function\") {\n const oldSend = (response as NextApiResponse).send;\n\n (response as NextApiResponse).send = (data) => {\n (response as NextApiResponse).send = oldSend;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n return (response as NextApiResponse).send(data);\n };\n } else {\n const oldEnd = response.end;\n\n // @ts-ignore\n response.end = (data, ...arguments_) => {\n response.end = oldEnd;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n // @ts-ignore\n return response.end(data, ...arguments_);\n };\n }\n\n return next();\n};\n\nexport type Serializers = {\n regex: RegExp;\n serializer: Serializer;\n}[];\nexport default serializersMiddleware;\n","import type { XmlElement } from \"jstoxml\";\nimport { toXML } from \"jstoxml\";\n\nimport type { Serializer } from \"./types\";\n\nconst xmlTransformer: Serializer = (data?: XmlElement | XmlElement[]) => toXML(data, {\n header: true,\n indent: \" \",\n});\n\nexport default xmlTransformer;\n","import type { NextHandler } from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { RateLimiterAbstract, RateLimiterRes } from \"rate-limiter-flexible\";\n\n// eslint-disable-next-line max-len\nconst getIP: (request: IncomingMessage & { ip?: string }) => string | undefined = (request) => request?.ip\n || (request.headers[\"x-forwarded-for\"] as string | undefined)\n || (request.headers[\"x-real-ip\"] as string | undefined)\n || request.connection.remoteAddress;\n\ntype HeaderValue = string | number | ReadonlyArray<string>;\n\n// eslint-disable-next-line max-len\nconst rateLimiterMiddleware = (rateLimiter: RateLimiterAbstract, headers?: (limiterResponse: RateLimiterRes) => { [key: string]: HeaderValue }) => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n const ip = getIP(request);\n\n if (ip === undefined) {\n throw createHttpError(400, \"Missing IP\");\n }\n\n try {\n const limiter = await rateLimiter.consume(ip);\n\n const mergedHeaders: { [key: string]: HeaderValue } = {\n \"Retry-After\": Math.round(limiter.msBeforeNext / 1000) || 1,\n \"X-RateLimit-Remaining\": limiter.remainingPoints,\n \"X-RateLimit-Reset\": new Date(Date.now() + limiter.msBeforeNext).toISOString(),\n ...headers,\n };\n\n Object.keys(mergedHeaders).forEach((key) => {\n response.setHeader(key, mergedHeaders[key] as HeaderValue);\n });\n\n await next();\n } catch {\n throw createHttpError(429, \"Too Many Requests\");\n }\n};\n\nexport default rateLimiterMiddleware;\n","import { expressWrapper } from \"@visulima/connect\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport type { CorsOptions, CorsOptionsDelegate } from \"cors\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport cors from \"cors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\n// eslint-disable-next-line max-len\nconst corsMiddleware = <Request extends IncomingMessage, Response extends ServerResponse>(options?: CorsOptions | CorsOptionsDelegate) => expressWrapper<Request, Response>(cors(options));\n\nexport default corsMiddleware;\n"]}
|