@oino-ts/common 0.21.2 → 1.0.0

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.
Files changed (69) hide show
  1. package/README.md +183 -0
  2. package/dist/cjs/OINOApi.js +322 -0
  3. package/dist/cjs/OINOConfig.js +104 -0
  4. package/dist/cjs/OINOConstants.js +42 -0
  5. package/dist/cjs/OINODataField.js +346 -0
  6. package/dist/cjs/OINODataModel.js +182 -0
  7. package/dist/cjs/OINODataSource.js +165 -0
  8. package/dist/cjs/OINOFormatter.js +6 -5
  9. package/dist/cjs/OINOHtmlTemplate.js +21 -18
  10. package/dist/cjs/OINOModelSet.js +333 -0
  11. package/dist/cjs/OINOParser.js +448 -0
  12. package/dist/cjs/OINOQueryParams.js +434 -0
  13. package/dist/cjs/OINORequest.js +21 -13
  14. package/dist/cjs/OINOResult.js +13 -12
  15. package/dist/cjs/OINOStr.js +11 -11
  16. package/dist/cjs/OINOSwagger.js +205 -0
  17. package/dist/cjs/index.js +57 -39
  18. package/dist/esm/OINOApi.js +315 -0
  19. package/dist/esm/OINOConfig.js +100 -0
  20. package/dist/esm/OINOConstants.js +39 -0
  21. package/dist/esm/OINODataField.js +337 -0
  22. package/dist/esm/OINODataModel.js +178 -0
  23. package/dist/esm/OINODataSource.js +159 -0
  24. package/dist/esm/OINOFormatter.js +2 -1
  25. package/dist/esm/OINOHtmlTemplate.js +4 -1
  26. package/dist/esm/OINOModelSet.js +329 -0
  27. package/dist/esm/OINOParser.js +444 -0
  28. package/dist/esm/OINOQueryParams.js +426 -0
  29. package/dist/esm/OINORequest.js +9 -1
  30. package/dist/esm/OINOResult.js +2 -1
  31. package/dist/esm/OINOStr.js +1 -1
  32. package/dist/esm/OINOSwagger.js +201 -0
  33. package/dist/esm/index.js +14 -32
  34. package/dist/types/OINOApi.d.ts +191 -0
  35. package/dist/types/OINOConfig.d.ts +63 -0
  36. package/dist/types/OINOConstants.d.ts +51 -0
  37. package/dist/types/OINODataField.d.ts +209 -0
  38. package/dist/types/OINODataModel.d.ts +78 -0
  39. package/dist/types/OINODataSource.d.ts +184 -0
  40. package/dist/types/OINOHtmlTemplate.d.ts +1 -1
  41. package/dist/types/OINOModelSet.d.ts +64 -0
  42. package/dist/types/OINOParser.d.ts +42 -0
  43. package/dist/types/OINOQueryParams.d.ts +270 -0
  44. package/dist/types/OINORequest.d.ts +4 -1
  45. package/dist/types/OINOResult.d.ts +1 -1
  46. package/dist/types/OINOStr.d.ts +1 -1
  47. package/dist/types/OINOSwagger.d.ts +25 -0
  48. package/dist/types/index.d.ts +14 -31
  49. package/package.json +32 -32
  50. package/src/OINOApi.ts +429 -0
  51. package/src/OINOBenchmark.ts +323 -323
  52. package/src/OINOConfig.ts +113 -0
  53. package/src/OINOConstants.ts +59 -0
  54. package/src/OINODataField.ts +371 -0
  55. package/src/OINODataModel.ts +187 -0
  56. package/src/OINODataSource.ts +280 -0
  57. package/src/OINOFormatter.ts +166 -165
  58. package/src/OINOHeaders.ts +51 -51
  59. package/src/OINOHtmlTemplate.test.ts +114 -114
  60. package/src/OINOHtmlTemplate.ts +225 -222
  61. package/src/OINOLog.ts +292 -292
  62. package/src/OINOModelSet.ts +359 -0
  63. package/src/OINOParser.ts +441 -0
  64. package/src/OINOQueryParams.ts +449 -0
  65. package/src/OINORequest.ts +204 -196
  66. package/src/OINOResult.ts +331 -330
  67. package/src/OINOStr.ts +254 -254
  68. package/src/OINOSwagger.ts +213 -0
  69. package/src/index.ts +18 -38
@@ -0,0 +1,444 @@
1
+ /*
2
+ * This Source Code Form is subject to the terms of the Mozilla Public
3
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
4
+ * file, You can obtain one at https://mozilla.org/MPL/2.0/.
5
+ */
6
+ import { Buffer } from "node:buffer";
7
+ import { OINOContentType, OINO_ERROR_PREFIX } from "./OINOConstants.js";
8
+ import { OINOStr } from "./OINOStr.js";
9
+ import { OINOLog } from "./OINOLog.js";
10
+ import { OINONumberDataField } from "./OINODataField.js";
11
+ /**
12
+ * Static factory class for easily creating things based on data
13
+ *
14
+ */
15
+ export class OINOParser {
16
+ /**
17
+ * Create data rows from request body based on the datamodel.
18
+ *
19
+ * @param datamodel datamodel of the api
20
+ * @param data data as either serialized string or unserialized JS object or OINODataRow-array or Buffer/Uint8Array binary data
21
+ * @param contentType content type of the data
22
+ * @param multipartBoundary multipart boundary for formdata parsing, if applicable
23
+ *
24
+ */
25
+ static createRows(datamodel, data, contentType, multipartBoundary) {
26
+ let result = [];
27
+ if (typeof data == "string") {
28
+ result = this._createRowsFromText(datamodel, data, contentType, multipartBoundary);
29
+ }
30
+ else if ((data instanceof Buffer) || (data instanceof Uint8Array)) {
31
+ result = this._createRowsFromBlob(datamodel, data, contentType, multipartBoundary);
32
+ }
33
+ else if (typeof data == "object") {
34
+ result = [this._createRowFromObject(datamodel, data)];
35
+ }
36
+ return result;
37
+ }
38
+ static _createRowsFromText(datamodel, data, contentType, multipartBoundary) {
39
+ if ((contentType == OINOContentType.json) || (contentType == undefined)) {
40
+ return this._createRowFromJson(datamodel, data);
41
+ }
42
+ else if (contentType == OINOContentType.csv) {
43
+ return this._createRowFromCsv(datamodel, data);
44
+ }
45
+ else if (contentType == OINOContentType.formdata) {
46
+ return this._createRowFromFormdata(datamodel, Buffer.from(data, "utf8"), multipartBoundary || "");
47
+ }
48
+ else if (contentType == OINOContentType.urlencode) {
49
+ return this._createRowFromUrlencoded(datamodel, data);
50
+ }
51
+ else if (contentType == OINOContentType.html) {
52
+ OINOLog.error("@oino-ts/db", "OINOParser", "createRowsFromText", "HTML can't be used as an input content type!", { contentType: OINOContentType.html });
53
+ return [];
54
+ }
55
+ else {
56
+ OINOLog.error("@oino-ts/db", "OINOParser", "createRowsFromText", "Unrecognized input content type!", { contentType: contentType });
57
+ return [];
58
+ }
59
+ }
60
+ static _createRowsFromBlob(datamodel, data, contentType, multipartBoundary) {
61
+ if (data instanceof Uint8Array && !(data instanceof Buffer)) {
62
+ data = Buffer.from(data);
63
+ }
64
+ if ((contentType == OINOContentType.json) || (contentType == undefined)) {
65
+ return this._createRowFromJson(datamodel, data.toString()); // JSON is always a string
66
+ }
67
+ else if (contentType == OINOContentType.csv) {
68
+ return this._createRowFromCsv(datamodel, data.toString()); // binary data has to be base64 encoded so it's a string
69
+ }
70
+ else if (contentType == OINOContentType.formdata) {
71
+ return this._createRowFromFormdata(datamodel, data, multipartBoundary || "");
72
+ }
73
+ else if (contentType == OINOContentType.urlencode) {
74
+ return this._createRowFromUrlencoded(datamodel, data.toString()); // data is urlencoded so it's a string
75
+ }
76
+ else if (contentType == OINOContentType.html) {
77
+ OINOLog.error("@oino-ts/db", "OINOParser", "createRowsFromBlob", "HTML can't be used as an input content type!", { contentType: OINOContentType.html });
78
+ return [];
79
+ }
80
+ else {
81
+ OINOLog.error("@oino-ts/db", "OINOParser", "createRowsFromBlob", "Unrecognized input content type!", { contentType: contentType });
82
+ return [];
83
+ }
84
+ }
85
+ /**
86
+ * Create one data row from javascript object based on the datamodel.
87
+ * NOTE! Data assumed to be unserialized i.e. of the native type (string, number, boolean, Buffer)
88
+ *
89
+ * @param datamodel datamodel of the api
90
+ * @param data data as javascript object
91
+ *
92
+ */
93
+ static _createRowFromObject(datamodel, data) {
94
+ const fields = datamodel.fields;
95
+ let result = new Array(fields.length);
96
+ for (let i = 0; i < fields.length; i++) {
97
+ result[i] = data[fields[i].name];
98
+ }
99
+ return result;
100
+ }
101
+ static _findCsvLineEnd(csvData, start) {
102
+ const n = csvData.length;
103
+ if (start >= n) {
104
+ return start;
105
+ }
106
+ let end = start;
107
+ let quote_open = false;
108
+ while (end < n) {
109
+ if (csvData[end] == "\"") {
110
+ if (!quote_open) {
111
+ quote_open = true;
112
+ }
113
+ else if ((end < n - 1) && (csvData[end + 1] == "\"")) {
114
+ end++;
115
+ }
116
+ else {
117
+ quote_open = false;
118
+ }
119
+ }
120
+ else if ((!quote_open) && (csvData[end] == "\r")) {
121
+ return end;
122
+ }
123
+ end++;
124
+ }
125
+ return n;
126
+ }
127
+ static _parseCsvLine(csvLine) {
128
+ let result = [];
129
+ const n = csvLine.length;
130
+ let start = 0;
131
+ let end = 0;
132
+ let quote_open = false;
133
+ let has_quotes = false;
134
+ let has_escaped_quotes = false;
135
+ let found_field = false;
136
+ while (end < n) {
137
+ if (csvLine[end] == "\"") {
138
+ if (!quote_open) {
139
+ quote_open = true;
140
+ }
141
+ else if ((end < n - 1) && (csvLine[end + 1] == "\"")) {
142
+ end++;
143
+ has_escaped_quotes = true;
144
+ }
145
+ else {
146
+ has_quotes = true;
147
+ quote_open = false;
148
+ }
149
+ }
150
+ if ((!quote_open) && ((end == n - 1) || (csvLine[end] == ","))) {
151
+ found_field = true;
152
+ if (end == n - 1) {
153
+ end++;
154
+ }
155
+ }
156
+ if (found_field) {
157
+ // console.log("OINODB_csvParseLine: next field=" + csvLine.substring(start,end) + ", start="+start+", end="+end)
158
+ let field_str;
159
+ if (has_quotes) {
160
+ field_str = csvLine.substring(start + 1, end - 1);
161
+ }
162
+ else if (start == end) {
163
+ field_str = undefined;
164
+ }
165
+ else {
166
+ field_str = csvLine.substring(start, end);
167
+ if (field_str == "null") {
168
+ field_str = null;
169
+ }
170
+ }
171
+ result.push(field_str);
172
+ has_quotes = false;
173
+ has_escaped_quotes = true;
174
+ found_field = false;
175
+ start = end + 1;
176
+ }
177
+ end++;
178
+ }
179
+ return result;
180
+ }
181
+ static _createRowFromCsv(datamodel, data) {
182
+ let result = [];
183
+ const n = data.length;
184
+ let start = 0;
185
+ let end = this._findCsvLineEnd(data, start);
186
+ const header_str = data.substring(start, end);
187
+ const headers = this._parseCsvLine(header_str);
188
+ let field_to_header_mapping = new Array(datamodel.fields.length);
189
+ let headers_found = false;
190
+ for (let i = 0; i < field_to_header_mapping.length; i++) {
191
+ field_to_header_mapping[i] = headers.indexOf(datamodel.fields[i].name);
192
+ headers_found = headers_found || (field_to_header_mapping[i] >= 0);
193
+ }
194
+ if (!headers_found) {
195
+ return result;
196
+ }
197
+ start = end + 1;
198
+ end = start;
199
+ while (end < n) {
200
+ while ((start < n) && ((data[start] == "\r") || (data[start] == "\n"))) {
201
+ start++;
202
+ }
203
+ if (start >= n) {
204
+ return result;
205
+ }
206
+ end = this._findCsvLineEnd(data, start);
207
+ const row_data = this._parseCsvLine(data.substring(start, end));
208
+ const row = new Array(field_to_header_mapping.length);
209
+ let has_data = false;
210
+ for (let i = 0; i < datamodel.fields.length; i++) {
211
+ const field = datamodel.fields[i];
212
+ let j = field_to_header_mapping[i];
213
+ let value = row_data[j];
214
+ if ((value === undefined) || (value === null)) { // null/undefined-decoding built into the parser
215
+ row[i] = value;
216
+ }
217
+ else if ((j >= 0) && (j < row_data.length)) {
218
+ value = OINOStr.decode(value, OINOContentType.csv);
219
+ if (value && (field.fieldParams.isPrimaryKey || field.fieldParams.isForeignKey) && (field instanceof OINONumberDataField) && (datamodel.api.hashid)) {
220
+ value = datamodel.api.hashid.decode(value);
221
+ }
222
+ row[i] = field.deserializeCell(value);
223
+ }
224
+ else {
225
+ row[i] = undefined;
226
+ }
227
+ has_data = has_data || (row[i] !== undefined);
228
+ }
229
+ // console.log("createRowFromCsv: next row=" + row)
230
+ if (has_data) {
231
+ result.push(row);
232
+ }
233
+ else {
234
+ OINOLog.warning("@oino-ts/db", "OINOParser", "_createRowFromCsv", "Empty row skipped", {});
235
+ }
236
+ start = end;
237
+ end = start;
238
+ }
239
+ return result;
240
+ }
241
+ static _createRowFromJsonObj(obj, datamodel) {
242
+ // console.log("createRowFromJsonObj: obj=" + JSON.stringify(obj))
243
+ const fields = datamodel.fields;
244
+ let result = new Array(fields.length);
245
+ let has_data = false;
246
+ // console.log("createRowFromJsonObj: " + result)
247
+ for (let i = 0; i < fields.length; i++) {
248
+ const field = fields[i];
249
+ let value = obj[field.name];
250
+ // console.log("createRowFromJsonObj: key=" + field.name + ", val=" + val)
251
+ if ((value === null) || (value === undefined)) { // must be checed first as null is an object
252
+ result[i] = value;
253
+ }
254
+ else if (Array.isArray(value) || typeof value === "object") {
255
+ result[i] = JSON.stringify(value); // store as proper JSON string so JSON.parse can recover it
256
+ }
257
+ else if (typeof value === "string") {
258
+ value = OINOStr.decode(value, OINOContentType.json);
259
+ if (value && (field.fieldParams.isPrimaryKey || field.fieldParams.isForeignKey) && (field instanceof OINONumberDataField) && (datamodel.api.hashid)) {
260
+ value = datamodel.api.hashid.decode(value);
261
+ }
262
+ result[i] = field.deserializeCell(value);
263
+ }
264
+ else {
265
+ result[i] = value; // value types are passed as-is
266
+ }
267
+ has_data = has_data || (result[i] !== undefined);
268
+ // console.log("createRowFromJsonObj: result["+i+"]=" + result[i])
269
+ }
270
+ // console.log("createRowFromJsonObj: " + result)
271
+ if (has_data) {
272
+ return result;
273
+ }
274
+ else {
275
+ OINOLog.warning("@oino-ts/db", "OINOParser", "_createRowFromJsonObj", "Empty row skipped", {});
276
+ return undefined;
277
+ }
278
+ }
279
+ static _createRowFromJson(datamodel, data) {
280
+ let result = [];
281
+ // console.log("OINORowFactoryJson: data=" + data)
282
+ const obj = JSON.parse(data);
283
+ if (Array.isArray(obj)) {
284
+ obj.forEach(row => {
285
+ const data_row = this._createRowFromJsonObj(row, datamodel);
286
+ if (data_row !== undefined) {
287
+ result.push(data_row);
288
+ }
289
+ });
290
+ }
291
+ else {
292
+ const data_row = this._createRowFromJsonObj(obj, datamodel);
293
+ if (data_row !== undefined) {
294
+ result.push(data_row);
295
+ }
296
+ }
297
+ return result;
298
+ }
299
+ static _findMultipartBoundary(formData, multipartBoundary, start) {
300
+ let n = formData.indexOf(multipartBoundary, start);
301
+ if (n >= 0) {
302
+ n += multipartBoundary.length + 2;
303
+ }
304
+ else {
305
+ n = formData.length;
306
+ }
307
+ return n;
308
+ }
309
+ static _parseMultipartLine(data, start) {
310
+ let line_end = data.indexOf('\r\n', start);
311
+ if (line_end >= start) {
312
+ return data.subarray(start, line_end).toString();
313
+ }
314
+ else {
315
+ return '';
316
+ }
317
+ }
318
+ static _multipartHeaderRegex = /Content-Disposition\: (form-data|file); name=\"([^\"]+)\"(; filename=.*)?/i;
319
+ static _createRowFromFormdata(datamodel, data, multipartBoundary) {
320
+ if (!multipartBoundary || (multipartBoundary.length == 0)) {
321
+ OINOLog.error("@oino-ts/db", "OINOParser", "_createRowFromFormdata", "Multipart boundary missing for formdata parsing!", {});
322
+ throw new Error(OINO_ERROR_PREFIX + "Multipart boundary missing for formdata parsing!");
323
+ }
324
+ let result = [];
325
+ try {
326
+ const n = data.length;
327
+ let start = this._findMultipartBoundary(data, multipartBoundary, 0);
328
+ let end = this._findMultipartBoundary(data, multipartBoundary, start);
329
+ const row = new Array(datamodel.fields.length);
330
+ let has_data = false;
331
+ while (end < n) {
332
+ let block_ok = true;
333
+ let l = this._parseMultipartLine(data, start);
334
+ start += l.length + 2;
335
+ const header_matches = OINOParser._multipartHeaderRegex.exec(l);
336
+ if (!header_matches) {
337
+ OINOLog.warning("@oino-ts/db", "OINOParser", "_createRowFromFormdata", "Unsupported block skipped", { header_line: l });
338
+ block_ok = false;
339
+ }
340
+ else {
341
+ const field_name = header_matches[2];
342
+ const is_file = header_matches[3] != null;
343
+ let is_base64 = false;
344
+ const field_index = datamodel.findFieldIndexByName(field_name);
345
+ if (field_index < 0) {
346
+ OINOLog.warning("@oino-ts/db", "OINOParser", "_createRowFromFormdata", "Form field not found and skipped!", { field_name: field_name });
347
+ block_ok = false;
348
+ }
349
+ else {
350
+ const field = datamodel.fields[field_index];
351
+ l = this._parseMultipartLine(data, start);
352
+ while (block_ok && (l != '')) {
353
+ if (l.startsWith('Content-Type:') && (l.indexOf('multipart/mixed') >= 0)) {
354
+ OINOLog.warning("@oino-ts/db", "OINOParser", "_createRowFromFormdata", "Mixed multipart files not supported and skipped!", { header_line: l });
355
+ block_ok = false;
356
+ }
357
+ else if (l.startsWith('Content-Transfer-Encoding:') && (l.indexOf('BASE64') >= 0)) {
358
+ is_base64 = true;
359
+ }
360
+ start += l.length + 2;
361
+ l = this._parseMultipartLine(data, start);
362
+ }
363
+ start += 2;
364
+ if (!block_ok) {
365
+ OINOLog.warning("@oino-ts/db", "OINOParser", "_createRowFromFormdata", "Invalid block skipped", { field_name: field_name });
366
+ }
367
+ else if (start + multipartBoundary.length + 2 >= end) {
368
+ row[field_index] = null;
369
+ }
370
+ else if (is_file) {
371
+ if (is_base64) {
372
+ const value = this._parseMultipartLine(data, start).trim();
373
+ row[field_index] = field.deserializeCell(OINOStr.decode(value, OINOContentType.formdata));
374
+ }
375
+ else {
376
+ const e = this._findMultipartBoundary(data, multipartBoundary, start);
377
+ const value = data.subarray(start, e - 2);
378
+ row[field_index] = value;
379
+ }
380
+ }
381
+ else {
382
+ let value = OINOStr.decode(this._parseMultipartLine(data, start).trim(), OINOContentType.formdata);
383
+ if (value && (field.fieldParams.isPrimaryKey || field.fieldParams.isForeignKey) && (field instanceof OINONumberDataField) && (datamodel.api.hashid)) {
384
+ value = datamodel.api.hashid.decode(value);
385
+ }
386
+ row[field_index] = field.deserializeCell(value);
387
+ }
388
+ has_data = has_data || (row[field_index] !== undefined);
389
+ }
390
+ }
391
+ start = end;
392
+ end = this._findMultipartBoundary(data, multipartBoundary, start);
393
+ }
394
+ if (has_data) {
395
+ result.push(row);
396
+ }
397
+ else {
398
+ OINOLog.warning("@oino-ts/db", "OINOParser", "_createRowFromFormdata", "Empty row skipped", {});
399
+ }
400
+ }
401
+ catch (e) {
402
+ OINOLog.exception("@oino-ts/db", "OINOParser", "_createRowFromFormdata", "Exception parsing formdata", { message: e.message, stack: e.stack });
403
+ }
404
+ return result;
405
+ }
406
+ static _createRowFromUrlencoded(datamodel, data) {
407
+ let result = [];
408
+ const row = new Array(datamodel.fields.length);
409
+ let has_data = false;
410
+ const data_parts = data.trim().split('&');
411
+ try {
412
+ for (let i = 0; i < data_parts.length; i++) {
413
+ const param_parts = data_parts[i].split('=');
414
+ if (param_parts.length == 2) {
415
+ const key = OINOStr.decodeUrlencode(param_parts[0]) || "";
416
+ const field_index = datamodel.findFieldIndexByName(key);
417
+ if (field_index < 0) {
418
+ OINOLog.info("@oino-ts/db", "OINOParser", "_createRowFromUrlencoded", "Param field not found", { field: key });
419
+ }
420
+ else {
421
+ const field = datamodel.fields[field_index];
422
+ let value = OINOStr.decode(param_parts[1], OINOContentType.urlencode);
423
+ if (value && (field.fieldParams.isPrimaryKey || field.fieldParams.isForeignKey) && (field instanceof OINONumberDataField) && (datamodel.api.hashid)) {
424
+ value = datamodel.api.hashid.decode(value);
425
+ }
426
+ row[field_index] = field.deserializeCell(value);
427
+ has_data = has_data || (row[field_index] !== undefined);
428
+ }
429
+ }
430
+ // const value = requestParams[]
431
+ }
432
+ if (has_data) {
433
+ result.push(row);
434
+ }
435
+ else {
436
+ OINOLog.warning("@oino-ts/db", "OINOParser", "_createRowFromUrlencoded", "Empty row skipped", {});
437
+ }
438
+ }
439
+ catch (e) {
440
+ OINOLog.exception("@oino-ts/db", "OINOParser", "_createRowFromUrlencoded", "Exception parsing urlencoded data", { message: e.message, stack: e.stack });
441
+ }
442
+ return result;
443
+ }
444
+ }