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,7 @@
1
+ import { Databases } from "node-appwrite";
2
+ /**
3
+ * Transfers all documents from one collection to another in a different database
4
+ * within the same Appwrite Project
5
+ */
6
+ export declare const transferDocumentsBetweenDbsLocalToLocal: (db: Databases, fromDbId: string, toDbId: string, fromCollId: string, toCollId: string) => Promise<void>;
7
+ export declare const transferDocumentsBetweenDbsLocalToRemote: (localDb: Databases, endpoint: string, projectId: string, apiKey: string, fromDbId: string, toDbId: string, fromCollId: string, toCollId: string) => Promise<void>;
@@ -0,0 +1,331 @@
1
+ import { Client, Databases, ID, Query, } from "node-appwrite";
2
+ import { tryAwaitWithRetry, delay, calculateExponentialBackoff } from "../utils/helperFunctions.js";
3
+ import { MessageFormatter } from "../shared/messageFormatter.js";
4
+ import { chunk } from "es-toolkit";
5
+ /**
6
+ * Transfers all documents from one collection to another in a different database
7
+ * within the same Appwrite Project
8
+ */
9
+ export const transferDocumentsBetweenDbsLocalToLocal = async (db, fromDbId, toDbId, fromCollId, toCollId) => {
10
+ let fromCollDocs = await tryAwaitWithRetry(async () => db.listDocuments(fromDbId, fromCollId, [Query.limit(50)]));
11
+ let totalDocumentsTransferred = 0;
12
+ if (fromCollDocs.documents.length === 0) {
13
+ MessageFormatter.info(`No documents found in collection ${fromCollId}`, { prefix: "Transfer" });
14
+ return;
15
+ }
16
+ else if (fromCollDocs.documents.length < 50) {
17
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
18
+ const toCreateObject = {
19
+ ...doc,
20
+ };
21
+ delete toCreateObject.$databaseId;
22
+ delete toCreateObject.$collectionId;
23
+ delete toCreateObject.$createdAt;
24
+ delete toCreateObject.$updatedAt;
25
+ delete toCreateObject.$id;
26
+ delete toCreateObject.$permissions;
27
+ return tryAwaitWithRetry(async () => await db.createDocument(toDbId, toCollId, doc.$id, toCreateObject, doc.$permissions));
28
+ });
29
+ await Promise.all(batchedPromises);
30
+ totalDocumentsTransferred += fromCollDocs.documents.length;
31
+ }
32
+ else {
33
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
34
+ const toCreateObject = {
35
+ ...doc,
36
+ };
37
+ delete toCreateObject.$databaseId;
38
+ delete toCreateObject.$collectionId;
39
+ delete toCreateObject.$createdAt;
40
+ delete toCreateObject.$updatedAt;
41
+ delete toCreateObject.$id;
42
+ delete toCreateObject.$permissions;
43
+ return tryAwaitWithRetry(async () => db.createDocument(toDbId, toCollId, doc.$id, toCreateObject, doc.$permissions));
44
+ });
45
+ await Promise.all(batchedPromises);
46
+ totalDocumentsTransferred += fromCollDocs.documents.length;
47
+ while (fromCollDocs.documents.length === 50) {
48
+ fromCollDocs = await tryAwaitWithRetry(async () => await db.listDocuments(fromDbId, fromCollId, [
49
+ Query.limit(50),
50
+ Query.cursorAfter(fromCollDocs.documents[fromCollDocs.documents.length - 1].$id),
51
+ ]));
52
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
53
+ const toCreateObject = {
54
+ ...doc,
55
+ };
56
+ delete toCreateObject.$databaseId;
57
+ delete toCreateObject.$collectionId;
58
+ delete toCreateObject.$createdAt;
59
+ delete toCreateObject.$updatedAt;
60
+ delete toCreateObject.$id;
61
+ delete toCreateObject.$permissions;
62
+ return tryAwaitWithRetry(async () => await db.createDocument(toDbId, toCollId, doc.$id, toCreateObject, doc.$permissions));
63
+ });
64
+ await Promise.all(batchedPromises);
65
+ totalDocumentsTransferred += fromCollDocs.documents.length;
66
+ }
67
+ }
68
+ MessageFormatter.success(`Transferred ${totalDocumentsTransferred} documents from database ${fromDbId} to database ${toDbId} -- collection ${fromCollId} to collection ${toCollId}`, { prefix: "Transfer" });
69
+ };
70
+ /**
71
+ * Enhanced document transfer with fault tolerance and exponential backoff
72
+ */
73
+ const transferDocumentWithRetry = async (db, dbId, collectionId, documentId, documentData, permissions, maxRetries = 3, retryCount = 0) => {
74
+ try {
75
+ await db.createDocument(dbId, collectionId, documentId, documentData, permissions);
76
+ return true;
77
+ }
78
+ catch (error) {
79
+ // Check if document already exists
80
+ if (error.code === 409 || error.message?.toLowerCase().includes('already exists')) {
81
+ await db.updateDocument(dbId, collectionId, documentId, documentData, permissions);
82
+ }
83
+ if (retryCount < maxRetries) {
84
+ // Calculate exponential backoff: 1s, 2s, 4s, max 8s
85
+ const exponentialDelay = calculateExponentialBackoff(retryCount, 1000, 8000);
86
+ MessageFormatter.progress(`Retrying document ${documentId} (attempt ${retryCount + 1}/${maxRetries}, backoff: ${exponentialDelay}ms)`, { prefix: "Transfer" });
87
+ await delay(exponentialDelay);
88
+ return await transferDocumentWithRetry(db, dbId, collectionId, documentId, documentData, permissions, maxRetries, retryCount + 1);
89
+ }
90
+ MessageFormatter.error(`Failed to transfer document ${documentId} after ${maxRetries} retries`, error, { prefix: "Transfer" });
91
+ return false;
92
+ }
93
+ };
94
+ /**
95
+ * Check if endpoint supports bulk operations (cloud.appwrite.io)
96
+ */
97
+ const supportsBulkOperations = (endpoint) => {
98
+ return endpoint.includes('cloud.appwrite.io');
99
+ };
100
+ /**
101
+ * Direct HTTP implementation of bulk upsert API
102
+ */
103
+ const bulkUpsertDocuments = async (client, dbId, collectionId, documents) => {
104
+ const apiPath = `/databases/${dbId}/collections/${collectionId}/documents`;
105
+ const url = new URL(client.config.endpoint + apiPath);
106
+ const headers = {
107
+ 'Content-Type': 'application/json',
108
+ 'X-Appwrite-Project': client.config.project,
109
+ 'X-Appwrite-Key': client.config.key
110
+ };
111
+ const response = await fetch(url.toString(), {
112
+ method: 'PUT',
113
+ headers,
114
+ body: JSON.stringify({ documents })
115
+ });
116
+ if (!response.ok) {
117
+ const errorData = await response.json().catch(() => ({ message: 'Unknown error' }));
118
+ throw new Error(`Bulk upsert failed: ${response.status} - ${errorData.message || 'Unknown error'}`);
119
+ }
120
+ return await response.json();
121
+ };
122
+ /**
123
+ * Direct HTTP implementation of bulk create API
124
+ */
125
+ const bulkCreateDocuments = async (client, dbId, collectionId, documents) => {
126
+ const apiPath = `/databases/${dbId}/collections/${collectionId}/documents`;
127
+ const url = new URL(client.config.endpoint + apiPath);
128
+ const headers = {
129
+ 'Content-Type': 'application/json',
130
+ 'X-Appwrite-Project': client.config.project,
131
+ 'X-Appwrite-Key': client.config.key
132
+ };
133
+ const response = await fetch(url.toString(), {
134
+ method: 'POST',
135
+ headers,
136
+ body: JSON.stringify({ documents })
137
+ });
138
+ if (!response.ok) {
139
+ const errorData = await response.json().catch(() => ({ message: 'Unknown error' }));
140
+ throw new Error(`Bulk create failed: ${response.status} - ${errorData.message || 'Unknown error'}`);
141
+ }
142
+ return await response.json();
143
+ };
144
+ /**
145
+ * Enhanced bulk document creation using direct HTTP calls
146
+ */
147
+ const transferDocumentsBulkUpsert = async (client, dbId, collectionId, documents, maxBatchSize = 1000) => {
148
+ let successful = 0;
149
+ let failed = 0;
150
+ // Prepare documents for bulk upsert
151
+ const preparedDocs = documents.map(doc => {
152
+ const toCreateObject = { ...doc };
153
+ delete toCreateObject.$databaseId;
154
+ delete toCreateObject.$collectionId;
155
+ delete toCreateObject.$createdAt;
156
+ delete toCreateObject.$updatedAt;
157
+ // Keep $id and $permissions for upsert functionality
158
+ return toCreateObject;
159
+ });
160
+ // Process in batches based on plan limits
161
+ const documentBatches = chunk(preparedDocs, maxBatchSize);
162
+ for (const batch of documentBatches) {
163
+ MessageFormatter.progress(`Bulk upserting ${batch.length} documents...`, { prefix: "Transfer" });
164
+ try {
165
+ // Try bulk upsert with direct HTTP call
166
+ const result = await bulkUpsertDocuments(client, dbId, collectionId, batch);
167
+ successful += result.documents?.length || batch.length;
168
+ MessageFormatter.success(`Bulk upserted ${result.documents?.length || batch.length} documents`, { prefix: "Transfer" });
169
+ }
170
+ catch (error) {
171
+ MessageFormatter.progress(`Bulk upsert failed, trying smaller batch size...`, { prefix: "Transfer" });
172
+ // If bulk upsert fails, try with smaller batch size (Pro plan limit)
173
+ if (maxBatchSize > 100) {
174
+ const smallerBatches = chunk(batch, 100);
175
+ for (const smallBatch of smallerBatches) {
176
+ try {
177
+ const result = await bulkUpsertDocuments(client, dbId, collectionId, smallBatch);
178
+ successful += result.documents?.length || smallBatch.length;
179
+ MessageFormatter.success(`Bulk upserted ${result.documents?.length || smallBatch.length} documents (smaller batch)`, { prefix: "Transfer" });
180
+ }
181
+ catch (smallBatchError) {
182
+ MessageFormatter.progress(`Smaller batch failed, falling back to individual transfers...`, { prefix: "Transfer" });
183
+ // Fall back to individual document transfer for this batch
184
+ const db = new Databases(client);
185
+ const { successful: indivSuccessful, failed: indivFailed } = await transferDocumentBatchWithRetryFallback(db, dbId, collectionId, smallBatch.map((doc, index) => ({
186
+ ...doc,
187
+ $id: documents[documentBatches.indexOf(batch) * maxBatchSize + smallerBatches.indexOf(smallBatch) * 100 + index]?.$id || ID.unique(),
188
+ $permissions: documents[documentBatches.indexOf(batch) * maxBatchSize + smallerBatches.indexOf(smallBatch) * 100 + index]?.$permissions || []
189
+ })));
190
+ successful += indivSuccessful;
191
+ failed += indivFailed;
192
+ }
193
+ // Add delay between batches
194
+ await delay(200);
195
+ }
196
+ }
197
+ else {
198
+ // Fall back to individual document transfer
199
+ const db = new Databases(client);
200
+ const { successful: indivSuccessful, failed: indivFailed } = await transferDocumentBatchWithRetryFallback(db, dbId, collectionId, batch.map((doc, index) => ({
201
+ ...doc,
202
+ $id: documents[documentBatches.indexOf(batch) * maxBatchSize + index]?.$id || ID.unique(),
203
+ $permissions: documents[documentBatches.indexOf(batch) * maxBatchSize + index]?.$permissions || []
204
+ })));
205
+ successful += indivSuccessful;
206
+ failed += indivFailed;
207
+ }
208
+ }
209
+ // Add delay between major batches
210
+ if (documentBatches.indexOf(batch) < documentBatches.length - 1) {
211
+ await delay(500);
212
+ }
213
+ }
214
+ return { successful, failed };
215
+ };
216
+ /**
217
+ * Fallback batch document transfer with individual retry logic
218
+ */
219
+ const transferDocumentBatchWithRetryFallback = async (db, dbId, collectionId, documents, batchSize = 10) => {
220
+ let successful = 0;
221
+ let failed = 0;
222
+ // Process documents in smaller batches to avoid overwhelming the server
223
+ const documentBatches = chunk(documents, batchSize);
224
+ for (const batch of documentBatches) {
225
+ MessageFormatter.progress(`Processing batch of ${batch.length} documents...`, { prefix: "Transfer" });
226
+ const batchPromises = batch.map(async (doc) => {
227
+ const toCreateObject = { ...doc };
228
+ delete toCreateObject.$databaseId;
229
+ delete toCreateObject.$collectionId;
230
+ delete toCreateObject.$createdAt;
231
+ delete toCreateObject.$updatedAt;
232
+ delete toCreateObject.$id;
233
+ delete toCreateObject.$permissions;
234
+ const result = await transferDocumentWithRetry(db, dbId, collectionId, doc.$id, toCreateObject, doc.$permissions || []);
235
+ return { docId: doc.$id, success: result };
236
+ });
237
+ const results = await Promise.allSettled(batchPromises);
238
+ results.forEach((result, index) => {
239
+ if (result.status === 'fulfilled') {
240
+ if (result.value.success) {
241
+ successful++;
242
+ }
243
+ else {
244
+ failed++;
245
+ }
246
+ }
247
+ else {
248
+ MessageFormatter.error(`Batch promise rejected for document ${batch[index].$id}`, new Error(String(result.reason)), { prefix: "Transfer" });
249
+ failed++;
250
+ }
251
+ });
252
+ // Add delay between batches to avoid rate limiting
253
+ if (documentBatches.indexOf(batch) < documentBatches.length - 1) {
254
+ await delay(500);
255
+ }
256
+ }
257
+ return { successful, failed };
258
+ };
259
+ /**
260
+ * Enhanced batch document transfer with fault tolerance and bulk API support
261
+ */
262
+ const transferDocumentBatchWithRetry = async (db, client, dbId, collectionId, documents, batchSize = 10) => {
263
+ // Check if we can use bulk operations
264
+ if (supportsBulkOperations(client.config.endpoint)) {
265
+ MessageFormatter.info(`Using bulk upsert API for faster document transfer`, { prefix: "Transfer" });
266
+ // Try with Scale plan limit first (2500), then Pro (1000), then Free (100)
267
+ const batchSizes = [1000, 100]; // Start with Pro plan, fallback to Free
268
+ for (const maxBatchSize of batchSizes) {
269
+ try {
270
+ return await transferDocumentsBulkUpsert(client, dbId, collectionId, documents, maxBatchSize);
271
+ }
272
+ catch (error) {
273
+ MessageFormatter.progress(`Bulk upsert with batch size ${maxBatchSize} failed, trying smaller size...`, { prefix: "Transfer" });
274
+ continue;
275
+ }
276
+ }
277
+ // If all bulk operations fail, fall back to individual transfers
278
+ MessageFormatter.progress(`All bulk operations failed, falling back to individual document transfers`, { prefix: "Transfer" });
279
+ }
280
+ // Fall back to individual document transfer
281
+ return await transferDocumentBatchWithRetryFallback(db, dbId, collectionId, documents, batchSize);
282
+ };
283
+ export const transferDocumentsBetweenDbsLocalToRemote = async (localDb, endpoint, projectId, apiKey, fromDbId, toDbId, fromCollId, toCollId) => {
284
+ MessageFormatter.info(`Starting enhanced document transfer from ${fromCollId} to ${toCollId}...`, { prefix: "Transfer" });
285
+ const client = new Client()
286
+ .setEndpoint(endpoint)
287
+ .setProject(projectId)
288
+ .setKey(apiKey);
289
+ const remoteDb = new Databases(client);
290
+ let totalDocumentsProcessed = 0;
291
+ let totalSuccessful = 0;
292
+ let totalFailed = 0;
293
+ // Fetch documents in larger batches (1000 at a time)
294
+ let hasMoreDocuments = true;
295
+ let lastDocumentId;
296
+ while (hasMoreDocuments) {
297
+ const queries = [Query.limit(1000)]; // Fetch 1000 documents at a time
298
+ if (lastDocumentId) {
299
+ queries.push(Query.cursorAfter(lastDocumentId));
300
+ }
301
+ const fromCollDocs = await tryAwaitWithRetry(async () => localDb.listDocuments(fromDbId, fromCollId, queries));
302
+ if (fromCollDocs.documents.length === 0) {
303
+ hasMoreDocuments = false;
304
+ break;
305
+ }
306
+ MessageFormatter.progress(`Fetched ${fromCollDocs.documents.length} documents, processing for transfer...`, { prefix: "Transfer" });
307
+ const { successful, failed } = await transferDocumentBatchWithRetry(remoteDb, client, toDbId, toCollId, fromCollDocs.documents);
308
+ totalDocumentsProcessed += fromCollDocs.documents.length;
309
+ totalSuccessful += successful;
310
+ totalFailed += failed;
311
+ // Check if we have more documents to process
312
+ if (fromCollDocs.documents.length < 1000) {
313
+ hasMoreDocuments = false;
314
+ }
315
+ else {
316
+ lastDocumentId = fromCollDocs.documents[fromCollDocs.documents.length - 1].$id;
317
+ }
318
+ MessageFormatter.debug(`Batch complete: ${successful} successful, ${failed} failed`, undefined, { prefix: "Transfer" });
319
+ }
320
+ if (totalDocumentsProcessed === 0) {
321
+ MessageFormatter.info(`No documents found in collection ${fromCollId}`, { prefix: "Transfer" });
322
+ return;
323
+ }
324
+ const message = `Total documents processed: ${totalDocumentsProcessed}, successful: ${totalSuccessful}, failed: ${totalFailed}`;
325
+ if (totalFailed > 0) {
326
+ MessageFormatter.warning(message, { prefix: "Transfer" });
327
+ }
328
+ else {
329
+ MessageFormatter.success(message, { prefix: "Transfer" });
330
+ }
331
+ };
@@ -0,0 +1,16 @@
1
+ import { Databases } from "node-appwrite";
2
+ import type { DatabaseAdapter } from "../adapters/DatabaseAdapter.js";
3
+ export declare const wipeDatabase: (database: Databases, databaseId: string) => Promise<{
4
+ collectionId: string;
5
+ collectionName: string;
6
+ }[]>;
7
+ export declare const wipeCollection: (database: Databases, databaseId: string, collectionId: string) => Promise<void>;
8
+ export declare const wipeAllTables: (adapter: DatabaseAdapter, databaseId: string) => Promise<{
9
+ tableId: string;
10
+ tableName: string;
11
+ }[]>;
12
+ /**
13
+ * Optimized streaming deletion of all rows from a table
14
+ * Uses bulk deletion when available, falls back to optimized individual deletion
15
+ */
16
+ export declare const wipeTableRows: (adapter: DatabaseAdapter, databaseId: string, tableId: string) => Promise<void>;