appwrite-utils-cli 1.11.0 → 1.12.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 (250) hide show
  1. package/{src/adapters/index.ts → dist/adapters/index.d.ts} +0 -1
  2. package/dist/adapters/index.js +10 -0
  3. package/dist/backups/operations/bucketBackup.d.ts +19 -0
  4. package/dist/backups/operations/bucketBackup.js +197 -0
  5. package/dist/backups/operations/collectionBackup.d.ts +30 -0
  6. package/dist/backups/operations/collectionBackup.js +201 -0
  7. package/dist/backups/operations/comprehensiveBackup.d.ts +25 -0
  8. package/dist/backups/operations/comprehensiveBackup.js +238 -0
  9. package/dist/backups/schemas/bucketManifest.d.ts +93 -0
  10. package/dist/backups/schemas/bucketManifest.js +33 -0
  11. package/dist/backups/schemas/comprehensiveManifest.d.ts +108 -0
  12. package/dist/backups/schemas/comprehensiveManifest.js +32 -0
  13. package/dist/backups/tracking/centralizedTracking.d.ts +34 -0
  14. package/dist/backups/tracking/centralizedTracking.js +274 -0
  15. package/dist/cli/commands/configCommands.d.ts +8 -0
  16. package/dist/cli/commands/configCommands.js +210 -0
  17. package/dist/cli/commands/databaseCommands.d.ts +14 -0
  18. package/dist/cli/commands/databaseCommands.js +696 -0
  19. package/dist/cli/commands/functionCommands.d.ts +7 -0
  20. package/dist/cli/commands/functionCommands.js +330 -0
  21. package/dist/cli/commands/importFileCommands.d.ts +7 -0
  22. package/dist/cli/commands/importFileCommands.js +674 -0
  23. package/dist/cli/commands/schemaCommands.d.ts +7 -0
  24. package/dist/cli/commands/schemaCommands.js +169 -0
  25. package/dist/cli/commands/storageCommands.d.ts +5 -0
  26. package/dist/cli/commands/storageCommands.js +142 -0
  27. package/dist/cli/commands/transferCommands.d.ts +5 -0
  28. package/dist/cli/commands/transferCommands.js +382 -0
  29. package/dist/collections/columns.d.ts +13 -0
  30. package/dist/collections/columns.js +1339 -0
  31. package/dist/collections/indexes.d.ts +12 -0
  32. package/dist/collections/indexes.js +215 -0
  33. package/dist/collections/methods.d.ts +19 -0
  34. package/dist/collections/methods.js +605 -0
  35. package/dist/collections/tableOperations.d.ts +87 -0
  36. package/dist/collections/tableOperations.js +466 -0
  37. package/dist/collections/transferOperations.d.ts +8 -0
  38. package/dist/collections/transferOperations.js +411 -0
  39. package/dist/collections/wipeOperations.d.ts +17 -0
  40. package/dist/collections/wipeOperations.js +306 -0
  41. package/dist/databases/methods.d.ts +6 -0
  42. package/dist/databases/methods.js +35 -0
  43. package/dist/databases/setup.d.ts +5 -0
  44. package/dist/databases/setup.js +45 -0
  45. package/dist/examples/yamlTerminologyExample.d.ts +42 -0
  46. package/dist/examples/yamlTerminologyExample.js +272 -0
  47. package/dist/functions/deployments.d.ts +4 -0
  48. package/dist/functions/deployments.js +146 -0
  49. package/dist/functions/fnConfigDiscovery.d.ts +3 -0
  50. package/dist/functions/fnConfigDiscovery.js +108 -0
  51. package/dist/functions/methods.d.ts +16 -0
  52. package/dist/functions/methods.js +174 -0
  53. package/dist/init.d.ts +2 -0
  54. package/dist/init.js +57 -0
  55. package/dist/interactiveCLI.d.ts +36 -0
  56. package/dist/interactiveCLI.js +952 -0
  57. package/dist/main.d.ts +2 -0
  58. package/dist/main.js +1125 -0
  59. package/dist/migrations/afterImportActions.d.ts +17 -0
  60. package/dist/migrations/afterImportActions.js +305 -0
  61. package/dist/migrations/appwriteToX.d.ts +211 -0
  62. package/dist/migrations/appwriteToX.js +493 -0
  63. package/dist/migrations/comprehensiveTransfer.d.ts +147 -0
  64. package/dist/migrations/comprehensiveTransfer.js +1315 -0
  65. package/dist/migrations/dataLoader.d.ts +755 -0
  66. package/dist/migrations/dataLoader.js +1272 -0
  67. package/dist/migrations/importController.d.ts +25 -0
  68. package/dist/migrations/importController.js +283 -0
  69. package/dist/migrations/importDataActions.d.ts +50 -0
  70. package/dist/migrations/importDataActions.js +230 -0
  71. package/dist/migrations/relationships.d.ts +29 -0
  72. package/dist/migrations/relationships.js +203 -0
  73. package/dist/migrations/services/DataTransformationService.d.ts +55 -0
  74. package/dist/migrations/services/DataTransformationService.js +158 -0
  75. package/dist/migrations/services/FileHandlerService.d.ts +75 -0
  76. package/dist/migrations/services/FileHandlerService.js +236 -0
  77. package/dist/migrations/services/ImportOrchestrator.d.ts +99 -0
  78. package/dist/migrations/services/ImportOrchestrator.js +493 -0
  79. package/dist/migrations/services/RateLimitManager.d.ts +138 -0
  80. package/dist/migrations/services/RateLimitManager.js +279 -0
  81. package/dist/migrations/services/RelationshipResolver.d.ts +120 -0
  82. package/dist/migrations/services/RelationshipResolver.js +332 -0
  83. package/dist/migrations/services/UserMappingService.d.ts +109 -0
  84. package/dist/migrations/services/UserMappingService.js +277 -0
  85. package/dist/migrations/services/ValidationService.d.ts +74 -0
  86. package/dist/migrations/services/ValidationService.js +260 -0
  87. package/dist/migrations/transfer.d.ts +30 -0
  88. package/dist/migrations/transfer.js +661 -0
  89. package/dist/migrations/yaml/YamlImportConfigLoader.d.ts +131 -0
  90. package/dist/migrations/yaml/YamlImportConfigLoader.js +383 -0
  91. package/dist/migrations/yaml/YamlImportIntegration.d.ts +93 -0
  92. package/dist/migrations/yaml/YamlImportIntegration.js +341 -0
  93. package/dist/migrations/yaml/generateImportSchemas.d.ts +30 -0
  94. package/dist/migrations/yaml/generateImportSchemas.js +1327 -0
  95. package/dist/schemas/authUser.d.ts +24 -0
  96. package/dist/schemas/authUser.js +17 -0
  97. package/dist/setup.d.ts +2 -0
  98. package/{src/setup.ts → dist/setup.js} +0 -3
  99. package/dist/setupCommands.d.ts +58 -0
  100. package/dist/setupCommands.js +489 -0
  101. package/dist/setupController.d.ts +9 -0
  102. package/dist/setupController.js +34 -0
  103. package/dist/shared/backupMetadataSchema.d.ts +94 -0
  104. package/dist/shared/backupMetadataSchema.js +38 -0
  105. package/dist/shared/backupTracking.d.ts +18 -0
  106. package/dist/shared/backupTracking.js +176 -0
  107. package/dist/shared/confirmationDialogs.d.ts +75 -0
  108. package/dist/shared/confirmationDialogs.js +236 -0
  109. package/dist/shared/migrationHelpers.d.ts +61 -0
  110. package/dist/shared/migrationHelpers.js +145 -0
  111. package/{src/shared/operationLogger.ts → dist/shared/operationLogger.d.ts} +1 -11
  112. package/dist/shared/operationLogger.js +12 -0
  113. package/dist/shared/operationQueue.d.ts +40 -0
  114. package/dist/shared/operationQueue.js +310 -0
  115. package/dist/shared/operationsTable.d.ts +26 -0
  116. package/dist/shared/operationsTable.js +287 -0
  117. package/dist/shared/operationsTableSchema.d.ts +48 -0
  118. package/dist/shared/operationsTableSchema.js +35 -0
  119. package/dist/shared/progressManager.d.ts +62 -0
  120. package/dist/shared/progressManager.js +215 -0
  121. package/dist/shared/relationshipExtractor.d.ts +56 -0
  122. package/dist/shared/relationshipExtractor.js +138 -0
  123. package/dist/shared/selectionDialogs.d.ts +220 -0
  124. package/dist/shared/selectionDialogs.js +588 -0
  125. package/dist/storage/backupCompression.d.ts +20 -0
  126. package/dist/storage/backupCompression.js +67 -0
  127. package/dist/storage/methods.d.ts +44 -0
  128. package/dist/storage/methods.js +475 -0
  129. package/dist/storage/schemas.d.ts +842 -0
  130. package/dist/storage/schemas.js +175 -0
  131. package/dist/tables/indexManager.d.ts +65 -0
  132. package/dist/tables/indexManager.js +294 -0
  133. package/{src/types.ts → dist/types.d.ts} +1 -6
  134. package/dist/types.js +3 -0
  135. package/dist/users/methods.d.ts +16 -0
  136. package/dist/users/methods.js +276 -0
  137. package/dist/utils/configMigration.d.ts +1 -0
  138. package/dist/utils/configMigration.js +261 -0
  139. package/dist/utils/index.js +2 -0
  140. package/dist/utils/loadConfigs.d.ts +50 -0
  141. package/dist/utils/loadConfigs.js +357 -0
  142. package/dist/utils/setupFiles.d.ts +4 -0
  143. package/dist/utils/setupFiles.js +1190 -0
  144. package/dist/utilsController.d.ts +114 -0
  145. package/dist/utilsController.js +898 -0
  146. package/package.json +6 -3
  147. package/CHANGELOG.md +0 -35
  148. package/CONFIG_TODO.md +0 -1189
  149. package/SELECTION_DIALOGS.md +0 -146
  150. package/SERVICE_IMPLEMENTATION_REPORT.md +0 -462
  151. package/scripts/copy-templates.ts +0 -23
  152. package/src/backups/operations/bucketBackup.ts +0 -277
  153. package/src/backups/operations/collectionBackup.ts +0 -310
  154. package/src/backups/operations/comprehensiveBackup.ts +0 -342
  155. package/src/backups/schemas/bucketManifest.ts +0 -78
  156. package/src/backups/schemas/comprehensiveManifest.ts +0 -76
  157. package/src/backups/tracking/centralizedTracking.ts +0 -352
  158. package/src/cli/commands/configCommands.ts +0 -265
  159. package/src/cli/commands/databaseCommands.ts +0 -931
  160. package/src/cli/commands/functionCommands.ts +0 -419
  161. package/src/cli/commands/importFileCommands.ts +0 -815
  162. package/src/cli/commands/schemaCommands.ts +0 -200
  163. package/src/cli/commands/storageCommands.ts +0 -151
  164. package/src/cli/commands/transferCommands.ts +0 -454
  165. package/src/collections/attributes.ts.backup +0 -1555
  166. package/src/collections/columns.ts +0 -2025
  167. package/src/collections/indexes.ts +0 -350
  168. package/src/collections/methods.ts +0 -714
  169. package/src/collections/tableOperations.ts +0 -542
  170. package/src/collections/transferOperations.ts +0 -589
  171. package/src/collections/wipeOperations.ts +0 -449
  172. package/src/databases/methods.ts +0 -49
  173. package/src/databases/setup.ts +0 -77
  174. package/src/examples/yamlTerminologyExample.ts +0 -346
  175. package/src/functions/deployments.ts +0 -221
  176. package/src/functions/fnConfigDiscovery.ts +0 -103
  177. package/src/functions/methods.ts +0 -284
  178. package/src/init.ts +0 -62
  179. package/src/interactiveCLI.ts +0 -1201
  180. package/src/main.ts +0 -1517
  181. package/src/migrations/afterImportActions.ts +0 -579
  182. package/src/migrations/appwriteToX.ts +0 -668
  183. package/src/migrations/comprehensiveTransfer.ts +0 -2285
  184. package/src/migrations/dataLoader.ts +0 -1729
  185. package/src/migrations/importController.ts +0 -440
  186. package/src/migrations/importDataActions.ts +0 -315
  187. package/src/migrations/relationships.ts +0 -333
  188. package/src/migrations/services/DataTransformationService.ts +0 -196
  189. package/src/migrations/services/FileHandlerService.ts +0 -311
  190. package/src/migrations/services/ImportOrchestrator.ts +0 -675
  191. package/src/migrations/services/RateLimitManager.ts +0 -363
  192. package/src/migrations/services/RelationshipResolver.ts +0 -461
  193. package/src/migrations/services/UserMappingService.ts +0 -345
  194. package/src/migrations/services/ValidationService.ts +0 -349
  195. package/src/migrations/transfer.ts +0 -1113
  196. package/src/migrations/yaml/YamlImportConfigLoader.ts +0 -439
  197. package/src/migrations/yaml/YamlImportIntegration.ts +0 -446
  198. package/src/migrations/yaml/generateImportSchemas.ts +0 -1354
  199. package/src/schemas/authUser.ts +0 -23
  200. package/src/setupCommands.ts +0 -602
  201. package/src/setupController.ts +0 -43
  202. package/src/shared/backupMetadataSchema.ts +0 -93
  203. package/src/shared/backupTracking.ts +0 -211
  204. package/src/shared/confirmationDialogs.ts +0 -327
  205. package/src/shared/migrationHelpers.ts +0 -232
  206. package/src/shared/operationQueue.ts +0 -376
  207. package/src/shared/operationsTable.ts +0 -338
  208. package/src/shared/operationsTableSchema.ts +0 -60
  209. package/src/shared/progressManager.ts +0 -278
  210. package/src/shared/relationshipExtractor.ts +0 -214
  211. package/src/shared/selectionDialogs.ts +0 -802
  212. package/src/storage/backupCompression.ts +0 -88
  213. package/src/storage/methods.ts +0 -711
  214. package/src/storage/schemas.ts +0 -205
  215. package/src/tables/indexManager.ts +0 -409
  216. package/src/types/node-appwrite-tablesdb.d.ts +0 -44
  217. package/src/users/methods.ts +0 -358
  218. package/src/utils/configMigration.ts +0 -348
  219. package/src/utils/loadConfigs.ts +0 -457
  220. package/src/utils/setupFiles.ts +0 -1236
  221. package/src/utilsController.ts +0 -1263
  222. package/tests/README.md +0 -497
  223. package/tests/adapters/AdapterFactory.test.ts +0 -277
  224. package/tests/integration/syncOperations.test.ts +0 -463
  225. package/tests/jest.config.js +0 -25
  226. package/tests/migration/configMigration.test.ts +0 -546
  227. package/tests/setup.ts +0 -62
  228. package/tests/testUtils.ts +0 -340
  229. package/tests/utils/loadConfigs.test.ts +0 -350
  230. package/tests/validation/configValidation.test.ts +0 -412
  231. package/tsconfig.json +0 -44
  232. /package/{src → dist}/functions/templates/count-docs-in-collection/README.md +0 -0
  233. /package/{src → dist}/functions/templates/count-docs-in-collection/src/main.ts +0 -0
  234. /package/{src → dist}/functions/templates/count-docs-in-collection/src/request.ts +0 -0
  235. /package/{src → dist}/functions/templates/hono-typescript/README.md +0 -0
  236. /package/{src → dist}/functions/templates/hono-typescript/src/adapters/request.ts +0 -0
  237. /package/{src → dist}/functions/templates/hono-typescript/src/adapters/response.ts +0 -0
  238. /package/{src → dist}/functions/templates/hono-typescript/src/app.ts +0 -0
  239. /package/{src → dist}/functions/templates/hono-typescript/src/context.ts +0 -0
  240. /package/{src → dist}/functions/templates/hono-typescript/src/main.ts +0 -0
  241. /package/{src → dist}/functions/templates/hono-typescript/src/middleware/appwrite.ts +0 -0
  242. /package/{src → dist}/functions/templates/typescript-node/README.md +0 -0
  243. /package/{src → dist}/functions/templates/typescript-node/src/context.ts +0 -0
  244. /package/{src → dist}/functions/templates/typescript-node/src/main.ts +0 -0
  245. /package/{src → dist}/functions/templates/uv/README.md +0 -0
  246. /package/{src → dist}/functions/templates/uv/pyproject.toml +0 -0
  247. /package/{src → dist}/functions/templates/uv/src/__init__.py +0 -0
  248. /package/{src → dist}/functions/templates/uv/src/context.py +0 -0
  249. /package/{src → dist}/functions/templates/uv/src/main.py +0 -0
  250. /package/{src/utils/index.ts → dist/utils/index.d.ts} +0 -0
@@ -0,0 +1,411 @@
1
+ import { Client, Databases, ID, Query, } from "node-appwrite";
2
+ import { tryAwaitWithRetry, delay, calculateExponentialBackoff, MessageFormatter } from "appwrite-utils-helpers";
3
+ import { chunk } from "es-toolkit";
4
+ import { isLegacyDatabases } from "appwrite-utils-helpers";
5
+ import { getAdapter } from "appwrite-utils-helpers";
6
+ /**
7
+ * Transfers all documents from one collection to another in a different database
8
+ * within the same Appwrite Project
9
+ */
10
+ export const transferDocumentsBetweenDbsLocalToLocal = async (db, fromDbId, toDbId, fromCollId, toCollId) => {
11
+ // Use adapter path when available for bulk operations
12
+ if (!isLegacyDatabases(db)) {
13
+ const adapter = db;
14
+ const pageSize = 1000;
15
+ let lastId;
16
+ let totalTransferred = 0;
17
+ while (true) {
18
+ const queries = [Query.limit(pageSize)];
19
+ if (lastId)
20
+ queries.push(Query.cursorAfter(lastId));
21
+ const result = await adapter.listRows({ databaseId: fromDbId, tableId: fromCollId, queries });
22
+ const rows = result.rows || result.documents || [];
23
+ if (!rows.length)
24
+ break;
25
+ // Prepare rows: strip system fields, keep $id and $permissions
26
+ const prepared = rows.map((doc) => {
27
+ const data = { ...doc };
28
+ delete data.$databaseId;
29
+ delete data.$collectionId;
30
+ delete data.$createdAt;
31
+ delete data.$updatedAt;
32
+ return data; // keep $id and $permissions for upsert
33
+ });
34
+ // Prefer bulk upsert, then bulk create, then individual
35
+ if (typeof adapter.bulkUpsertRows === 'function' && adapter.supportsBulkOperations()) {
36
+ await adapter.bulkUpsertRows({ databaseId: toDbId, tableId: toCollId, rows: prepared });
37
+ }
38
+ else if (typeof adapter.bulkCreateRows === 'function' && adapter.supportsBulkOperations()) {
39
+ await adapter.bulkCreateRows({ databaseId: toDbId, tableId: toCollId, rows: prepared });
40
+ }
41
+ else {
42
+ for (const row of prepared) {
43
+ const id = row.$id || ID.unique();
44
+ const permissions = row.$permissions || [];
45
+ const { $id, $permissions, ...data } = row;
46
+ await adapter.createRow({ databaseId: toDbId, tableId: toCollId, id, data, permissions });
47
+ }
48
+ }
49
+ totalTransferred += rows.length;
50
+ if (rows.length < pageSize)
51
+ break;
52
+ lastId = rows[rows.length - 1].$id;
53
+ }
54
+ MessageFormatter.success(`Transferred ${totalTransferred} rows from ${fromDbId}/${fromCollId} to ${toDbId}/${toCollId}`, { prefix: "Transfer" });
55
+ return;
56
+ }
57
+ // Legacy path (Databases) – keep existing behavior
58
+ const legacyDb = db;
59
+ let fromCollDocs = await tryAwaitWithRetry(async () => legacyDb.listDocuments(fromDbId, fromCollId, [Query.limit(50)]));
60
+ let totalDocumentsTransferred = 0;
61
+ if (fromCollDocs.documents.length === 0) {
62
+ MessageFormatter.info(`No documents found in collection ${fromCollId}`, { prefix: "Transfer" });
63
+ return;
64
+ }
65
+ else if (fromCollDocs.documents.length < 50) {
66
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
67
+ const toCreateObject = {
68
+ ...doc,
69
+ };
70
+ delete toCreateObject.$databaseId;
71
+ delete toCreateObject.$collectionId;
72
+ delete toCreateObject.$createdAt;
73
+ delete toCreateObject.$updatedAt;
74
+ delete toCreateObject.$id;
75
+ delete toCreateObject.$permissions;
76
+ return tryAwaitWithRetry(async () => await legacyDb.createDocument(toDbId, toCollId, doc.$id, toCreateObject, doc.$permissions));
77
+ });
78
+ await Promise.all(batchedPromises);
79
+ totalDocumentsTransferred += fromCollDocs.documents.length;
80
+ }
81
+ else {
82
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
83
+ const toCreateObject = {
84
+ ...doc,
85
+ };
86
+ delete toCreateObject.$databaseId;
87
+ delete toCreateObject.$collectionId;
88
+ delete toCreateObject.$createdAt;
89
+ delete toCreateObject.$updatedAt;
90
+ delete toCreateObject.$id;
91
+ delete toCreateObject.$permissions;
92
+ return tryAwaitWithRetry(async () => legacyDb.createDocument(toDbId, toCollId, doc.$id, toCreateObject, doc.$permissions));
93
+ });
94
+ await Promise.all(batchedPromises);
95
+ totalDocumentsTransferred += fromCollDocs.documents.length;
96
+ while (fromCollDocs.documents.length === 50) {
97
+ fromCollDocs = await tryAwaitWithRetry(async () => await legacyDb.listDocuments(fromDbId, fromCollId, [
98
+ Query.limit(50),
99
+ Query.cursorAfter(fromCollDocs.documents[fromCollDocs.documents.length - 1].$id),
100
+ ]));
101
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
102
+ const toCreateObject = {
103
+ ...doc,
104
+ };
105
+ delete toCreateObject.$databaseId;
106
+ delete toCreateObject.$collectionId;
107
+ delete toCreateObject.$createdAt;
108
+ delete toCreateObject.$updatedAt;
109
+ delete toCreateObject.$id;
110
+ delete toCreateObject.$permissions;
111
+ return tryAwaitWithRetry(async () => await legacyDb.createDocument(toDbId, toCollId, doc.$id, toCreateObject, doc.$permissions));
112
+ });
113
+ await Promise.all(batchedPromises);
114
+ totalDocumentsTransferred += fromCollDocs.documents.length;
115
+ }
116
+ }
117
+ MessageFormatter.success(`Transferred ${totalDocumentsTransferred} documents from database ${fromDbId} to database ${toDbId} -- collection ${fromCollId} to collection ${toCollId}`, { prefix: "Transfer" });
118
+ };
119
+ /**
120
+ * Enhanced document transfer with fault tolerance and exponential backoff
121
+ */
122
+ const transferDocumentWithRetry = async (db, dbId, collectionId, documentId, documentData, permissions, maxRetries = 3, retryCount = 0) => {
123
+ try {
124
+ await db.createDocument(dbId, collectionId, documentId, documentData, permissions);
125
+ return true;
126
+ }
127
+ catch (error) {
128
+ // Check if document already exists
129
+ if (error.code === 409 || error.message?.toLowerCase().includes('already exists')) {
130
+ await db.updateDocument(dbId, collectionId, documentId, documentData, permissions);
131
+ }
132
+ if (retryCount < maxRetries) {
133
+ // Calculate exponential backoff: 1s, 2s, 4s, max 8s
134
+ const exponentialDelay = calculateExponentialBackoff(retryCount, 1000, 8000);
135
+ MessageFormatter.progress(`Retrying document ${documentId} (attempt ${retryCount + 1}/${maxRetries}, backoff: ${exponentialDelay}ms)`, { prefix: "Transfer" });
136
+ await delay(exponentialDelay);
137
+ return await transferDocumentWithRetry(db, dbId, collectionId, documentId, documentData, permissions, maxRetries, retryCount + 1);
138
+ }
139
+ MessageFormatter.error(`Failed to transfer document ${documentId} after ${maxRetries} retries`, error, { prefix: "Transfer" });
140
+ return false;
141
+ }
142
+ };
143
+ /**
144
+ * Check if endpoint supports bulk operations (cloud.appwrite.io)
145
+ */
146
+ const supportsBulkOperations = (endpoint) => {
147
+ return endpoint.includes('cloud.appwrite.io');
148
+ };
149
+ /**
150
+ * Direct HTTP implementation of bulk upsert API
151
+ */
152
+ const bulkUpsertDocuments = async (client, dbId, collectionId, documents) => {
153
+ const apiPath = `/databases/${dbId}/collections/${collectionId}/documents`;
154
+ const url = new URL(client.config.endpoint + apiPath);
155
+ const headers = {
156
+ 'Content-Type': 'application/json',
157
+ 'X-Appwrite-Project': client.config.project,
158
+ 'X-Appwrite-Key': client.config.key
159
+ };
160
+ const response = await fetch(url.toString(), {
161
+ method: 'PUT',
162
+ headers,
163
+ body: JSON.stringify({ documents })
164
+ });
165
+ if (!response.ok) {
166
+ const errorData = await response.json().catch(() => ({ message: 'Unknown error' }));
167
+ throw new Error(`Bulk upsert failed: ${response.status} - ${errorData.message || 'Unknown error'}`);
168
+ }
169
+ return await response.json();
170
+ };
171
+ /**
172
+ * Direct HTTP implementation of bulk create API
173
+ */
174
+ const bulkCreateDocuments = async (client, dbId, collectionId, documents) => {
175
+ const apiPath = `/databases/${dbId}/collections/${collectionId}/documents`;
176
+ const url = new URL(client.config.endpoint + apiPath);
177
+ const headers = {
178
+ 'Content-Type': 'application/json',
179
+ 'X-Appwrite-Project': client.config.project,
180
+ 'X-Appwrite-Key': client.config.key
181
+ };
182
+ const response = await fetch(url.toString(), {
183
+ method: 'POST',
184
+ headers,
185
+ body: JSON.stringify({ documents })
186
+ });
187
+ if (!response.ok) {
188
+ const errorData = await response.json().catch(() => ({ message: 'Unknown error' }));
189
+ throw new Error(`Bulk create failed: ${response.status} - ${errorData.message || 'Unknown error'}`);
190
+ }
191
+ return await response.json();
192
+ };
193
+ /**
194
+ * Enhanced bulk document creation using direct HTTP calls
195
+ */
196
+ const transferDocumentsBulkUpsert = async (client, dbId, collectionId, documents, maxBatchSize = 1000) => {
197
+ let successful = 0;
198
+ let failed = 0;
199
+ // Prepare documents for bulk upsert
200
+ const preparedDocs = documents.map(doc => {
201
+ const toCreateObject = { ...doc };
202
+ delete toCreateObject.$databaseId;
203
+ delete toCreateObject.$collectionId;
204
+ delete toCreateObject.$createdAt;
205
+ delete toCreateObject.$updatedAt;
206
+ // Keep $id and $permissions for upsert functionality
207
+ return toCreateObject;
208
+ });
209
+ // Process in batches based on plan limits
210
+ const documentBatches = chunk(preparedDocs, maxBatchSize);
211
+ for (const batch of documentBatches) {
212
+ MessageFormatter.progress(`Bulk upserting ${batch.length} documents...`, { prefix: "Transfer" });
213
+ try {
214
+ // Try bulk upsert with direct HTTP call
215
+ const result = await bulkUpsertDocuments(client, dbId, collectionId, batch);
216
+ successful += result.documents?.length || batch.length;
217
+ MessageFormatter.success(`Bulk upserted ${result.documents?.length || batch.length} documents`, { prefix: "Transfer" });
218
+ }
219
+ catch (error) {
220
+ MessageFormatter.progress(`Bulk upsert failed, trying smaller batch size...`, { prefix: "Transfer" });
221
+ // If bulk upsert fails, try with smaller batch size (Pro plan limit)
222
+ if (maxBatchSize > 100) {
223
+ const smallerBatches = chunk(batch, 100);
224
+ for (const smallBatch of smallerBatches) {
225
+ try {
226
+ const result = await bulkUpsertDocuments(client, dbId, collectionId, smallBatch);
227
+ successful += result.documents?.length || smallBatch.length;
228
+ MessageFormatter.success(`Bulk upserted ${result.documents?.length || smallBatch.length} documents (smaller batch)`, { prefix: "Transfer" });
229
+ }
230
+ catch (smallBatchError) {
231
+ MessageFormatter.progress(`Smaller batch failed, falling back to individual transfers...`, { prefix: "Transfer" });
232
+ // Fall back to individual document transfer for this batch
233
+ const db = new Databases(client);
234
+ const { successful: indivSuccessful, failed: indivFailed } = await transferDocumentBatchWithRetryFallback(db, dbId, collectionId, smallBatch.map((doc, index) => ({
235
+ ...doc,
236
+ $id: documents[documentBatches.indexOf(batch) * maxBatchSize + smallerBatches.indexOf(smallBatch) * 100 + index]?.$id || ID.unique(),
237
+ $permissions: documents[documentBatches.indexOf(batch) * maxBatchSize + smallerBatches.indexOf(smallBatch) * 100 + index]?.$permissions || []
238
+ })));
239
+ successful += indivSuccessful;
240
+ failed += indivFailed;
241
+ }
242
+ // Add delay between batches
243
+ await delay(200);
244
+ }
245
+ }
246
+ else {
247
+ // Fall back to individual document transfer
248
+ const db = new Databases(client);
249
+ const { successful: indivSuccessful, failed: indivFailed } = await transferDocumentBatchWithRetryFallback(db, dbId, collectionId, batch.map((doc, index) => ({
250
+ ...doc,
251
+ $id: documents[documentBatches.indexOf(batch) * maxBatchSize + index]?.$id || ID.unique(),
252
+ $permissions: documents[documentBatches.indexOf(batch) * maxBatchSize + index]?.$permissions || []
253
+ })));
254
+ successful += indivSuccessful;
255
+ failed += indivFailed;
256
+ }
257
+ }
258
+ // Add delay between major batches
259
+ if (documentBatches.indexOf(batch) < documentBatches.length - 1) {
260
+ await delay(500);
261
+ }
262
+ }
263
+ return { successful, failed };
264
+ };
265
+ /**
266
+ * Fallback batch document transfer with individual retry logic
267
+ */
268
+ const transferDocumentBatchWithRetryFallback = async (db, dbId, collectionId, documents, batchSize = 10) => {
269
+ let successful = 0;
270
+ let failed = 0;
271
+ // Process documents in smaller batches to avoid overwhelming the server
272
+ const documentBatches = chunk(documents, batchSize);
273
+ for (const batch of documentBatches) {
274
+ MessageFormatter.progress(`Processing batch of ${batch.length} documents...`, { prefix: "Transfer" });
275
+ const batchPromises = batch.map(async (doc) => {
276
+ const toCreateObject = { ...doc };
277
+ delete toCreateObject.$databaseId;
278
+ delete toCreateObject.$collectionId;
279
+ delete toCreateObject.$createdAt;
280
+ delete toCreateObject.$updatedAt;
281
+ delete toCreateObject.$id;
282
+ delete toCreateObject.$permissions;
283
+ const result = await transferDocumentWithRetry(db, dbId, collectionId, doc.$id, toCreateObject, doc.$permissions || []);
284
+ return { docId: doc.$id, success: result };
285
+ });
286
+ const results = await Promise.allSettled(batchPromises);
287
+ results.forEach((result, index) => {
288
+ if (result.status === 'fulfilled') {
289
+ if (result.value.success) {
290
+ successful++;
291
+ }
292
+ else {
293
+ failed++;
294
+ }
295
+ }
296
+ else {
297
+ MessageFormatter.error(`Batch promise rejected for document ${batch[index].$id}`, new Error(String(result.reason)), { prefix: "Transfer" });
298
+ failed++;
299
+ }
300
+ });
301
+ // Add delay between batches to avoid rate limiting
302
+ if (documentBatches.indexOf(batch) < documentBatches.length - 1) {
303
+ await delay(500);
304
+ }
305
+ }
306
+ return { successful, failed };
307
+ };
308
+ /**
309
+ * Enhanced batch document transfer with fault tolerance and bulk API support
310
+ */
311
+ const transferDocumentBatchWithRetry = async (db, client, dbId, collectionId, documents, batchSize = 10) => {
312
+ // Check if we can use bulk operations
313
+ if (supportsBulkOperations(client.config.endpoint)) {
314
+ MessageFormatter.info(`Using bulk upsert API for faster document transfer`, { prefix: "Transfer" });
315
+ // Try with Scale plan limit first (2500), then Pro (1000), then Free (100)
316
+ const batchSizes = [1000, 100]; // Start with Pro plan, fallback to Free
317
+ for (const maxBatchSize of batchSizes) {
318
+ try {
319
+ return await transferDocumentsBulkUpsert(client, dbId, collectionId, documents, maxBatchSize);
320
+ }
321
+ catch (error) {
322
+ MessageFormatter.progress(`Bulk upsert with batch size ${maxBatchSize} failed, trying smaller size...`, { prefix: "Transfer" });
323
+ continue;
324
+ }
325
+ }
326
+ // If all bulk operations fail, fall back to individual transfers
327
+ MessageFormatter.progress(`All bulk operations failed, falling back to individual document transfers`, { prefix: "Transfer" });
328
+ }
329
+ // Fall back to individual document transfer
330
+ return await transferDocumentBatchWithRetryFallback(db, dbId, collectionId, documents, batchSize);
331
+ };
332
+ export const transferDocumentsBetweenDbsLocalToRemote = async (localDb, endpoint, projectId, apiKey, fromDbId, toDbId, fromCollId, toCollId) => {
333
+ MessageFormatter.info(`Starting enhanced document transfer from ${fromCollId} to ${toCollId}...`, { prefix: "Transfer" });
334
+ // Prefer adapter for remote to enable bulk operations
335
+ const { adapter: remoteAdapter, client } = await getAdapter(endpoint, projectId, apiKey, 'auto');
336
+ const remoteDb = new Databases(client); // Legacy fallback for HTTP/individual
337
+ let totalDocumentsProcessed = 0;
338
+ let totalSuccessful = 0;
339
+ let totalFailed = 0;
340
+ // Fetch documents in larger batches (1000 at a time)
341
+ let hasMoreDocuments = true;
342
+ let lastDocumentId;
343
+ while (hasMoreDocuments) {
344
+ const queries = [Query.limit(1000)]; // Fetch 1000 documents at a time
345
+ if (lastDocumentId) {
346
+ queries.push(Query.cursorAfter(lastDocumentId));
347
+ }
348
+ const fromCollDocs = await tryAwaitWithRetry(async () => {
349
+ if (isLegacyDatabases(localDb)) {
350
+ return localDb.listDocuments(fromDbId, fromCollId, queries);
351
+ }
352
+ else {
353
+ const res = await localDb.listRows({ databaseId: fromDbId, tableId: fromCollId, queries });
354
+ const rows = res.rows || res.documents || [];
355
+ return { documents: rows };
356
+ }
357
+ });
358
+ if (fromCollDocs.documents.length === 0) {
359
+ hasMoreDocuments = false;
360
+ break;
361
+ }
362
+ MessageFormatter.progress(`Fetched ${fromCollDocs.documents.length} documents, processing for transfer...`, { prefix: "Transfer" });
363
+ // Prefer remote adapter bulk upsert if available
364
+ const prepared = fromCollDocs.documents.map((doc) => {
365
+ const data = { ...doc };
366
+ delete data.$databaseId;
367
+ delete data.$collectionId;
368
+ delete data.$createdAt;
369
+ delete data.$updatedAt;
370
+ return data; // Keep $id and $permissions for upsert
371
+ });
372
+ let successful = 0;
373
+ let failed = 0;
374
+ if (typeof remoteAdapter.bulkUpsertRows === 'function' && remoteAdapter.supportsBulkOperations()) {
375
+ try {
376
+ await remoteAdapter.bulkUpsertRows({ databaseId: toDbId, tableId: toCollId, rows: prepared });
377
+ successful = prepared.length;
378
+ }
379
+ catch (e) {
380
+ MessageFormatter.warning('Remote adapter bulk upsert failed, falling back to HTTP/individual', { prefix: 'Transfer' });
381
+ }
382
+ }
383
+ if (successful === 0) {
384
+ const res = await transferDocumentBatchWithRetry(remoteDb, client, toDbId, toCollId, fromCollDocs.documents);
385
+ successful = res.successful;
386
+ failed = res.failed;
387
+ }
388
+ totalDocumentsProcessed += fromCollDocs.documents.length;
389
+ totalSuccessful += successful;
390
+ totalFailed += failed;
391
+ // Check if we have more documents to process
392
+ if (fromCollDocs.documents.length < 1000) {
393
+ hasMoreDocuments = false;
394
+ }
395
+ else {
396
+ lastDocumentId = fromCollDocs.documents[fromCollDocs.documents.length - 1].$id;
397
+ }
398
+ MessageFormatter.debug(`Batch complete: ${successful} successful, ${failed} failed`, undefined, { prefix: "Transfer" });
399
+ }
400
+ if (totalDocumentsProcessed === 0) {
401
+ MessageFormatter.info(`No documents found in collection ${fromCollId}`, { prefix: "Transfer" });
402
+ return;
403
+ }
404
+ const message = `Total documents processed: ${totalDocumentsProcessed}, successful: ${totalSuccessful}, failed: ${totalFailed}`;
405
+ if (totalFailed > 0) {
406
+ MessageFormatter.warning(message, { prefix: "Transfer" });
407
+ }
408
+ else {
409
+ MessageFormatter.success(message, { prefix: "Transfer" });
410
+ }
411
+ };
@@ -0,0 +1,17 @@
1
+ import { Databases } from "node-appwrite";
2
+ import type { DatabaseAdapter } from "appwrite-utils-helpers";
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 deletion of all rows from a table.
14
+ * Uses bulk deletion when possible, but falls back to individual row deletion
15
+ * for tables with relationship columns (bulk delete not supported for those).
16
+ */
17
+ export declare const wipeTableRows: (adapter: DatabaseAdapter, databaseId: string, tableId: string) => Promise<void>;