@njdamstra/appwrite-utils-cli 1.8.9

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 (392) hide show
  1. package/CHANGELOG.md +19 -0
  2. package/README.md +1133 -0
  3. package/dist/adapters/AdapterFactory.d.ts +94 -0
  4. package/dist/adapters/AdapterFactory.js +405 -0
  5. package/dist/adapters/DatabaseAdapter.d.ts +233 -0
  6. package/dist/adapters/DatabaseAdapter.js +50 -0
  7. package/dist/adapters/LegacyAdapter.d.ts +50 -0
  8. package/dist/adapters/LegacyAdapter.js +612 -0
  9. package/dist/adapters/TablesDBAdapter.d.ts +45 -0
  10. package/dist/adapters/TablesDBAdapter.js +571 -0
  11. package/dist/adapters/index.d.ts +11 -0
  12. package/dist/adapters/index.js +12 -0
  13. package/dist/backups/operations/bucketBackup.d.ts +19 -0
  14. package/dist/backups/operations/bucketBackup.js +197 -0
  15. package/dist/backups/operations/collectionBackup.d.ts +30 -0
  16. package/dist/backups/operations/collectionBackup.js +201 -0
  17. package/dist/backups/operations/comprehensiveBackup.d.ts +25 -0
  18. package/dist/backups/operations/comprehensiveBackup.js +238 -0
  19. package/dist/backups/schemas/bucketManifest.d.ts +93 -0
  20. package/dist/backups/schemas/bucketManifest.js +33 -0
  21. package/dist/backups/schemas/comprehensiveManifest.d.ts +108 -0
  22. package/dist/backups/schemas/comprehensiveManifest.js +32 -0
  23. package/dist/backups/tracking/centralizedTracking.d.ts +34 -0
  24. package/dist/backups/tracking/centralizedTracking.js +274 -0
  25. package/dist/cli/commands/configCommands.d.ts +8 -0
  26. package/dist/cli/commands/configCommands.js +166 -0
  27. package/dist/cli/commands/databaseCommands.d.ts +13 -0
  28. package/dist/cli/commands/databaseCommands.js +554 -0
  29. package/dist/cli/commands/functionCommands.d.ts +7 -0
  30. package/dist/cli/commands/functionCommands.js +330 -0
  31. package/dist/cli/commands/schemaCommands.d.ts +7 -0
  32. package/dist/cli/commands/schemaCommands.js +169 -0
  33. package/dist/cli/commands/storageCommands.d.ts +5 -0
  34. package/dist/cli/commands/storageCommands.js +143 -0
  35. package/dist/cli/commands/transferCommands.d.ts +5 -0
  36. package/dist/cli/commands/transferCommands.js +384 -0
  37. package/dist/collections/attributes.d.ts +13 -0
  38. package/dist/collections/attributes.js +1364 -0
  39. package/dist/collections/indexes.d.ts +12 -0
  40. package/dist/collections/indexes.js +217 -0
  41. package/dist/collections/methods.d.ts +19 -0
  42. package/dist/collections/methods.js +682 -0
  43. package/dist/collections/tableOperations.d.ts +86 -0
  44. package/dist/collections/tableOperations.js +434 -0
  45. package/dist/collections/transferOperations.d.ts +8 -0
  46. package/dist/collections/transferOperations.js +412 -0
  47. package/dist/collections/wipeOperations.d.ts +16 -0
  48. package/dist/collections/wipeOperations.js +233 -0
  49. package/dist/config/ConfigManager.d.ts +445 -0
  50. package/dist/config/ConfigManager.js +625 -0
  51. package/dist/config/configMigration.d.ts +87 -0
  52. package/dist/config/configMigration.js +390 -0
  53. package/dist/config/configValidation.d.ts +66 -0
  54. package/dist/config/configValidation.js +358 -0
  55. package/dist/config/index.d.ts +8 -0
  56. package/dist/config/index.js +7 -0
  57. package/dist/config/services/ConfigDiscoveryService.d.ts +126 -0
  58. package/dist/config/services/ConfigDiscoveryService.js +374 -0
  59. package/dist/config/services/ConfigLoaderService.d.ts +129 -0
  60. package/dist/config/services/ConfigLoaderService.js +540 -0
  61. package/dist/config/services/ConfigMergeService.d.ts +208 -0
  62. package/dist/config/services/ConfigMergeService.js +308 -0
  63. package/dist/config/services/ConfigValidationService.d.ts +214 -0
  64. package/dist/config/services/ConfigValidationService.js +310 -0
  65. package/dist/config/services/SessionAuthService.d.ts +225 -0
  66. package/dist/config/services/SessionAuthService.js +456 -0
  67. package/dist/config/services/__tests__/ConfigMergeService.test.d.ts +1 -0
  68. package/dist/config/services/__tests__/ConfigMergeService.test.js +271 -0
  69. package/dist/config/services/index.d.ts +13 -0
  70. package/dist/config/services/index.js +10 -0
  71. package/dist/config/yamlConfig.d.ts +722 -0
  72. package/dist/config/yamlConfig.js +702 -0
  73. package/dist/databases/methods.d.ts +6 -0
  74. package/dist/databases/methods.js +35 -0
  75. package/dist/databases/setup.d.ts +5 -0
  76. package/dist/databases/setup.js +45 -0
  77. package/dist/examples/yamlTerminologyExample.d.ts +42 -0
  78. package/dist/examples/yamlTerminologyExample.js +272 -0
  79. package/dist/functions/deployments.d.ts +4 -0
  80. package/dist/functions/deployments.js +146 -0
  81. package/dist/functions/fnConfigDiscovery.d.ts +3 -0
  82. package/dist/functions/fnConfigDiscovery.js +108 -0
  83. package/dist/functions/methods.d.ts +16 -0
  84. package/dist/functions/methods.js +162 -0
  85. package/dist/functions/pathResolution.d.ts +37 -0
  86. package/dist/functions/pathResolution.js +185 -0
  87. package/dist/functions/templates/count-docs-in-collection/README.md +54 -0
  88. package/dist/functions/templates/count-docs-in-collection/src/main.ts +159 -0
  89. package/dist/functions/templates/count-docs-in-collection/src/request.ts +9 -0
  90. package/dist/functions/templates/hono-typescript/README.md +286 -0
  91. package/dist/functions/templates/hono-typescript/src/adapters/request.ts +74 -0
  92. package/dist/functions/templates/hono-typescript/src/adapters/response.ts +106 -0
  93. package/dist/functions/templates/hono-typescript/src/app.ts +180 -0
  94. package/dist/functions/templates/hono-typescript/src/context.ts +103 -0
  95. package/dist/functions/templates/hono-typescript/src/index.ts +54 -0
  96. package/dist/functions/templates/hono-typescript/src/middleware/appwrite.ts +119 -0
  97. package/dist/functions/templates/typescript-node/README.md +32 -0
  98. package/dist/functions/templates/typescript-node/src/context.ts +103 -0
  99. package/dist/functions/templates/typescript-node/src/index.ts +29 -0
  100. package/dist/functions/templates/uv/README.md +31 -0
  101. package/dist/functions/templates/uv/pyproject.toml +30 -0
  102. package/dist/functions/templates/uv/src/__init__.py +0 -0
  103. package/dist/functions/templates/uv/src/context.py +125 -0
  104. package/dist/functions/templates/uv/src/index.py +46 -0
  105. package/dist/init.d.ts +2 -0
  106. package/dist/init.js +57 -0
  107. package/dist/interactiveCLI.d.ts +31 -0
  108. package/dist/interactiveCLI.js +898 -0
  109. package/dist/main.d.ts +2 -0
  110. package/dist/main.js +1172 -0
  111. package/dist/migrations/afterImportActions.d.ts +17 -0
  112. package/dist/migrations/afterImportActions.js +306 -0
  113. package/dist/migrations/appwriteToX.d.ts +211 -0
  114. package/dist/migrations/appwriteToX.js +491 -0
  115. package/dist/migrations/comprehensiveTransfer.d.ts +147 -0
  116. package/dist/migrations/comprehensiveTransfer.js +1317 -0
  117. package/dist/migrations/dataLoader.d.ts +753 -0
  118. package/dist/migrations/dataLoader.js +1250 -0
  119. package/dist/migrations/importController.d.ts +23 -0
  120. package/dist/migrations/importController.js +268 -0
  121. package/dist/migrations/importDataActions.d.ts +50 -0
  122. package/dist/migrations/importDataActions.js +230 -0
  123. package/dist/migrations/relationships.d.ts +29 -0
  124. package/dist/migrations/relationships.js +204 -0
  125. package/dist/migrations/services/DataTransformationService.d.ts +55 -0
  126. package/dist/migrations/services/DataTransformationService.js +158 -0
  127. package/dist/migrations/services/FileHandlerService.d.ts +75 -0
  128. package/dist/migrations/services/FileHandlerService.js +236 -0
  129. package/dist/migrations/services/ImportOrchestrator.d.ts +97 -0
  130. package/dist/migrations/services/ImportOrchestrator.js +485 -0
  131. package/dist/migrations/services/RateLimitManager.d.ts +138 -0
  132. package/dist/migrations/services/RateLimitManager.js +279 -0
  133. package/dist/migrations/services/RelationshipResolver.d.ts +120 -0
  134. package/dist/migrations/services/RelationshipResolver.js +332 -0
  135. package/dist/migrations/services/UserMappingService.d.ts +109 -0
  136. package/dist/migrations/services/UserMappingService.js +277 -0
  137. package/dist/migrations/services/ValidationService.d.ts +74 -0
  138. package/dist/migrations/services/ValidationService.js +260 -0
  139. package/dist/migrations/transfer.d.ts +26 -0
  140. package/dist/migrations/transfer.js +608 -0
  141. package/dist/migrations/yaml/YamlImportConfigLoader.d.ts +131 -0
  142. package/dist/migrations/yaml/YamlImportConfigLoader.js +383 -0
  143. package/dist/migrations/yaml/YamlImportIntegration.d.ts +93 -0
  144. package/dist/migrations/yaml/YamlImportIntegration.js +341 -0
  145. package/dist/migrations/yaml/generateImportSchemas.d.ts +30 -0
  146. package/dist/migrations/yaml/generateImportSchemas.js +1327 -0
  147. package/dist/schemas/authUser.d.ts +24 -0
  148. package/dist/schemas/authUser.js +17 -0
  149. package/dist/setup.d.ts +2 -0
  150. package/dist/setup.js +5 -0
  151. package/dist/setupCommands.d.ts +58 -0
  152. package/dist/setupCommands.js +490 -0
  153. package/dist/setupController.d.ts +9 -0
  154. package/dist/setupController.js +34 -0
  155. package/dist/shared/attributeMapper.d.ts +20 -0
  156. package/dist/shared/attributeMapper.js +203 -0
  157. package/dist/shared/backupMetadataSchema.d.ts +94 -0
  158. package/dist/shared/backupMetadataSchema.js +38 -0
  159. package/dist/shared/backupTracking.d.ts +18 -0
  160. package/dist/shared/backupTracking.js +176 -0
  161. package/dist/shared/confirmationDialogs.d.ts +75 -0
  162. package/dist/shared/confirmationDialogs.js +236 -0
  163. package/dist/shared/errorUtils.d.ts +54 -0
  164. package/dist/shared/errorUtils.js +95 -0
  165. package/dist/shared/functionManager.d.ts +48 -0
  166. package/dist/shared/functionManager.js +336 -0
  167. package/dist/shared/indexManager.d.ts +24 -0
  168. package/dist/shared/indexManager.js +151 -0
  169. package/dist/shared/jsonSchemaGenerator.d.ts +50 -0
  170. package/dist/shared/jsonSchemaGenerator.js +290 -0
  171. package/dist/shared/logging.d.ts +61 -0
  172. package/dist/shared/logging.js +116 -0
  173. package/dist/shared/messageFormatter.d.ts +39 -0
  174. package/dist/shared/messageFormatter.js +162 -0
  175. package/dist/shared/migrationHelpers.d.ts +61 -0
  176. package/dist/shared/migrationHelpers.js +145 -0
  177. package/dist/shared/operationLogger.d.ts +10 -0
  178. package/dist/shared/operationLogger.js +12 -0
  179. package/dist/shared/operationQueue.d.ts +40 -0
  180. package/dist/shared/operationQueue.js +311 -0
  181. package/dist/shared/operationsTable.d.ts +26 -0
  182. package/dist/shared/operationsTable.js +286 -0
  183. package/dist/shared/operationsTableSchema.d.ts +48 -0
  184. package/dist/shared/operationsTableSchema.js +35 -0
  185. package/dist/shared/progressManager.d.ts +62 -0
  186. package/dist/shared/progressManager.js +215 -0
  187. package/dist/shared/pydanticModelGenerator.d.ts +17 -0
  188. package/dist/shared/pydanticModelGenerator.js +615 -0
  189. package/dist/shared/relationshipExtractor.d.ts +56 -0
  190. package/dist/shared/relationshipExtractor.js +138 -0
  191. package/dist/shared/schemaGenerator.d.ts +40 -0
  192. package/dist/shared/schemaGenerator.js +556 -0
  193. package/dist/shared/selectionDialogs.d.ts +214 -0
  194. package/dist/shared/selectionDialogs.js +544 -0
  195. package/dist/storage/backupCompression.d.ts +20 -0
  196. package/dist/storage/backupCompression.js +67 -0
  197. package/dist/storage/methods.d.ts +32 -0
  198. package/dist/storage/methods.js +472 -0
  199. package/dist/storage/schemas.d.ts +842 -0
  200. package/dist/storage/schemas.js +175 -0
  201. package/dist/types.d.ts +4 -0
  202. package/dist/types.js +3 -0
  203. package/dist/users/methods.d.ts +16 -0
  204. package/dist/users/methods.js +277 -0
  205. package/dist/utils/ClientFactory.d.ts +87 -0
  206. package/dist/utils/ClientFactory.js +212 -0
  207. package/dist/utils/configDiscovery.d.ts +78 -0
  208. package/dist/utils/configDiscovery.js +472 -0
  209. package/dist/utils/configMigration.d.ts +1 -0
  210. package/dist/utils/configMigration.js +261 -0
  211. package/dist/utils/constantsGenerator.d.ts +31 -0
  212. package/dist/utils/constantsGenerator.js +321 -0
  213. package/dist/utils/dataConverters.d.ts +46 -0
  214. package/dist/utils/dataConverters.js +139 -0
  215. package/dist/utils/directoryUtils.d.ts +22 -0
  216. package/dist/utils/directoryUtils.js +59 -0
  217. package/dist/utils/getClientFromConfig.d.ts +39 -0
  218. package/dist/utils/getClientFromConfig.js +199 -0
  219. package/dist/utils/helperFunctions.d.ts +63 -0
  220. package/dist/utils/helperFunctions.js +156 -0
  221. package/dist/utils/index.d.ts +2 -0
  222. package/dist/utils/index.js +2 -0
  223. package/dist/utils/loadConfigs.d.ts +50 -0
  224. package/dist/utils/loadConfigs.js +358 -0
  225. package/dist/utils/pathResolvers.d.ts +53 -0
  226. package/dist/utils/pathResolvers.js +72 -0
  227. package/dist/utils/projectConfig.d.ts +119 -0
  228. package/dist/utils/projectConfig.js +171 -0
  229. package/dist/utils/retryFailedPromises.d.ts +2 -0
  230. package/dist/utils/retryFailedPromises.js +23 -0
  231. package/dist/utils/sessionAuth.d.ts +48 -0
  232. package/dist/utils/sessionAuth.js +164 -0
  233. package/dist/utils/setupFiles.d.ts +4 -0
  234. package/dist/utils/setupFiles.js +1192 -0
  235. package/dist/utils/typeGuards.d.ts +35 -0
  236. package/dist/utils/typeGuards.js +57 -0
  237. package/dist/utils/validationRules.d.ts +43 -0
  238. package/dist/utils/validationRules.js +42 -0
  239. package/dist/utils/versionDetection.d.ts +58 -0
  240. package/dist/utils/versionDetection.js +251 -0
  241. package/dist/utils/yamlConverter.d.ts +100 -0
  242. package/dist/utils/yamlConverter.js +428 -0
  243. package/dist/utils/yamlLoader.d.ts +70 -0
  244. package/dist/utils/yamlLoader.js +267 -0
  245. package/dist/utilsController.d.ts +106 -0
  246. package/dist/utilsController.js +863 -0
  247. package/package.json +75 -0
  248. package/scripts/copy-templates.ts +23 -0
  249. package/src/adapters/AdapterFactory.ts +510 -0
  250. package/src/adapters/DatabaseAdapter.ts +306 -0
  251. package/src/adapters/LegacyAdapter.ts +841 -0
  252. package/src/adapters/TablesDBAdapter.ts +773 -0
  253. package/src/adapters/index.ts +37 -0
  254. package/src/backups/operations/bucketBackup.ts +277 -0
  255. package/src/backups/operations/collectionBackup.ts +310 -0
  256. package/src/backups/operations/comprehensiveBackup.ts +342 -0
  257. package/src/backups/schemas/bucketManifest.ts +78 -0
  258. package/src/backups/schemas/comprehensiveManifest.ts +76 -0
  259. package/src/backups/tracking/centralizedTracking.ts +352 -0
  260. package/src/cli/commands/configCommands.ts +201 -0
  261. package/src/cli/commands/databaseCommands.ts +749 -0
  262. package/src/cli/commands/functionCommands.ts +418 -0
  263. package/src/cli/commands/schemaCommands.ts +200 -0
  264. package/src/cli/commands/storageCommands.ts +152 -0
  265. package/src/cli/commands/transferCommands.ts +457 -0
  266. package/src/collections/attributes.ts +2054 -0
  267. package/src/collections/attributes.ts.backup +1555 -0
  268. package/src/collections/indexes.ts +352 -0
  269. package/src/collections/methods.ts +745 -0
  270. package/src/collections/tableOperations.ts +506 -0
  271. package/src/collections/transferOperations.ts +590 -0
  272. package/src/collections/wipeOperations.ts +346 -0
  273. package/src/config/ConfigManager.ts +808 -0
  274. package/src/config/README.md +274 -0
  275. package/src/config/configMigration.ts +575 -0
  276. package/src/config/configValidation.ts +445 -0
  277. package/src/config/index.ts +10 -0
  278. package/src/config/services/ConfigDiscoveryService.ts +463 -0
  279. package/src/config/services/ConfigLoaderService.ts +740 -0
  280. package/src/config/services/ConfigMergeService.ts +388 -0
  281. package/src/config/services/ConfigValidationService.ts +394 -0
  282. package/src/config/services/SessionAuthService.ts +565 -0
  283. package/src/config/services/__tests__/ConfigMergeService.test.ts +351 -0
  284. package/src/config/services/index.ts +29 -0
  285. package/src/config/yamlConfig.ts +761 -0
  286. package/src/databases/methods.ts +49 -0
  287. package/src/databases/setup.ts +77 -0
  288. package/src/examples/yamlTerminologyExample.ts +346 -0
  289. package/src/functions/deployments.ts +220 -0
  290. package/src/functions/fnConfigDiscovery.ts +103 -0
  291. package/src/functions/methods.ts +271 -0
  292. package/src/functions/pathResolution.ts +227 -0
  293. package/src/functions/templates/count-docs-in-collection/README.md +54 -0
  294. package/src/functions/templates/count-docs-in-collection/src/main.ts +159 -0
  295. package/src/functions/templates/count-docs-in-collection/src/request.ts +9 -0
  296. package/src/functions/templates/hono-typescript/README.md +286 -0
  297. package/src/functions/templates/hono-typescript/src/adapters/request.ts +74 -0
  298. package/src/functions/templates/hono-typescript/src/adapters/response.ts +106 -0
  299. package/src/functions/templates/hono-typescript/src/app.ts +180 -0
  300. package/src/functions/templates/hono-typescript/src/context.ts +103 -0
  301. package/src/functions/templates/hono-typescript/src/index.ts +54 -0
  302. package/src/functions/templates/hono-typescript/src/middleware/appwrite.ts +119 -0
  303. package/src/functions/templates/typescript-node/README.md +32 -0
  304. package/src/functions/templates/typescript-node/src/context.ts +103 -0
  305. package/src/functions/templates/typescript-node/src/index.ts +29 -0
  306. package/src/functions/templates/uv/README.md +31 -0
  307. package/src/functions/templates/uv/pyproject.toml +30 -0
  308. package/src/functions/templates/uv/src/__init__.py +0 -0
  309. package/src/functions/templates/uv/src/context.py +125 -0
  310. package/src/functions/templates/uv/src/index.py +46 -0
  311. package/src/init.ts +62 -0
  312. package/src/interactiveCLI.ts +1136 -0
  313. package/src/main.ts +1661 -0
  314. package/src/migrations/afterImportActions.ts +580 -0
  315. package/src/migrations/appwriteToX.ts +664 -0
  316. package/src/migrations/comprehensiveTransfer.ts +2285 -0
  317. package/src/migrations/dataLoader.ts +1702 -0
  318. package/src/migrations/importController.ts +428 -0
  319. package/src/migrations/importDataActions.ts +315 -0
  320. package/src/migrations/relationships.ts +334 -0
  321. package/src/migrations/services/DataTransformationService.ts +196 -0
  322. package/src/migrations/services/FileHandlerService.ts +311 -0
  323. package/src/migrations/services/ImportOrchestrator.ts +666 -0
  324. package/src/migrations/services/RateLimitManager.ts +363 -0
  325. package/src/migrations/services/RelationshipResolver.ts +461 -0
  326. package/src/migrations/services/UserMappingService.ts +345 -0
  327. package/src/migrations/services/ValidationService.ts +349 -0
  328. package/src/migrations/transfer.ts +1068 -0
  329. package/src/migrations/yaml/YamlImportConfigLoader.ts +439 -0
  330. package/src/migrations/yaml/YamlImportIntegration.ts +446 -0
  331. package/src/migrations/yaml/generateImportSchemas.ts +1354 -0
  332. package/src/schemas/authUser.ts +23 -0
  333. package/src/setup.ts +8 -0
  334. package/src/setupCommands.ts +603 -0
  335. package/src/setupController.ts +43 -0
  336. package/src/shared/attributeMapper.ts +229 -0
  337. package/src/shared/backupMetadataSchema.ts +93 -0
  338. package/src/shared/backupTracking.ts +211 -0
  339. package/src/shared/confirmationDialogs.ts +327 -0
  340. package/src/shared/errorUtils.ts +110 -0
  341. package/src/shared/functionManager.ts +525 -0
  342. package/src/shared/indexManager.ts +254 -0
  343. package/src/shared/jsonSchemaGenerator.ts +383 -0
  344. package/src/shared/logging.ts +149 -0
  345. package/src/shared/messageFormatter.ts +208 -0
  346. package/src/shared/migrationHelpers.ts +232 -0
  347. package/src/shared/operationLogger.ts +20 -0
  348. package/src/shared/operationQueue.ts +377 -0
  349. package/src/shared/operationsTable.ts +338 -0
  350. package/src/shared/operationsTableSchema.ts +60 -0
  351. package/src/shared/progressManager.ts +278 -0
  352. package/src/shared/pydanticModelGenerator.ts +618 -0
  353. package/src/shared/relationshipExtractor.ts +214 -0
  354. package/src/shared/schemaGenerator.ts +644 -0
  355. package/src/shared/selectionDialogs.ts +749 -0
  356. package/src/storage/backupCompression.ts +88 -0
  357. package/src/storage/methods.ts +698 -0
  358. package/src/storage/schemas.ts +205 -0
  359. package/src/types/node-appwrite-tablesdb.d.ts +44 -0
  360. package/src/types.ts +9 -0
  361. package/src/users/methods.ts +359 -0
  362. package/src/utils/ClientFactory.ts +240 -0
  363. package/src/utils/configDiscovery.ts +557 -0
  364. package/src/utils/configMigration.ts +348 -0
  365. package/src/utils/constantsGenerator.ts +369 -0
  366. package/src/utils/dataConverters.ts +159 -0
  367. package/src/utils/directoryUtils.ts +61 -0
  368. package/src/utils/getClientFromConfig.ts +257 -0
  369. package/src/utils/helperFunctions.ts +228 -0
  370. package/src/utils/index.ts +2 -0
  371. package/src/utils/loadConfigs.ts +449 -0
  372. package/src/utils/pathResolvers.ts +81 -0
  373. package/src/utils/projectConfig.ts +299 -0
  374. package/src/utils/retryFailedPromises.ts +29 -0
  375. package/src/utils/sessionAuth.ts +230 -0
  376. package/src/utils/setupFiles.ts +1238 -0
  377. package/src/utils/typeGuards.ts +65 -0
  378. package/src/utils/validationRules.ts +88 -0
  379. package/src/utils/versionDetection.ts +292 -0
  380. package/src/utils/yamlConverter.ts +542 -0
  381. package/src/utils/yamlLoader.ts +371 -0
  382. package/src/utilsController.ts +1203 -0
  383. package/tests/README.md +497 -0
  384. package/tests/adapters/AdapterFactory.test.ts +277 -0
  385. package/tests/integration/syncOperations.test.ts +463 -0
  386. package/tests/jest.config.js +25 -0
  387. package/tests/migration/configMigration.test.ts +546 -0
  388. package/tests/setup.ts +62 -0
  389. package/tests/testUtils.ts +340 -0
  390. package/tests/utils/loadConfigs.test.ts +350 -0
  391. package/tests/validation/configValidation.test.ts +412 -0
  392. package/tsconfig.json +44 -0
@@ -0,0 +1,412 @@
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
+ import { isLegacyDatabases } from "../utils/typeGuards.js";
6
+ import { getAdapter } from "../utils/getClientFromConfig.js";
7
+ /**
8
+ * Transfers all documents from one collection to another in a different database
9
+ * within the same Appwrite Project
10
+ */
11
+ export const transferDocumentsBetweenDbsLocalToLocal = async (db, fromDbId, toDbId, fromCollId, toCollId) => {
12
+ // Use adapter path when available for bulk operations
13
+ if (!isLegacyDatabases(db)) {
14
+ const adapter = db;
15
+ const pageSize = 1000;
16
+ let lastId;
17
+ let totalTransferred = 0;
18
+ while (true) {
19
+ const queries = [Query.limit(pageSize)];
20
+ if (lastId)
21
+ queries.push(Query.cursorAfter(lastId));
22
+ const result = await adapter.listRows({ databaseId: fromDbId, tableId: fromCollId, queries });
23
+ const rows = result.rows || result.documents || [];
24
+ if (!rows.length)
25
+ break;
26
+ // Prepare rows: strip system fields, keep $id and $permissions
27
+ const prepared = rows.map((doc) => {
28
+ const data = { ...doc };
29
+ delete data.$databaseId;
30
+ delete data.$collectionId;
31
+ delete data.$createdAt;
32
+ delete data.$updatedAt;
33
+ return data; // keep $id and $permissions for upsert
34
+ });
35
+ // Prefer bulk upsert, then bulk create, then individual
36
+ if (typeof adapter.bulkUpsertRows === 'function' && adapter.supportsBulkOperations()) {
37
+ await adapter.bulkUpsertRows({ databaseId: toDbId, tableId: toCollId, rows: prepared });
38
+ }
39
+ else if (typeof adapter.bulkCreateRows === 'function' && adapter.supportsBulkOperations()) {
40
+ await adapter.bulkCreateRows({ databaseId: toDbId, tableId: toCollId, rows: prepared });
41
+ }
42
+ else {
43
+ for (const row of prepared) {
44
+ const id = row.$id || ID.unique();
45
+ const permissions = row.$permissions || [];
46
+ const { $id, $permissions, ...data } = row;
47
+ await adapter.createRow({ databaseId: toDbId, tableId: toCollId, id, data, permissions });
48
+ }
49
+ }
50
+ totalTransferred += rows.length;
51
+ if (rows.length < pageSize)
52
+ break;
53
+ lastId = rows[rows.length - 1].$id;
54
+ }
55
+ MessageFormatter.success(`Transferred ${totalTransferred} rows from ${fromDbId}/${fromCollId} to ${toDbId}/${toCollId}`, { prefix: "Transfer" });
56
+ return;
57
+ }
58
+ // Legacy path (Databases) – keep existing behavior
59
+ const legacyDb = db;
60
+ let fromCollDocs = await tryAwaitWithRetry(async () => legacyDb.listDocuments(fromDbId, fromCollId, [Query.limit(50)]));
61
+ let totalDocumentsTransferred = 0;
62
+ if (fromCollDocs.documents.length === 0) {
63
+ MessageFormatter.info(`No documents found in collection ${fromCollId}`, { prefix: "Transfer" });
64
+ return;
65
+ }
66
+ else if (fromCollDocs.documents.length < 50) {
67
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
68
+ const toCreateObject = {
69
+ ...doc,
70
+ };
71
+ delete toCreateObject.$databaseId;
72
+ delete toCreateObject.$collectionId;
73
+ delete toCreateObject.$createdAt;
74
+ delete toCreateObject.$updatedAt;
75
+ delete toCreateObject.$id;
76
+ delete toCreateObject.$permissions;
77
+ return tryAwaitWithRetry(async () => await legacyDb.createDocument(toDbId, toCollId, doc.$id, toCreateObject, doc.$permissions));
78
+ });
79
+ await Promise.all(batchedPromises);
80
+ totalDocumentsTransferred += fromCollDocs.documents.length;
81
+ }
82
+ else {
83
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
84
+ const toCreateObject = {
85
+ ...doc,
86
+ };
87
+ delete toCreateObject.$databaseId;
88
+ delete toCreateObject.$collectionId;
89
+ delete toCreateObject.$createdAt;
90
+ delete toCreateObject.$updatedAt;
91
+ delete toCreateObject.$id;
92
+ delete toCreateObject.$permissions;
93
+ return tryAwaitWithRetry(async () => legacyDb.createDocument(toDbId, toCollId, doc.$id, toCreateObject, doc.$permissions));
94
+ });
95
+ await Promise.all(batchedPromises);
96
+ totalDocumentsTransferred += fromCollDocs.documents.length;
97
+ while (fromCollDocs.documents.length === 50) {
98
+ fromCollDocs = await tryAwaitWithRetry(async () => await legacyDb.listDocuments(fromDbId, fromCollId, [
99
+ Query.limit(50),
100
+ Query.cursorAfter(fromCollDocs.documents[fromCollDocs.documents.length - 1].$id),
101
+ ]));
102
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
103
+ const toCreateObject = {
104
+ ...doc,
105
+ };
106
+ delete toCreateObject.$databaseId;
107
+ delete toCreateObject.$collectionId;
108
+ delete toCreateObject.$createdAt;
109
+ delete toCreateObject.$updatedAt;
110
+ delete toCreateObject.$id;
111
+ delete toCreateObject.$permissions;
112
+ return tryAwaitWithRetry(async () => await legacyDb.createDocument(toDbId, toCollId, doc.$id, toCreateObject, doc.$permissions));
113
+ });
114
+ await Promise.all(batchedPromises);
115
+ totalDocumentsTransferred += fromCollDocs.documents.length;
116
+ }
117
+ }
118
+ MessageFormatter.success(`Transferred ${totalDocumentsTransferred} documents from database ${fromDbId} to database ${toDbId} -- collection ${fromCollId} to collection ${toCollId}`, { prefix: "Transfer" });
119
+ };
120
+ /**
121
+ * Enhanced document transfer with fault tolerance and exponential backoff
122
+ */
123
+ const transferDocumentWithRetry = async (db, dbId, collectionId, documentId, documentData, permissions, maxRetries = 3, retryCount = 0) => {
124
+ try {
125
+ await db.createDocument(dbId, collectionId, documentId, documentData, permissions);
126
+ return true;
127
+ }
128
+ catch (error) {
129
+ // Check if document already exists
130
+ if (error.code === 409 || error.message?.toLowerCase().includes('already exists')) {
131
+ await db.updateDocument(dbId, collectionId, documentId, documentData, permissions);
132
+ }
133
+ if (retryCount < maxRetries) {
134
+ // Calculate exponential backoff: 1s, 2s, 4s, max 8s
135
+ const exponentialDelay = calculateExponentialBackoff(retryCount, 1000, 8000);
136
+ MessageFormatter.progress(`Retrying document ${documentId} (attempt ${retryCount + 1}/${maxRetries}, backoff: ${exponentialDelay}ms)`, { prefix: "Transfer" });
137
+ await delay(exponentialDelay);
138
+ return await transferDocumentWithRetry(db, dbId, collectionId, documentId, documentData, permissions, maxRetries, retryCount + 1);
139
+ }
140
+ MessageFormatter.error(`Failed to transfer document ${documentId} after ${maxRetries} retries`, error, { prefix: "Transfer" });
141
+ return false;
142
+ }
143
+ };
144
+ /**
145
+ * Check if endpoint supports bulk operations (cloud.appwrite.io)
146
+ */
147
+ const supportsBulkOperations = (endpoint) => {
148
+ return endpoint.includes('cloud.appwrite.io');
149
+ };
150
+ /**
151
+ * Direct HTTP implementation of bulk upsert API
152
+ */
153
+ const bulkUpsertDocuments = async (client, dbId, collectionId, documents) => {
154
+ const apiPath = `/databases/${dbId}/collections/${collectionId}/documents`;
155
+ const url = new URL(client.config.endpoint + apiPath);
156
+ const headers = {
157
+ 'Content-Type': 'application/json',
158
+ 'X-Appwrite-Project': client.config.project,
159
+ 'X-Appwrite-Key': client.config.key
160
+ };
161
+ const response = await fetch(url.toString(), {
162
+ method: 'PUT',
163
+ headers,
164
+ body: JSON.stringify({ documents })
165
+ });
166
+ if (!response.ok) {
167
+ const errorData = await response.json().catch(() => ({ message: 'Unknown error' }));
168
+ throw new Error(`Bulk upsert failed: ${response.status} - ${errorData.message || 'Unknown error'}`);
169
+ }
170
+ return await response.json();
171
+ };
172
+ /**
173
+ * Direct HTTP implementation of bulk create API
174
+ */
175
+ const bulkCreateDocuments = async (client, dbId, collectionId, documents) => {
176
+ const apiPath = `/databases/${dbId}/collections/${collectionId}/documents`;
177
+ const url = new URL(client.config.endpoint + apiPath);
178
+ const headers = {
179
+ 'Content-Type': 'application/json',
180
+ 'X-Appwrite-Project': client.config.project,
181
+ 'X-Appwrite-Key': client.config.key
182
+ };
183
+ const response = await fetch(url.toString(), {
184
+ method: 'POST',
185
+ headers,
186
+ body: JSON.stringify({ documents })
187
+ });
188
+ if (!response.ok) {
189
+ const errorData = await response.json().catch(() => ({ message: 'Unknown error' }));
190
+ throw new Error(`Bulk create failed: ${response.status} - ${errorData.message || 'Unknown error'}`);
191
+ }
192
+ return await response.json();
193
+ };
194
+ /**
195
+ * Enhanced bulk document creation using direct HTTP calls
196
+ */
197
+ const transferDocumentsBulkUpsert = async (client, dbId, collectionId, documents, maxBatchSize = 1000) => {
198
+ let successful = 0;
199
+ let failed = 0;
200
+ // Prepare documents for bulk upsert
201
+ const preparedDocs = documents.map(doc => {
202
+ const toCreateObject = { ...doc };
203
+ delete toCreateObject.$databaseId;
204
+ delete toCreateObject.$collectionId;
205
+ delete toCreateObject.$createdAt;
206
+ delete toCreateObject.$updatedAt;
207
+ // Keep $id and $permissions for upsert functionality
208
+ return toCreateObject;
209
+ });
210
+ // Process in batches based on plan limits
211
+ const documentBatches = chunk(preparedDocs, maxBatchSize);
212
+ for (const batch of documentBatches) {
213
+ MessageFormatter.progress(`Bulk upserting ${batch.length} documents...`, { prefix: "Transfer" });
214
+ try {
215
+ // Try bulk upsert with direct HTTP call
216
+ const result = await bulkUpsertDocuments(client, dbId, collectionId, batch);
217
+ successful += result.documents?.length || batch.length;
218
+ MessageFormatter.success(`Bulk upserted ${result.documents?.length || batch.length} documents`, { prefix: "Transfer" });
219
+ }
220
+ catch (error) {
221
+ MessageFormatter.progress(`Bulk upsert failed, trying smaller batch size...`, { prefix: "Transfer" });
222
+ // If bulk upsert fails, try with smaller batch size (Pro plan limit)
223
+ if (maxBatchSize > 100) {
224
+ const smallerBatches = chunk(batch, 100);
225
+ for (const smallBatch of smallerBatches) {
226
+ try {
227
+ const result = await bulkUpsertDocuments(client, dbId, collectionId, smallBatch);
228
+ successful += result.documents?.length || smallBatch.length;
229
+ MessageFormatter.success(`Bulk upserted ${result.documents?.length || smallBatch.length} documents (smaller batch)`, { prefix: "Transfer" });
230
+ }
231
+ catch (smallBatchError) {
232
+ MessageFormatter.progress(`Smaller batch failed, falling back to individual transfers...`, { prefix: "Transfer" });
233
+ // Fall back to individual document transfer for this batch
234
+ const db = new Databases(client);
235
+ const { successful: indivSuccessful, failed: indivFailed } = await transferDocumentBatchWithRetryFallback(db, dbId, collectionId, smallBatch.map((doc, index) => ({
236
+ ...doc,
237
+ $id: documents[documentBatches.indexOf(batch) * maxBatchSize + smallerBatches.indexOf(smallBatch) * 100 + index]?.$id || ID.unique(),
238
+ $permissions: documents[documentBatches.indexOf(batch) * maxBatchSize + smallerBatches.indexOf(smallBatch) * 100 + index]?.$permissions || []
239
+ })));
240
+ successful += indivSuccessful;
241
+ failed += indivFailed;
242
+ }
243
+ // Add delay between batches
244
+ await delay(200);
245
+ }
246
+ }
247
+ else {
248
+ // Fall back to individual document transfer
249
+ const db = new Databases(client);
250
+ const { successful: indivSuccessful, failed: indivFailed } = await transferDocumentBatchWithRetryFallback(db, dbId, collectionId, batch.map((doc, index) => ({
251
+ ...doc,
252
+ $id: documents[documentBatches.indexOf(batch) * maxBatchSize + index]?.$id || ID.unique(),
253
+ $permissions: documents[documentBatches.indexOf(batch) * maxBatchSize + index]?.$permissions || []
254
+ })));
255
+ successful += indivSuccessful;
256
+ failed += indivFailed;
257
+ }
258
+ }
259
+ // Add delay between major batches
260
+ if (documentBatches.indexOf(batch) < documentBatches.length - 1) {
261
+ await delay(500);
262
+ }
263
+ }
264
+ return { successful, failed };
265
+ };
266
+ /**
267
+ * Fallback batch document transfer with individual retry logic
268
+ */
269
+ const transferDocumentBatchWithRetryFallback = async (db, dbId, collectionId, documents, batchSize = 10) => {
270
+ let successful = 0;
271
+ let failed = 0;
272
+ // Process documents in smaller batches to avoid overwhelming the server
273
+ const documentBatches = chunk(documents, batchSize);
274
+ for (const batch of documentBatches) {
275
+ MessageFormatter.progress(`Processing batch of ${batch.length} documents...`, { prefix: "Transfer" });
276
+ const batchPromises = batch.map(async (doc) => {
277
+ const toCreateObject = { ...doc };
278
+ delete toCreateObject.$databaseId;
279
+ delete toCreateObject.$collectionId;
280
+ delete toCreateObject.$createdAt;
281
+ delete toCreateObject.$updatedAt;
282
+ delete toCreateObject.$id;
283
+ delete toCreateObject.$permissions;
284
+ const result = await transferDocumentWithRetry(db, dbId, collectionId, doc.$id, toCreateObject, doc.$permissions || []);
285
+ return { docId: doc.$id, success: result };
286
+ });
287
+ const results = await Promise.allSettled(batchPromises);
288
+ results.forEach((result, index) => {
289
+ if (result.status === 'fulfilled') {
290
+ if (result.value.success) {
291
+ successful++;
292
+ }
293
+ else {
294
+ failed++;
295
+ }
296
+ }
297
+ else {
298
+ MessageFormatter.error(`Batch promise rejected for document ${batch[index].$id}`, new Error(String(result.reason)), { prefix: "Transfer" });
299
+ failed++;
300
+ }
301
+ });
302
+ // Add delay between batches to avoid rate limiting
303
+ if (documentBatches.indexOf(batch) < documentBatches.length - 1) {
304
+ await delay(500);
305
+ }
306
+ }
307
+ return { successful, failed };
308
+ };
309
+ /**
310
+ * Enhanced batch document transfer with fault tolerance and bulk API support
311
+ */
312
+ const transferDocumentBatchWithRetry = async (db, client, dbId, collectionId, documents, batchSize = 10) => {
313
+ // Check if we can use bulk operations
314
+ if (supportsBulkOperations(client.config.endpoint)) {
315
+ MessageFormatter.info(`Using bulk upsert API for faster document transfer`, { prefix: "Transfer" });
316
+ // Try with Scale plan limit first (2500), then Pro (1000), then Free (100)
317
+ const batchSizes = [1000, 100]; // Start with Pro plan, fallback to Free
318
+ for (const maxBatchSize of batchSizes) {
319
+ try {
320
+ return await transferDocumentsBulkUpsert(client, dbId, collectionId, documents, maxBatchSize);
321
+ }
322
+ catch (error) {
323
+ MessageFormatter.progress(`Bulk upsert with batch size ${maxBatchSize} failed, trying smaller size...`, { prefix: "Transfer" });
324
+ continue;
325
+ }
326
+ }
327
+ // If all bulk operations fail, fall back to individual transfers
328
+ MessageFormatter.progress(`All bulk operations failed, falling back to individual document transfers`, { prefix: "Transfer" });
329
+ }
330
+ // Fall back to individual document transfer
331
+ return await transferDocumentBatchWithRetryFallback(db, dbId, collectionId, documents, batchSize);
332
+ };
333
+ export const transferDocumentsBetweenDbsLocalToRemote = async (localDb, endpoint, projectId, apiKey, fromDbId, toDbId, fromCollId, toCollId) => {
334
+ MessageFormatter.info(`Starting enhanced document transfer from ${fromCollId} to ${toCollId}...`, { prefix: "Transfer" });
335
+ // Prefer adapter for remote to enable bulk operations
336
+ const { adapter: remoteAdapter, client } = await getAdapter(endpoint, projectId, apiKey, 'auto');
337
+ const remoteDb = new Databases(client); // Legacy fallback for HTTP/individual
338
+ let totalDocumentsProcessed = 0;
339
+ let totalSuccessful = 0;
340
+ let totalFailed = 0;
341
+ // Fetch documents in larger batches (1000 at a time)
342
+ let hasMoreDocuments = true;
343
+ let lastDocumentId;
344
+ while (hasMoreDocuments) {
345
+ const queries = [Query.limit(1000)]; // Fetch 1000 documents at a time
346
+ if (lastDocumentId) {
347
+ queries.push(Query.cursorAfter(lastDocumentId));
348
+ }
349
+ const fromCollDocs = await tryAwaitWithRetry(async () => {
350
+ if (isLegacyDatabases(localDb)) {
351
+ return localDb.listDocuments(fromDbId, fromCollId, queries);
352
+ }
353
+ else {
354
+ const res = await localDb.listRows({ databaseId: fromDbId, tableId: fromCollId, queries });
355
+ const rows = res.rows || res.documents || [];
356
+ return { documents: rows };
357
+ }
358
+ });
359
+ if (fromCollDocs.documents.length === 0) {
360
+ hasMoreDocuments = false;
361
+ break;
362
+ }
363
+ MessageFormatter.progress(`Fetched ${fromCollDocs.documents.length} documents, processing for transfer...`, { prefix: "Transfer" });
364
+ // Prefer remote adapter bulk upsert if available
365
+ const prepared = fromCollDocs.documents.map((doc) => {
366
+ const data = { ...doc };
367
+ delete data.$databaseId;
368
+ delete data.$collectionId;
369
+ delete data.$createdAt;
370
+ delete data.$updatedAt;
371
+ return data; // Keep $id and $permissions for upsert
372
+ });
373
+ let successful = 0;
374
+ let failed = 0;
375
+ if (typeof remoteAdapter.bulkUpsertRows === 'function' && remoteAdapter.supportsBulkOperations()) {
376
+ try {
377
+ await remoteAdapter.bulkUpsertRows({ databaseId: toDbId, tableId: toCollId, rows: prepared });
378
+ successful = prepared.length;
379
+ }
380
+ catch (e) {
381
+ MessageFormatter.warning('Remote adapter bulk upsert failed, falling back to HTTP/individual', { prefix: 'Transfer' });
382
+ }
383
+ }
384
+ if (successful === 0) {
385
+ const res = await transferDocumentBatchWithRetry(remoteDb, client, toDbId, toCollId, fromCollDocs.documents);
386
+ successful = res.successful;
387
+ failed = res.failed;
388
+ }
389
+ totalDocumentsProcessed += fromCollDocs.documents.length;
390
+ totalSuccessful += successful;
391
+ totalFailed += failed;
392
+ // Check if we have more documents to process
393
+ if (fromCollDocs.documents.length < 1000) {
394
+ hasMoreDocuments = false;
395
+ }
396
+ else {
397
+ lastDocumentId = fromCollDocs.documents[fromCollDocs.documents.length - 1].$id;
398
+ }
399
+ MessageFormatter.debug(`Batch complete: ${successful} successful, ${failed} failed`, undefined, { prefix: "Transfer" });
400
+ }
401
+ if (totalDocumentsProcessed === 0) {
402
+ MessageFormatter.info(`No documents found in collection ${fromCollId}`, { prefix: "Transfer" });
403
+ return;
404
+ }
405
+ const message = `Total documents processed: ${totalDocumentsProcessed}, successful: ${totalSuccessful}, failed: ${totalFailed}`;
406
+ if (totalFailed > 0) {
407
+ MessageFormatter.warning(message, { prefix: "Transfer" });
408
+ }
409
+ else {
410
+ MessageFormatter.success(message, { prefix: "Transfer" });
411
+ }
412
+ };
@@ -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 deletion of all rows from a table using direct bulk deletion
14
+ * Uses Query.limit() to delete rows without fetching IDs first
15
+ */
16
+ export declare const wipeTableRows: (adapter: DatabaseAdapter, databaseId: string, tableId: string) => Promise<void>;
@@ -0,0 +1,233 @@
1
+ import { Databases, Query, } from "node-appwrite";
2
+ import { tryAwaitWithRetry } from "../utils/helperFunctions.js";
3
+ import { MessageFormatter } from "../shared/messageFormatter.js";
4
+ import { ProgressManager } from "../shared/progressManager.js";
5
+ import { isRetryableError, isCriticalError } from "../shared/errorUtils.js";
6
+ import { delay } from "../utils/helperFunctions.js";
7
+ import { chunk } from "es-toolkit";
8
+ import pLimit from "p-limit";
9
+ import { fetchAllCollections } from "./methods.js";
10
+ /**
11
+ * Optimized streaming deletion of all documents from a collection
12
+ * Uses memory-efficient pagination instead of loading all documents into memory
13
+ */
14
+ async function wipeDocumentsFromCollection(database, databaseId, collectionId) {
15
+ try {
16
+ // Use streaming deletion pattern - fetch and delete in batches without accumulating
17
+ const FETCH_BATCH_SIZE = 1000; // How many to fetch per query
18
+ const DELETE_BATCH_SIZE = 200; // How many to delete concurrently
19
+ const MAX_CONCURRENT_DELETIONS = 10; // Concurrent deletion operations
20
+ let totalDeleted = 0;
21
+ let cursor;
22
+ let hasMoreDocuments = true;
23
+ MessageFormatter.info("Starting optimized document deletion...", { prefix: "Wipe" });
24
+ // Create progress tracker (we'll update the total as we discover more documents)
25
+ const progress = ProgressManager.create(`delete-${collectionId}`, 1, // Start with 1, will update as we go
26
+ { title: "Deleting documents" });
27
+ while (hasMoreDocuments) {
28
+ // Fetch next batch of documents
29
+ const queries = [Query.limit(FETCH_BATCH_SIZE)];
30
+ if (cursor) {
31
+ queries.push(Query.cursorAfter(cursor));
32
+ }
33
+ const response = await database.listDocuments(databaseId, collectionId, queries);
34
+ const documents = response.documents;
35
+ if (documents.length === 0) {
36
+ hasMoreDocuments = false;
37
+ break;
38
+ }
39
+ // Update progress total as we discover more documents
40
+ if (documents.length === FETCH_BATCH_SIZE) {
41
+ // There might be more documents, update progress total
42
+ progress.setTotal(totalDeleted + documents.length + 1000); // Estimate more
43
+ }
44
+ MessageFormatter.progress(`Processing batch: ${documents.length} documents (${totalDeleted + documents.length} total so far)`, { prefix: "Wipe" });
45
+ // Delete this batch using optimized concurrent deletion
46
+ const documentBatches = chunk(documents, DELETE_BATCH_SIZE);
47
+ const limit = pLimit(MAX_CONCURRENT_DELETIONS);
48
+ const deletePromises = documentBatches.map((batch) => limit(async () => {
49
+ const batchDeletePromises = batch.map(async (doc) => {
50
+ try {
51
+ await tryAwaitWithRetry(async () => database.deleteDocument(databaseId, collectionId, doc.$id));
52
+ totalDeleted++;
53
+ progress.update(totalDeleted);
54
+ }
55
+ catch (error) {
56
+ const errorMessage = error.message || String(error);
57
+ // Enhanced error handling for document deletion
58
+ if (errorMessage.includes("Document with the requested ID could not be found")) {
59
+ // Document already deleted, skip silently
60
+ totalDeleted++;
61
+ progress.update(totalDeleted);
62
+ }
63
+ else if (isCriticalError(errorMessage)) {
64
+ // Critical error, log and rethrow to stop operation
65
+ MessageFormatter.error(`Critical error deleting document ${doc.$id}: ${errorMessage}`, error, { prefix: "Wipe" });
66
+ throw error;
67
+ }
68
+ else if (isRetryableError(errorMessage)) {
69
+ // Retryable error, will be handled by tryAwaitWithRetry
70
+ MessageFormatter.progress(`Retryable error for document ${doc.$id}, will retry`, { prefix: "Wipe" });
71
+ totalDeleted++;
72
+ progress.update(totalDeleted);
73
+ }
74
+ else {
75
+ // Other non-critical errors, log but continue
76
+ MessageFormatter.error(`Failed to delete document ${doc.$id}: ${errorMessage}`, error, { prefix: "Wipe" });
77
+ totalDeleted++;
78
+ progress.update(totalDeleted);
79
+ }
80
+ }
81
+ });
82
+ await Promise.all(batchDeletePromises);
83
+ }));
84
+ await Promise.all(deletePromises);
85
+ // Set up cursor for next iteration
86
+ if (documents.length < FETCH_BATCH_SIZE) {
87
+ hasMoreDocuments = false;
88
+ }
89
+ else {
90
+ cursor = documents[documents.length - 1].$id;
91
+ }
92
+ // Small delay between fetch cycles to be respectful to the API
93
+ await delay(10);
94
+ }
95
+ // Update final progress total
96
+ progress.setTotal(totalDeleted);
97
+ progress.stop();
98
+ if (totalDeleted === 0) {
99
+ MessageFormatter.info("No documents found to delete", { prefix: "Wipe" });
100
+ }
101
+ else {
102
+ MessageFormatter.success(`Successfully deleted ${totalDeleted} documents from collection ${collectionId}`, { prefix: "Wipe" });
103
+ }
104
+ }
105
+ catch (error) {
106
+ MessageFormatter.error(`Error wiping documents from collection ${collectionId}`, error instanceof Error ? error : new Error(String(error)), { prefix: "Wipe" });
107
+ throw error;
108
+ }
109
+ }
110
+ export const wipeDatabase = async (database, databaseId) => {
111
+ MessageFormatter.info(`Wiping database: ${databaseId}`, { prefix: "Wipe" });
112
+ const existingCollections = await fetchAllCollections(databaseId, database);
113
+ let collectionsDeleted = [];
114
+ if (existingCollections.length === 0) {
115
+ MessageFormatter.info("No collections to delete", { prefix: "Wipe" });
116
+ return collectionsDeleted;
117
+ }
118
+ const progress = ProgressManager.create(`wipe-db-${databaseId}`, existingCollections.length, { title: "Deleting collections" });
119
+ let processed = 0;
120
+ for (const { $id: collectionId, name: name } of existingCollections) {
121
+ MessageFormatter.progress(`Deleting collection: ${collectionId}`, { prefix: "Wipe" });
122
+ collectionsDeleted.push({
123
+ collectionId: collectionId,
124
+ collectionName: name,
125
+ });
126
+ tryAwaitWithRetry(async () => await database.deleteCollection(databaseId, collectionId)); // Try to delete the collection and ignore errors if it doesn't exist or if it's already being deleted
127
+ processed++;
128
+ progress.update(processed);
129
+ await delay(100);
130
+ }
131
+ progress.stop();
132
+ MessageFormatter.success(`Deleted ${collectionsDeleted.length} collections from database`, { prefix: "Wipe" });
133
+ return collectionsDeleted;
134
+ };
135
+ export const wipeCollection = async (database, databaseId, collectionId) => {
136
+ const collections = await database.listCollections(databaseId, [
137
+ Query.equal("$id", collectionId),
138
+ ]);
139
+ if (collections.total === 0) {
140
+ MessageFormatter.warning(`Collection ${collectionId} not found`, { prefix: "Wipe" });
141
+ return;
142
+ }
143
+ const collection = collections.collections[0];
144
+ await wipeDocumentsFromCollection(database, databaseId, collection.$id);
145
+ };
146
+ // TablesDB helpers for wiping
147
+ export const wipeAllTables = async (adapter, databaseId) => {
148
+ MessageFormatter.info(`Wiping tables in database: ${databaseId}`, { prefix: 'Wipe' });
149
+ const res = await adapter.listTables({ databaseId, queries: [Query.limit(500)] });
150
+ const tables = res.tables || [];
151
+ const deleted = [];
152
+ const progress = ProgressManager.create(`wipe-db-${databaseId}`, tables.length, { title: 'Deleting tables' });
153
+ let processed = 0;
154
+ for (const t of tables) {
155
+ try {
156
+ await adapter.deleteTable({ databaseId, tableId: t.$id });
157
+ deleted.push({ tableId: t.$id, tableName: t.name });
158
+ }
159
+ catch (e) {
160
+ MessageFormatter.error(`Failed deleting table ${t.$id}`, e instanceof Error ? e : new Error(String(e)), { prefix: 'Wipe' });
161
+ }
162
+ processed++;
163
+ progress.update(processed);
164
+ await delay(100);
165
+ }
166
+ progress.stop();
167
+ return deleted;
168
+ };
169
+ /**
170
+ * Optimized deletion of all rows from a table using direct bulk deletion
171
+ * Uses Query.limit() to delete rows without fetching IDs first
172
+ */
173
+ export const wipeTableRows = async (adapter, databaseId, tableId) => {
174
+ try {
175
+ // Check if bulk deletion is available
176
+ if (!adapter.bulkDeleteRows) {
177
+ MessageFormatter.error("Bulk deletion not available for this adapter - wipe operation not supported", new Error("bulkDeleteRows not available"), { prefix: "Wipe" });
178
+ throw new Error("Bulk deletion required for wipe operations");
179
+ }
180
+ const DELETE_BATCH_SIZE = 250; // How many rows to delete per batch
181
+ let totalDeleted = 0;
182
+ let hasMoreRows = true;
183
+ MessageFormatter.info("Starting optimized table row deletion...", { prefix: "Wipe" });
184
+ const progress = ProgressManager.create(`delete-${tableId}`, 1, // Start with 1, will update as we discover more
185
+ { title: "Deleting table rows" });
186
+ while (hasMoreRows) {
187
+ try {
188
+ // Delete next batch using Query.limit() - no fetching needed!
189
+ const result = await tryAwaitWithRetry(async () => adapter.bulkDeleteRows({
190
+ databaseId,
191
+ tableId,
192
+ rowIds: [], // Empty array signals we want to use Query.limit instead
193
+ batchSize: DELETE_BATCH_SIZE
194
+ }));
195
+ const deletedCount = result.total || 0;
196
+ if (deletedCount === 0) {
197
+ hasMoreRows = false;
198
+ break;
199
+ }
200
+ totalDeleted += deletedCount;
201
+ progress.setTotal(totalDeleted + 100); // Estimate more rows exist
202
+ progress.update(totalDeleted);
203
+ MessageFormatter.progress(`Deleted ${deletedCount} rows (${totalDeleted} total so far)`, { prefix: "Wipe" });
204
+ // Small delay between batches to be respectful to the API
205
+ await delay(10);
206
+ }
207
+ catch (error) {
208
+ const errorMessage = error.message || String(error);
209
+ if (isCriticalError(errorMessage)) {
210
+ MessageFormatter.error(`Critical error during bulk deletion: ${errorMessage}`, error, { prefix: "Wipe" });
211
+ throw error;
212
+ }
213
+ else {
214
+ MessageFormatter.error(`Error during deletion batch: ${errorMessage}`, error, { prefix: "Wipe" });
215
+ // Continue trying with next batch
216
+ }
217
+ }
218
+ }
219
+ // Update final progress total
220
+ progress.setTotal(totalDeleted);
221
+ progress.stop();
222
+ if (totalDeleted === 0) {
223
+ MessageFormatter.info("No rows found to delete", { prefix: "Wipe" });
224
+ }
225
+ else {
226
+ MessageFormatter.success(`Successfully deleted ${totalDeleted} rows from table ${tableId}`, { prefix: "Wipe" });
227
+ }
228
+ }
229
+ catch (error) {
230
+ MessageFormatter.error(`Error wiping rows from table ${tableId}`, error instanceof Error ? error : new Error(String(error)), { prefix: "Wipe" });
231
+ throw error;
232
+ }
233
+ };