appflare 0.2.43 → 0.2.45

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 (139) hide show
  1. package/Documentation.md +898 -898
  2. package/cli/commands/index.ts +247 -247
  3. package/cli/generate.ts +360 -360
  4. package/cli/index.ts +120 -120
  5. package/cli/load-config.ts +184 -184
  6. package/cli/schema-compiler.ts +1366 -1363
  7. package/cli/templates/auth/README.md +156 -156
  8. package/cli/templates/auth/config.ts +61 -61
  9. package/cli/templates/auth/route-config.ts +1 -1
  10. package/cli/templates/auth/route-handler.ts +1 -1
  11. package/cli/templates/auth/route-request-utils.ts +5 -5
  12. package/cli/templates/auth/route.config.ts +18 -18
  13. package/cli/templates/auth/route.handler.ts +18 -18
  14. package/cli/templates/auth/route.request-utils.ts +55 -55
  15. package/cli/templates/auth/route.ts +14 -14
  16. package/cli/templates/core/README.md +266 -266
  17. package/cli/templates/core/app-creation.ts +19 -19
  18. package/cli/templates/core/client/appflare.ts +112 -112
  19. package/cli/templates/core/client/handlers/index.ts +763 -763
  20. package/cli/templates/core/client/handlers.ts +1 -1
  21. package/cli/templates/core/client/index.ts +7 -7
  22. package/cli/templates/core/client/storage.ts +195 -195
  23. package/cli/templates/core/client/types.ts +187 -187
  24. package/cli/templates/core/client-modules/appflare.ts +1 -1
  25. package/cli/templates/core/client-modules/handlers.ts +1 -1
  26. package/cli/templates/core/client-modules/index.ts +1 -1
  27. package/cli/templates/core/client-modules/storage.ts +1 -1
  28. package/cli/templates/core/client-modules/types.ts +1 -1
  29. package/cli/templates/core/client.artifacts.ts +39 -39
  30. package/cli/templates/core/client.ts +4 -4
  31. package/cli/templates/core/drizzle.ts +15 -15
  32. package/cli/templates/core/export.ts +14 -14
  33. package/cli/templates/core/handlers.route.ts +24 -24
  34. package/cli/templates/core/handlers.ts +1 -1
  35. package/cli/templates/core/imports.ts +9 -9
  36. package/cli/templates/core/server.ts +38 -38
  37. package/cli/templates/core/types.ts +6 -6
  38. package/cli/templates/core/wrangler.ts +109 -109
  39. package/cli/templates/dashboard/builders/functions/index.ts +17 -17
  40. package/cli/templates/dashboard/builders/functions/render-page/header.ts +20 -20
  41. package/cli/templates/dashboard/builders/functions/render-page/index.ts +33 -33
  42. package/cli/templates/dashboard/builders/functions/render-page/request-panel.ts +171 -171
  43. package/cli/templates/dashboard/builders/functions/render-page/result-panel.ts +85 -85
  44. package/cli/templates/dashboard/builders/functions/render-page/scripts.ts +554 -554
  45. package/cli/templates/dashboard/builders/functions/tree-builder.ts +47 -47
  46. package/cli/templates/dashboard/builders/navigation.ts +155 -155
  47. package/cli/templates/dashboard/builders/storage/index.ts +13 -13
  48. package/cli/templates/dashboard/builders/storage/routes/create-directory-route.ts +29 -29
  49. package/cli/templates/dashboard/builders/storage/routes/delete-route.ts +18 -18
  50. package/cli/templates/dashboard/builders/storage/routes/download-route.ts +23 -23
  51. package/cli/templates/dashboard/builders/storage/routes/index.ts +22 -22
  52. package/cli/templates/dashboard/builders/storage/routes/list-route.ts +25 -25
  53. package/cli/templates/dashboard/builders/storage/routes/preview-route.ts +21 -21
  54. package/cli/templates/dashboard/builders/storage/routes/upload-route.ts +21 -21
  55. package/cli/templates/dashboard/builders/storage/runtime/helpers.ts +72 -72
  56. package/cli/templates/dashboard/builders/storage/runtime/storage-page.ts +130 -130
  57. package/cli/templates/dashboard/builders/table-routes/common/drawer-panel.ts +27 -27
  58. package/cli/templates/dashboard/builders/table-routes/common/pagination.ts +30 -30
  59. package/cli/templates/dashboard/builders/table-routes/common/search-bar.ts +23 -23
  60. package/cli/templates/dashboard/builders/table-routes/fragments.ts +217 -217
  61. package/cli/templates/dashboard/builders/table-routes/helpers.ts +45 -45
  62. package/cli/templates/dashboard/builders/table-routes/index.ts +8 -8
  63. package/cli/templates/dashboard/builders/table-routes/table/actions-cell.ts +71 -71
  64. package/cli/templates/dashboard/builders/table-routes/table/get-route.ts +291 -291
  65. package/cli/templates/dashboard/builders/table-routes/table/index.ts +80 -80
  66. package/cli/templates/dashboard/builders/table-routes/table/post-routes.ts +163 -163
  67. package/cli/templates/dashboard/builders/table-routes/table-route.ts +7 -7
  68. package/cli/templates/dashboard/builders/table-routes/users/get-route.ts +69 -69
  69. package/cli/templates/dashboard/builders/table-routes/users/html/modals.ts +57 -57
  70. package/cli/templates/dashboard/builders/table-routes/users/html/page.ts +27 -27
  71. package/cli/templates/dashboard/builders/table-routes/users/html/table.ts +128 -128
  72. package/cli/templates/dashboard/builders/table-routes/users/index.ts +32 -32
  73. package/cli/templates/dashboard/builders/table-routes/users/post-routes.ts +150 -150
  74. package/cli/templates/dashboard/builders/table-routes/users/redirect.ts +14 -14
  75. package/cli/templates/dashboard/builders/table-routes/users-route.ts +10 -10
  76. package/cli/templates/dashboard/components/dashboard-home.ts +23 -23
  77. package/cli/templates/dashboard/components/layout.ts +420 -420
  78. package/cli/templates/dashboard/components/login-page.ts +65 -65
  79. package/cli/templates/dashboard/index.ts +61 -61
  80. package/cli/templates/dashboard/types.ts +9 -9
  81. package/cli/templates/handlers/README.md +353 -353
  82. package/cli/templates/handlers/auth.ts +37 -37
  83. package/cli/templates/handlers/execution.ts +42 -42
  84. package/cli/templates/handlers/generators/context/context-creation.ts +101 -101
  85. package/cli/templates/handlers/generators/context/error-helpers.ts +11 -11
  86. package/cli/templates/handlers/generators/context/scheduler.ts +24 -24
  87. package/cli/templates/handlers/generators/context/storage-api.ts +82 -82
  88. package/cli/templates/handlers/generators/context/storage-helpers.ts +59 -59
  89. package/cli/templates/handlers/generators/context/types.ts +40 -40
  90. package/cli/templates/handlers/generators/context.ts +43 -43
  91. package/cli/templates/handlers/generators/execution.ts +15 -15
  92. package/cli/templates/handlers/generators/handlers.ts +14 -14
  93. package/cli/templates/handlers/generators/registration/modules/cron.ts +26 -26
  94. package/cli/templates/handlers/generators/registration/modules/realtime/auth.ts +75 -75
  95. package/cli/templates/handlers/generators/registration/modules/realtime/durable-object.ts +144 -144
  96. package/cli/templates/handlers/generators/registration/modules/realtime/index.ts +14 -14
  97. package/cli/templates/handlers/generators/registration/modules/realtime/publisher.ts +102 -102
  98. package/cli/templates/handlers/generators/registration/modules/realtime/routes.ts +164 -164
  99. package/cli/templates/handlers/generators/registration/modules/realtime/types.ts +30 -30
  100. package/cli/templates/handlers/generators/registration/modules/realtime/utils.ts +510 -510
  101. package/cli/templates/handlers/generators/registration/modules/scheduler.ts +56 -56
  102. package/cli/templates/handlers/generators/registration/modules/storage.ts +199 -199
  103. package/cli/templates/handlers/generators/registration/sections.ts +210 -210
  104. package/cli/templates/handlers/generators/types/context.ts +121 -121
  105. package/cli/templates/handlers/generators/types/core.ts +106 -106
  106. package/cli/templates/handlers/generators/types/operations.ts +135 -135
  107. package/cli/templates/handlers/generators/types/query-definitions/filter-and-where-types.ts +291 -291
  108. package/cli/templates/handlers/generators/types/query-definitions/query-api-types.ts +135 -135
  109. package/cli/templates/handlers/generators/types/query-definitions/query-helper-functions.ts +1382 -1382
  110. package/cli/templates/handlers/generators/types/query-definitions/schema-and-table-types.ts +278 -278
  111. package/cli/templates/handlers/generators/types/query-definitions.ts +13 -13
  112. package/cli/templates/handlers/generators/types/query-runtime/handled-error.ts +13 -13
  113. package/cli/templates/handlers/generators/types/query-runtime/runtime-aggregate-and-footer.ts +174 -174
  114. package/cli/templates/handlers/generators/types/query-runtime/runtime-read.ts +157 -157
  115. package/cli/templates/handlers/generators/types/query-runtime/runtime-setup.ts +45 -45
  116. package/cli/templates/handlers/generators/types/query-runtime/runtime-write.ts +958 -958
  117. package/cli/templates/handlers/generators/types/query-runtime.ts +15 -15
  118. package/cli/templates/handlers/index.ts +47 -47
  119. package/cli/templates/handlers/operations.ts +116 -116
  120. package/cli/templates/handlers/registration.ts +91 -91
  121. package/cli/templates/handlers/types.ts +17 -17
  122. package/cli/templates/handlers/utils.ts +48 -48
  123. package/cli/types.ts +110 -110
  124. package/cli/utils/handler-discovery.ts +466 -466
  125. package/cli/utils/json-utils.ts +24 -24
  126. package/cli/utils/path-utils.ts +19 -19
  127. package/cli/utils/schema-discovery.ts +399 -399
  128. package/dist/cli/index.js +21 -21
  129. package/dist/cli/index.mjs +21 -21
  130. package/index.ts +18 -18
  131. package/package.json +58 -58
  132. package/react/index.ts +5 -5
  133. package/react/use-infinite-query.ts +255 -255
  134. package/react/use-mutation.ts +89 -89
  135. package/react/use-query.ts +210 -210
  136. package/schema.ts +641 -641
  137. package/test-better-auth-hash.ts +2 -2
  138. package/tsconfig.json +6 -6
  139. package/tsup.config.ts +82 -82
@@ -1,247 +1,247 @@
1
- import chokidar from "chokidar";
2
- import { existsSync } from "node:fs";
3
- import { dirname, resolve } from "node:path";
4
- import { generateArtifacts } from "../generate";
5
- import { loadConfig } from "../load-config";
6
-
7
- type MigrateOptions = {
8
- local?: boolean;
9
- remote?: boolean;
10
- preview?: boolean;
11
- };
12
-
13
- function findNearestPackageDir(startDir: string): string {
14
- let currentDir = startDir;
15
-
16
- while (true) {
17
- if (existsSync(resolve(currentDir, "package.json"))) {
18
- return currentDir;
19
- }
20
-
21
- const parentDir = dirname(currentDir);
22
- if (parentDir === currentDir) {
23
- return startDir;
24
- }
25
-
26
- currentDir = parentDir;
27
- }
28
- }
29
-
30
- export async function runBuild(
31
- configPath?: string,
32
- options: { build?: boolean } = {},
33
- ): Promise<void> {
34
- const loadedConfig = await loadConfig(configPath);
35
- if (options.build !== undefined) {
36
- loadedConfig.config.build = options.build;
37
- }
38
- await generateArtifacts(loadedConfig);
39
- if (loadedConfig.wranglerOutDirAbs === loadedConfig.outDirAbs) {
40
- process.stdout.write(
41
- `✅ Generated artifacts in ${loadedConfig.outDirAbs}\n`,
42
- );
43
- return;
44
- }
45
-
46
- process.stdout.write(
47
- `✅ Generated server/client in ${loadedConfig.outDirAbs} and wrangler.json in ${loadedConfig.wranglerOutDirAbs}\n`,
48
- );
49
- }
50
-
51
- export async function runDev(
52
- configPath?: string,
53
- options: { watch?: boolean; build?: boolean } = {},
54
- ): Promise<void> {
55
- await runBuild(configPath, { build: options.build });
56
-
57
- if (!options.watch) {
58
- return;
59
- }
60
-
61
- const loadedConfig = await loadConfig(configPath);
62
- let isRunning = false;
63
- let shouldRerun = false;
64
-
65
- const rebuild = async (): Promise<void> => {
66
- if (isRunning) {
67
- shouldRerun = true;
68
- return;
69
- }
70
-
71
- isRunning = true;
72
- try {
73
- await runBuild(configPath, { build: options.build });
74
- } catch (error) {
75
- process.stderr.write(`❌ Build failed: ${(error as Error).message}\n`);
76
- } finally {
77
- isRunning = false;
78
- if (shouldRerun) {
79
- shouldRerun = false;
80
- await rebuild();
81
- }
82
- }
83
- };
84
-
85
- const watcher = chokidar.watch(loadedConfig.scanDirAbs, {
86
- ignoreInitial: true,
87
- ignored: (path: string) => {
88
- return (
89
- path.includes("/_generated/") ||
90
- path.includes("/dist/") ||
91
- path.includes("/out/") ||
92
- path.includes("/node_modules/") ||
93
- path.endsWith(".d.ts")
94
- );
95
- },
96
- });
97
-
98
- watcher.on("all", async (_eventName, changedPath) => {
99
- process.stdout.write(`🔄 Change detected: ${changedPath}\n`);
100
- await rebuild();
101
- });
102
-
103
- process.stdout.write(`👀 Watching ${loadedConfig.scanDirAbs}\n`);
104
- }
105
-
106
- export async function runMigrate(
107
- configPath?: string,
108
- options: MigrateOptions = {},
109
- ): Promise<void> {
110
- const loadedConfig = await loadConfig(configPath);
111
- const packageDir = findNearestPackageDir(process.cwd());
112
- const selectedTargetCount = [
113
- Boolean(options.local),
114
- Boolean(options.remote),
115
- Boolean(options.preview),
116
- ].filter(Boolean).length;
117
-
118
- if (selectedTargetCount > 1) {
119
- throw new Error("Only one of --local, --remote, or --preview can be set.");
120
- }
121
-
122
- const drizzleConfigPath = resolve(
123
- loadedConfig.outDirAbs,
124
- "drizzle.config.js",
125
- );
126
- const npxCmd = process.platform === "win32" ? "npx.cmd" : "npx";
127
- const drizzleGenerate = Bun.spawn(
128
- [npxCmd, "drizzle-kit", "generate", "--config", drizzleConfigPath],
129
- {
130
- cwd: packageDir,
131
- stdin: "inherit",
132
- stdout: "inherit",
133
- stderr: "inherit",
134
- },
135
- );
136
-
137
- const drizzleExitCode = await drizzleGenerate.exited;
138
- if (drizzleExitCode !== 0) {
139
- throw new Error(
140
- `drizzle-kit generate failed with exit code ${drizzleExitCode}`,
141
- );
142
- }
143
-
144
- const databaseName = loadedConfig.config.database[0].databaseName;
145
- const wranglerArgs = [
146
- npxCmd,
147
- "wrangler",
148
- "d1",
149
- "migrations",
150
- "apply",
151
- databaseName,
152
- ];
153
-
154
- if (options.local) {
155
- wranglerArgs.push("--local");
156
- } else if (options.remote) {
157
- wranglerArgs.push("--remote");
158
- } else if (options.preview) {
159
- wranglerArgs.push("--preview");
160
- }
161
-
162
- const wranglerApply = Bun.spawn(wranglerArgs, {
163
- cwd: loadedConfig.configDir,
164
- stdin: "inherit",
165
- stdout: "inherit",
166
- stderr: "inherit",
167
- });
168
-
169
- const wranglerExitCode = await wranglerApply.exited;
170
- if (wranglerExitCode !== 0) {
171
- throw new Error(
172
- `wrangler d1 migrations apply failed with exit code ${wranglerExitCode}`,
173
- );
174
- }
175
- }
176
-
177
- export async function runAddAdmin(
178
- configPath?: string,
179
- options: {
180
- name: string;
181
- email: string;
182
- password: string;
183
- local?: boolean;
184
- remote?: boolean;
185
- } = { name: "", email: "", password: "" },
186
- ): Promise<void> {
187
- const loadedConfig = await loadConfig(configPath);
188
-
189
- const selectedTargetCount = [
190
- Boolean(options.local),
191
- Boolean(options.remote),
192
- ].filter(Boolean).length;
193
-
194
- if (selectedTargetCount > 1) {
195
- throw new Error("Only one of --local or --remote can be set.");
196
- }
197
-
198
- const { hashPassword } = await import("better-auth/crypto");
199
- const passwordHash = await hashPassword(options.password);
200
-
201
- const userId = crypto.randomUUID();
202
- const accountId = crypto.randomUUID();
203
- const now = Date.now();
204
-
205
- const safeName = options.name.replace(/'/g, "''");
206
- const safeEmail = options.email.replace(/'/g, "''");
207
-
208
- const sqlQuery = [
209
- "INSERT INTO users (id, name, email, email_verified, created_at, updated_at, role, banned)",
210
- `VALUES ('${userId}', '${safeName}', '${safeEmail}', 1, ${now}, ${now}, 'admin', 0);`,
211
- "INSERT INTO accounts (id, account_id, provider_id, user_id, password, created_at, updated_at)",
212
- `VALUES ('${accountId}', '${safeEmail}', 'credential', '${userId}', '${passwordHash}', ${now}, ${now});`,
213
- ].join(" ");
214
-
215
- const databaseName = loadedConfig.config.database[0].databaseName;
216
- const npxCmd = process.platform === "win32" ? "npx.cmd" : "npx";
217
- const wranglerArgs = [
218
- npxCmd,
219
- "wrangler",
220
- "d1",
221
- "execute",
222
- databaseName,
223
- `--command=${sqlQuery}`,
224
- ];
225
-
226
- if (options.local) {
227
- wranglerArgs.push("--local");
228
- } else if (options.remote) {
229
- wranglerArgs.push("--remote");
230
- }
231
-
232
- const wranglerExecute = Bun.spawn(wranglerArgs, {
233
- cwd: loadedConfig.configDir,
234
- stdin: "inherit",
235
- stdout: "inherit",
236
- stderr: "inherit",
237
- });
238
-
239
- const wranglerExitCode = await wranglerExecute.exited;
240
- if (wranglerExitCode !== 0) {
241
- throw new Error(
242
- `Failed to add admin user. wrangler d1 execute exited with code ${wranglerExitCode}`,
243
- );
244
- }
245
-
246
- console.log("✅ Admin user " + options.email + " created successfully!");
247
- }
1
+ import chokidar from "chokidar";
2
+ import { existsSync } from "node:fs";
3
+ import { dirname, resolve } from "node:path";
4
+ import { generateArtifacts } from "../generate";
5
+ import { loadConfig } from "../load-config";
6
+
7
+ type MigrateOptions = {
8
+ local?: boolean;
9
+ remote?: boolean;
10
+ preview?: boolean;
11
+ };
12
+
13
+ function findNearestPackageDir(startDir: string): string {
14
+ let currentDir = startDir;
15
+
16
+ while (true) {
17
+ if (existsSync(resolve(currentDir, "package.json"))) {
18
+ return currentDir;
19
+ }
20
+
21
+ const parentDir = dirname(currentDir);
22
+ if (parentDir === currentDir) {
23
+ return startDir;
24
+ }
25
+
26
+ currentDir = parentDir;
27
+ }
28
+ }
29
+
30
+ export async function runBuild(
31
+ configPath?: string,
32
+ options: { build?: boolean } = {},
33
+ ): Promise<void> {
34
+ const loadedConfig = await loadConfig(configPath);
35
+ if (options.build !== undefined) {
36
+ loadedConfig.config.build = options.build;
37
+ }
38
+ await generateArtifacts(loadedConfig);
39
+ if (loadedConfig.wranglerOutDirAbs === loadedConfig.outDirAbs) {
40
+ process.stdout.write(
41
+ `✅ Generated artifacts in ${loadedConfig.outDirAbs}\n`,
42
+ );
43
+ return;
44
+ }
45
+
46
+ process.stdout.write(
47
+ `✅ Generated server/client in ${loadedConfig.outDirAbs} and wrangler.json in ${loadedConfig.wranglerOutDirAbs}\n`,
48
+ );
49
+ }
50
+
51
+ export async function runDev(
52
+ configPath?: string,
53
+ options: { watch?: boolean; build?: boolean } = {},
54
+ ): Promise<void> {
55
+ await runBuild(configPath, { build: options.build });
56
+
57
+ if (!options.watch) {
58
+ return;
59
+ }
60
+
61
+ const loadedConfig = await loadConfig(configPath);
62
+ let isRunning = false;
63
+ let shouldRerun = false;
64
+
65
+ const rebuild = async (): Promise<void> => {
66
+ if (isRunning) {
67
+ shouldRerun = true;
68
+ return;
69
+ }
70
+
71
+ isRunning = true;
72
+ try {
73
+ await runBuild(configPath, { build: options.build });
74
+ } catch (error) {
75
+ process.stderr.write(`❌ Build failed: ${(error as Error).message}\n`);
76
+ } finally {
77
+ isRunning = false;
78
+ if (shouldRerun) {
79
+ shouldRerun = false;
80
+ await rebuild();
81
+ }
82
+ }
83
+ };
84
+
85
+ const watcher = chokidar.watch(loadedConfig.scanDirAbs, {
86
+ ignoreInitial: true,
87
+ ignored: (path: string) => {
88
+ return (
89
+ path.includes("/_generated/") ||
90
+ path.includes("/dist/") ||
91
+ path.includes("/out/") ||
92
+ path.includes("/node_modules/") ||
93
+ path.endsWith(".d.ts")
94
+ );
95
+ },
96
+ });
97
+
98
+ watcher.on("all", async (_eventName, changedPath) => {
99
+ process.stdout.write(`🔄 Change detected: ${changedPath}\n`);
100
+ await rebuild();
101
+ });
102
+
103
+ process.stdout.write(`👀 Watching ${loadedConfig.scanDirAbs}\n`);
104
+ }
105
+
106
+ export async function runMigrate(
107
+ configPath?: string,
108
+ options: MigrateOptions = {},
109
+ ): Promise<void> {
110
+ const loadedConfig = await loadConfig(configPath);
111
+ const packageDir = findNearestPackageDir(process.cwd());
112
+ const selectedTargetCount = [
113
+ Boolean(options.local),
114
+ Boolean(options.remote),
115
+ Boolean(options.preview),
116
+ ].filter(Boolean).length;
117
+
118
+ if (selectedTargetCount > 1) {
119
+ throw new Error("Only one of --local, --remote, or --preview can be set.");
120
+ }
121
+
122
+ const drizzleConfigPath = resolve(
123
+ loadedConfig.outDirAbs,
124
+ "drizzle.config.js",
125
+ );
126
+ const npxCmd = process.platform === "win32" ? "npx.cmd" : "npx";
127
+ const drizzleGenerate = Bun.spawn(
128
+ [npxCmd, "drizzle-kit", "generate", "--config", drizzleConfigPath],
129
+ {
130
+ cwd: packageDir,
131
+ stdin: "inherit",
132
+ stdout: "inherit",
133
+ stderr: "inherit",
134
+ },
135
+ );
136
+
137
+ const drizzleExitCode = await drizzleGenerate.exited;
138
+ if (drizzleExitCode !== 0) {
139
+ throw new Error(
140
+ `drizzle-kit generate failed with exit code ${drizzleExitCode}`,
141
+ );
142
+ }
143
+
144
+ const databaseName = loadedConfig.config.database[0].databaseName;
145
+ const wranglerArgs = [
146
+ npxCmd,
147
+ "wrangler",
148
+ "d1",
149
+ "migrations",
150
+ "apply",
151
+ databaseName,
152
+ ];
153
+
154
+ if (options.local) {
155
+ wranglerArgs.push("--local");
156
+ } else if (options.remote) {
157
+ wranglerArgs.push("--remote");
158
+ } else if (options.preview) {
159
+ wranglerArgs.push("--preview");
160
+ }
161
+
162
+ const wranglerApply = Bun.spawn(wranglerArgs, {
163
+ cwd: loadedConfig.configDir,
164
+ stdin: "inherit",
165
+ stdout: "inherit",
166
+ stderr: "inherit",
167
+ });
168
+
169
+ const wranglerExitCode = await wranglerApply.exited;
170
+ if (wranglerExitCode !== 0) {
171
+ throw new Error(
172
+ `wrangler d1 migrations apply failed with exit code ${wranglerExitCode}`,
173
+ );
174
+ }
175
+ }
176
+
177
+ export async function runAddAdmin(
178
+ configPath?: string,
179
+ options: {
180
+ name: string;
181
+ email: string;
182
+ password: string;
183
+ local?: boolean;
184
+ remote?: boolean;
185
+ } = { name: "", email: "", password: "" },
186
+ ): Promise<void> {
187
+ const loadedConfig = await loadConfig(configPath);
188
+
189
+ const selectedTargetCount = [
190
+ Boolean(options.local),
191
+ Boolean(options.remote),
192
+ ].filter(Boolean).length;
193
+
194
+ if (selectedTargetCount > 1) {
195
+ throw new Error("Only one of --local or --remote can be set.");
196
+ }
197
+
198
+ const { hashPassword } = await import("better-auth/crypto");
199
+ const passwordHash = await hashPassword(options.password);
200
+
201
+ const userId = crypto.randomUUID();
202
+ const accountId = crypto.randomUUID();
203
+ const now = Date.now();
204
+
205
+ const safeName = options.name.replace(/'/g, "''");
206
+ const safeEmail = options.email.replace(/'/g, "''");
207
+
208
+ const sqlQuery = [
209
+ "INSERT INTO users (id, name, email, email_verified, created_at, updated_at, role, banned)",
210
+ `VALUES ('${userId}', '${safeName}', '${safeEmail}', 1, ${now}, ${now}, 'admin', 0);`,
211
+ "INSERT INTO accounts (id, account_id, provider_id, user_id, password, created_at, updated_at)",
212
+ `VALUES ('${accountId}', '${safeEmail}', 'credential', '${userId}', '${passwordHash}', ${now}, ${now});`,
213
+ ].join(" ");
214
+
215
+ const databaseName = loadedConfig.config.database[0].databaseName;
216
+ const npxCmd = process.platform === "win32" ? "npx.cmd" : "npx";
217
+ const wranglerArgs = [
218
+ npxCmd,
219
+ "wrangler",
220
+ "d1",
221
+ "execute",
222
+ databaseName,
223
+ `--command=${sqlQuery}`,
224
+ ];
225
+
226
+ if (options.local) {
227
+ wranglerArgs.push("--local");
228
+ } else if (options.remote) {
229
+ wranglerArgs.push("--remote");
230
+ }
231
+
232
+ const wranglerExecute = Bun.spawn(wranglerArgs, {
233
+ cwd: loadedConfig.configDir,
234
+ stdin: "inherit",
235
+ stdout: "inherit",
236
+ stderr: "inherit",
237
+ });
238
+
239
+ const wranglerExitCode = await wranglerExecute.exited;
240
+ if (wranglerExitCode !== 0) {
241
+ throw new Error(
242
+ `Failed to add admin user. wrangler d1 execute exited with code ${wranglerExitCode}`,
243
+ );
244
+ }
245
+
246
+ console.log("✅ Admin user " + options.email + " created successfully!");
247
+ }