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.
- package/CHANGELOG.md +199 -0
- package/README.md +251 -29
- package/dist/adapters/AdapterFactory.d.ts +10 -3
- package/dist/adapters/AdapterFactory.js +213 -17
- package/dist/adapters/TablesDBAdapter.js +60 -17
- package/dist/backups/operations/bucketBackup.d.ts +19 -0
- package/dist/backups/operations/bucketBackup.js +197 -0
- package/dist/backups/operations/collectionBackup.d.ts +30 -0
- package/dist/backups/operations/collectionBackup.js +201 -0
- package/dist/backups/operations/comprehensiveBackup.d.ts +25 -0
- package/dist/backups/operations/comprehensiveBackup.js +238 -0
- package/dist/backups/schemas/bucketManifest.d.ts +93 -0
- package/dist/backups/schemas/bucketManifest.js +33 -0
- package/dist/backups/schemas/comprehensiveManifest.d.ts +108 -0
- package/dist/backups/schemas/comprehensiveManifest.js +32 -0
- package/dist/backups/tracking/centralizedTracking.d.ts +34 -0
- package/dist/backups/tracking/centralizedTracking.js +274 -0
- package/dist/cli/commands/configCommands.d.ts +8 -0
- package/dist/cli/commands/configCommands.js +160 -0
- package/dist/cli/commands/databaseCommands.d.ts +13 -0
- package/dist/cli/commands/databaseCommands.js +478 -0
- package/dist/cli/commands/functionCommands.d.ts +7 -0
- package/dist/cli/commands/functionCommands.js +289 -0
- package/dist/cli/commands/schemaCommands.d.ts +7 -0
- package/dist/cli/commands/schemaCommands.js +134 -0
- package/dist/cli/commands/transferCommands.d.ts +5 -0
- package/dist/cli/commands/transferCommands.js +384 -0
- package/dist/collections/attributes.d.ts +5 -4
- package/dist/collections/attributes.js +539 -246
- package/dist/collections/indexes.js +39 -37
- package/dist/collections/methods.d.ts +2 -16
- package/dist/collections/methods.js +90 -538
- package/dist/collections/transferOperations.d.ts +7 -0
- package/dist/collections/transferOperations.js +331 -0
- package/dist/collections/wipeOperations.d.ts +16 -0
- package/dist/collections/wipeOperations.js +328 -0
- package/dist/config/configMigration.d.ts +87 -0
- package/dist/config/configMigration.js +390 -0
- package/dist/config/configValidation.d.ts +66 -0
- package/dist/config/configValidation.js +358 -0
- package/dist/config/yamlConfig.d.ts +455 -1
- package/dist/config/yamlConfig.js +145 -52
- package/dist/databases/methods.js +3 -2
- package/dist/databases/setup.d.ts +1 -2
- package/dist/databases/setup.js +9 -87
- package/dist/examples/yamlTerminologyExample.d.ts +42 -0
- package/dist/examples/yamlTerminologyExample.js +269 -0
- package/dist/functions/deployments.js +11 -10
- package/dist/functions/methods.d.ts +1 -1
- package/dist/functions/methods.js +5 -4
- package/dist/init.js +9 -9
- package/dist/interactiveCLI.d.ts +8 -17
- package/dist/interactiveCLI.js +181 -1172
- package/dist/main.js +364 -21
- package/dist/migrations/afterImportActions.js +22 -30
- package/dist/migrations/appwriteToX.js +71 -25
- package/dist/migrations/dataLoader.js +35 -26
- package/dist/migrations/importController.js +29 -30
- package/dist/migrations/relationships.js +13 -12
- package/dist/migrations/services/ImportOrchestrator.js +16 -19
- package/dist/migrations/transfer.js +46 -46
- package/dist/migrations/yaml/YamlImportConfigLoader.d.ts +3 -1
- package/dist/migrations/yaml/YamlImportConfigLoader.js +6 -3
- package/dist/migrations/yaml/YamlImportIntegration.d.ts +9 -3
- package/dist/migrations/yaml/YamlImportIntegration.js +22 -11
- package/dist/migrations/yaml/generateImportSchemas.d.ts +14 -1
- package/dist/migrations/yaml/generateImportSchemas.js +736 -7
- package/dist/schemas/authUser.d.ts +1 -1
- package/dist/setupController.js +3 -2
- package/dist/shared/backupMetadataSchema.d.ts +94 -0
- package/dist/shared/backupMetadataSchema.js +38 -0
- package/dist/shared/backupTracking.d.ts +18 -0
- package/dist/shared/backupTracking.js +176 -0
- package/dist/shared/confirmationDialogs.js +15 -15
- package/dist/shared/errorUtils.d.ts +54 -0
- package/dist/shared/errorUtils.js +95 -0
- package/dist/shared/functionManager.js +20 -19
- package/dist/shared/indexManager.js +12 -11
- package/dist/shared/jsonSchemaGenerator.js +10 -26
- package/dist/shared/logging.d.ts +51 -0
- package/dist/shared/logging.js +70 -0
- package/dist/shared/messageFormatter.d.ts +2 -0
- package/dist/shared/messageFormatter.js +10 -0
- package/dist/shared/migrationHelpers.d.ts +6 -16
- package/dist/shared/migrationHelpers.js +24 -21
- package/dist/shared/operationLogger.d.ts +8 -1
- package/dist/shared/operationLogger.js +11 -24
- package/dist/shared/operationQueue.d.ts +28 -1
- package/dist/shared/operationQueue.js +268 -66
- package/dist/shared/operationsTable.d.ts +26 -0
- package/dist/shared/operationsTable.js +286 -0
- package/dist/shared/operationsTableSchema.d.ts +48 -0
- package/dist/shared/operationsTableSchema.js +35 -0
- package/dist/shared/relationshipExtractor.d.ts +56 -0
- package/dist/shared/relationshipExtractor.js +138 -0
- package/dist/shared/schemaGenerator.d.ts +19 -1
- package/dist/shared/schemaGenerator.js +56 -75
- package/dist/storage/backupCompression.d.ts +20 -0
- package/dist/storage/backupCompression.js +67 -0
- package/dist/storage/methods.d.ts +16 -2
- package/dist/storage/methods.js +98 -14
- package/dist/users/methods.js +9 -8
- package/dist/utils/configDiscovery.d.ts +78 -0
- package/dist/utils/configDiscovery.js +430 -0
- package/dist/utils/directoryUtils.d.ts +22 -0
- package/dist/utils/directoryUtils.js +59 -0
- package/dist/utils/getClientFromConfig.d.ts +17 -8
- package/dist/utils/getClientFromConfig.js +162 -17
- package/dist/utils/helperFunctions.d.ts +16 -2
- package/dist/utils/helperFunctions.js +19 -5
- package/dist/utils/loadConfigs.d.ts +34 -9
- package/dist/utils/loadConfigs.js +236 -316
- package/dist/utils/pathResolvers.d.ts +53 -0
- package/dist/utils/pathResolvers.js +72 -0
- package/dist/utils/projectConfig.d.ts +119 -0
- package/dist/utils/projectConfig.js +171 -0
- package/dist/utils/retryFailedPromises.js +4 -2
- package/dist/utils/sessionAuth.d.ts +48 -0
- package/dist/utils/sessionAuth.js +164 -0
- package/dist/utils/sessionPreservationExample.d.ts +1666 -0
- package/dist/utils/sessionPreservationExample.js +101 -0
- package/dist/utils/setupFiles.js +301 -41
- package/dist/utils/typeGuards.d.ts +35 -0
- package/dist/utils/typeGuards.js +57 -0
- package/dist/utils/versionDetection.js +145 -9
- package/dist/utils/yamlConverter.d.ts +53 -3
- package/dist/utils/yamlConverter.js +232 -13
- package/dist/utils/yamlLoader.d.ts +70 -0
- package/dist/utils/yamlLoader.js +263 -0
- package/dist/utilsController.d.ts +36 -3
- package/dist/utilsController.js +186 -56
- package/package.json +12 -2
- package/src/adapters/AdapterFactory.ts +263 -35
- package/src/adapters/TablesDBAdapter.ts +225 -36
- package/src/backups/operations/bucketBackup.ts +277 -0
- package/src/backups/operations/collectionBackup.ts +310 -0
- package/src/backups/operations/comprehensiveBackup.ts +342 -0
- package/src/backups/schemas/bucketManifest.ts +78 -0
- package/src/backups/schemas/comprehensiveManifest.ts +76 -0
- package/src/backups/tracking/centralizedTracking.ts +352 -0
- package/src/cli/commands/configCommands.ts +194 -0
- package/src/cli/commands/databaseCommands.ts +635 -0
- package/src/cli/commands/functionCommands.ts +379 -0
- package/src/cli/commands/schemaCommands.ts +163 -0
- package/src/cli/commands/transferCommands.ts +457 -0
- package/src/collections/attributes.ts +900 -621
- package/src/collections/attributes.ts.backup +1555 -0
- package/src/collections/indexes.ts +116 -114
- package/src/collections/methods.ts +295 -968
- package/src/collections/transferOperations.ts +516 -0
- package/src/collections/wipeOperations.ts +501 -0
- package/src/config/README.md +274 -0
- package/src/config/configMigration.ts +575 -0
- package/src/config/configValidation.ts +445 -0
- package/src/config/yamlConfig.ts +168 -55
- package/src/databases/methods.ts +3 -2
- package/src/databases/setup.ts +11 -138
- package/src/examples/yamlTerminologyExample.ts +341 -0
- package/src/functions/deployments.ts +14 -12
- package/src/functions/methods.ts +11 -11
- package/src/functions/templates/hono-typescript/README.md +286 -0
- package/src/functions/templates/hono-typescript/package.json +26 -0
- package/src/functions/templates/hono-typescript/src/adapters/request.ts +74 -0
- package/src/functions/templates/hono-typescript/src/adapters/response.ts +106 -0
- package/src/functions/templates/hono-typescript/src/app.ts +180 -0
- package/src/functions/templates/hono-typescript/src/context.ts +103 -0
- package/src/functions/templates/hono-typescript/src/index.ts +54 -0
- package/src/functions/templates/hono-typescript/src/middleware/appwrite.ts +119 -0
- package/src/functions/templates/hono-typescript/tsconfig.json +20 -0
- package/src/functions/templates/typescript-node/package.json +2 -1
- package/src/functions/templates/typescript-node/src/context.ts +103 -0
- package/src/functions/templates/typescript-node/src/index.ts +18 -12
- package/src/functions/templates/uv/pyproject.toml +1 -0
- package/src/functions/templates/uv/src/context.py +125 -0
- package/src/functions/templates/uv/src/index.py +35 -5
- package/src/init.ts +9 -11
- package/src/interactiveCLI.ts +278 -1596
- package/src/main.ts +418 -24
- package/src/migrations/afterImportActions.ts +71 -44
- package/src/migrations/appwriteToX.ts +100 -34
- package/src/migrations/dataLoader.ts +48 -34
- package/src/migrations/importController.ts +44 -39
- package/src/migrations/relationships.ts +28 -18
- package/src/migrations/services/ImportOrchestrator.ts +24 -27
- package/src/migrations/transfer.ts +159 -121
- package/src/migrations/yaml/YamlImportConfigLoader.ts +11 -4
- package/src/migrations/yaml/YamlImportIntegration.ts +47 -20
- package/src/migrations/yaml/generateImportSchemas.ts +751 -12
- package/src/setupController.ts +3 -2
- package/src/shared/backupMetadataSchema.ts +93 -0
- package/src/shared/backupTracking.ts +211 -0
- package/src/shared/confirmationDialogs.ts +19 -19
- package/src/shared/errorUtils.ts +110 -0
- package/src/shared/functionManager.ts +21 -20
- package/src/shared/indexManager.ts +12 -11
- package/src/shared/jsonSchemaGenerator.ts +38 -52
- package/src/shared/logging.ts +75 -0
- package/src/shared/messageFormatter.ts +14 -1
- package/src/shared/migrationHelpers.ts +45 -38
- package/src/shared/operationLogger.ts +11 -36
- package/src/shared/operationQueue.ts +322 -93
- package/src/shared/operationsTable.ts +338 -0
- package/src/shared/operationsTableSchema.ts +60 -0
- package/src/shared/relationshipExtractor.ts +214 -0
- package/src/shared/schemaGenerator.ts +179 -219
- package/src/storage/backupCompression.ts +88 -0
- package/src/storage/methods.ts +131 -34
- package/src/users/methods.ts +11 -9
- package/src/utils/configDiscovery.ts +502 -0
- package/src/utils/directoryUtils.ts +61 -0
- package/src/utils/getClientFromConfig.ts +205 -22
- package/src/utils/helperFunctions.ts +23 -5
- package/src/utils/loadConfigs.ts +313 -345
- package/src/utils/pathResolvers.ts +81 -0
- package/src/utils/projectConfig.ts +299 -0
- package/src/utils/retryFailedPromises.ts +4 -2
- package/src/utils/sessionAuth.ts +230 -0
- package/src/utils/setupFiles.ts +322 -54
- package/src/utils/typeGuards.ts +65 -0
- package/src/utils/versionDetection.ts +218 -64
- package/src/utils/yamlConverter.ts +296 -13
- package/src/utils/yamlLoader.ts +364 -0
- package/src/utilsController.ts +314 -110
- package/tests/README.md +497 -0
- package/tests/adapters/AdapterFactory.test.ts +277 -0
- package/tests/integration/syncOperations.test.ts +463 -0
- package/tests/jest.config.js +25 -0
- package/tests/migration/configMigration.test.ts +546 -0
- package/tests/setup.ts +62 -0
- package/tests/testUtils.ts +340 -0
- package/tests/utils/loadConfigs.test.ts +350 -0
- package/tests/validation/configValidation.test.ts +412 -0
- 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
|
+
}
|