@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,590 @@
1
+ import {
2
+ Client,
3
+ Databases,
4
+ ID,
5
+ Query,
6
+ } from "node-appwrite";
7
+ import { tryAwaitWithRetry, delay, calculateExponentialBackoff } from "../utils/helperFunctions.js";
8
+ import { MessageFormatter } from "../shared/messageFormatter.js";
9
+ import { chunk } from "es-toolkit";
10
+ import type { DatabaseAdapter } from "../adapters/DatabaseAdapter.js";
11
+ import { isLegacyDatabases } from "../utils/typeGuards.js";
12
+ import { getAdapter } from "../utils/getClientFromConfig.js";
13
+
14
+ /**
15
+ * Transfers all documents from one collection to another in a different database
16
+ * within the same Appwrite Project
17
+ */
18
+ export const transferDocumentsBetweenDbsLocalToLocal = async (
19
+ db: Databases | DatabaseAdapter,
20
+ fromDbId: string,
21
+ toDbId: string,
22
+ fromCollId: string,
23
+ toCollId: string
24
+ ) => {
25
+ // Use adapter path when available for bulk operations
26
+ if (!isLegacyDatabases(db)) {
27
+ const adapter = db as DatabaseAdapter;
28
+
29
+ const pageSize = 1000;
30
+ let lastId: string | undefined;
31
+ let totalTransferred = 0;
32
+
33
+ while (true) {
34
+ const queries = [Query.limit(pageSize)];
35
+ if (lastId) queries.push(Query.cursorAfter(lastId));
36
+
37
+ const result = await adapter.listRows({ databaseId: fromDbId, tableId: fromCollId, queries });
38
+ const rows: any[] = (result as any).rows || (result as any).documents || [];
39
+ if (!rows.length) break;
40
+
41
+ // Prepare rows: strip system fields, keep $id and $permissions
42
+ const prepared = rows.map((doc) => {
43
+ const data: any = { ...doc };
44
+ delete data.$databaseId;
45
+ delete data.$collectionId;
46
+ delete data.$createdAt;
47
+ delete data.$updatedAt;
48
+ return data; // keep $id and $permissions for upsert
49
+ });
50
+
51
+ // Prefer bulk upsert, then bulk create, then individual
52
+ if (typeof (adapter as any).bulkUpsertRows === 'function' && adapter.supportsBulkOperations()) {
53
+ await (adapter as any).bulkUpsertRows({ databaseId: toDbId, tableId: toCollId, rows: prepared });
54
+ } else if (typeof (adapter as any).bulkCreateRows === 'function' && adapter.supportsBulkOperations()) {
55
+ await (adapter as any).bulkCreateRows({ databaseId: toDbId, tableId: toCollId, rows: prepared });
56
+ } else {
57
+ for (const row of prepared) {
58
+ const id = row.$id || ID.unique();
59
+ const permissions = row.$permissions || [];
60
+ const { $id, $permissions, ...data } = row;
61
+ await adapter.createRow({ databaseId: toDbId, tableId: toCollId, id, data, permissions });
62
+ }
63
+ }
64
+
65
+ totalTransferred += rows.length;
66
+ if (rows.length < pageSize) break;
67
+ lastId = rows[rows.length - 1].$id;
68
+ }
69
+
70
+ MessageFormatter.success(
71
+ `Transferred ${totalTransferred} rows from ${fromDbId}/${fromCollId} to ${toDbId}/${toCollId}`,
72
+ { prefix: "Transfer" }
73
+ );
74
+ return;
75
+ }
76
+
77
+ // Legacy path (Databases) – keep existing behavior
78
+ const legacyDb = db as Databases;
79
+ let fromCollDocs = await tryAwaitWithRetry(async () =>
80
+ legacyDb.listDocuments(fromDbId, fromCollId, [Query.limit(50)])
81
+ );
82
+ let totalDocumentsTransferred = 0;
83
+
84
+ if (fromCollDocs.documents.length === 0) {
85
+ MessageFormatter.info(`No documents found in collection ${fromCollId}`, { prefix: "Transfer" });
86
+ return;
87
+ } else if (fromCollDocs.documents.length < 50) {
88
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
89
+ const toCreateObject: any = {
90
+ ...doc,
91
+ };
92
+ delete toCreateObject.$databaseId;
93
+ delete toCreateObject.$collectionId;
94
+ delete toCreateObject.$createdAt;
95
+ delete toCreateObject.$updatedAt;
96
+ delete toCreateObject.$id;
97
+ delete toCreateObject.$permissions;
98
+ return tryAwaitWithRetry(
99
+ async () =>
100
+ await legacyDb.createDocument(
101
+ toDbId,
102
+ toCollId,
103
+ doc.$id,
104
+ toCreateObject,
105
+ doc.$permissions
106
+ )
107
+ );
108
+ });
109
+ await Promise.all(batchedPromises);
110
+ totalDocumentsTransferred += fromCollDocs.documents.length;
111
+ } else {
112
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
113
+ const toCreateObject: any = {
114
+ ...doc,
115
+ };
116
+ delete toCreateObject.$databaseId;
117
+ delete toCreateObject.$collectionId;
118
+ delete toCreateObject.$createdAt;
119
+ delete toCreateObject.$updatedAt;
120
+ delete toCreateObject.$id;
121
+ delete toCreateObject.$permissions;
122
+ return tryAwaitWithRetry(async () =>
123
+ legacyDb.createDocument(
124
+ toDbId,
125
+ toCollId,
126
+ doc.$id,
127
+ toCreateObject,
128
+ doc.$permissions
129
+ )
130
+ );
131
+ });
132
+ await Promise.all(batchedPromises);
133
+ totalDocumentsTransferred += fromCollDocs.documents.length;
134
+ while (fromCollDocs.documents.length === 50) {
135
+ fromCollDocs = await tryAwaitWithRetry(
136
+ async () =>
137
+ await legacyDb.listDocuments(fromDbId, fromCollId, [
138
+ Query.limit(50),
139
+ Query.cursorAfter(
140
+ fromCollDocs.documents[fromCollDocs.documents.length - 1].$id
141
+ ),
142
+ ])
143
+ );
144
+ const batchedPromises = fromCollDocs.documents.map((doc) => {
145
+ const toCreateObject: any = {
146
+ ...doc,
147
+ };
148
+ delete toCreateObject.$databaseId;
149
+ delete toCreateObject.$collectionId;
150
+ delete toCreateObject.$createdAt;
151
+ delete toCreateObject.$updatedAt;
152
+ delete toCreateObject.$id;
153
+ delete toCreateObject.$permissions;
154
+ return tryAwaitWithRetry(
155
+ async () =>
156
+ await legacyDb.createDocument(
157
+ toDbId,
158
+ toCollId,
159
+ doc.$id,
160
+ toCreateObject,
161
+ doc.$permissions
162
+ )
163
+ );
164
+ });
165
+ await Promise.all(batchedPromises);
166
+ totalDocumentsTransferred += fromCollDocs.documents.length;
167
+ }
168
+ }
169
+
170
+ MessageFormatter.success(
171
+ `Transferred ${totalDocumentsTransferred} documents from database ${fromDbId} to database ${toDbId} -- collection ${fromCollId} to collection ${toCollId}`,
172
+ { prefix: "Transfer" }
173
+ );
174
+ };
175
+
176
+ /**
177
+ * Enhanced document transfer with fault tolerance and exponential backoff
178
+ */
179
+ const transferDocumentWithRetry = async (
180
+ db: Databases,
181
+ dbId: string,
182
+ collectionId: string,
183
+ documentId: string,
184
+ documentData: any,
185
+ permissions: string[],
186
+ maxRetries: number = 3,
187
+ retryCount: number = 0
188
+ ): Promise<boolean> => {
189
+ try {
190
+ await db.createDocument(
191
+ dbId,
192
+ collectionId,
193
+ documentId,
194
+ documentData,
195
+ permissions
196
+ );
197
+ return true;
198
+ } catch (error: any) {
199
+ // Check if document already exists
200
+ if (error.code === 409 || error.message?.toLowerCase().includes('already exists')) {
201
+ await db.updateDocument(
202
+ dbId,
203
+ collectionId,
204
+ documentId,
205
+ documentData,
206
+ permissions
207
+ );
208
+ }
209
+
210
+ if (retryCount < maxRetries) {
211
+ // Calculate exponential backoff: 1s, 2s, 4s, max 8s
212
+ const exponentialDelay = calculateExponentialBackoff(retryCount, 1000, 8000);
213
+ MessageFormatter.progress(`Retrying document ${documentId} (attempt ${retryCount + 1}/${maxRetries}, backoff: ${exponentialDelay}ms)`, { prefix: "Transfer" });
214
+
215
+ await delay(exponentialDelay);
216
+
217
+ return await transferDocumentWithRetry(
218
+ db,
219
+ dbId,
220
+ collectionId,
221
+ documentId,
222
+ documentData,
223
+ permissions,
224
+ maxRetries,
225
+ retryCount + 1
226
+ );
227
+ }
228
+
229
+ MessageFormatter.error(`Failed to transfer document ${documentId} after ${maxRetries} retries`, error, { prefix: "Transfer" });
230
+ return false;
231
+ }
232
+ };
233
+
234
+ /**
235
+ * Check if endpoint supports bulk operations (cloud.appwrite.io)
236
+ */
237
+ const supportsBulkOperations = (endpoint: string): boolean => {
238
+ return endpoint.includes('cloud.appwrite.io');
239
+ };
240
+
241
+ /**
242
+ * Direct HTTP implementation of bulk upsert API
243
+ */
244
+ const bulkUpsertDocuments = async (
245
+ client: any,
246
+ dbId: string,
247
+ collectionId: string,
248
+ documents: any[]
249
+ ): Promise<any> => {
250
+ const apiPath = `/databases/${dbId}/collections/${collectionId}/documents`;
251
+ const url = new URL(client.config.endpoint + apiPath);
252
+
253
+ const headers = {
254
+ 'Content-Type': 'application/json',
255
+ 'X-Appwrite-Project': client.config.project,
256
+ 'X-Appwrite-Key': client.config.key
257
+ };
258
+
259
+ const response = await fetch(url.toString(), {
260
+ method: 'PUT',
261
+ headers,
262
+ body: JSON.stringify({ documents })
263
+ });
264
+
265
+ if (!response.ok) {
266
+ const errorData: any = await response.json().catch(() => ({ message: 'Unknown error' }));
267
+ throw new Error(`Bulk upsert failed: ${response.status} - ${errorData.message || 'Unknown error'}`);
268
+ }
269
+
270
+ return await response.json();
271
+ };
272
+
273
+ /**
274
+ * Direct HTTP implementation of bulk create API
275
+ */
276
+ const bulkCreateDocuments = async (
277
+ client: any,
278
+ dbId: string,
279
+ collectionId: string,
280
+ documents: any[]
281
+ ): Promise<any> => {
282
+ const apiPath = `/databases/${dbId}/collections/${collectionId}/documents`;
283
+ const url = new URL(client.config.endpoint + apiPath);
284
+
285
+ const headers = {
286
+ 'Content-Type': 'application/json',
287
+ 'X-Appwrite-Project': client.config.project,
288
+ 'X-Appwrite-Key': client.config.key
289
+ };
290
+
291
+ const response = await fetch(url.toString(), {
292
+ method: 'POST',
293
+ headers,
294
+ body: JSON.stringify({ documents })
295
+ });
296
+
297
+ if (!response.ok) {
298
+ const errorData: any = await response.json().catch(() => ({ message: 'Unknown error' }));
299
+ throw new Error(`Bulk create failed: ${response.status} - ${errorData.message || 'Unknown error'}`);
300
+ }
301
+
302
+ return await response.json();
303
+ };
304
+
305
+ /**
306
+ * Enhanced bulk document creation using direct HTTP calls
307
+ */
308
+ const transferDocumentsBulkUpsert = async (
309
+ client: any,
310
+ dbId: string,
311
+ collectionId: string,
312
+ documents: any[],
313
+ maxBatchSize: number = 1000
314
+ ): Promise<{ successful: number; failed: number }> => {
315
+ let successful = 0;
316
+ let failed = 0;
317
+
318
+ // Prepare documents for bulk upsert
319
+ const preparedDocs = documents.map(doc => {
320
+ const toCreateObject: any = { ...doc };
321
+ delete toCreateObject.$databaseId;
322
+ delete toCreateObject.$collectionId;
323
+ delete toCreateObject.$createdAt;
324
+ delete toCreateObject.$updatedAt;
325
+
326
+ // Keep $id and $permissions for upsert functionality
327
+ return toCreateObject;
328
+ });
329
+
330
+ // Process in batches based on plan limits
331
+ const documentBatches = chunk(preparedDocs, maxBatchSize);
332
+
333
+ for (const batch of documentBatches) {
334
+ MessageFormatter.progress(`Bulk upserting ${batch.length} documents...`, { prefix: "Transfer" });
335
+
336
+ try {
337
+ // Try bulk upsert with direct HTTP call
338
+ const result = await bulkUpsertDocuments(client, dbId, collectionId, batch);
339
+ successful += result.documents?.length || batch.length;
340
+ MessageFormatter.success(`Bulk upserted ${result.documents?.length || batch.length} documents`, { prefix: "Transfer" });
341
+
342
+ } catch (error: any) {
343
+ MessageFormatter.progress(`Bulk upsert failed, trying smaller batch size...`, { prefix: "Transfer" });
344
+
345
+ // If bulk upsert fails, try with smaller batch size (Pro plan limit)
346
+ if (maxBatchSize > 100) {
347
+ const smallerBatches = chunk(batch, 100);
348
+
349
+ for (const smallBatch of smallerBatches) {
350
+ try {
351
+ const result = await bulkUpsertDocuments(client, dbId, collectionId, smallBatch);
352
+ successful += result.documents?.length || smallBatch.length;
353
+ MessageFormatter.success(`Bulk upserted ${result.documents?.length || smallBatch.length} documents (smaller batch)`, { prefix: "Transfer" });
354
+ } catch (smallBatchError: any) {
355
+ MessageFormatter.progress(`Smaller batch failed, falling back to individual transfers...`, { prefix: "Transfer" });
356
+
357
+ // Fall back to individual document transfer for this batch
358
+ const db = new Databases(client);
359
+ const { successful: indivSuccessful, failed: indivFailed } = await transferDocumentBatchWithRetryFallback(
360
+ db, dbId, collectionId, smallBatch.map((doc, index) => ({
361
+ ...doc,
362
+ $id: documents[documentBatches.indexOf(batch) * maxBatchSize + smallerBatches.indexOf(smallBatch) * 100 + index]?.$id || ID.unique(),
363
+ $permissions: documents[documentBatches.indexOf(batch) * maxBatchSize + smallerBatches.indexOf(smallBatch) * 100 + index]?.$permissions || []
364
+ }))
365
+ );
366
+ successful += indivSuccessful;
367
+ failed += indivFailed;
368
+ }
369
+
370
+ // Add delay between batches
371
+ await delay(200);
372
+ }
373
+ } else {
374
+ // Fall back to individual document transfer
375
+ const db = new Databases(client);
376
+ const { successful: indivSuccessful, failed: indivFailed } = await transferDocumentBatchWithRetryFallback(
377
+ db, dbId, collectionId, batch.map((doc, index) => ({
378
+ ...doc,
379
+ $id: documents[documentBatches.indexOf(batch) * maxBatchSize + index]?.$id || ID.unique(),
380
+ $permissions: documents[documentBatches.indexOf(batch) * maxBatchSize + index]?.$permissions || []
381
+ }))
382
+ );
383
+ successful += indivSuccessful;
384
+ failed += indivFailed;
385
+ }
386
+ }
387
+
388
+ // Add delay between major batches
389
+ if (documentBatches.indexOf(batch) < documentBatches.length - 1) {
390
+ await delay(500);
391
+ }
392
+ }
393
+
394
+ return { successful, failed };
395
+ };
396
+
397
+ /**
398
+ * Fallback batch document transfer with individual retry logic
399
+ */
400
+ const transferDocumentBatchWithRetryFallback = async (
401
+ db: Databases,
402
+ dbId: string,
403
+ collectionId: string,
404
+ documents: any[],
405
+ batchSize: number = 10
406
+ ): Promise<{ successful: number; failed: number }> => {
407
+ let successful = 0;
408
+ let failed = 0;
409
+
410
+ // Process documents in smaller batches to avoid overwhelming the server
411
+ const documentBatches = chunk(documents, batchSize);
412
+
413
+ for (const batch of documentBatches) {
414
+ MessageFormatter.progress(`Processing batch of ${batch.length} documents...`, { prefix: "Transfer" });
415
+
416
+ const batchPromises = batch.map(async (doc) => {
417
+ const toCreateObject: Partial<typeof doc> = { ...doc };
418
+ delete toCreateObject.$databaseId;
419
+ delete toCreateObject.$collectionId;
420
+ delete toCreateObject.$createdAt;
421
+ delete toCreateObject.$updatedAt;
422
+ delete toCreateObject.$id;
423
+ delete toCreateObject.$permissions;
424
+
425
+ const result = await transferDocumentWithRetry(
426
+ db,
427
+ dbId,
428
+ collectionId,
429
+ doc.$id,
430
+ toCreateObject,
431
+ doc.$permissions || []
432
+ );
433
+
434
+ return { docId: doc.$id, success: result };
435
+ });
436
+
437
+ const results = await Promise.allSettled(batchPromises);
438
+
439
+ results.forEach((result, index) => {
440
+ if (result.status === 'fulfilled') {
441
+ if (result.value.success) {
442
+ successful++;
443
+ } else {
444
+ failed++;
445
+ }
446
+ } else {
447
+ MessageFormatter.error(`Batch promise rejected for document ${batch[index].$id}`, new Error(String(result.reason)), { prefix: "Transfer" });
448
+ failed++;
449
+ }
450
+ });
451
+
452
+ // Add delay between batches to avoid rate limiting
453
+ if (documentBatches.indexOf(batch) < documentBatches.length - 1) {
454
+ await delay(500);
455
+ }
456
+ }
457
+
458
+ return { successful, failed };
459
+ };
460
+
461
+ /**
462
+ * Enhanced batch document transfer with fault tolerance and bulk API support
463
+ */
464
+ const transferDocumentBatchWithRetry = async (
465
+ db: Databases,
466
+ client: any,
467
+ dbId: string,
468
+ collectionId: string,
469
+ documents: any[],
470
+ batchSize: number = 10
471
+ ): Promise<{ successful: number; failed: number }> => {
472
+ // Check if we can use bulk operations
473
+ if (supportsBulkOperations(client.config.endpoint)) {
474
+ MessageFormatter.info(`Using bulk upsert API for faster document transfer`, { prefix: "Transfer" });
475
+
476
+ // Try with Scale plan limit first (2500), then Pro (1000), then Free (100)
477
+ const batchSizes = [1000, 100]; // Start with Pro plan, fallback to Free
478
+
479
+ for (const maxBatchSize of batchSizes) {
480
+ try {
481
+ return await transferDocumentsBulkUpsert(client, dbId, collectionId, documents, maxBatchSize);
482
+ } catch (error: any) {
483
+ MessageFormatter.progress(`Bulk upsert with batch size ${maxBatchSize} failed, trying smaller size...`, { prefix: "Transfer" });
484
+ continue;
485
+ }
486
+ }
487
+
488
+ // If all bulk operations fail, fall back to individual transfers
489
+ MessageFormatter.progress(`All bulk operations failed, falling back to individual document transfers`, { prefix: "Transfer" });
490
+ }
491
+
492
+ // Fall back to individual document transfer
493
+ return await transferDocumentBatchWithRetryFallback(db, dbId, collectionId, documents, batchSize);
494
+ };
495
+
496
+ export const transferDocumentsBetweenDbsLocalToRemote = async (
497
+ localDb: Databases | DatabaseAdapter,
498
+ endpoint: string,
499
+ projectId: string,
500
+ apiKey: string,
501
+ fromDbId: string,
502
+ toDbId: string,
503
+ fromCollId: string,
504
+ toCollId: string
505
+ ) => {
506
+ MessageFormatter.info(`Starting enhanced document transfer from ${fromCollId} to ${toCollId}...`, { prefix: "Transfer" });
507
+
508
+ // Prefer adapter for remote to enable bulk operations
509
+ const { adapter: remoteAdapter, client } = await getAdapter(endpoint, projectId, apiKey, 'auto');
510
+ const remoteDb = new Databases(client); // Legacy fallback for HTTP/individual
511
+ let totalDocumentsProcessed = 0;
512
+ let totalSuccessful = 0;
513
+ let totalFailed = 0;
514
+
515
+ // Fetch documents in larger batches (1000 at a time)
516
+ let hasMoreDocuments = true;
517
+ let lastDocumentId: string | undefined;
518
+
519
+ while (hasMoreDocuments) {
520
+ const queries = [Query.limit(1000)]; // Fetch 1000 documents at a time
521
+ if (lastDocumentId) {
522
+ queries.push(Query.cursorAfter(lastDocumentId));
523
+ }
524
+
525
+ const fromCollDocs = await tryAwaitWithRetry(async () => {
526
+ if (isLegacyDatabases(localDb)) {
527
+ return localDb.listDocuments(fromDbId, fromCollId, queries);
528
+ } else {
529
+ const res = await (localDb as DatabaseAdapter).listRows({ databaseId: fromDbId, tableId: fromCollId, queries });
530
+ const rows = (res as any).rows || (res as any).documents || [];
531
+ return { documents: rows } as any;
532
+ }
533
+ });
534
+
535
+ if (fromCollDocs.documents.length === 0) {
536
+ hasMoreDocuments = false;
537
+ break;
538
+ }
539
+
540
+ MessageFormatter.progress(`Fetched ${fromCollDocs.documents.length} documents, processing for transfer...`, { prefix: "Transfer" });
541
+
542
+ // Prefer remote adapter bulk upsert if available
543
+ const prepared = fromCollDocs.documents.map((doc: any) => {
544
+ const data: any = { ...doc };
545
+ delete data.$databaseId; delete data.$collectionId; delete data.$createdAt; delete data.$updatedAt;
546
+ return data; // Keep $id and $permissions for upsert
547
+ });
548
+
549
+ let successful = 0; let failed = 0;
550
+ if (typeof (remoteAdapter as any).bulkUpsertRows === 'function' && remoteAdapter.supportsBulkOperations()) {
551
+ try {
552
+ await (remoteAdapter as any).bulkUpsertRows({ databaseId: toDbId, tableId: toCollId, rows: prepared });
553
+ successful = prepared.length;
554
+ } catch (e) {
555
+ MessageFormatter.warning('Remote adapter bulk upsert failed, falling back to HTTP/individual', { prefix: 'Transfer' });
556
+ }
557
+ }
558
+
559
+ if (successful === 0) {
560
+ const res = await transferDocumentBatchWithRetry(remoteDb, client, toDbId, toCollId, fromCollDocs.documents);
561
+ successful = res.successful; failed = res.failed;
562
+ }
563
+
564
+ totalDocumentsProcessed += fromCollDocs.documents.length;
565
+ totalSuccessful += successful;
566
+ totalFailed += failed;
567
+
568
+ // Check if we have more documents to process
569
+ if (fromCollDocs.documents.length < 1000) {
570
+ hasMoreDocuments = false;
571
+ } else {
572
+ lastDocumentId = fromCollDocs.documents[fromCollDocs.documents.length - 1].$id;
573
+ }
574
+
575
+ MessageFormatter.debug(`Batch complete: ${successful} successful, ${failed} failed`, undefined, { prefix: "Transfer" });
576
+ }
577
+
578
+ if (totalDocumentsProcessed === 0) {
579
+ MessageFormatter.info(`No documents found in collection ${fromCollId}`, { prefix: "Transfer" });
580
+ return;
581
+ }
582
+
583
+ const message = `Total documents processed: ${totalDocumentsProcessed}, successful: ${totalSuccessful}, failed: ${totalFailed}`;
584
+
585
+ if (totalFailed > 0) {
586
+ MessageFormatter.warning(message, { prefix: "Transfer" });
587
+ } else {
588
+ MessageFormatter.success(message, { prefix: "Transfer" });
589
+ }
590
+ };