@dyrected/core 2.5.25 → 2.5.26
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/dist/__tests__/app.test.d.ts +2 -0
- package/dist/__tests__/app.test.d.ts.map +1 -0
- package/dist/__tests__/app.test.js +27 -0
- package/dist/__tests__/app.test.js.map +1 -0
- package/dist/__tests__/config.test.d.ts +2 -0
- package/dist/__tests__/config.test.d.ts.map +1 -0
- package/dist/__tests__/config.test.js +34 -0
- package/dist/__tests__/config.test.js.map +1 -0
- package/dist/__tests__/deleteMany.test.d.ts +2 -0
- package/dist/__tests__/deleteMany.test.d.ts.map +1 -0
- package/dist/__tests__/deleteMany.test.js +75 -0
- package/dist/__tests__/deleteMany.test.js.map +1 -0
- package/dist/__tests__/depth.test.d.ts +2 -0
- package/dist/__tests__/depth.test.d.ts.map +1 -0
- package/dist/__tests__/depth.test.js +81 -0
- package/dist/__tests__/depth.test.js.map +1 -0
- package/dist/__tests__/dynamic-options.test.d.ts +2 -0
- package/dist/__tests__/dynamic-options.test.d.ts.map +1 -0
- package/dist/__tests__/dynamic-options.test.js +132 -0
- package/dist/__tests__/dynamic-options.test.js.map +1 -0
- package/dist/__tests__/field-inference.test-types.d.ts +24 -0
- package/dist/__tests__/field-inference.test-types.d.ts.map +1 -0
- package/dist/__tests__/field-inference.test-types.js +87 -0
- package/dist/__tests__/field-inference.test-types.js.map +1 -0
- package/dist/__tests__/hooks.test.d.ts +2 -0
- package/dist/__tests__/hooks.test.d.ts.map +1 -0
- package/dist/__tests__/hooks.test.js +320 -0
- package/dist/__tests__/hooks.test.js.map +1 -0
- package/dist/__tests__/mocks.d.ts +68 -0
- package/dist/__tests__/mocks.d.ts.map +1 -0
- package/dist/__tests__/mocks.js +151 -0
- package/dist/__tests__/mocks.js.map +1 -0
- package/dist/__tests__/router.test.d.ts +2 -0
- package/dist/__tests__/router.test.d.ts.map +1 -0
- package/dist/__tests__/router.test.js +48 -0
- package/dist/__tests__/router.test.js.map +1 -0
- package/dist/__tests__/where.test.d.ts +2 -0
- package/dist/__tests__/where.test.d.ts.map +1 -0
- package/dist/__tests__/where.test.js +97 -0
- package/dist/__tests__/where.test.js.map +1 -0
- package/dist/app-BOrsS7Tz.d.cts +1771 -0
- package/dist/app-BOrsS7Tz.d.ts +1771 -0
- package/dist/app.d.ts +21 -0
- package/dist/app.d.ts.map +1 -0
- package/dist/app.js +56 -0
- package/dist/app.js.map +1 -0
- package/dist/auth/jexl.d.ts +10 -0
- package/dist/auth/jexl.d.ts.map +1 -0
- package/dist/auth/jexl.js +22 -0
- package/dist/auth/jexl.js.map +1 -0
- package/dist/auth/password.d.ts +10 -0
- package/dist/auth/password.d.ts.map +1 -0
- package/dist/auth/password.js +28 -0
- package/dist/auth/password.js.map +1 -0
- package/dist/auth/token.d.ts +20 -0
- package/dist/auth/token.d.ts.map +1 -0
- package/dist/auth/token.js +40 -0
- package/dist/auth/token.js.map +1 -0
- package/dist/chunk-SUGK7UYL.js +311 -0
- package/dist/chunk-ZFAOBRHT.js +2709 -0
- package/dist/controllers/auth.controller.d.ts +125 -0
- package/dist/controllers/auth.controller.d.ts.map +1 -0
- package/dist/controllers/auth.controller.js +323 -0
- package/dist/controllers/auth.controller.js.map +1 -0
- package/dist/controllers/collection.controller.d.ts +88 -0
- package/dist/controllers/collection.controller.d.ts.map +1 -0
- package/dist/controllers/collection.controller.js +554 -0
- package/dist/controllers/collection.controller.js.map +1 -0
- package/dist/controllers/global.controller.d.ts +17 -0
- package/dist/controllers/global.controller.d.ts.map +1 -0
- package/dist/controllers/global.controller.js +116 -0
- package/dist/controllers/global.controller.js.map +1 -0
- package/dist/controllers/media.controller.d.ts +36 -0
- package/dist/controllers/media.controller.d.ts.map +1 -0
- package/dist/controllers/media.controller.js +155 -0
- package/dist/controllers/media.controller.js.map +1 -0
- package/dist/controllers/preview.controller.d.ts +37 -0
- package/dist/controllers/preview.controller.d.ts.map +1 -0
- package/dist/controllers/preview.controller.js +48 -0
- package/dist/controllers/preview.controller.js.map +1 -0
- package/dist/index-Bp7PDOYG.d.cts +1750 -0
- package/dist/index-Bp7PDOYG.d.ts +1750 -0
- package/dist/index-DfAmTZXk.d.cts +1749 -0
- package/dist/index-DfAmTZXk.d.ts +1749 -0
- package/dist/index.cjs +2 -2396
- package/dist/index.d.cts +2 -4
- package/dist/index.d.ts +2 -4
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +3 -5
- package/dist/index.js.map +1 -0
- package/dist/middleware/auth.d.ts +18 -0
- package/dist/middleware/auth.d.ts.map +1 -0
- package/dist/middleware/auth.js +45 -0
- package/dist/middleware/auth.js.map +1 -0
- package/dist/router.d.ts +8 -0
- package/dist/router.d.ts.map +1 -0
- package/dist/router.js +463 -0
- package/dist/router.js.map +1 -0
- package/dist/server.cjs +103 -0
- package/dist/server.d.cts +22 -4
- package/dist/server.d.ts +22 -4
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +2429 -8
- package/dist/server.js.map +1 -0
- package/dist/services/audit.service.d.ts +23 -0
- package/dist/services/audit.service.d.ts.map +1 -0
- package/dist/services/audit.service.js +28 -0
- package/dist/services/audit.service.js.map +1 -0
- package/dist/services/defaults.service.d.ts +8 -0
- package/dist/services/defaults.service.d.ts.map +1 -0
- package/dist/services/defaults.service.js +55 -0
- package/dist/services/defaults.service.js.map +1 -0
- package/dist/services/email.service.d.ts +33 -0
- package/dist/services/email.service.d.ts.map +1 -0
- package/dist/services/email.service.js +219 -0
- package/dist/services/email.service.js.map +1 -0
- package/dist/services/media.service.d.ts +20 -0
- package/dist/services/media.service.d.ts.map +1 -0
- package/dist/services/media.service.js +49 -0
- package/dist/services/media.service.js.map +1 -0
- package/dist/services/population.service.d.ts +20 -0
- package/dist/services/population.service.d.ts.map +1 -0
- package/dist/services/population.service.js +168 -0
- package/dist/services/population.service.js.map +1 -0
- package/dist/types/index.d.ts +1749 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +3 -0
- package/dist/types/index.js.map +1 -0
- package/dist/utils/config.d.ts +8 -0
- package/dist/utils/config.d.ts.map +1 -0
- package/dist/utils/config.js +153 -0
- package/dist/utils/config.js.map +1 -0
- package/dist/utils/hooks.d.ts +41 -0
- package/dist/utils/hooks.d.ts.map +1 -0
- package/dist/utils/hooks.js +169 -0
- package/dist/utils/hooks.js.map +1 -0
- package/dist/utils/openapi.d.ts +6 -0
- package/dist/utils/openapi.d.ts.map +1 -0
- package/dist/utils/openapi.js +331 -0
- package/dist/utils/openapi.js.map +1 -0
- package/dist/utils/parse-where.d.ts +63 -0
- package/dist/utils/parse-where.d.ts.map +1 -0
- package/dist/utils/parse-where.js +196 -0
- package/dist/utils/parse-where.js.map +1 -0
- package/dist/utils/readonly-db.d.ts +9 -0
- package/dist/utils/readonly-db.d.ts.map +1 -0
- package/dist/utils/readonly-db.js +21 -0
- package/dist/utils/readonly-db.js.map +1 -0
- package/dist/utils/setup-prompt.d.ts +11 -0
- package/dist/utils/setup-prompt.d.ts.map +1 -0
- package/dist/utils/setup-prompt.js +863 -0
- package/dist/utils/setup-prompt.js.map +1 -0
- package/dist/utils/swagger.d.ts +5 -0
- package/dist/utils/swagger.d.ts.map +1 -0
- package/dist/utils/swagger.js +51 -0
- package/dist/utils/swagger.js.map +1 -0
- package/dist/utils/where-sanitizer.d.ts +10 -0
- package/dist/utils/where-sanitizer.d.ts.map +1 -0
- package/dist/utils/where-sanitizer.js +63 -0
- package/dist/utils/where-sanitizer.js.map +1 -0
- package/dist/where-sanitizer-DQIWTQZW.js +50 -0
- package/package.json +1 -1
|
@@ -0,0 +1,331 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generate an OpenAPI 3.0 specification based on the Dyrected configuration.
|
|
3
|
+
*/
|
|
4
|
+
export function generateOpenApi(config) {
|
|
5
|
+
const spec = {
|
|
6
|
+
openapi: "3.0.0",
|
|
7
|
+
info: {
|
|
8
|
+
title: "Dyrected API",
|
|
9
|
+
version: "1.0.0",
|
|
10
|
+
description: "Automatically generated OpenAPI specification for the Dyrected project.",
|
|
11
|
+
},
|
|
12
|
+
components: {
|
|
13
|
+
schemas: {},
|
|
14
|
+
securitySchemes: {
|
|
15
|
+
ApiKeyAuth: {
|
|
16
|
+
type: "apiKey",
|
|
17
|
+
in: "header",
|
|
18
|
+
name: "x-api-key",
|
|
19
|
+
},
|
|
20
|
+
},
|
|
21
|
+
},
|
|
22
|
+
paths: {},
|
|
23
|
+
security: [{ ApiKeyAuth: [] }],
|
|
24
|
+
};
|
|
25
|
+
// 1. Generate Schemas for Collections
|
|
26
|
+
for (const collection of config.collections) {
|
|
27
|
+
spec.components.schemas[collection.slug] = collectionToSchema(collection);
|
|
28
|
+
}
|
|
29
|
+
// 2. Generate Schemas for Globals
|
|
30
|
+
for (const global of config.globals) {
|
|
31
|
+
spec.components.schemas[global.slug] = globalToSchema(global);
|
|
32
|
+
}
|
|
33
|
+
// 3. Generate Paths for Collections
|
|
34
|
+
for (const collection of config.collections) {
|
|
35
|
+
const slug = collection.slug;
|
|
36
|
+
const path = `/api/collections/${slug}`;
|
|
37
|
+
const labels = collection.labels || { singular: slug, plural: `${slug}s` };
|
|
38
|
+
spec.paths[path] = {
|
|
39
|
+
get: {
|
|
40
|
+
tags: ["Collections"],
|
|
41
|
+
summary: `Find ${labels.plural}`,
|
|
42
|
+
parameters: [
|
|
43
|
+
{ name: "limit", in: "query", schema: { type: "integer", default: 10 } },
|
|
44
|
+
{ name: "page", in: "query", schema: { type: "integer", default: 1 } },
|
|
45
|
+
{ name: "where", in: "query", schema: { type: "string" }, description: "JSON filter" },
|
|
46
|
+
{ name: "sort", in: "query", schema: { type: "string" }, description: "Sort field (e.g. -createdAt)" },
|
|
47
|
+
],
|
|
48
|
+
responses: {
|
|
49
|
+
200: {
|
|
50
|
+
description: "Success",
|
|
51
|
+
content: {
|
|
52
|
+
"application/json": {
|
|
53
|
+
schema: {
|
|
54
|
+
type: "object",
|
|
55
|
+
properties: {
|
|
56
|
+
docs: { type: "array", items: { $ref: `#/components/schemas/${slug}` } },
|
|
57
|
+
total: { type: "integer" },
|
|
58
|
+
limit: { type: "integer" },
|
|
59
|
+
page: { type: "integer" },
|
|
60
|
+
},
|
|
61
|
+
},
|
|
62
|
+
},
|
|
63
|
+
},
|
|
64
|
+
},
|
|
65
|
+
},
|
|
66
|
+
},
|
|
67
|
+
post: {
|
|
68
|
+
tags: ["Collections"],
|
|
69
|
+
summary: `Create ${labels.singular}`,
|
|
70
|
+
requestBody: {
|
|
71
|
+
required: true,
|
|
72
|
+
content: {
|
|
73
|
+
"application/json": {
|
|
74
|
+
schema: { $ref: `#/components/schemas/${slug}` },
|
|
75
|
+
},
|
|
76
|
+
},
|
|
77
|
+
},
|
|
78
|
+
responses: {
|
|
79
|
+
201: {
|
|
80
|
+
description: "Created",
|
|
81
|
+
content: {
|
|
82
|
+
"application/json": {
|
|
83
|
+
schema: { $ref: `#/components/schemas/${slug}` },
|
|
84
|
+
},
|
|
85
|
+
},
|
|
86
|
+
},
|
|
87
|
+
},
|
|
88
|
+
},
|
|
89
|
+
};
|
|
90
|
+
spec.paths[`${path}/{id}`] = {
|
|
91
|
+
get: {
|
|
92
|
+
tags: ["Collections"],
|
|
93
|
+
summary: `Get a single ${labels.singular}`,
|
|
94
|
+
parameters: [{ name: "id", in: "path", required: true, schema: { type: "string" } }],
|
|
95
|
+
responses: {
|
|
96
|
+
200: {
|
|
97
|
+
description: "Success",
|
|
98
|
+
content: {
|
|
99
|
+
"application/json": {
|
|
100
|
+
schema: { $ref: `#/components/schemas/${slug}` },
|
|
101
|
+
},
|
|
102
|
+
},
|
|
103
|
+
},
|
|
104
|
+
},
|
|
105
|
+
},
|
|
106
|
+
patch: {
|
|
107
|
+
tags: ["Collections"],
|
|
108
|
+
summary: `Update ${labels.singular}`,
|
|
109
|
+
parameters: [{ name: "id", in: "path", required: true, schema: { type: "string" } }],
|
|
110
|
+
requestBody: {
|
|
111
|
+
required: true,
|
|
112
|
+
content: {
|
|
113
|
+
"application/json": {
|
|
114
|
+
schema: { $ref: `#/components/schemas/${slug}` },
|
|
115
|
+
},
|
|
116
|
+
},
|
|
117
|
+
},
|
|
118
|
+
responses: {
|
|
119
|
+
200: {
|
|
120
|
+
description: "Updated",
|
|
121
|
+
content: {
|
|
122
|
+
"application/json": {
|
|
123
|
+
schema: { $ref: `#/components/schemas/${slug}` },
|
|
124
|
+
},
|
|
125
|
+
},
|
|
126
|
+
},
|
|
127
|
+
},
|
|
128
|
+
},
|
|
129
|
+
delete: {
|
|
130
|
+
tags: ["Collections"],
|
|
131
|
+
summary: `Delete ${labels.singular}`,
|
|
132
|
+
parameters: [{ name: "id", in: "path", required: true, schema: { type: "string" } }],
|
|
133
|
+
responses: {
|
|
134
|
+
204: { description: "Deleted" },
|
|
135
|
+
},
|
|
136
|
+
},
|
|
137
|
+
};
|
|
138
|
+
}
|
|
139
|
+
// 4. Generate Paths for Globals
|
|
140
|
+
for (const global of config.globals) {
|
|
141
|
+
const slug = global.slug;
|
|
142
|
+
const path = `/api/globals/${slug}`;
|
|
143
|
+
spec.paths[path] = {
|
|
144
|
+
get: {
|
|
145
|
+
tags: ["Globals"],
|
|
146
|
+
summary: `Get ${global.label || slug}`,
|
|
147
|
+
responses: {
|
|
148
|
+
200: {
|
|
149
|
+
description: "Success",
|
|
150
|
+
content: {
|
|
151
|
+
"application/json": {
|
|
152
|
+
schema: { $ref: `#/components/schemas/${slug}` },
|
|
153
|
+
},
|
|
154
|
+
},
|
|
155
|
+
},
|
|
156
|
+
},
|
|
157
|
+
},
|
|
158
|
+
patch: {
|
|
159
|
+
tags: ["Globals"],
|
|
160
|
+
summary: `Update ${global.label || slug}`,
|
|
161
|
+
requestBody: {
|
|
162
|
+
required: true,
|
|
163
|
+
content: {
|
|
164
|
+
"application/json": {
|
|
165
|
+
schema: { $ref: `#/components/schemas/${slug}` },
|
|
166
|
+
},
|
|
167
|
+
},
|
|
168
|
+
},
|
|
169
|
+
responses: {
|
|
170
|
+
200: {
|
|
171
|
+
description: "Updated",
|
|
172
|
+
content: {
|
|
173
|
+
"application/json": {
|
|
174
|
+
schema: { $ref: `#/components/schemas/${slug}` },
|
|
175
|
+
},
|
|
176
|
+
},
|
|
177
|
+
},
|
|
178
|
+
},
|
|
179
|
+
},
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
// 5. Generate Paths for Media (if storage is configured)
|
|
183
|
+
if (config.storage) {
|
|
184
|
+
spec.paths["/api/media"] = {
|
|
185
|
+
get: {
|
|
186
|
+
tags: ["Media"],
|
|
187
|
+
summary: "List Media",
|
|
188
|
+
responses: {
|
|
189
|
+
200: {
|
|
190
|
+
description: "Success",
|
|
191
|
+
content: {
|
|
192
|
+
"application/json": {
|
|
193
|
+
schema: {
|
|
194
|
+
type: "object",
|
|
195
|
+
properties: {
|
|
196
|
+
docs: { type: "array", items: { type: "object", additionalProperties: true } },
|
|
197
|
+
},
|
|
198
|
+
},
|
|
199
|
+
},
|
|
200
|
+
},
|
|
201
|
+
},
|
|
202
|
+
},
|
|
203
|
+
},
|
|
204
|
+
post: {
|
|
205
|
+
tags: ["Media"],
|
|
206
|
+
summary: "Upload Media",
|
|
207
|
+
requestBody: {
|
|
208
|
+
content: {
|
|
209
|
+
"multipart/form-data": {
|
|
210
|
+
schema: {
|
|
211
|
+
type: "object",
|
|
212
|
+
properties: {
|
|
213
|
+
file: { type: "string", format: "binary" },
|
|
214
|
+
},
|
|
215
|
+
},
|
|
216
|
+
},
|
|
217
|
+
},
|
|
218
|
+
},
|
|
219
|
+
responses: {
|
|
220
|
+
201: { description: "Uploaded" },
|
|
221
|
+
},
|
|
222
|
+
},
|
|
223
|
+
};
|
|
224
|
+
}
|
|
225
|
+
return spec;
|
|
226
|
+
}
|
|
227
|
+
function collectionToSchema(collection) {
|
|
228
|
+
const { properties, required } = fieldsToProperties(collection.fields);
|
|
229
|
+
return {
|
|
230
|
+
type: "object",
|
|
231
|
+
properties: {
|
|
232
|
+
id: { type: "string" },
|
|
233
|
+
createdAt: { type: "string", format: "date-time" },
|
|
234
|
+
updatedAt: { type: "string", format: "date-time" },
|
|
235
|
+
...properties,
|
|
236
|
+
},
|
|
237
|
+
required: ["id", ...required],
|
|
238
|
+
};
|
|
239
|
+
}
|
|
240
|
+
function globalToSchema(global) {
|
|
241
|
+
const { properties, required } = fieldsToProperties(global.fields);
|
|
242
|
+
return {
|
|
243
|
+
type: "object",
|
|
244
|
+
properties,
|
|
245
|
+
required,
|
|
246
|
+
};
|
|
247
|
+
}
|
|
248
|
+
function fieldsToProperties(fields) {
|
|
249
|
+
const props = {};
|
|
250
|
+
const required = [];
|
|
251
|
+
for (const field of fields) {
|
|
252
|
+
if (!field.name || field.type === 'join' || field.type === 'row')
|
|
253
|
+
continue;
|
|
254
|
+
props[field.name] = fieldToSchema(field);
|
|
255
|
+
if (field.required) {
|
|
256
|
+
required.push(field.name);
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
return { properties: props, required };
|
|
260
|
+
}
|
|
261
|
+
function fieldToSchema(field) {
|
|
262
|
+
let schema = {};
|
|
263
|
+
switch (field.type) {
|
|
264
|
+
case "text":
|
|
265
|
+
case "textarea":
|
|
266
|
+
case "email":
|
|
267
|
+
case "url":
|
|
268
|
+
schema = { type: "string" };
|
|
269
|
+
break;
|
|
270
|
+
case "number":
|
|
271
|
+
schema = { type: "number" };
|
|
272
|
+
break;
|
|
273
|
+
case "boolean":
|
|
274
|
+
schema = { type: "boolean" };
|
|
275
|
+
break;
|
|
276
|
+
case "date":
|
|
277
|
+
schema = { type: "string", format: "date-time" };
|
|
278
|
+
break;
|
|
279
|
+
case "select":
|
|
280
|
+
case "radio":
|
|
281
|
+
schema = { type: "string", enum: Array.isArray(field.options) ? field.options.map((o) => (typeof o === "string" ? o : o.value)) : undefined };
|
|
282
|
+
break;
|
|
283
|
+
case "multiSelect":
|
|
284
|
+
schema = {
|
|
285
|
+
type: "array",
|
|
286
|
+
items: { type: "string", enum: Array.isArray(field.options) ? field.options.map((o) => (typeof o === "string" ? o : o.value)) : undefined },
|
|
287
|
+
};
|
|
288
|
+
break;
|
|
289
|
+
case "relationship":
|
|
290
|
+
schema = { type: 'string', description: `ID of a ${field.relationTo} record` };
|
|
291
|
+
break;
|
|
292
|
+
case "object": {
|
|
293
|
+
const { properties, required } = fieldsToProperties(field.fields || []);
|
|
294
|
+
schema = { type: "object", properties, required };
|
|
295
|
+
break;
|
|
296
|
+
}
|
|
297
|
+
case "array": {
|
|
298
|
+
const { properties, required } = fieldsToProperties(field.fields || []);
|
|
299
|
+
schema = { type: "array", items: { type: "object", properties, required } };
|
|
300
|
+
break;
|
|
301
|
+
}
|
|
302
|
+
case "json":
|
|
303
|
+
case "richText":
|
|
304
|
+
schema = { type: "object", additionalProperties: true };
|
|
305
|
+
break;
|
|
306
|
+
case "blocks":
|
|
307
|
+
schema = {
|
|
308
|
+
type: "array",
|
|
309
|
+
items: {
|
|
310
|
+
oneOf: field.blocks?.map((block) => {
|
|
311
|
+
const { properties, required } = fieldsToProperties(block.fields);
|
|
312
|
+
return {
|
|
313
|
+
type: "object",
|
|
314
|
+
properties: {
|
|
315
|
+
blockType: { type: "string", enum: [block.slug] },
|
|
316
|
+
...properties,
|
|
317
|
+
},
|
|
318
|
+
required: ["blockType", ...required],
|
|
319
|
+
};
|
|
320
|
+
}),
|
|
321
|
+
},
|
|
322
|
+
};
|
|
323
|
+
break;
|
|
324
|
+
default:
|
|
325
|
+
schema = { type: "string" };
|
|
326
|
+
}
|
|
327
|
+
if (field.label)
|
|
328
|
+
schema.description = field.label;
|
|
329
|
+
return schema;
|
|
330
|
+
}
|
|
331
|
+
//# sourceMappingURL=openapi.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"openapi.js","sourceRoot":"","sources":["../../src/utils/openapi.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Dyrected Where Clause DSL — shared query language across all database adapters.
|
|
3
|
+
*
|
|
4
|
+
* Adapters translate this DSL into their native format:
|
|
5
|
+
* - SQL adapters (SQLite, Postgres, MySQL): use parseSqlWhere()
|
|
6
|
+
* - MongoDB adapter: use parseMongoWhere()
|
|
7
|
+
* - Future adapters: implement their own translator against WhereClause
|
|
8
|
+
*
|
|
9
|
+
* Exhaustive operator handling is enforced at compile time via `assertNever`,
|
|
10
|
+
* so adding a new operator without handling it will produce a TypeScript error
|
|
11
|
+
* in every adapter translator.
|
|
12
|
+
*/
|
|
13
|
+
export type WhereOperatorName = 'equals' | 'not_equals' | 'in' | 'not_in' | 'gt' | 'gte' | 'lt' | 'lte' | 'contains' | 'starts_with' | 'exists';
|
|
14
|
+
export type WhereOperator = {
|
|
15
|
+
equals: any;
|
|
16
|
+
} | {
|
|
17
|
+
not_equals: any;
|
|
18
|
+
} | {
|
|
19
|
+
in: any[];
|
|
20
|
+
} | {
|
|
21
|
+
not_in: any[];
|
|
22
|
+
} | {
|
|
23
|
+
gt: any;
|
|
24
|
+
} | {
|
|
25
|
+
gte: any;
|
|
26
|
+
} | {
|
|
27
|
+
lt: any;
|
|
28
|
+
} | {
|
|
29
|
+
lte: any;
|
|
30
|
+
} | {
|
|
31
|
+
contains: string;
|
|
32
|
+
} | {
|
|
33
|
+
starts_with: string;
|
|
34
|
+
} | {
|
|
35
|
+
exists: boolean;
|
|
36
|
+
};
|
|
37
|
+
/** Top-level where clause. Fields map to operator objects or shorthand scalar values. */
|
|
38
|
+
export type WhereClause = {
|
|
39
|
+
[field: string]: WhereOperator | any;
|
|
40
|
+
OR?: WhereClause[];
|
|
41
|
+
AND?: WhereClause[];
|
|
42
|
+
};
|
|
43
|
+
export interface SqlWhereResult {
|
|
44
|
+
sql: string;
|
|
45
|
+
params: any[];
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Translates a WhereClause into a parameterized SQL WHERE fragment.
|
|
49
|
+
*
|
|
50
|
+
* @param where The Dyrected where DSL object
|
|
51
|
+
* @param getJsonField Returns the SQL expression for a JSON-stored field (dialect-specific):
|
|
52
|
+
* SQLite: (f) => `json_extract(data, '$.${f}')`
|
|
53
|
+
* Postgres: (f) => `data->>'${f}'`
|
|
54
|
+
* MySQL: (f) => `JSON_UNQUOTE(JSON_EXTRACT(data, '$.${f}'))`
|
|
55
|
+
* @param placeholder '?' for SQLite/MySQL, 'pg' for auto-incrementing Postgres $1/$2/…
|
|
56
|
+
*/
|
|
57
|
+
export declare function parseSqlWhere(where: WhereClause, getJsonField: (field: string) => string, placeholder?: '?' | 'pg'): SqlWhereResult;
|
|
58
|
+
/**
|
|
59
|
+
* Translates a WhereClause into a MongoDB filter object.
|
|
60
|
+
* Handles nested OR/AND via $or/$and, and all operators via $eq/$in/$gt etc.
|
|
61
|
+
*/
|
|
62
|
+
export declare function parseMongoWhere(where: WhereClause): Record<string, any>;
|
|
63
|
+
//# sourceMappingURL=parse-where.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"parse-where.d.ts","sourceRoot":"","sources":["../../src/utils/parse-where.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAIH,MAAM,MAAM,iBAAiB,GACzB,QAAQ,GACR,YAAY,GACZ,IAAI,GACJ,QAAQ,GACR,IAAI,GACJ,KAAK,GACL,IAAI,GACJ,KAAK,GACL,UAAU,GACV,aAAa,GACb,QAAQ,CAAC;AAEb,MAAM,MAAM,aAAa,GACrB;IAAE,MAAM,EAAE,GAAG,CAAA;CAAE,GACf;IAAE,UAAU,EAAE,GAAG,CAAA;CAAE,GACnB;IAAE,EAAE,EAAE,GAAG,EAAE,CAAA;CAAE,GACb;IAAE,MAAM,EAAE,GAAG,EAAE,CAAA;CAAE,GACjB;IAAE,EAAE,EAAE,GAAG,CAAA;CAAE,GACX;IAAE,GAAG,EAAE,GAAG,CAAA;CAAE,GACZ;IAAE,EAAE,EAAE,GAAG,CAAA;CAAE,GACX;IAAE,GAAG,EAAE,GAAG,CAAA;CAAE,GACZ;IAAE,QAAQ,EAAE,MAAM,CAAA;CAAE,GACpB;IAAE,WAAW,EAAE,MAAM,CAAA;CAAE,GACvB;IAAE,MAAM,EAAE,OAAO,CAAA;CAAE,CAAC;AAExB,yFAAyF;AACzF,MAAM,MAAM,WAAW,GAAG;IACxB,CAAC,KAAK,EAAE,MAAM,GAAG,aAAa,GAAG,GAAG,CAAC;IACrC,EAAE,CAAC,EAAE,WAAW,EAAE,CAAC;IACnB,GAAG,CAAC,EAAE,WAAW,EAAE,CAAC;CACrB,CAAC;AAUF,MAAM,WAAW,cAAc;IAC7B,GAAG,EAAE,MAAM,CAAC;IACZ,MAAM,EAAE,GAAG,EAAE,CAAC;CACf;AAED;;;;;;;;;GASG;AACH,wBAAgB,aAAa,CAC3B,KAAK,EAAE,WAAW,EAClB,YAAY,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,MAAM,EACvC,WAAW,GAAE,GAAG,GAAG,IAAU,GAC5B,cAAc,CA6GhB;AAKD;;;GAGG;AACH,wBAAgB,eAAe,CAAC,KAAK,EAAE,WAAW,GAAG,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAkFvE"}
|
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Dyrected Where Clause DSL — shared query language across all database adapters.
|
|
3
|
+
*
|
|
4
|
+
* Adapters translate this DSL into their native format:
|
|
5
|
+
* - SQL adapters (SQLite, Postgres, MySQL): use parseSqlWhere()
|
|
6
|
+
* - MongoDB adapter: use parseMongoWhere()
|
|
7
|
+
* - Future adapters: implement their own translator against WhereClause
|
|
8
|
+
*
|
|
9
|
+
* Exhaustive operator handling is enforced at compile time via `assertNever`,
|
|
10
|
+
* so adding a new operator without handling it will produce a TypeScript error
|
|
11
|
+
* in every adapter translator.
|
|
12
|
+
*/
|
|
13
|
+
/** Compile-time exhaustiveness guard. Any unhandled operator becomes a type error. */
|
|
14
|
+
function assertNever(op, context) {
|
|
15
|
+
throw new Error(`[dyrected/core] Unhandled where operator "${op}" in ${context}`);
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Translates a WhereClause into a parameterized SQL WHERE fragment.
|
|
19
|
+
*
|
|
20
|
+
* @param where The Dyrected where DSL object
|
|
21
|
+
* @param getJsonField Returns the SQL expression for a JSON-stored field (dialect-specific):
|
|
22
|
+
* SQLite: (f) => `json_extract(data, '$.${f}')`
|
|
23
|
+
* Postgres: (f) => `data->>'${f}'`
|
|
24
|
+
* MySQL: (f) => `JSON_UNQUOTE(JSON_EXTRACT(data, '$.${f}'))`
|
|
25
|
+
* @param placeholder '?' for SQLite/MySQL, 'pg' for auto-incrementing Postgres $1/$2/…
|
|
26
|
+
*/
|
|
27
|
+
export function parseSqlWhere(where, getJsonField, placeholder = '?') {
|
|
28
|
+
const params = [];
|
|
29
|
+
let pgIndex = 1;
|
|
30
|
+
function next() {
|
|
31
|
+
return placeholder === 'pg' ? `$${pgIndex++}` : '?';
|
|
32
|
+
}
|
|
33
|
+
function col(field) {
|
|
34
|
+
return field === 'id' ? 'id' : getJsonField(field);
|
|
35
|
+
}
|
|
36
|
+
function buildOperator(field, value) {
|
|
37
|
+
const c = col(field);
|
|
38
|
+
// Shorthand scalar: { slug: 'hello' } → treat as equals
|
|
39
|
+
if (typeof value !== 'object' || value === null || Array.isArray(value)) {
|
|
40
|
+
params.push(value);
|
|
41
|
+
return `${c} = ${next()}`;
|
|
42
|
+
}
|
|
43
|
+
const entries = Object.entries(value);
|
|
44
|
+
if (entries.length !== 1) {
|
|
45
|
+
// Multiple keys inside an operator object — treat as AND of conditions on same field
|
|
46
|
+
return entries
|
|
47
|
+
.map(([op, operand]) => buildSingleOp(c, op, operand))
|
|
48
|
+
.join(' AND ');
|
|
49
|
+
}
|
|
50
|
+
const [op, operand] = entries[0];
|
|
51
|
+
return buildSingleOp(c, op, operand);
|
|
52
|
+
}
|
|
53
|
+
function buildSingleOp(c, op, operand) {
|
|
54
|
+
switch (op) {
|
|
55
|
+
case 'equals':
|
|
56
|
+
params.push(typeof operand === 'boolean' ? String(operand) : operand);
|
|
57
|
+
return `${c} = ${next()}`;
|
|
58
|
+
case 'not_equals':
|
|
59
|
+
params.push(typeof operand === 'boolean' ? String(operand) : operand);
|
|
60
|
+
return `${c} != ${next()}`;
|
|
61
|
+
case 'in': {
|
|
62
|
+
const vals = Array.isArray(operand) ? operand : [operand];
|
|
63
|
+
if (vals.length === 0)
|
|
64
|
+
return '1=0'; // IN () is invalid SQL
|
|
65
|
+
const placeholders = vals.map((v) => { params.push(v); return next(); });
|
|
66
|
+
return `${c} IN (${placeholders.join(', ')})`;
|
|
67
|
+
}
|
|
68
|
+
case 'not_in': {
|
|
69
|
+
const vals = Array.isArray(operand) ? operand : [operand];
|
|
70
|
+
if (vals.length === 0)
|
|
71
|
+
return '1=1';
|
|
72
|
+
const placeholders = vals.map((v) => { params.push(v); return next(); });
|
|
73
|
+
return `${c} NOT IN (${placeholders.join(', ')})`;
|
|
74
|
+
}
|
|
75
|
+
case 'gt':
|
|
76
|
+
params.push(operand);
|
|
77
|
+
return `${c} > ${next()}`;
|
|
78
|
+
case 'gte':
|
|
79
|
+
params.push(operand);
|
|
80
|
+
return `${c} >= ${next()}`;
|
|
81
|
+
case 'lt':
|
|
82
|
+
params.push(operand);
|
|
83
|
+
return `${c} < ${next()}`;
|
|
84
|
+
case 'lte':
|
|
85
|
+
params.push(operand);
|
|
86
|
+
return `${c} <= ${next()}`;
|
|
87
|
+
case 'contains':
|
|
88
|
+
params.push(`%${operand}%`);
|
|
89
|
+
return `${c} LIKE ${next()}`;
|
|
90
|
+
case 'starts_with':
|
|
91
|
+
params.push(`${operand}%`);
|
|
92
|
+
return `${c} LIKE ${next()}`;
|
|
93
|
+
case 'exists':
|
|
94
|
+
return operand ? `${c} IS NOT NULL` : `${c} IS NULL`;
|
|
95
|
+
default:
|
|
96
|
+
// If you hit this, you added an operator to WhereOperatorName but forgot to handle it here.
|
|
97
|
+
return assertNever(op, 'parseSqlWhere');
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
function buildClause(w) {
|
|
101
|
+
const parts = [];
|
|
102
|
+
for (const [field, value] of Object.entries(w)) {
|
|
103
|
+
const upperField = field.toUpperCase();
|
|
104
|
+
if (upperField === 'OR' && Array.isArray(value)) {
|
|
105
|
+
const sub = value.map((v) => `(${buildClause(v)})`).join(' OR ');
|
|
106
|
+
parts.push(`(${sub})`);
|
|
107
|
+
}
|
|
108
|
+
else if (upperField === 'AND' && Array.isArray(value)) {
|
|
109
|
+
const sub = value.map((v) => `(${buildClause(v)})`).join(' AND ');
|
|
110
|
+
parts.push(`(${sub})`);
|
|
111
|
+
}
|
|
112
|
+
else {
|
|
113
|
+
parts.push(buildOperator(field, value));
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
return parts.length ? parts.join(' AND ') : '1=1';
|
|
117
|
+
}
|
|
118
|
+
const sql = buildClause(where);
|
|
119
|
+
return { sql, params };
|
|
120
|
+
}
|
|
121
|
+
// ─── MongoDB Translator ──────────────────────────────────────────────────────
|
|
122
|
+
// Used by: @dyrected/db-mongodb
|
|
123
|
+
/**
|
|
124
|
+
* Translates a WhereClause into a MongoDB filter object.
|
|
125
|
+
* Handles nested OR/AND via $or/$and, and all operators via $eq/$in/$gt etc.
|
|
126
|
+
*/
|
|
127
|
+
export function parseMongoWhere(where) {
|
|
128
|
+
function buildOperator(field, value) {
|
|
129
|
+
// Shorthand scalar
|
|
130
|
+
if (typeof value !== 'object' || value === null || Array.isArray(value)) {
|
|
131
|
+
return { [field]: { $eq: value } };
|
|
132
|
+
}
|
|
133
|
+
const entries = Object.entries(value);
|
|
134
|
+
if (entries.length !== 1) {
|
|
135
|
+
// Multiple operators on same field — merge into one field condition
|
|
136
|
+
const merged = {};
|
|
137
|
+
for (const [op, operand] of entries) {
|
|
138
|
+
Object.assign(merged, buildSingleOp(field, op, operand)[field]);
|
|
139
|
+
}
|
|
140
|
+
return { [field]: merged };
|
|
141
|
+
}
|
|
142
|
+
const [op, operand] = entries[0];
|
|
143
|
+
return buildSingleOp(field, op, operand);
|
|
144
|
+
}
|
|
145
|
+
function buildSingleOp(field, op, operand) {
|
|
146
|
+
switch (op) {
|
|
147
|
+
case 'equals':
|
|
148
|
+
return { [field]: { $eq: operand } };
|
|
149
|
+
case 'not_equals':
|
|
150
|
+
return { [field]: { $ne: operand } };
|
|
151
|
+
case 'in':
|
|
152
|
+
return { [field]: { $in: Array.isArray(operand) ? operand : [operand] } };
|
|
153
|
+
case 'not_in':
|
|
154
|
+
return { [field]: { $nin: Array.isArray(operand) ? operand : [operand] } };
|
|
155
|
+
case 'gt':
|
|
156
|
+
return { [field]: { $gt: operand } };
|
|
157
|
+
case 'gte':
|
|
158
|
+
return { [field]: { $gte: operand } };
|
|
159
|
+
case 'lt':
|
|
160
|
+
return { [field]: { $lt: operand } };
|
|
161
|
+
case 'lte':
|
|
162
|
+
return { [field]: { $lte: operand } };
|
|
163
|
+
case 'contains':
|
|
164
|
+
return { [field]: { $regex: operand, $options: 'i' } };
|
|
165
|
+
case 'starts_with':
|
|
166
|
+
return { [field]: { $regex: `^${operand}`, $options: 'i' } };
|
|
167
|
+
case 'exists':
|
|
168
|
+
return { [field]: { $exists: operand } };
|
|
169
|
+
default:
|
|
170
|
+
// Compile-time exhaustiveness: adding an operator without handling it here = build error.
|
|
171
|
+
return assertNever(op, 'parseMongoWhere');
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
function buildClause(w) {
|
|
175
|
+
const conditions = [];
|
|
176
|
+
for (const [field, value] of Object.entries(w)) {
|
|
177
|
+
const upperField = field.toUpperCase();
|
|
178
|
+
if (upperField === 'OR' && Array.isArray(value)) {
|
|
179
|
+
conditions.push({ $or: value.map(buildClause) });
|
|
180
|
+
}
|
|
181
|
+
else if (upperField === 'AND' && Array.isArray(value)) {
|
|
182
|
+
conditions.push({ $and: value.map(buildClause) });
|
|
183
|
+
}
|
|
184
|
+
else {
|
|
185
|
+
conditions.push(buildOperator(field, value));
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
if (conditions.length === 0)
|
|
189
|
+
return {};
|
|
190
|
+
if (conditions.length === 1)
|
|
191
|
+
return conditions[0];
|
|
192
|
+
return { $and: conditions };
|
|
193
|
+
}
|
|
194
|
+
return buildClause(where);
|
|
195
|
+
}
|
|
196
|
+
//# sourceMappingURL=parse-where.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"parse-where.js","sourceRoot":"","sources":["../../src/utils/parse-where.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import type { DatabaseAdapter, ReadonlyDatabaseAdapter } from "../types/index.js";
|
|
2
|
+
/**
|
|
3
|
+
* Wrap a DatabaseAdapter so that write operations throw at runtime.
|
|
4
|
+
*
|
|
5
|
+
* Passed to `beforeChange`, `beforeDelete`, `beforeRead`, `afterRead`,
|
|
6
|
+
* and field-level hooks — phases where only reads are permitted.
|
|
7
|
+
*/
|
|
8
|
+
export declare function createReadonlyDb(db: DatabaseAdapter): ReadonlyDatabaseAdapter;
|
|
9
|
+
//# sourceMappingURL=readonly-db.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"readonly-db.d.ts","sourceRoot":"","sources":["../../src/utils/readonly-db.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,uBAAuB,EAAE,MAAM,mBAAmB,CAAC;AAIlF;;;;;GAKG;AACH,wBAAgB,gBAAgB,CAAC,EAAE,EAAE,eAAe,GAAG,uBAAuB,CAc7E"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
const WRITE_METHODS = ["create", "update", "delete", "updateGlobal", "execute"];
|
|
2
|
+
/**
|
|
3
|
+
* Wrap a DatabaseAdapter so that write operations throw at runtime.
|
|
4
|
+
*
|
|
5
|
+
* Passed to `beforeChange`, `beforeDelete`, `beforeRead`, `afterRead`,
|
|
6
|
+
* and field-level hooks — phases where only reads are permitted.
|
|
7
|
+
*/
|
|
8
|
+
export function createReadonlyDb(db) {
|
|
9
|
+
return new Proxy(db, {
|
|
10
|
+
get(target, prop) {
|
|
11
|
+
if (WRITE_METHODS.includes(prop)) {
|
|
12
|
+
return () => {
|
|
13
|
+
throw new Error(`[dyrected] Write operation "${String(prop)}" is not allowed in this hook phase. ` +
|
|
14
|
+
`Use afterChange/afterDelete hooks for write operations.`);
|
|
15
|
+
};
|
|
16
|
+
}
|
|
17
|
+
return Reflect.get(target, prop);
|
|
18
|
+
},
|
|
19
|
+
});
|
|
20
|
+
}
|
|
21
|
+
//# sourceMappingURL=readonly-db.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"readonly-db.js","sourceRoot":"","sources":["../../src/utils/readonly-db.ts"],"names":[],"mappings":"AAEA,MAAM,aAAa,GAAG,CAAC,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,cAAc,EAAE,SAAS,CAAC,CAAC;AAEhF;;;;;GAKG;AACH,MAAM,UAAU,gBAAgB,CAAC,EAAmB;IAClD,OAAO,IAAI,KAAK,CAAC,EAAE,EAAE;QACnB,GAAG,CAAC,MAAM,EAAE,IAAI;YACd,IAAI,aAAa,CAAC,QAAQ,CAAC,IAAc,CAAC,EAAE,CAAC;gBAC3C,OAAO,GAAG,EAAE;oBACV,MAAM,IAAI,KAAK,CACb,+BAA+B,MAAM,CAAC,IAAI,CAAC,uCAAuC;wBAClF,yDAAyD,CAC1D,CAAC;gBACJ,CAAC,CAAC;YACJ,CAAC;YACD,OAAO,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;QACnC,CAAC;KACF,CAAC,CAAC;AACL,CAAC"}
|