appwrite-utils-cli 1.5.2 → 1.6.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 (233) hide show
  1. package/CHANGELOG.md +199 -0
  2. package/README.md +251 -29
  3. package/dist/adapters/AdapterFactory.d.ts +10 -3
  4. package/dist/adapters/AdapterFactory.js +213 -17
  5. package/dist/adapters/TablesDBAdapter.js +60 -17
  6. package/dist/backups/operations/bucketBackup.d.ts +19 -0
  7. package/dist/backups/operations/bucketBackup.js +197 -0
  8. package/dist/backups/operations/collectionBackup.d.ts +30 -0
  9. package/dist/backups/operations/collectionBackup.js +201 -0
  10. package/dist/backups/operations/comprehensiveBackup.d.ts +25 -0
  11. package/dist/backups/operations/comprehensiveBackup.js +238 -0
  12. package/dist/backups/schemas/bucketManifest.d.ts +93 -0
  13. package/dist/backups/schemas/bucketManifest.js +33 -0
  14. package/dist/backups/schemas/comprehensiveManifest.d.ts +108 -0
  15. package/dist/backups/schemas/comprehensiveManifest.js +32 -0
  16. package/dist/backups/tracking/centralizedTracking.d.ts +34 -0
  17. package/dist/backups/tracking/centralizedTracking.js +274 -0
  18. package/dist/cli/commands/configCommands.d.ts +8 -0
  19. package/dist/cli/commands/configCommands.js +160 -0
  20. package/dist/cli/commands/databaseCommands.d.ts +13 -0
  21. package/dist/cli/commands/databaseCommands.js +478 -0
  22. package/dist/cli/commands/functionCommands.d.ts +7 -0
  23. package/dist/cli/commands/functionCommands.js +289 -0
  24. package/dist/cli/commands/schemaCommands.d.ts +7 -0
  25. package/dist/cli/commands/schemaCommands.js +134 -0
  26. package/dist/cli/commands/transferCommands.d.ts +5 -0
  27. package/dist/cli/commands/transferCommands.js +384 -0
  28. package/dist/collections/attributes.d.ts +5 -4
  29. package/dist/collections/attributes.js +539 -246
  30. package/dist/collections/indexes.js +39 -37
  31. package/dist/collections/methods.d.ts +2 -16
  32. package/dist/collections/methods.js +90 -538
  33. package/dist/collections/transferOperations.d.ts +7 -0
  34. package/dist/collections/transferOperations.js +331 -0
  35. package/dist/collections/wipeOperations.d.ts +16 -0
  36. package/dist/collections/wipeOperations.js +328 -0
  37. package/dist/config/configMigration.d.ts +87 -0
  38. package/dist/config/configMigration.js +390 -0
  39. package/dist/config/configValidation.d.ts +66 -0
  40. package/dist/config/configValidation.js +358 -0
  41. package/dist/config/yamlConfig.d.ts +455 -1
  42. package/dist/config/yamlConfig.js +145 -52
  43. package/dist/databases/methods.js +3 -2
  44. package/dist/databases/setup.d.ts +1 -2
  45. package/dist/databases/setup.js +9 -87
  46. package/dist/examples/yamlTerminologyExample.d.ts +42 -0
  47. package/dist/examples/yamlTerminologyExample.js +269 -0
  48. package/dist/functions/deployments.js +11 -10
  49. package/dist/functions/methods.d.ts +1 -1
  50. package/dist/functions/methods.js +5 -4
  51. package/dist/init.js +9 -9
  52. package/dist/interactiveCLI.d.ts +8 -17
  53. package/dist/interactiveCLI.js +181 -1172
  54. package/dist/main.js +364 -21
  55. package/dist/migrations/afterImportActions.js +22 -30
  56. package/dist/migrations/appwriteToX.js +71 -25
  57. package/dist/migrations/dataLoader.js +35 -26
  58. package/dist/migrations/importController.js +29 -30
  59. package/dist/migrations/relationships.js +13 -12
  60. package/dist/migrations/services/ImportOrchestrator.js +16 -19
  61. package/dist/migrations/transfer.js +46 -46
  62. package/dist/migrations/yaml/YamlImportConfigLoader.d.ts +3 -1
  63. package/dist/migrations/yaml/YamlImportConfigLoader.js +6 -3
  64. package/dist/migrations/yaml/YamlImportIntegration.d.ts +9 -3
  65. package/dist/migrations/yaml/YamlImportIntegration.js +22 -11
  66. package/dist/migrations/yaml/generateImportSchemas.d.ts +14 -1
  67. package/dist/migrations/yaml/generateImportSchemas.js +736 -7
  68. package/dist/schemas/authUser.d.ts +1 -1
  69. package/dist/setupController.js +3 -2
  70. package/dist/shared/backupMetadataSchema.d.ts +94 -0
  71. package/dist/shared/backupMetadataSchema.js +38 -0
  72. package/dist/shared/backupTracking.d.ts +18 -0
  73. package/dist/shared/backupTracking.js +176 -0
  74. package/dist/shared/confirmationDialogs.js +15 -15
  75. package/dist/shared/errorUtils.d.ts +54 -0
  76. package/dist/shared/errorUtils.js +95 -0
  77. package/dist/shared/functionManager.js +20 -19
  78. package/dist/shared/indexManager.js +12 -11
  79. package/dist/shared/jsonSchemaGenerator.js +10 -26
  80. package/dist/shared/logging.d.ts +51 -0
  81. package/dist/shared/logging.js +70 -0
  82. package/dist/shared/messageFormatter.d.ts +2 -0
  83. package/dist/shared/messageFormatter.js +10 -0
  84. package/dist/shared/migrationHelpers.d.ts +6 -16
  85. package/dist/shared/migrationHelpers.js +24 -21
  86. package/dist/shared/operationLogger.d.ts +8 -1
  87. package/dist/shared/operationLogger.js +11 -24
  88. package/dist/shared/operationQueue.d.ts +28 -1
  89. package/dist/shared/operationQueue.js +268 -66
  90. package/dist/shared/operationsTable.d.ts +26 -0
  91. package/dist/shared/operationsTable.js +286 -0
  92. package/dist/shared/operationsTableSchema.d.ts +48 -0
  93. package/dist/shared/operationsTableSchema.js +35 -0
  94. package/dist/shared/relationshipExtractor.d.ts +56 -0
  95. package/dist/shared/relationshipExtractor.js +138 -0
  96. package/dist/shared/schemaGenerator.d.ts +19 -1
  97. package/dist/shared/schemaGenerator.js +56 -75
  98. package/dist/storage/backupCompression.d.ts +20 -0
  99. package/dist/storage/backupCompression.js +67 -0
  100. package/dist/storage/methods.d.ts +16 -2
  101. package/dist/storage/methods.js +98 -14
  102. package/dist/users/methods.js +9 -8
  103. package/dist/utils/configDiscovery.d.ts +78 -0
  104. package/dist/utils/configDiscovery.js +430 -0
  105. package/dist/utils/directoryUtils.d.ts +22 -0
  106. package/dist/utils/directoryUtils.js +59 -0
  107. package/dist/utils/getClientFromConfig.d.ts +17 -8
  108. package/dist/utils/getClientFromConfig.js +162 -17
  109. package/dist/utils/helperFunctions.d.ts +16 -2
  110. package/dist/utils/helperFunctions.js +19 -5
  111. package/dist/utils/loadConfigs.d.ts +34 -9
  112. package/dist/utils/loadConfigs.js +236 -316
  113. package/dist/utils/pathResolvers.d.ts +53 -0
  114. package/dist/utils/pathResolvers.js +72 -0
  115. package/dist/utils/projectConfig.d.ts +119 -0
  116. package/dist/utils/projectConfig.js +171 -0
  117. package/dist/utils/retryFailedPromises.js +4 -2
  118. package/dist/utils/sessionAuth.d.ts +48 -0
  119. package/dist/utils/sessionAuth.js +164 -0
  120. package/dist/utils/sessionPreservationExample.d.ts +1666 -0
  121. package/dist/utils/sessionPreservationExample.js +101 -0
  122. package/dist/utils/setupFiles.js +301 -41
  123. package/dist/utils/typeGuards.d.ts +35 -0
  124. package/dist/utils/typeGuards.js +57 -0
  125. package/dist/utils/versionDetection.js +145 -9
  126. package/dist/utils/yamlConverter.d.ts +53 -3
  127. package/dist/utils/yamlConverter.js +232 -13
  128. package/dist/utils/yamlLoader.d.ts +70 -0
  129. package/dist/utils/yamlLoader.js +263 -0
  130. package/dist/utilsController.d.ts +36 -3
  131. package/dist/utilsController.js +186 -56
  132. package/package.json +12 -2
  133. package/src/adapters/AdapterFactory.ts +263 -35
  134. package/src/adapters/TablesDBAdapter.ts +225 -36
  135. package/src/backups/operations/bucketBackup.ts +277 -0
  136. package/src/backups/operations/collectionBackup.ts +310 -0
  137. package/src/backups/operations/comprehensiveBackup.ts +342 -0
  138. package/src/backups/schemas/bucketManifest.ts +78 -0
  139. package/src/backups/schemas/comprehensiveManifest.ts +76 -0
  140. package/src/backups/tracking/centralizedTracking.ts +352 -0
  141. package/src/cli/commands/configCommands.ts +194 -0
  142. package/src/cli/commands/databaseCommands.ts +635 -0
  143. package/src/cli/commands/functionCommands.ts +379 -0
  144. package/src/cli/commands/schemaCommands.ts +163 -0
  145. package/src/cli/commands/transferCommands.ts +457 -0
  146. package/src/collections/attributes.ts +900 -621
  147. package/src/collections/attributes.ts.backup +1555 -0
  148. package/src/collections/indexes.ts +116 -114
  149. package/src/collections/methods.ts +295 -968
  150. package/src/collections/transferOperations.ts +516 -0
  151. package/src/collections/wipeOperations.ts +501 -0
  152. package/src/config/README.md +274 -0
  153. package/src/config/configMigration.ts +575 -0
  154. package/src/config/configValidation.ts +445 -0
  155. package/src/config/yamlConfig.ts +168 -55
  156. package/src/databases/methods.ts +3 -2
  157. package/src/databases/setup.ts +11 -138
  158. package/src/examples/yamlTerminologyExample.ts +341 -0
  159. package/src/functions/deployments.ts +14 -12
  160. package/src/functions/methods.ts +11 -11
  161. package/src/functions/templates/hono-typescript/README.md +286 -0
  162. package/src/functions/templates/hono-typescript/package.json +26 -0
  163. package/src/functions/templates/hono-typescript/src/adapters/request.ts +74 -0
  164. package/src/functions/templates/hono-typescript/src/adapters/response.ts +106 -0
  165. package/src/functions/templates/hono-typescript/src/app.ts +180 -0
  166. package/src/functions/templates/hono-typescript/src/context.ts +103 -0
  167. package/src/functions/templates/hono-typescript/src/index.ts +54 -0
  168. package/src/functions/templates/hono-typescript/src/middleware/appwrite.ts +119 -0
  169. package/src/functions/templates/hono-typescript/tsconfig.json +20 -0
  170. package/src/functions/templates/typescript-node/package.json +2 -1
  171. package/src/functions/templates/typescript-node/src/context.ts +103 -0
  172. package/src/functions/templates/typescript-node/src/index.ts +18 -12
  173. package/src/functions/templates/uv/pyproject.toml +1 -0
  174. package/src/functions/templates/uv/src/context.py +125 -0
  175. package/src/functions/templates/uv/src/index.py +35 -5
  176. package/src/init.ts +9 -11
  177. package/src/interactiveCLI.ts +278 -1596
  178. package/src/main.ts +418 -24
  179. package/src/migrations/afterImportActions.ts +71 -44
  180. package/src/migrations/appwriteToX.ts +100 -34
  181. package/src/migrations/dataLoader.ts +48 -34
  182. package/src/migrations/importController.ts +44 -39
  183. package/src/migrations/relationships.ts +28 -18
  184. package/src/migrations/services/ImportOrchestrator.ts +24 -27
  185. package/src/migrations/transfer.ts +159 -121
  186. package/src/migrations/yaml/YamlImportConfigLoader.ts +11 -4
  187. package/src/migrations/yaml/YamlImportIntegration.ts +47 -20
  188. package/src/migrations/yaml/generateImportSchemas.ts +751 -12
  189. package/src/setupController.ts +3 -2
  190. package/src/shared/backupMetadataSchema.ts +93 -0
  191. package/src/shared/backupTracking.ts +211 -0
  192. package/src/shared/confirmationDialogs.ts +19 -19
  193. package/src/shared/errorUtils.ts +110 -0
  194. package/src/shared/functionManager.ts +21 -20
  195. package/src/shared/indexManager.ts +12 -11
  196. package/src/shared/jsonSchemaGenerator.ts +38 -52
  197. package/src/shared/logging.ts +75 -0
  198. package/src/shared/messageFormatter.ts +14 -1
  199. package/src/shared/migrationHelpers.ts +45 -38
  200. package/src/shared/operationLogger.ts +11 -36
  201. package/src/shared/operationQueue.ts +322 -93
  202. package/src/shared/operationsTable.ts +338 -0
  203. package/src/shared/operationsTableSchema.ts +60 -0
  204. package/src/shared/relationshipExtractor.ts +214 -0
  205. package/src/shared/schemaGenerator.ts +179 -219
  206. package/src/storage/backupCompression.ts +88 -0
  207. package/src/storage/methods.ts +131 -34
  208. package/src/users/methods.ts +11 -9
  209. package/src/utils/configDiscovery.ts +502 -0
  210. package/src/utils/directoryUtils.ts +61 -0
  211. package/src/utils/getClientFromConfig.ts +205 -22
  212. package/src/utils/helperFunctions.ts +23 -5
  213. package/src/utils/loadConfigs.ts +313 -345
  214. package/src/utils/pathResolvers.ts +81 -0
  215. package/src/utils/projectConfig.ts +299 -0
  216. package/src/utils/retryFailedPromises.ts +4 -2
  217. package/src/utils/sessionAuth.ts +230 -0
  218. package/src/utils/setupFiles.ts +322 -54
  219. package/src/utils/typeGuards.ts +65 -0
  220. package/src/utils/versionDetection.ts +218 -64
  221. package/src/utils/yamlConverter.ts +296 -13
  222. package/src/utils/yamlLoader.ts +364 -0
  223. package/src/utilsController.ts +314 -110
  224. package/tests/README.md +497 -0
  225. package/tests/adapters/AdapterFactory.test.ts +277 -0
  226. package/tests/integration/syncOperations.test.ts +463 -0
  227. package/tests/jest.config.js +25 -0
  228. package/tests/migration/configMigration.test.ts +546 -0
  229. package/tests/setup.ts +62 -0
  230. package/tests/testUtils.ts +340 -0
  231. package/tests/utils/loadConfigs.test.ts +350 -0
  232. package/tests/validation/configValidation.test.ts +412 -0
  233. package/src/utils/schemaStrings.ts +0 -517
@@ -0,0 +1,501 @@
1
+ import {
2
+ Databases,
3
+ Query,
4
+ type Models,
5
+ } from "node-appwrite";
6
+ import type { DatabaseAdapter } from "../adapters/DatabaseAdapter.js";
7
+ import { tryAwaitWithRetry } from "../utils/helperFunctions.js";
8
+ import { MessageFormatter } from "../shared/messageFormatter.js";
9
+ import { ProgressManager } from "../shared/progressManager.js";
10
+ import { isRetryableError, isBulkNotSupportedError, isCriticalError } from "../shared/errorUtils.js";
11
+ import { delay } from "../utils/helperFunctions.js";
12
+ import { chunk } from "es-toolkit";
13
+ import pLimit from "p-limit";
14
+ import { fetchAllCollections } from "./methods.js";
15
+
16
+ /**
17
+ * Optimized streaming deletion of all documents from a collection
18
+ * Uses memory-efficient pagination instead of loading all documents into memory
19
+ */
20
+ async function wipeDocumentsFromCollection(
21
+ database: Databases,
22
+ databaseId: string,
23
+ collectionId: string
24
+ ) {
25
+ try {
26
+ // Use streaming deletion pattern - fetch and delete in batches without accumulating
27
+ const FETCH_BATCH_SIZE = 1000; // How many to fetch per query
28
+ const DELETE_BATCH_SIZE = 200; // How many to delete concurrently
29
+ const MAX_CONCURRENT_DELETIONS = 10; // Concurrent deletion operations
30
+
31
+ let totalDeleted = 0;
32
+ let cursor: string | undefined;
33
+ let hasMoreDocuments = true;
34
+
35
+ MessageFormatter.info("Starting optimized document deletion...", { prefix: "Wipe" });
36
+
37
+ // Create progress tracker (we'll update the total as we discover more documents)
38
+ const progress = ProgressManager.create(
39
+ `delete-${collectionId}`,
40
+ 1, // Start with 1, will update as we go
41
+ { title: "Deleting documents" }
42
+ );
43
+
44
+ while (hasMoreDocuments) {
45
+ // Fetch next batch of documents
46
+ const queries = [Query.limit(FETCH_BATCH_SIZE)];
47
+ if (cursor) {
48
+ queries.push(Query.cursorAfter(cursor));
49
+ }
50
+
51
+ const response = await database.listDocuments(databaseId, collectionId, queries);
52
+ const documents = response.documents;
53
+
54
+ if (documents.length === 0) {
55
+ hasMoreDocuments = false;
56
+ break;
57
+ }
58
+
59
+ // Update progress total as we discover more documents
60
+ if (documents.length === FETCH_BATCH_SIZE) {
61
+ // There might be more documents, update progress total
62
+ progress.setTotal(totalDeleted + documents.length + 1000); // Estimate more
63
+ }
64
+
65
+ MessageFormatter.progress(
66
+ `Processing batch: ${documents.length} documents (${totalDeleted + documents.length} total so far)`,
67
+ { prefix: "Wipe" }
68
+ );
69
+
70
+ // Delete this batch using optimized concurrent deletion
71
+ const documentBatches = chunk(documents, DELETE_BATCH_SIZE);
72
+ const limit = pLimit(MAX_CONCURRENT_DELETIONS);
73
+
74
+ const deletePromises = documentBatches.map((batch) =>
75
+ limit(async () => {
76
+ const batchDeletePromises = batch.map(async (doc) => {
77
+ try {
78
+ await tryAwaitWithRetry(async () =>
79
+ database.deleteDocument(databaseId, collectionId, doc.$id)
80
+ );
81
+ totalDeleted++;
82
+ progress.update(totalDeleted);
83
+ } catch (error: any) {
84
+ const errorMessage = error.message || String(error);
85
+
86
+ // Enhanced error handling for document deletion
87
+ if (errorMessage.includes("Document with the requested ID could not be found")) {
88
+ // Document already deleted, skip silently
89
+ totalDeleted++;
90
+ progress.update(totalDeleted);
91
+ } else if (isCriticalError(errorMessage)) {
92
+ // Critical error, log and rethrow to stop operation
93
+ MessageFormatter.error(
94
+ `Critical error deleting document ${doc.$id}: ${errorMessage}`,
95
+ error,
96
+ { prefix: "Wipe" }
97
+ );
98
+ throw error;
99
+ } else if (isRetryableError(errorMessage)) {
100
+ // Retryable error, will be handled by tryAwaitWithRetry
101
+ MessageFormatter.progress(
102
+ `Retryable error for document ${doc.$id}, will retry`,
103
+ { prefix: "Wipe" }
104
+ );
105
+ totalDeleted++;
106
+ progress.update(totalDeleted);
107
+ } else {
108
+ // Other non-critical errors, log but continue
109
+ MessageFormatter.error(
110
+ `Failed to delete document ${doc.$id}: ${errorMessage}`,
111
+ error,
112
+ { prefix: "Wipe" }
113
+ );
114
+ totalDeleted++;
115
+ progress.update(totalDeleted);
116
+ }
117
+ }
118
+ });
119
+
120
+ await Promise.all(batchDeletePromises);
121
+ })
122
+ );
123
+
124
+ await Promise.all(deletePromises);
125
+
126
+ // Set up cursor for next iteration
127
+ if (documents.length < FETCH_BATCH_SIZE) {
128
+ hasMoreDocuments = false;
129
+ } else {
130
+ cursor = documents[documents.length - 1].$id;
131
+ }
132
+
133
+ // Small delay between fetch cycles to be respectful to the API
134
+ await delay(10);
135
+ }
136
+
137
+ // Update final progress total
138
+ progress.setTotal(totalDeleted);
139
+ progress.stop();
140
+
141
+ if (totalDeleted === 0) {
142
+ MessageFormatter.info("No documents found to delete", { prefix: "Wipe" });
143
+ } else {
144
+ MessageFormatter.success(
145
+ `Successfully deleted ${totalDeleted} documents from collection ${collectionId}`,
146
+ { prefix: "Wipe" }
147
+ );
148
+ }
149
+
150
+ } catch (error) {
151
+ MessageFormatter.error(
152
+ `Error wiping documents from collection ${collectionId}`,
153
+ error instanceof Error ? error : new Error(String(error)),
154
+ { prefix: "Wipe" }
155
+ );
156
+ throw error;
157
+ }
158
+ }
159
+
160
+ export const wipeDatabase = async (
161
+ database: Databases,
162
+ databaseId: string
163
+ ): Promise<{ collectionId: string; collectionName: string }[]> => {
164
+ MessageFormatter.info(`Wiping database: ${databaseId}`, { prefix: "Wipe" });
165
+ const existingCollections = await fetchAllCollections(databaseId, database);
166
+ let collectionsDeleted: { collectionId: string; collectionName: string }[] =
167
+ [];
168
+
169
+ if (existingCollections.length === 0) {
170
+ MessageFormatter.info("No collections to delete", { prefix: "Wipe" });
171
+ return collectionsDeleted;
172
+ }
173
+
174
+ const progress = ProgressManager.create(
175
+ `wipe-db-${databaseId}`,
176
+ existingCollections.length,
177
+ { title: "Deleting collections" }
178
+ );
179
+
180
+ let processed = 0;
181
+ for (const { $id: collectionId, name: name } of existingCollections) {
182
+ MessageFormatter.progress(`Deleting collection: ${collectionId}`, { prefix: "Wipe" });
183
+ collectionsDeleted.push({
184
+ collectionId: collectionId,
185
+ collectionName: name,
186
+ });
187
+ tryAwaitWithRetry(
188
+ async () => await database.deleteCollection(databaseId, collectionId)
189
+ ); // Try to delete the collection and ignore errors if it doesn't exist or if it's already being deleted
190
+ processed++;
191
+ progress.update(processed);
192
+ await delay(100);
193
+ }
194
+
195
+ progress.stop();
196
+ MessageFormatter.success(`Deleted ${collectionsDeleted.length} collections from database`, { prefix: "Wipe" });
197
+ return collectionsDeleted;
198
+ };
199
+
200
+ export const wipeCollection = async (
201
+ database: Databases,
202
+ databaseId: string,
203
+ collectionId: string
204
+ ): Promise<void> => {
205
+ const collections = await database.listCollections(databaseId, [
206
+ Query.equal("$id", collectionId),
207
+ ]);
208
+ if (collections.total === 0) {
209
+ MessageFormatter.warning(`Collection ${collectionId} not found`, { prefix: "Wipe" });
210
+ return;
211
+ }
212
+ const collection = collections.collections[0];
213
+ await wipeDocumentsFromCollection(database, databaseId, collection.$id);
214
+ };
215
+
216
+ // TablesDB helpers for wiping
217
+ export const wipeAllTables = async (
218
+ adapter: DatabaseAdapter,
219
+ databaseId: string
220
+ ): Promise<{ tableId: string; tableName: string }[]> => {
221
+ MessageFormatter.info(`Wiping tables in database: ${databaseId}`, { prefix: 'Wipe' });
222
+ const res = await adapter.listTables({ databaseId, queries: [Query.limit(500)] });
223
+ const tables: any[] = (res as any).tables || [];
224
+ const deleted: { tableId: string; tableName: string }[] = [];
225
+ const progress = ProgressManager.create(`wipe-db-${databaseId}`, tables.length, { title: 'Deleting tables' });
226
+ let processed = 0;
227
+ for (const t of tables) {
228
+ try {
229
+ await adapter.deleteTable({ databaseId, tableId: t.$id });
230
+ deleted.push({ tableId: t.$id, tableName: t.name });
231
+ } catch (e) {
232
+ MessageFormatter.error(`Failed deleting table ${t.$id}`, e instanceof Error ? e : new Error(String(e)), { prefix: 'Wipe' });
233
+ }
234
+ processed++; progress.update(processed);
235
+ await delay(100);
236
+ }
237
+ progress.stop();
238
+ return deleted;
239
+ };
240
+
241
+ /**
242
+ * Optimized streaming deletion of all rows from a table
243
+ * Uses bulk deletion when available, falls back to optimized individual deletion
244
+ */
245
+ export const wipeTableRows = async (
246
+ adapter: DatabaseAdapter,
247
+ databaseId: string,
248
+ tableId: string
249
+ ): Promise<void> => {
250
+ try {
251
+ // Configuration for optimized deletion
252
+ const FETCH_BATCH_SIZE = 1000; // How many to fetch per query
253
+ const BULK_DELETE_BATCH_SIZE = 500; // How many to bulk delete at once
254
+ const INDIVIDUAL_DELETE_BATCH_SIZE = 200; // For fallback individual deletion
255
+ const MAX_CONCURRENT_OPERATIONS = 10; // Concurrent bulk/individual operations
256
+
257
+ let totalDeleted = 0;
258
+ let cursor: string | undefined;
259
+ let hasMoreRows = true;
260
+
261
+ MessageFormatter.info("Starting optimized table row deletion...", { prefix: "Wipe" });
262
+
263
+ // Create progress tracker (we'll update the total as we discover more rows)
264
+ const progress = ProgressManager.create(
265
+ `delete-${tableId}`,
266
+ 1, // Start with 1, will update as we go
267
+ { title: "Deleting table rows" }
268
+ );
269
+
270
+ while (hasMoreRows) {
271
+ // Fetch next batch of rows
272
+ const queries = [Query.limit(FETCH_BATCH_SIZE)];
273
+ if (cursor) {
274
+ queries.push(Query.cursorAfter(cursor));
275
+ }
276
+
277
+ const response = await adapter.listRows({ databaseId, tableId, queries });
278
+ const rows: any[] = (response as any).rows || [];
279
+
280
+ if (rows.length === 0) {
281
+ hasMoreRows = false;
282
+ break;
283
+ }
284
+
285
+ // Update progress total as we discover more rows
286
+ if (rows.length === FETCH_BATCH_SIZE) {
287
+ // There might be more rows, update progress total
288
+ progress.setTotal(totalDeleted + rows.length + 1000); // Estimate more
289
+ }
290
+
291
+ MessageFormatter.progress(
292
+ `Processing batch: ${rows.length} rows (${totalDeleted + rows.length} total so far)`,
293
+ { prefix: "Wipe" }
294
+ );
295
+
296
+ // Try to use bulk deletion first, fall back to individual deletion
297
+ const rowIds = rows.map((row: any) => row.$id);
298
+
299
+ // Check if bulk deletion is available and try it first
300
+ if (adapter.bulkDeleteRows) {
301
+ try {
302
+ // Attempt bulk deletion (available in TablesDB)
303
+ await tryBulkDeletion(adapter, databaseId, tableId, rowIds, BULK_DELETE_BATCH_SIZE, MAX_CONCURRENT_OPERATIONS);
304
+ totalDeleted += rows.length;
305
+ progress.update(totalDeleted);
306
+ } catch (bulkError) {
307
+ // Enhanced error handling: categorize the error and decide on fallback strategy
308
+ const errorMessage = bulkError instanceof Error ? bulkError.message : String(bulkError);
309
+
310
+ if (isRetryableError(errorMessage)) {
311
+ MessageFormatter.progress(
312
+ `Bulk deletion encountered retryable error, retrying with individual deletion for ${rows.length} rows`,
313
+ { prefix: "Wipe" }
314
+ );
315
+ } else if (isBulkNotSupportedError(errorMessage)) {
316
+ MessageFormatter.progress(
317
+ `Bulk deletion not supported by server, switching to individual deletion for ${rows.length} rows`,
318
+ { prefix: "Wipe" }
319
+ );
320
+ } else {
321
+ MessageFormatter.progress(
322
+ `Bulk deletion failed (${errorMessage}), falling back to individual deletion for ${rows.length} rows`,
323
+ { prefix: "Wipe" }
324
+ );
325
+ }
326
+
327
+ await tryIndividualDeletion(
328
+ adapter,
329
+ databaseId,
330
+ tableId,
331
+ rows,
332
+ INDIVIDUAL_DELETE_BATCH_SIZE,
333
+ MAX_CONCURRENT_OPERATIONS,
334
+ progress,
335
+ totalDeleted
336
+ );
337
+ totalDeleted += rows.length;
338
+ }
339
+ } else {
340
+ // Bulk deletion not available, use optimized individual deletion
341
+ MessageFormatter.progress(
342
+ `Using individual deletion for ${rows.length} rows (bulk deletion not available)`,
343
+ { prefix: "Wipe" }
344
+ );
345
+
346
+ await tryIndividualDeletion(
347
+ adapter,
348
+ databaseId,
349
+ tableId,
350
+ rows,
351
+ INDIVIDUAL_DELETE_BATCH_SIZE,
352
+ MAX_CONCURRENT_OPERATIONS,
353
+ progress,
354
+ totalDeleted
355
+ );
356
+ totalDeleted += rows.length;
357
+ }
358
+
359
+ // Set up cursor for next iteration
360
+ if (rows.length < FETCH_BATCH_SIZE) {
361
+ hasMoreRows = false;
362
+ } else {
363
+ cursor = rows[rows.length - 1].$id;
364
+ }
365
+
366
+ // Small delay between fetch cycles to be respectful to the API
367
+ await delay(10);
368
+ }
369
+
370
+ // Update final progress total
371
+ progress.setTotal(totalDeleted);
372
+ progress.stop();
373
+
374
+ if (totalDeleted === 0) {
375
+ MessageFormatter.info("No rows found to delete", { prefix: "Wipe" });
376
+ } else {
377
+ MessageFormatter.success(
378
+ `Successfully deleted ${totalDeleted} rows from table ${tableId}`,
379
+ { prefix: "Wipe" }
380
+ );
381
+ }
382
+
383
+ } catch (error) {
384
+ MessageFormatter.error(
385
+ `Error wiping rows from table ${tableId}`,
386
+ error instanceof Error ? error : new Error(String(error)),
387
+ { prefix: "Wipe" }
388
+ );
389
+ throw error;
390
+ }
391
+ };
392
+
393
+ /**
394
+ * Helper function to attempt bulk deletion of row IDs
395
+ */
396
+ async function tryBulkDeletion(
397
+ adapter: DatabaseAdapter,
398
+ databaseId: string,
399
+ tableId: string,
400
+ rowIds: string[],
401
+ batchSize: number,
402
+ maxConcurrent: number
403
+ ): Promise<void> {
404
+ if (!adapter.bulkDeleteRows) {
405
+ throw new Error("Bulk deletion not available on this adapter");
406
+ }
407
+
408
+ const limit = pLimit(maxConcurrent);
409
+ const batches = chunk(rowIds, batchSize);
410
+
411
+ const deletePromises = batches.map((batch) =>
412
+ limit(async () => {
413
+ try {
414
+ await tryAwaitWithRetry(async () =>
415
+ adapter.bulkDeleteRows!({ databaseId, tableId, rowIds: batch })
416
+ );
417
+ } catch (error: any) {
418
+ const errorMessage = error.message || String(error);
419
+
420
+ // Enhanced error handling for bulk deletion
421
+ if (isCriticalError(errorMessage)) {
422
+ MessageFormatter.error(
423
+ `Critical error in bulk deletion batch: ${errorMessage}`,
424
+ error,
425
+ { prefix: "Wipe" }
426
+ );
427
+ throw error;
428
+ } else {
429
+ // For non-critical errors in bulk deletion, re-throw to trigger fallback
430
+ throw new Error(`Bulk deletion batch failed: ${errorMessage}`);
431
+ }
432
+ }
433
+ })
434
+ );
435
+
436
+ await Promise.all(deletePromises);
437
+ }
438
+
439
+ /**
440
+ * Helper function for fallback individual deletion
441
+ */
442
+ async function tryIndividualDeletion(
443
+ adapter: DatabaseAdapter,
444
+ databaseId: string,
445
+ tableId: string,
446
+ rows: any[],
447
+ batchSize: number,
448
+ maxConcurrent: number,
449
+ progress: any,
450
+ baseDeleted: number
451
+ ): Promise<void> {
452
+ const limit = pLimit(maxConcurrent);
453
+ const batches = chunk(rows, batchSize);
454
+ let processedInBatch = 0;
455
+
456
+ const deletePromises = batches.map((batch) =>
457
+ limit(async () => {
458
+ const batchDeletePromises = batch.map(async (row: any) => {
459
+ try {
460
+ await tryAwaitWithRetry(async () =>
461
+ adapter.deleteRow({ databaseId, tableId, id: row.$id })
462
+ );
463
+ } catch (error: any) {
464
+ const errorMessage = error.message || String(error);
465
+
466
+ // Enhanced error handling for row deletion
467
+ if (errorMessage.includes("Row with the requested ID could not be found")) {
468
+ // Row already deleted, skip silently
469
+ } else if (isCriticalError(errorMessage)) {
470
+ // Critical error, log and rethrow to stop operation
471
+ MessageFormatter.error(
472
+ `Critical error deleting row ${row.$id}: ${errorMessage}`,
473
+ error,
474
+ { prefix: "Wipe" }
475
+ );
476
+ throw error;
477
+ } else if (isRetryableError(errorMessage)) {
478
+ // Retryable error, will be handled by tryAwaitWithRetry
479
+ MessageFormatter.progress(
480
+ `Retryable error for row ${row.$id}, will retry`,
481
+ { prefix: "Wipe" }
482
+ );
483
+ } else {
484
+ // Other non-critical errors, log but continue
485
+ MessageFormatter.error(
486
+ `Failed to delete row ${row.$id}: ${errorMessage}`,
487
+ error,
488
+ { prefix: "Wipe" }
489
+ );
490
+ }
491
+ }
492
+ processedInBatch++;
493
+ progress.update(baseDeleted + processedInBatch);
494
+ });
495
+
496
+ await Promise.all(batchDeletePromises);
497
+ })
498
+ );
499
+
500
+ await Promise.all(deletePromises);
501
+ }