appflare 0.2.30 → 0.2.31
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/Documentation.md +758 -758
- package/cli/commands/index.ts +238 -238
- package/cli/generate.ts +178 -178
- package/cli/index.ts +120 -120
- package/cli/load-config.ts +184 -184
- package/cli/schema-compiler.ts +1183 -1183
- package/cli/templates/auth/README.md +156 -156
- package/cli/templates/auth/config.ts +61 -61
- package/cli/templates/auth/route-config.ts +1 -1
- package/cli/templates/auth/route-handler.ts +1 -1
- package/cli/templates/auth/route-request-utils.ts +5 -5
- package/cli/templates/auth/route.config.ts +18 -18
- package/cli/templates/auth/route.handler.ts +18 -18
- package/cli/templates/auth/route.request-utils.ts +55 -55
- package/cli/templates/auth/route.ts +14 -14
- package/cli/templates/core/README.md +266 -266
- package/cli/templates/core/app-creation.ts +19 -19
- package/cli/templates/core/client/appflare.ts +112 -112
- package/cli/templates/core/client/handlers/index.ts +748 -748
- package/cli/templates/core/client/handlers.ts +1 -1
- package/cli/templates/core/client/index.ts +7 -7
- package/cli/templates/core/client/storage.ts +195 -195
- package/cli/templates/core/client/types.ts +186 -186
- package/cli/templates/core/client-modules/appflare.ts +1 -1
- package/cli/templates/core/client-modules/handlers.ts +1 -1
- package/cli/templates/core/client-modules/index.ts +1 -1
- package/cli/templates/core/client-modules/storage.ts +1 -1
- package/cli/templates/core/client-modules/types.ts +1 -1
- package/cli/templates/core/client.artifacts.ts +39 -39
- package/cli/templates/core/client.ts +4 -4
- package/cli/templates/core/drizzle.ts +15 -15
- package/cli/templates/core/export.ts +14 -14
- package/cli/templates/core/handlers.route.ts +24 -24
- package/cli/templates/core/handlers.ts +1 -1
- package/cli/templates/core/imports.ts +9 -9
- package/cli/templates/core/server.ts +38 -38
- package/cli/templates/core/types.ts +6 -6
- package/cli/templates/core/wrangler.ts +109 -109
- package/cli/templates/dashboard/builders/functions/index.ts +17 -17
- package/cli/templates/dashboard/builders/functions/render-page/header.ts +20 -20
- package/cli/templates/dashboard/builders/functions/render-page/index.ts +33 -33
- package/cli/templates/dashboard/builders/functions/render-page/request-panel.ts +171 -171
- package/cli/templates/dashboard/builders/functions/render-page/result-panel.ts +85 -85
- package/cli/templates/dashboard/builders/functions/render-page/scripts.ts +554 -554
- package/cli/templates/dashboard/builders/navigation.ts +122 -122
- package/cli/templates/dashboard/builders/storage/index.ts +13 -13
- package/cli/templates/dashboard/builders/storage/routes/create-directory-route.ts +29 -29
- package/cli/templates/dashboard/builders/storage/routes/delete-route.ts +18 -18
- package/cli/templates/dashboard/builders/storage/routes/download-route.ts +23 -23
- package/cli/templates/dashboard/builders/storage/routes/index.ts +22 -22
- package/cli/templates/dashboard/builders/storage/routes/list-route.ts +25 -25
- package/cli/templates/dashboard/builders/storage/routes/preview-route.ts +21 -21
- package/cli/templates/dashboard/builders/storage/routes/upload-route.ts +21 -21
- package/cli/templates/dashboard/builders/storage/runtime/helpers.ts +72 -72
- package/cli/templates/dashboard/builders/storage/runtime/storage-page.ts +130 -130
- package/cli/templates/dashboard/builders/table-routes/common/drawer-panel.ts +27 -27
- package/cli/templates/dashboard/builders/table-routes/common/pagination.ts +30 -30
- package/cli/templates/dashboard/builders/table-routes/common/search-bar.ts +23 -23
- package/cli/templates/dashboard/builders/table-routes/fragments.ts +217 -217
- package/cli/templates/dashboard/builders/table-routes/helpers.ts +45 -45
- package/cli/templates/dashboard/builders/table-routes/index.ts +8 -8
- package/cli/templates/dashboard/builders/table-routes/table/actions-cell.ts +71 -71
- package/cli/templates/dashboard/builders/table-routes/table/get-route.ts +291 -291
- package/cli/templates/dashboard/builders/table-routes/table/index.ts +80 -80
- package/cli/templates/dashboard/builders/table-routes/table/post-routes.ts +163 -163
- package/cli/templates/dashboard/builders/table-routes/table-route.ts +7 -7
- package/cli/templates/dashboard/builders/table-routes/users/get-route.ts +69 -69
- package/cli/templates/dashboard/builders/table-routes/users/html/modals.ts +57 -57
- package/cli/templates/dashboard/builders/table-routes/users/html/page.ts +27 -27
- package/cli/templates/dashboard/builders/table-routes/users/html/table.ts +128 -128
- package/cli/templates/dashboard/builders/table-routes/users/index.ts +32 -32
- package/cli/templates/dashboard/builders/table-routes/users/post-routes.ts +150 -150
- package/cli/templates/dashboard/builders/table-routes/users/redirect.ts +14 -14
- package/cli/templates/dashboard/builders/table-routes/users-route.ts +10 -10
- package/cli/templates/dashboard/components/dashboard-home.ts +23 -23
- package/cli/templates/dashboard/components/layout.ts +388 -388
- package/cli/templates/dashboard/components/login-page.ts +65 -65
- package/cli/templates/dashboard/index.ts +61 -61
- package/cli/templates/dashboard/types.ts +9 -9
- package/cli/templates/handlers/README.md +353 -353
- package/cli/templates/handlers/auth.ts +37 -37
- package/cli/templates/handlers/execution.ts +42 -42
- package/cli/templates/handlers/generators/context/context-creation.ts +101 -101
- package/cli/templates/handlers/generators/context/error-helpers.ts +11 -11
- package/cli/templates/handlers/generators/context/scheduler.ts +24 -24
- package/cli/templates/handlers/generators/context/storage-api.ts +82 -82
- package/cli/templates/handlers/generators/context/storage-helpers.ts +59 -59
- package/cli/templates/handlers/generators/context/types.ts +40 -40
- package/cli/templates/handlers/generators/context.ts +43 -43
- package/cli/templates/handlers/generators/execution.ts +15 -15
- package/cli/templates/handlers/generators/handlers.ts +13 -13
- package/cli/templates/handlers/generators/registration/modules/cron.ts +26 -26
- package/cli/templates/handlers/generators/registration/modules/realtime/auth.ts +75 -75
- package/cli/templates/handlers/generators/registration/modules/realtime/durable-object.ts +144 -144
- package/cli/templates/handlers/generators/registration/modules/realtime/index.ts +14 -14
- package/cli/templates/handlers/generators/registration/modules/realtime/publisher.ts +102 -102
- package/cli/templates/handlers/generators/registration/modules/realtime/routes.ts +164 -164
- package/cli/templates/handlers/generators/registration/modules/realtime/types.ts +30 -30
- package/cli/templates/handlers/generators/registration/modules/realtime/utils.ts +516 -516
- package/cli/templates/handlers/generators/registration/modules/scheduler.ts +56 -56
- package/cli/templates/handlers/generators/registration/modules/storage.ts +199 -199
- package/cli/templates/handlers/generators/registration/sections.ts +210 -210
- package/cli/templates/handlers/generators/types/context.ts +92 -92
- package/cli/templates/handlers/generators/types/core.ts +106 -106
- package/cli/templates/handlers/generators/types/operations.ts +135 -135
- package/cli/templates/handlers/generators/types/query-definitions/filter-and-where-types.ts +281 -259
- package/cli/templates/handlers/generators/types/query-definitions/query-api-types.ts +135 -135
- package/cli/templates/handlers/generators/types/query-definitions/query-helper-functions.ts +1103 -1031
- package/cli/templates/handlers/generators/types/query-definitions/schema-and-table-types.ts +278 -246
- package/cli/templates/handlers/generators/types/query-definitions.ts +13 -13
- package/cli/templates/handlers/generators/types/query-runtime/handled-error.ts +13 -13
- package/cli/templates/handlers/generators/types/query-runtime/runtime-aggregate-and-footer.ts +174 -174
- package/cli/templates/handlers/generators/types/query-runtime/runtime-read.ts +157 -121
- package/cli/templates/handlers/generators/types/query-runtime/runtime-setup.ts +45 -45
- package/cli/templates/handlers/generators/types/query-runtime/runtime-write.ts +697 -697
- package/cli/templates/handlers/generators/types/query-runtime.ts +15 -15
- package/cli/templates/handlers/index.ts +43 -43
- package/cli/templates/handlers/operations.ts +116 -116
- package/cli/templates/handlers/registration.ts +91 -91
- package/cli/templates/handlers/types.ts +15 -15
- package/cli/templates/handlers/utils.ts +48 -48
- package/cli/types.ts +110 -110
- package/cli/utils/handler-discovery.ts +466 -466
- package/cli/utils/json-utils.ts +24 -24
- package/cli/utils/path-utils.ts +19 -19
- package/cli/utils/schema-discovery.ts +399 -399
- package/dist/cli/index.d.mts +2 -0
- package/dist/cli/index.d.ts +2 -0
- package/dist/cli/index.js +270 -108
- package/dist/cli/index.mjs +270 -108
- package/index.ts +18 -18
- package/package.json +58 -58
- package/react/index.ts +5 -5
- package/react/use-infinite-query.ts +252 -252
- package/react/use-mutation.ts +89 -89
- package/react/use-query.ts +207 -207
- package/schema.ts +415 -415
- package/test-better-auth-hash.ts +2 -2
- package/tsconfig.json +6 -6
- package/tsup.config.ts +82 -82
|
@@ -1,217 +1,217 @@
|
|
|
1
|
-
import { DiscoveredTable } from "../../types";
|
|
2
|
-
|
|
3
|
-
export function buildSearchConditions(table: DiscoveredTable): string {
|
|
4
|
-
return table.columns
|
|
5
|
-
.filter((column) => column.type === "string")
|
|
6
|
-
.map(
|
|
7
|
-
(column) => `
|
|
8
|
-
\t\t\ttry { searchConditions.push(like(tableSchema.${column.name}, \`%\${search}%\`)); } catch (e) {}
|
|
9
|
-
\t\t\t`,
|
|
10
|
-
)
|
|
11
|
-
.join("");
|
|
12
|
-
}
|
|
13
|
-
|
|
14
|
-
/**
|
|
15
|
-
* Returns an Iconify icon name for a column type.
|
|
16
|
-
*/
|
|
17
|
-
function columnTypeIcon(type: string): string {
|
|
18
|
-
switch (type) {
|
|
19
|
-
case "number":
|
|
20
|
-
return "mdi:pound";
|
|
21
|
-
case "boolean":
|
|
22
|
-
return "mdi:toggle-switch-outline";
|
|
23
|
-
case "date":
|
|
24
|
-
return "mdi:calendar";
|
|
25
|
-
default:
|
|
26
|
-
return "mdi:format-text";
|
|
27
|
-
}
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
export function buildColumnHeaders(
|
|
31
|
-
table: DiscoveredTable,
|
|
32
|
-
columns: string[],
|
|
33
|
-
): string {
|
|
34
|
-
return columns
|
|
35
|
-
.map((column) => {
|
|
36
|
-
const col = table.columns.find((c) => c.name === column);
|
|
37
|
-
const icon = col ? columnTypeIcon(col.type) : "mdi:format-text";
|
|
38
|
-
return `
|
|
39
|
-
\t\t\t\t\t\t\t<th>
|
|
40
|
-
\t\t\t\t\t\t\t\t<a href="#"
|
|
41
|
-
\t\t\t\t\t\t\t\t hx-get="/admin/table/${table.exportName}?page=\${page}&search=\${search}&sort=${column}&order=\${sort === '${column}' && order === 'asc' ? 'desc' : 'asc'}"
|
|
42
|
-
\t\t\t\t\t\t\t\t hx-target="#main-content"
|
|
43
|
-
\t\t\t\t\t\t\t\t hx-push-url="true"
|
|
44
|
-
\t\t\t\t\t\t\t\t class="hover:text-primary flex items-center gap-1.5 transition-colors whitespace-nowrap">
|
|
45
|
-
\t\t\t\t\t\t\t\t <iconify-icon icon="${icon}" width="14" height="14" class="opacity-40"></iconify-icon>
|
|
46
|
-
\t\t\t\t\t\t\t\t ${column}
|
|
47
|
-
\t\t\t\t\t\t\t\t <span class="text-[10px] opacity-30">\${sort === '${column}' ? (order === 'asc' ? '▲' : '▼') : ''}</span>
|
|
48
|
-
\t\t\t\t\t\t\t\t</a>
|
|
49
|
-
\t\t\t\t\t\t\t</th>
|
|
50
|
-
\t\t\t\t\t\t\t`;
|
|
51
|
-
})
|
|
52
|
-
.join("");
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
export function buildRowCells(columns: string[], primaryKey?: string): string {
|
|
56
|
-
return columns
|
|
57
|
-
.map((column) => {
|
|
58
|
-
if (primaryKey && column === primaryKey) {
|
|
59
|
-
return `<td><button type="button" class="truncate max-w-[200px] text-sm font-mono text-xs opacity-70 hover:opacity-100 cursor-copy text-left" title="Click to copy: \${String((row as any).${column} ?? '')}" data-copy-value="\${String((row as any).${column} ?? '')}" onclick="navigator.clipboard?.writeText(this.dataset.copyValue || '')">\${String((row as any).${column} ?? '')}</button></td>`;
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
return `<td><div class="truncate max-w-[200px] text-sm" title="\${String((row as any).${column} ?? '')}">\${String((row as any).${column} ?? '')}</div></td>`;
|
|
63
|
-
})
|
|
64
|
-
.join("");
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
export function buildFieldInput(
|
|
68
|
-
table: DiscoveredTable,
|
|
69
|
-
columnName: string,
|
|
70
|
-
mode: "create" | "edit",
|
|
71
|
-
): string {
|
|
72
|
-
const column = table.columns.find((item) => item.name === columnName);
|
|
73
|
-
if (!column) {
|
|
74
|
-
return "";
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
const requiredAttr = !column.optional ? " required" : "";
|
|
78
|
-
const inputType =
|
|
79
|
-
column.type === "number"
|
|
80
|
-
? "number"
|
|
81
|
-
: column.type === "date"
|
|
82
|
-
? "date"
|
|
83
|
-
: "text";
|
|
84
|
-
|
|
85
|
-
if (column.type === "boolean") {
|
|
86
|
-
if (mode === "edit") {
|
|
87
|
-
return `
|
|
88
|
-
\t\t<div class="form-control">
|
|
89
|
-
\t\t\t<label class="label cursor-pointer justify-start gap-3">
|
|
90
|
-
\t\t\t\t<input type="checkbox" name="${columnName}" value="true" class="checkbox checkbox-sm checkbox-primary" \${(row as any).${columnName} ? 'checked' : ''} />
|
|
91
|
-
\t\t\t\t<span class="label-text text-sm">${columnName}</span>
|
|
92
|
-
\t\t\t</label>
|
|
93
|
-
\t\t</div>
|
|
94
|
-
\t\t`;
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
return `
|
|
98
|
-
\t\t<div class="form-control">
|
|
99
|
-
\t\t\t<label class="label cursor-pointer justify-start gap-3">
|
|
100
|
-
\t\t\t\t<input type="checkbox" name="${columnName}" value="true" class="checkbox checkbox-sm checkbox-primary" />
|
|
101
|
-
\t\t\t\t<span class="label-text text-sm">${columnName}</span>
|
|
102
|
-
\t\t\t</label>
|
|
103
|
-
\t\t</div>
|
|
104
|
-
\t\t`;
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
if (mode === "edit") {
|
|
108
|
-
const valueExpression =
|
|
109
|
-
column.type === "date"
|
|
110
|
-
? `\${(() => {
|
|
111
|
-
const value = (row as any).${columnName};
|
|
112
|
-
if (value == null || value === '') return '';
|
|
113
|
-
|
|
114
|
-
const date =
|
|
115
|
-
value instanceof Date
|
|
116
|
-
? value
|
|
117
|
-
: new Date(typeof value === 'number' ? value : String(value));
|
|
118
|
-
if (Number.isNaN(date.getTime())) return '';
|
|
119
|
-
|
|
120
|
-
const year = date.getFullYear();
|
|
121
|
-
const month = String(date.getMonth() + 1).padStart(2, '0');
|
|
122
|
-
const day = String(date.getDate()).padStart(2, '0');
|
|
123
|
-
return String(year) + '-' + month + '-' + day;
|
|
124
|
-
})()}`
|
|
125
|
-
: `\${String((row as any).${columnName} ?? '')}`;
|
|
126
|
-
|
|
127
|
-
return `
|
|
128
|
-
\t\t<div class="form-control">
|
|
129
|
-
\t\t\t<label class="label"><span class="label-text text-sm font-medium">${columnName}</span></label>
|
|
130
|
-
\t\t\t<input type="${inputType}" name="${columnName}" class="input input-bordered w-full text-sm" value="${valueExpression}"${requiredAttr} />
|
|
131
|
-
\t\t</div>
|
|
132
|
-
\t\t`;
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
return `
|
|
136
|
-
\t\t<div class="form-control">
|
|
137
|
-
\t\t\t<label class="label"><span class="label-text text-sm font-medium">${columnName}</span></label>
|
|
138
|
-
\t\t\t<input type="${inputType}" name="${columnName}" class="input input-bordered w-full text-sm"${requiredAttr} />
|
|
139
|
-
\t\t</div>
|
|
140
|
-
\t\t`;
|
|
141
|
-
}
|
|
142
|
-
|
|
143
|
-
export function buildPayloadAssignments(
|
|
144
|
-
table: DiscoveredTable,
|
|
145
|
-
fields: string[],
|
|
146
|
-
): string {
|
|
147
|
-
return fields
|
|
148
|
-
.map((field) => {
|
|
149
|
-
const column = table.columns.find((item) => item.name === field);
|
|
150
|
-
if (!column) {
|
|
151
|
-
return "";
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
const requiredCheck = !column.optional
|
|
155
|
-
? `
|
|
156
|
-
\t\tif (raw_${field} === '') {
|
|
157
|
-
\t\t\treturn c.text('${field} is required', 400);
|
|
158
|
-
\t\t}
|
|
159
|
-
\t\t`
|
|
160
|
-
: "";
|
|
161
|
-
|
|
162
|
-
if (column.type === "number") {
|
|
163
|
-
return `
|
|
164
|
-
\t\tconst raw_${field} = getValue(body['${field}']);
|
|
165
|
-
\t\t${requiredCheck}
|
|
166
|
-
\t\tif (raw_${field} !== '') {
|
|
167
|
-
\t\t\tconst parsed_${field} = Number(raw_${field});
|
|
168
|
-
\t\t\tif (Number.isNaN(parsed_${field})) {
|
|
169
|
-
\t\t\t\treturn c.text('${field} must be a valid number', 400);
|
|
170
|
-
\t\t\t}
|
|
171
|
-
\t\t\tpayload.${field} = parsed_${field};
|
|
172
|
-
\t\t}
|
|
173
|
-
\t\t`;
|
|
174
|
-
}
|
|
175
|
-
|
|
176
|
-
if (column.type === "boolean") {
|
|
177
|
-
return `
|
|
178
|
-
\t\tconst raw_${field} = getValue(body['${field}']);
|
|
179
|
-
\t\tpayload.${field} = raw_${field} === 'true' || raw_${field} === 'on' || raw_${field} === '1';
|
|
180
|
-
\t\t`;
|
|
181
|
-
}
|
|
182
|
-
|
|
183
|
-
if (column.type === "date") {
|
|
184
|
-
return `
|
|
185
|
-
const raw_${field} = getValue(body['${field}']);
|
|
186
|
-
${requiredCheck}
|
|
187
|
-
if (raw_${field} !== '') {
|
|
188
|
-
if (!/^\\d{4}-\\d{2}-\\d{2}$/.test(raw_${field})) {
|
|
189
|
-
return c.text('${field} must be a valid date (YYYY-MM-DD)', 400);
|
|
190
|
-
}
|
|
191
|
-
|
|
192
|
-
const [year_${field}, month_${field}, day_${field}] = raw_${field}.split('-').map(Number);
|
|
193
|
-
const parsed_${field} = new Date(year_${field}, month_${field} - 1, day_${field});
|
|
194
|
-
if (
|
|
195
|
-
Number.isNaN(parsed_${field}.getTime()) ||
|
|
196
|
-
parsed_${field}.getFullYear() !== year_${field} ||
|
|
197
|
-
parsed_${field}.getMonth() !== month_${field} - 1 ||
|
|
198
|
-
parsed_${field}.getDate() !== day_${field}
|
|
199
|
-
) {
|
|
200
|
-
return c.text('${field} must be a valid date', 400);
|
|
201
|
-
}
|
|
202
|
-
|
|
203
|
-
payload.${field} = parsed_${field};
|
|
204
|
-
}
|
|
205
|
-
`;
|
|
206
|
-
}
|
|
207
|
-
|
|
208
|
-
return `
|
|
209
|
-
\t\tconst raw_${field} = getValue(body['${field}']);
|
|
210
|
-
\t\t${requiredCheck}
|
|
211
|
-
\t\tif (raw_${field} !== '') {
|
|
212
|
-
\t\t\tpayload.${field} = raw_${field};
|
|
213
|
-
\t\t}
|
|
214
|
-
\t\t`;
|
|
215
|
-
})
|
|
216
|
-
.join("\n");
|
|
217
|
-
}
|
|
1
|
+
import { DiscoveredTable } from "../../types";
|
|
2
|
+
|
|
3
|
+
export function buildSearchConditions(table: DiscoveredTable): string {
|
|
4
|
+
return table.columns
|
|
5
|
+
.filter((column) => column.type === "string")
|
|
6
|
+
.map(
|
|
7
|
+
(column) => `
|
|
8
|
+
\t\t\ttry { searchConditions.push(like(tableSchema.${column.name}, \`%\${search}%\`)); } catch (e) {}
|
|
9
|
+
\t\t\t`,
|
|
10
|
+
)
|
|
11
|
+
.join("");
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Returns an Iconify icon name for a column type.
|
|
16
|
+
*/
|
|
17
|
+
function columnTypeIcon(type: string): string {
|
|
18
|
+
switch (type) {
|
|
19
|
+
case "number":
|
|
20
|
+
return "mdi:pound";
|
|
21
|
+
case "boolean":
|
|
22
|
+
return "mdi:toggle-switch-outline";
|
|
23
|
+
case "date":
|
|
24
|
+
return "mdi:calendar";
|
|
25
|
+
default:
|
|
26
|
+
return "mdi:format-text";
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export function buildColumnHeaders(
|
|
31
|
+
table: DiscoveredTable,
|
|
32
|
+
columns: string[],
|
|
33
|
+
): string {
|
|
34
|
+
return columns
|
|
35
|
+
.map((column) => {
|
|
36
|
+
const col = table.columns.find((c) => c.name === column);
|
|
37
|
+
const icon = col ? columnTypeIcon(col.type) : "mdi:format-text";
|
|
38
|
+
return `
|
|
39
|
+
\t\t\t\t\t\t\t<th>
|
|
40
|
+
\t\t\t\t\t\t\t\t<a href="#"
|
|
41
|
+
\t\t\t\t\t\t\t\t hx-get="/admin/table/${table.exportName}?page=\${page}&search=\${search}&sort=${column}&order=\${sort === '${column}' && order === 'asc' ? 'desc' : 'asc'}"
|
|
42
|
+
\t\t\t\t\t\t\t\t hx-target="#main-content"
|
|
43
|
+
\t\t\t\t\t\t\t\t hx-push-url="true"
|
|
44
|
+
\t\t\t\t\t\t\t\t class="hover:text-primary flex items-center gap-1.5 transition-colors whitespace-nowrap">
|
|
45
|
+
\t\t\t\t\t\t\t\t <iconify-icon icon="${icon}" width="14" height="14" class="opacity-40"></iconify-icon>
|
|
46
|
+
\t\t\t\t\t\t\t\t ${column}
|
|
47
|
+
\t\t\t\t\t\t\t\t <span class="text-[10px] opacity-30">\${sort === '${column}' ? (order === 'asc' ? '▲' : '▼') : ''}</span>
|
|
48
|
+
\t\t\t\t\t\t\t\t</a>
|
|
49
|
+
\t\t\t\t\t\t\t</th>
|
|
50
|
+
\t\t\t\t\t\t\t`;
|
|
51
|
+
})
|
|
52
|
+
.join("");
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
export function buildRowCells(columns: string[], primaryKey?: string): string {
|
|
56
|
+
return columns
|
|
57
|
+
.map((column) => {
|
|
58
|
+
if (primaryKey && column === primaryKey) {
|
|
59
|
+
return `<td><button type="button" class="truncate max-w-[200px] text-sm font-mono text-xs opacity-70 hover:opacity-100 cursor-copy text-left" title="Click to copy: \${String((row as any).${column} ?? '')}" data-copy-value="\${String((row as any).${column} ?? '')}" onclick="navigator.clipboard?.writeText(this.dataset.copyValue || '')">\${String((row as any).${column} ?? '')}</button></td>`;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
return `<td><div class="truncate max-w-[200px] text-sm" title="\${String((row as any).${column} ?? '')}">\${String((row as any).${column} ?? '')}</div></td>`;
|
|
63
|
+
})
|
|
64
|
+
.join("");
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
export function buildFieldInput(
|
|
68
|
+
table: DiscoveredTable,
|
|
69
|
+
columnName: string,
|
|
70
|
+
mode: "create" | "edit",
|
|
71
|
+
): string {
|
|
72
|
+
const column = table.columns.find((item) => item.name === columnName);
|
|
73
|
+
if (!column) {
|
|
74
|
+
return "";
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
const requiredAttr = !column.optional ? " required" : "";
|
|
78
|
+
const inputType =
|
|
79
|
+
column.type === "number"
|
|
80
|
+
? "number"
|
|
81
|
+
: column.type === "date"
|
|
82
|
+
? "date"
|
|
83
|
+
: "text";
|
|
84
|
+
|
|
85
|
+
if (column.type === "boolean") {
|
|
86
|
+
if (mode === "edit") {
|
|
87
|
+
return `
|
|
88
|
+
\t\t<div class="form-control">
|
|
89
|
+
\t\t\t<label class="label cursor-pointer justify-start gap-3">
|
|
90
|
+
\t\t\t\t<input type="checkbox" name="${columnName}" value="true" class="checkbox checkbox-sm checkbox-primary" \${(row as any).${columnName} ? 'checked' : ''} />
|
|
91
|
+
\t\t\t\t<span class="label-text text-sm">${columnName}</span>
|
|
92
|
+
\t\t\t</label>
|
|
93
|
+
\t\t</div>
|
|
94
|
+
\t\t`;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
return `
|
|
98
|
+
\t\t<div class="form-control">
|
|
99
|
+
\t\t\t<label class="label cursor-pointer justify-start gap-3">
|
|
100
|
+
\t\t\t\t<input type="checkbox" name="${columnName}" value="true" class="checkbox checkbox-sm checkbox-primary" />
|
|
101
|
+
\t\t\t\t<span class="label-text text-sm">${columnName}</span>
|
|
102
|
+
\t\t\t</label>
|
|
103
|
+
\t\t</div>
|
|
104
|
+
\t\t`;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
if (mode === "edit") {
|
|
108
|
+
const valueExpression =
|
|
109
|
+
column.type === "date"
|
|
110
|
+
? `\${(() => {
|
|
111
|
+
const value = (row as any).${columnName};
|
|
112
|
+
if (value == null || value === '') return '';
|
|
113
|
+
|
|
114
|
+
const date =
|
|
115
|
+
value instanceof Date
|
|
116
|
+
? value
|
|
117
|
+
: new Date(typeof value === 'number' ? value : String(value));
|
|
118
|
+
if (Number.isNaN(date.getTime())) return '';
|
|
119
|
+
|
|
120
|
+
const year = date.getFullYear();
|
|
121
|
+
const month = String(date.getMonth() + 1).padStart(2, '0');
|
|
122
|
+
const day = String(date.getDate()).padStart(2, '0');
|
|
123
|
+
return String(year) + '-' + month + '-' + day;
|
|
124
|
+
})()}`
|
|
125
|
+
: `\${String((row as any).${columnName} ?? '')}`;
|
|
126
|
+
|
|
127
|
+
return `
|
|
128
|
+
\t\t<div class="form-control">
|
|
129
|
+
\t\t\t<label class="label"><span class="label-text text-sm font-medium">${columnName}</span></label>
|
|
130
|
+
\t\t\t<input type="${inputType}" name="${columnName}" class="input input-bordered w-full text-sm" value="${valueExpression}"${requiredAttr} />
|
|
131
|
+
\t\t</div>
|
|
132
|
+
\t\t`;
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
return `
|
|
136
|
+
\t\t<div class="form-control">
|
|
137
|
+
\t\t\t<label class="label"><span class="label-text text-sm font-medium">${columnName}</span></label>
|
|
138
|
+
\t\t\t<input type="${inputType}" name="${columnName}" class="input input-bordered w-full text-sm"${requiredAttr} />
|
|
139
|
+
\t\t</div>
|
|
140
|
+
\t\t`;
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
export function buildPayloadAssignments(
|
|
144
|
+
table: DiscoveredTable,
|
|
145
|
+
fields: string[],
|
|
146
|
+
): string {
|
|
147
|
+
return fields
|
|
148
|
+
.map((field) => {
|
|
149
|
+
const column = table.columns.find((item) => item.name === field);
|
|
150
|
+
if (!column) {
|
|
151
|
+
return "";
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
const requiredCheck = !column.optional
|
|
155
|
+
? `
|
|
156
|
+
\t\tif (raw_${field} === '') {
|
|
157
|
+
\t\t\treturn c.text('${field} is required', 400);
|
|
158
|
+
\t\t}
|
|
159
|
+
\t\t`
|
|
160
|
+
: "";
|
|
161
|
+
|
|
162
|
+
if (column.type === "number") {
|
|
163
|
+
return `
|
|
164
|
+
\t\tconst raw_${field} = getValue(body['${field}']);
|
|
165
|
+
\t\t${requiredCheck}
|
|
166
|
+
\t\tif (raw_${field} !== '') {
|
|
167
|
+
\t\t\tconst parsed_${field} = Number(raw_${field});
|
|
168
|
+
\t\t\tif (Number.isNaN(parsed_${field})) {
|
|
169
|
+
\t\t\t\treturn c.text('${field} must be a valid number', 400);
|
|
170
|
+
\t\t\t}
|
|
171
|
+
\t\t\tpayload.${field} = parsed_${field};
|
|
172
|
+
\t\t}
|
|
173
|
+
\t\t`;
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
if (column.type === "boolean") {
|
|
177
|
+
return `
|
|
178
|
+
\t\tconst raw_${field} = getValue(body['${field}']);
|
|
179
|
+
\t\tpayload.${field} = raw_${field} === 'true' || raw_${field} === 'on' || raw_${field} === '1';
|
|
180
|
+
\t\t`;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
if (column.type === "date") {
|
|
184
|
+
return `
|
|
185
|
+
const raw_${field} = getValue(body['${field}']);
|
|
186
|
+
${requiredCheck}
|
|
187
|
+
if (raw_${field} !== '') {
|
|
188
|
+
if (!/^\\d{4}-\\d{2}-\\d{2}$/.test(raw_${field})) {
|
|
189
|
+
return c.text('${field} must be a valid date (YYYY-MM-DD)', 400);
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
const [year_${field}, month_${field}, day_${field}] = raw_${field}.split('-').map(Number);
|
|
193
|
+
const parsed_${field} = new Date(year_${field}, month_${field} - 1, day_${field});
|
|
194
|
+
if (
|
|
195
|
+
Number.isNaN(parsed_${field}.getTime()) ||
|
|
196
|
+
parsed_${field}.getFullYear() !== year_${field} ||
|
|
197
|
+
parsed_${field}.getMonth() !== month_${field} - 1 ||
|
|
198
|
+
parsed_${field}.getDate() !== day_${field}
|
|
199
|
+
) {
|
|
200
|
+
return c.text('${field} must be a valid date', 400);
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
payload.${field} = parsed_${field};
|
|
204
|
+
}
|
|
205
|
+
`;
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
return `
|
|
209
|
+
\t\tconst raw_${field} = getValue(body['${field}']);
|
|
210
|
+
\t\t${requiredCheck}
|
|
211
|
+
\t\tif (raw_${field} !== '') {
|
|
212
|
+
\t\t\tpayload.${field} = raw_${field};
|
|
213
|
+
\t\t}
|
|
214
|
+
\t\t`;
|
|
215
|
+
})
|
|
216
|
+
.join("\n");
|
|
217
|
+
}
|
|
@@ -1,45 +1,45 @@
|
|
|
1
|
-
import { DiscoveredTable } from "../../types";
|
|
2
|
-
|
|
3
|
-
export function resolvePrimaryKey(table: DiscoveredTable): string {
|
|
4
|
-
return (
|
|
5
|
-
table.columns.find((column) => column.primaryKey)?.name ||
|
|
6
|
-
table.columns[0]?.name ||
|
|
7
|
-
""
|
|
8
|
-
);
|
|
9
|
-
}
|
|
10
|
-
|
|
11
|
-
export function shouldIncludeCreateField(
|
|
12
|
-
table: DiscoveredTable,
|
|
13
|
-
columnName: string,
|
|
14
|
-
): boolean {
|
|
15
|
-
const column = table.columns.find((item) => item.name === columnName);
|
|
16
|
-
if (!column) {
|
|
17
|
-
return false;
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
if (column.autoIncrement) {
|
|
21
|
-
return false;
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
if (column.primaryKey) {
|
|
25
|
-
return false;
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
return true;
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
export function shouldIncludeEditField(
|
|
32
|
-
table: DiscoveredTable,
|
|
33
|
-
columnName: string,
|
|
34
|
-
): boolean {
|
|
35
|
-
const column = table.columns.find((item) => item.name === columnName);
|
|
36
|
-
if (!column) {
|
|
37
|
-
return false;
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
if (column.primaryKey || column.autoIncrement) {
|
|
41
|
-
return false;
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
return true;
|
|
45
|
-
}
|
|
1
|
+
import { DiscoveredTable } from "../../types";
|
|
2
|
+
|
|
3
|
+
export function resolvePrimaryKey(table: DiscoveredTable): string {
|
|
4
|
+
return (
|
|
5
|
+
table.columns.find((column) => column.primaryKey)?.name ||
|
|
6
|
+
table.columns[0]?.name ||
|
|
7
|
+
""
|
|
8
|
+
);
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export function shouldIncludeCreateField(
|
|
12
|
+
table: DiscoveredTable,
|
|
13
|
+
columnName: string,
|
|
14
|
+
): boolean {
|
|
15
|
+
const column = table.columns.find((item) => item.name === columnName);
|
|
16
|
+
if (!column) {
|
|
17
|
+
return false;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
if (column.autoIncrement) {
|
|
21
|
+
return false;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
if (column.primaryKey) {
|
|
25
|
+
return false;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
return true;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export function shouldIncludeEditField(
|
|
32
|
+
table: DiscoveredTable,
|
|
33
|
+
columnName: string,
|
|
34
|
+
): boolean {
|
|
35
|
+
const column = table.columns.find((item) => item.name === columnName);
|
|
36
|
+
if (!column) {
|
|
37
|
+
return false;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
if (column.primaryKey || column.autoIncrement) {
|
|
41
|
+
return false;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
return true;
|
|
45
|
+
}
|
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
import { DiscoveredSchema } from "../../../../utils/schema-discovery";
|
|
2
|
-
import { buildTableRoute } from "./table-route";
|
|
3
|
-
import { buildUsersRoute } from "./users-route";
|
|
4
|
-
|
|
5
|
-
export function buildTableRoutes(schema: DiscoveredSchema): string {
|
|
6
|
-
return `${schema.tables.map((table) => buildTableRoute(table)).join("\n")}
|
|
7
|
-
${buildUsersRoute()}`;
|
|
8
|
-
}
|
|
1
|
+
import { DiscoveredSchema } from "../../../../utils/schema-discovery";
|
|
2
|
+
import { buildTableRoute } from "./table-route";
|
|
3
|
+
import { buildUsersRoute } from "./users-route";
|
|
4
|
+
|
|
5
|
+
export function buildTableRoutes(schema: DiscoveredSchema): string {
|
|
6
|
+
return `${schema.tables.map((table) => buildTableRoute(table)).join("\n")}
|
|
7
|
+
${buildUsersRoute()}`;
|
|
8
|
+
}
|