@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,749 @@
1
+ import inquirer from "inquirer";
2
+ import chalk from "chalk";
3
+ import type { Models } from "node-appwrite";
4
+ import { MessageFormatter } from "./messageFormatter.js";
5
+ import { logger } from "./logging.js";
6
+
7
+ /**
8
+ * Interface for sync selection summary
9
+ */
10
+ export interface SyncSelectionSummary {
11
+ databases: DatabaseSelection[];
12
+ buckets: BucketSelection[];
13
+ totalDatabases: number;
14
+ totalTables: number;
15
+ totalBuckets: number;
16
+ newItems: {
17
+ databases: number;
18
+ tables: number;
19
+ buckets: number;
20
+ };
21
+ existingItems: {
22
+ databases: number;
23
+ tables: number;
24
+ buckets: number;
25
+ };
26
+ }
27
+
28
+ /**
29
+ * Database selection with associated tables
30
+ */
31
+ export interface DatabaseSelection {
32
+ databaseId: string;
33
+ databaseName: string;
34
+ tableIds: string[];
35
+ tableNames: string[];
36
+ isNew: boolean;
37
+ }
38
+
39
+ /**
40
+ * Bucket selection with associated database
41
+ */
42
+ export interface BucketSelection {
43
+ bucketId: string;
44
+ bucketName: string;
45
+ databaseId?: string;
46
+ databaseName?: string;
47
+ isNew: boolean;
48
+ }
49
+
50
+ /**
51
+ * Options for database selection
52
+ */
53
+ export interface DatabaseSelectionOptions {
54
+ showSelectAll?: boolean;
55
+ allowNewOnly?: boolean;
56
+ defaultSelected?: string[];
57
+ }
58
+
59
+ /**
60
+ * Options for table selection
61
+ */
62
+ export interface TableSelectionOptions {
63
+ showSelectAll?: boolean;
64
+ allowNewOnly?: boolean;
65
+ defaultSelected?: string[];
66
+ showDatabaseContext?: boolean;
67
+ }
68
+
69
+ /**
70
+ * Options for bucket selection
71
+ */
72
+ export interface BucketSelectionOptions {
73
+ showSelectAll?: boolean;
74
+ allowNewOnly?: boolean;
75
+ defaultSelected?: string[];
76
+ groupByDatabase?: boolean;
77
+ }
78
+
79
+ /**
80
+ * Response from existing config prompt
81
+ */
82
+ export interface ExistingConfigResponse {
83
+ syncExisting: boolean;
84
+ modifyConfiguration: boolean;
85
+ }
86
+
87
+ /**
88
+ * Comprehensive selection dialog system for enhanced sync flow
89
+ *
90
+ * This class provides interactive dialogs for selecting databases, tables/collections,
91
+ * and storage buckets during sync operations. It supports both new and existing
92
+ * configurations with visual indicators and comprehensive confirmation flows.
93
+ *
94
+ * @example
95
+ * ```typescript
96
+ * import { SelectionDialogs } from './shared/selectionDialogs.js';
97
+ * import type { Models } from 'node-appwrite';
98
+ *
99
+ * // Example usage in a sync command
100
+ * const availableDatabases: Models.Database[] = await getAvailableDatabases();
101
+ * const configuredDatabases = config.databases || [];
102
+ *
103
+ * // Prompt about existing configuration
104
+ * const { syncExisting, modifyConfiguration } = await SelectionDialogs.promptForExistingConfig(configuredDatabases);
105
+ *
106
+ * if (modifyConfiguration) {
107
+ * // Select databases
108
+ * const selectedDatabaseIds = await SelectionDialogs.selectDatabases(
109
+ * availableDatabases,
110
+ * configuredDatabases,
111
+ * { showSelectAll: true, allowNewOnly: !syncExisting }
112
+ * );
113
+ *
114
+ * // For each database, select tables
115
+ * const tableSelectionsMap = new Map<string, string[]>();
116
+ * const availableTablesMap = new Map<string, any[]>();
117
+ *
118
+ * for (const databaseId of selectedDatabaseIds) {
119
+ * const database = availableDatabases.find(db => db.$id === databaseId)!;
120
+ * const availableTables = await getTablesForDatabase(databaseId);
121
+ * const configuredTables = getConfiguredTablesForDatabase(databaseId);
122
+ *
123
+ * availableTablesMap.set(databaseId, availableTables);
124
+ *
125
+ * const selectedTableIds = await SelectionDialogs.selectTablesForDatabase(
126
+ * databaseId,
127
+ * database.name,
128
+ * availableTables,
129
+ * configuredTables,
130
+ * { showSelectAll: true, allowNewOnly: !syncExisting }
131
+ * );
132
+ *
133
+ * tableSelectionsMap.set(databaseId, selectedTableIds);
134
+ * }
135
+ *
136
+ * // Select buckets
137
+ * const availableBuckets = await getAvailableBuckets();
138
+ * const configuredBuckets = config.buckets || [];
139
+ * const selectedBucketIds = await SelectionDialogs.selectBucketsForDatabases(
140
+ * selectedDatabaseIds,
141
+ * availableBuckets,
142
+ * configuredBuckets,
143
+ * { showSelectAll: true, groupByDatabase: true }
144
+ * );
145
+ *
146
+ * // Create selection objects
147
+ * const databaseSelections = SelectionDialogs.createDatabaseSelection(
148
+ * selectedDatabaseIds,
149
+ * availableDatabases,
150
+ * tableSelectionsMap,
151
+ * configuredDatabases,
152
+ * availableTablesMap
153
+ * );
154
+ *
155
+ * const bucketSelections = SelectionDialogs.createBucketSelection(
156
+ * selectedBucketIds,
157
+ * availableBuckets,
158
+ * configuredBuckets,
159
+ * availableDatabases
160
+ * );
161
+ *
162
+ * // Show final confirmation
163
+ * const selectionSummary = SelectionDialogs.createSyncSelectionSummary(
164
+ * databaseSelections,
165
+ * bucketSelections
166
+ * );
167
+ *
168
+ * const confirmed = await SelectionDialogs.confirmSyncSelection(selectionSummary);
169
+ *
170
+ * if (confirmed) {
171
+ * // Proceed with sync operation
172
+ * await performSync(databaseSelections, bucketSelections);
173
+ * }
174
+ * }
175
+ * ```
176
+ */
177
+ export class SelectionDialogs {
178
+ /**
179
+ * Prompts user about existing configuration
180
+ */
181
+ static async promptForExistingConfig(configuredItems: any[]): Promise<ExistingConfigResponse> {
182
+ if (configuredItems.length === 0) {
183
+ return { syncExisting: false, modifyConfiguration: true };
184
+ }
185
+
186
+ MessageFormatter.section("Existing Configuration Found");
187
+ MessageFormatter.info(`Found ${configuredItems.length} configured items.`, { skipLogging: true });
188
+
189
+ const { syncExisting } = await inquirer.prompt([{
190
+ type: 'confirm',
191
+ name: 'syncExisting',
192
+ message: 'Sync existing configured items?',
193
+ default: true
194
+ }]);
195
+
196
+ if (!syncExisting) {
197
+ return { syncExisting: false, modifyConfiguration: true };
198
+ }
199
+
200
+ const { modifyConfiguration } = await inquirer.prompt([{
201
+ type: 'confirm',
202
+ name: 'modifyConfiguration',
203
+ message: 'Add/remove items from configuration?',
204
+ default: false
205
+ }]);
206
+
207
+ return { syncExisting, modifyConfiguration };
208
+ }
209
+
210
+ /**
211
+ * Shows database selection dialog with indicators for configured vs new databases
212
+ */
213
+ static async selectDatabases(
214
+ availableDatabases: Models.Database[],
215
+ configuredDatabases: any[],
216
+ options: DatabaseSelectionOptions = {}
217
+ ): Promise<string[]> {
218
+ const {
219
+ showSelectAll = true,
220
+ allowNewOnly = false,
221
+ defaultSelected = []
222
+ } = options;
223
+
224
+ MessageFormatter.section("Database Selection");
225
+
226
+ const configuredIds = new Set(configuredDatabases.map(db => db.$id || db.id));
227
+
228
+ let choices: any[] = [];
229
+
230
+ if (showSelectAll && availableDatabases.length > 1) {
231
+ choices.push({
232
+ name: chalk.green.bold('📋 Select All Databases'),
233
+ value: '__SELECT_ALL__',
234
+ short: 'All databases'
235
+ });
236
+ }
237
+
238
+ availableDatabases.forEach(database => {
239
+ const isConfigured = configuredIds.has(database.$id);
240
+ const status = isConfigured ? chalk.green('✅') : chalk.blue('○');
241
+ const name = `${status} ${database.name} (${database.$id})`;
242
+
243
+ if (allowNewOnly && isConfigured) {
244
+ return; // Skip configured databases if only allowing new ones
245
+ }
246
+
247
+ choices.push({
248
+ name,
249
+ value: database.$id,
250
+ short: database.name,
251
+ // Do not preselect anything unless explicitly provided
252
+ checked: defaultSelected.includes(database.$id)
253
+ });
254
+ });
255
+
256
+ if (choices.length === 0) {
257
+ MessageFormatter.warning("No databases available for selection.", { skipLogging: true });
258
+ return [];
259
+ }
260
+
261
+ const { selectedDatabaseIds } = await inquirer.prompt([{
262
+ type: 'checkbox',
263
+ name: 'selectedDatabaseIds',
264
+ message: 'Select databases to sync:',
265
+ choices,
266
+ validate: (input: string[]) => {
267
+ if (input.length === 0) {
268
+ return chalk.red('Please select at least one database.');
269
+ }
270
+ if (input.includes('__SELECT_ALL__') && input.length > 1) {
271
+ return chalk.red('Cannot select "Select All" with individual databases.');
272
+ }
273
+ return true;
274
+ }
275
+ }]);
276
+
277
+ // Handle select all
278
+ if (selectedDatabaseIds.includes('__SELECT_ALL__')) {
279
+ const allIds = availableDatabases.map(db => db.$id);
280
+ if (allowNewOnly) {
281
+ return allIds.filter(id => !configuredIds.has(id));
282
+ }
283
+ return allIds;
284
+ }
285
+
286
+ return selectedDatabaseIds;
287
+ }
288
+
289
+ /**
290
+ * Shows table/collection selection dialog for a specific database
291
+ */
292
+ static async selectTablesForDatabase(
293
+ databaseId: string,
294
+ databaseName: string,
295
+ availableTables: any[],
296
+ configuredTables: any[],
297
+ options: TableSelectionOptions = {}
298
+ ): Promise<string[]> {
299
+ const {
300
+ showSelectAll = true,
301
+ allowNewOnly = false,
302
+ defaultSelected = [],
303
+ showDatabaseContext = true
304
+ } = options;
305
+
306
+ if (showDatabaseContext) {
307
+ MessageFormatter.section(`Table Selection for ${databaseName}`);
308
+ }
309
+
310
+ const configuredIds = new Set(configuredTables.map(table => table.$id || table.id));
311
+
312
+ let choices: any[] = [];
313
+
314
+ if (showSelectAll && availableTables.length > 1) {
315
+ choices.push({
316
+ name: chalk.green.bold('📋 Select All Tables'),
317
+ value: '__SELECT_ALL__',
318
+ short: 'All tables'
319
+ });
320
+ }
321
+
322
+ availableTables.forEach(table => {
323
+ const isConfigured = configuredIds.has(table.$id);
324
+ const status = isConfigured ? chalk.green('✅') : chalk.blue('○');
325
+ const name = `${status} ${table.name} (${table.$id})`;
326
+
327
+ if (allowNewOnly && isConfigured) {
328
+ return; // Skip configured tables if only allowing new ones
329
+ }
330
+
331
+ choices.push({
332
+ name,
333
+ value: table.$id,
334
+ short: table.name,
335
+ // Do not preselect anything unless explicitly provided
336
+ checked: defaultSelected.includes(table.$id)
337
+ });
338
+ });
339
+
340
+ if (choices.length === 0) {
341
+ MessageFormatter.warning(`No tables available for database: ${databaseName}`, { skipLogging: true });
342
+ return [];
343
+ }
344
+
345
+ const { selectedTableIds } = await inquirer.prompt([{
346
+ type: 'checkbox',
347
+ name: 'selectedTableIds',
348
+ message: `Select tables to sync for ${databaseName}:`,
349
+ choices,
350
+ validate: (input: string[]) => {
351
+ if (input.length === 0) {
352
+ return chalk.red('Please select at least one table.');
353
+ }
354
+ if (input.includes('__SELECT_ALL__') && input.length > 1) {
355
+ return chalk.red('Cannot select "Select All" with individual tables.');
356
+ }
357
+ return true;
358
+ }
359
+ }]);
360
+
361
+ // Handle select all
362
+ if (selectedTableIds.includes('__SELECT_ALL__')) {
363
+ const allIds = availableTables.map(table => table.$id);
364
+ if (allowNewOnly) {
365
+ return allIds.filter(id => !configuredIds.has(id));
366
+ }
367
+ return allIds;
368
+ }
369
+
370
+ return selectedTableIds;
371
+ }
372
+
373
+ /**
374
+ * Shows bucket selection dialog for selected databases
375
+ */
376
+ static async selectBucketsForDatabases(
377
+ selectedDatabaseIds: string[],
378
+ availableBuckets: any[],
379
+ configuredBuckets: any[],
380
+ options: BucketSelectionOptions = {}
381
+ ): Promise<string[]> {
382
+ const {
383
+ showSelectAll = true,
384
+ allowNewOnly = false,
385
+ defaultSelected = [],
386
+ groupByDatabase = true
387
+ } = options;
388
+
389
+ MessageFormatter.section("Storage Bucket Selection");
390
+
391
+ const configuredIds = new Set(configuredBuckets.map(bucket => bucket.$id || bucket.id));
392
+
393
+ // Filter buckets that are associated with selected databases
394
+ const relevantBuckets = availableBuckets.filter(bucket => {
395
+ if (selectedDatabaseIds.length === 0) return true; // If no databases selected, show all buckets
396
+ return selectedDatabaseIds.includes(bucket.databaseId) || !bucket.databaseId;
397
+ });
398
+
399
+ if (relevantBuckets.length === 0) {
400
+ MessageFormatter.warning("No storage buckets available for selected databases.", { skipLogging: true });
401
+ return [];
402
+ }
403
+
404
+ let choices: any[] = [];
405
+
406
+ if (showSelectAll && relevantBuckets.length > 1) {
407
+ choices.push({
408
+ name: chalk.green.bold('📋 Select All Buckets'),
409
+ value: '__SELECT_ALL__',
410
+ short: 'All buckets'
411
+ });
412
+ }
413
+
414
+ if (groupByDatabase) {
415
+ // Group buckets by database
416
+ const bucketsByDatabase = new Map<string, any[]>();
417
+
418
+ relevantBuckets.forEach(bucket => {
419
+ const dbId = bucket.databaseId || 'ungrouped';
420
+ if (!bucketsByDatabase.has(dbId)) {
421
+ bucketsByDatabase.set(dbId, []);
422
+ }
423
+ bucketsByDatabase.get(dbId)!.push(bucket);
424
+ });
425
+
426
+ // Add buckets grouped by database
427
+ selectedDatabaseIds.forEach(dbId => {
428
+ const buckets = bucketsByDatabase.get(dbId) || [];
429
+ if (buckets.length > 0) {
430
+ choices.push(new inquirer.Separator(chalk.cyan(`📁 Database: ${dbId}`)));
431
+
432
+ buckets.forEach(bucket => {
433
+ const isConfigured = configuredIds.has(bucket.$id);
434
+ const status = isConfigured ? chalk.green('✅') : chalk.blue('○');
435
+ const name = `${status} ${bucket.name} (${bucket.$id})`;
436
+
437
+ if (allowNewOnly && isConfigured) {
438
+ return; // Skip configured buckets if only allowing new ones
439
+ }
440
+
441
+ choices.push({
442
+ name: ` ${name}`,
443
+ value: bucket.$id,
444
+ short: bucket.name,
445
+ // Do not preselect anything unless explicitly provided
446
+ checked: defaultSelected.includes(bucket.$id)
447
+ });
448
+ });
449
+ }
450
+ });
451
+
452
+ // Add ungrouped buckets
453
+ const ungroupedBuckets = bucketsByDatabase.get('ungrouped') || [];
454
+ if (ungroupedBuckets.length > 0) {
455
+ choices.push(new inquirer.Separator(chalk.cyan('📁 General Storage')));
456
+
457
+ ungroupedBuckets.forEach(bucket => {
458
+ const isConfigured = configuredIds.has(bucket.$id);
459
+ const status = isConfigured ? chalk.green('✅') : chalk.blue('○');
460
+ const name = `${status} ${bucket.name} (${bucket.$id})`;
461
+
462
+ if (allowNewOnly && isConfigured) {
463
+ return; // Skip configured buckets if only allowing new ones
464
+ }
465
+
466
+ choices.push({
467
+ name: ` ${name}`,
468
+ value: bucket.$id,
469
+ short: bucket.name,
470
+ checked: defaultSelected.includes(bucket.$id) || (!allowNewOnly && isConfigured)
471
+ });
472
+ });
473
+ }
474
+ } else {
475
+ // Flat list of buckets
476
+ relevantBuckets.forEach(bucket => {
477
+ const isConfigured = configuredIds.has(bucket.$id);
478
+ const status = isConfigured ? chalk.green('✅') : chalk.blue('○');
479
+ const dbContext = bucket.databaseId ? ` [${bucket.databaseId}]` : '';
480
+ const name = `${status} ${bucket.name} (${bucket.$id})${dbContext}`;
481
+
482
+ if (allowNewOnly && isConfigured) {
483
+ return; // Skip configured buckets if only allowing new ones
484
+ }
485
+
486
+ choices.push({
487
+ name,
488
+ value: bucket.$id,
489
+ short: bucket.name,
490
+ // Do not preselect anything unless explicitly provided
491
+ checked: defaultSelected.includes(bucket.$id)
492
+ });
493
+ });
494
+ }
495
+
496
+ const { selectedBucketIds } = await inquirer.prompt([{
497
+ type: 'checkbox',
498
+ name: 'selectedBucketIds',
499
+ message: 'Select storage buckets to sync:',
500
+ choices,
501
+ validate: (input: string[]) => {
502
+ if (input.length === 0) {
503
+ return chalk.yellow('No storage buckets selected. Continue with databases only?') || true;
504
+ }
505
+ if (input.includes('__SELECT_ALL__') && input.length > 1) {
506
+ return chalk.red('Cannot select "Select All" with individual buckets.');
507
+ }
508
+ return true;
509
+ }
510
+ }]);
511
+
512
+ // Handle select all
513
+ if (selectedBucketIds && selectedBucketIds.includes('__SELECT_ALL__')) {
514
+ const allIds = relevantBuckets.map(bucket => bucket.$id);
515
+ if (allowNewOnly) {
516
+ return allIds.filter(id => !configuredIds.has(id));
517
+ }
518
+ return allIds;
519
+ }
520
+
521
+ return selectedBucketIds || [];
522
+ }
523
+
524
+ /**
525
+ * Shows final confirmation dialog with sync selection summary
526
+ */
527
+ static async confirmSyncSelection(
528
+ selectionSummary: SyncSelectionSummary,
529
+ operationType: 'push' | 'pull' | 'sync' = 'sync'
530
+ ): Promise<boolean> {
531
+ const labels = {
532
+ push: {
533
+ banner: "Push Selection Summary",
534
+ subtitle: "Review selections before pushing to Appwrite",
535
+ confirm: "Proceed with push operation?",
536
+ success: "Push operation confirmed.",
537
+ cancel: "Push operation cancelled."
538
+ },
539
+ pull: {
540
+ banner: "Pull Selection Summary",
541
+ subtitle: "Review selections before pulling from Appwrite",
542
+ confirm: "Proceed with pull operation?",
543
+ success: "Pull operation confirmed.",
544
+ cancel: "Pull operation cancelled."
545
+ },
546
+ sync: {
547
+ banner: "Sync Selection Summary",
548
+ subtitle: "Review your selections before proceeding",
549
+ confirm: "Proceed with sync operation?",
550
+ success: "Sync operation confirmed.",
551
+ cancel: "Sync operation cancelled."
552
+ }
553
+ };
554
+
555
+ const label = labels[operationType];
556
+
557
+ MessageFormatter.banner(label.banner, label.subtitle);
558
+
559
+ // Database summary
560
+ console.log(chalk.bold.cyan("\n📊 Databases:"));
561
+ console.log(` Total: ${selectionSummary.totalDatabases}`);
562
+ console.log(` ${chalk.green('✅ Configured')}: ${selectionSummary.existingItems.databases}`);
563
+ console.log(` ${chalk.blue('○ New')}: ${selectionSummary.newItems.databases}`);
564
+
565
+ if (selectionSummary.databases.length > 0) {
566
+ console.log(chalk.gray("\n Selected databases:"));
567
+ selectionSummary.databases.forEach(db => {
568
+ const status = db.isNew ? chalk.blue('○') : chalk.green('✅');
569
+ console.log(` ${status} ${db.databaseName} (${db.tableNames.length} tables)`);
570
+ });
571
+ }
572
+
573
+ // Table summary
574
+ console.log(chalk.bold.cyan("\n📋 Tables/Collections:"));
575
+ console.log(` Total: ${selectionSummary.totalTables}`);
576
+ console.log(` ${chalk.green('✅ Configured')}: ${selectionSummary.existingItems.tables}`);
577
+ console.log(` ${chalk.blue('○ New')}: ${selectionSummary.newItems.tables}`);
578
+
579
+ // Bucket summary
580
+ console.log(chalk.bold.cyan("\n🪣 Storage Buckets:"));
581
+ console.log(` Total: ${selectionSummary.totalBuckets}`);
582
+ console.log(` ${chalk.green('✅ Configured')}: ${selectionSummary.existingItems.buckets}`);
583
+ console.log(` ${chalk.blue('○ New')}: ${selectionSummary.newItems.buckets}`);
584
+
585
+ if (selectionSummary.buckets.length > 0) {
586
+ console.log(chalk.gray("\n Selected buckets:"));
587
+ selectionSummary.buckets.forEach(bucket => {
588
+ const status = bucket.isNew ? chalk.blue('○') : chalk.green('✅');
589
+ const dbContext = bucket.databaseName ? ` [${bucket.databaseName}]` : '';
590
+ console.log(` ${status} ${bucket.bucketName}${dbContext}`);
591
+ });
592
+ }
593
+
594
+ console.log(); // Add spacing
595
+
596
+ const { confirmed } = await inquirer.prompt([{
597
+ type: 'confirm',
598
+ name: 'confirmed',
599
+ message: chalk.green.bold(label.confirm),
600
+ default: true
601
+ }]);
602
+
603
+ if (confirmed) {
604
+ MessageFormatter.success(label.success, { skipLogging: true });
605
+ logger.info(`${operationType} selection confirmed`, {
606
+ databases: selectionSummary.totalDatabases,
607
+ tables: selectionSummary.totalTables,
608
+ buckets: selectionSummary.totalBuckets
609
+ });
610
+ } else {
611
+ MessageFormatter.warning(label.cancel, { skipLogging: true });
612
+ logger.info(`${operationType} selection cancelled by user`);
613
+ }
614
+
615
+ return confirmed;
616
+ }
617
+
618
+ /**
619
+ * Creates a sync selection summary from selected items
620
+ */
621
+ static createSyncSelectionSummary(
622
+ databaseSelections: DatabaseSelection[],
623
+ bucketSelections: BucketSelection[]
624
+ ): SyncSelectionSummary {
625
+ const totalDatabases = databaseSelections.length;
626
+ const totalTables = databaseSelections.reduce((sum, db) => sum + db.tableIds.length, 0);
627
+ const totalBuckets = bucketSelections.length;
628
+
629
+ const newDatabases = databaseSelections.filter(db => db.isNew).length;
630
+ const newTables = databaseSelections.reduce((sum, db) =>
631
+ sum + db.tableIds.length, 0); // TODO: Track which tables are new
632
+ const newBuckets = bucketSelections.filter(bucket => bucket.isNew).length;
633
+
634
+ const existingDatabases = totalDatabases - newDatabases;
635
+ const existingTables = totalTables - newTables;
636
+ const existingBuckets = totalBuckets - newBuckets;
637
+
638
+ return {
639
+ databases: databaseSelections,
640
+ buckets: bucketSelections,
641
+ totalDatabases,
642
+ totalTables,
643
+ totalBuckets,
644
+ newItems: {
645
+ databases: newDatabases,
646
+ tables: newTables,
647
+ buckets: newBuckets
648
+ },
649
+ existingItems: {
650
+ databases: existingDatabases,
651
+ tables: existingTables,
652
+ buckets: existingBuckets
653
+ }
654
+ };
655
+ }
656
+
657
+ /**
658
+ * Helper method to create database selection objects
659
+ */
660
+ static createDatabaseSelection(
661
+ selectedDatabaseIds: string[],
662
+ availableDatabases: Models.Database[],
663
+ tableSelectionsMap: Map<string, string[]>,
664
+ configuredDatabases: any[],
665
+ availableTablesMap: Map<string, any[]> = new Map()
666
+ ): DatabaseSelection[] {
667
+ const configuredIds = new Set(configuredDatabases.map(db => db.$id || db.id));
668
+
669
+ return selectedDatabaseIds.map(databaseId => {
670
+ const database = availableDatabases.find(db => db.$id === databaseId);
671
+ if (!database) {
672
+ throw new Error(`Database with ID ${databaseId} not found in available databases`);
673
+ }
674
+
675
+ const tableIds = tableSelectionsMap.get(databaseId) || [];
676
+ const tables = availableTablesMap.get(databaseId) || [];
677
+ const tableNames: string[] = tables.map(table => table.name || table.$id || `Table-${table.$id}`);
678
+
679
+ return {
680
+ databaseId,
681
+ databaseName: database.name,
682
+ tableIds,
683
+ tableNames,
684
+ isNew: !configuredIds.has(databaseId)
685
+ };
686
+ });
687
+ }
688
+
689
+ /**
690
+ * Helper method to create bucket selection objects
691
+ */
692
+ static createBucketSelection(
693
+ selectedBucketIds: string[],
694
+ availableBuckets: any[],
695
+ configuredBuckets: any[],
696
+ availableDatabases: Models.Database[]
697
+ ): BucketSelection[] {
698
+ const configuredIds = new Set(configuredBuckets.map(bucket => bucket.$id || bucket.id));
699
+
700
+ return selectedBucketIds.map(bucketId => {
701
+ const bucket = availableBuckets.find(b => b.$id === bucketId);
702
+ if (!bucket) {
703
+ throw new Error(`Bucket with ID ${bucketId} not found in available buckets`);
704
+ }
705
+
706
+ const database = bucket.databaseId ?
707
+ availableDatabases.find(db => db.$id === bucket.databaseId) : undefined;
708
+
709
+ return {
710
+ bucketId,
711
+ bucketName: bucket.name,
712
+ databaseId: bucket.databaseId,
713
+ databaseName: database?.name,
714
+ isNew: !configuredIds.has(bucketId)
715
+ };
716
+ });
717
+ }
718
+
719
+ /**
720
+ * Shows a progress message during selection operations
721
+ */
722
+ static showProgress(message: string): void {
723
+ MessageFormatter.progress(message, { skipLogging: true });
724
+ }
725
+
726
+ /**
727
+ * Shows an error message and handles graceful cancellation
728
+ */
729
+ static showError(message: string, error?: Error): void {
730
+ MessageFormatter.error(message, error, { skipLogging: true });
731
+ logger.error(`Selection dialog error: ${message}`, { error: error?.message });
732
+ }
733
+
734
+ /**
735
+ * Shows a warning message
736
+ */
737
+ static showWarning(message: string): void {
738
+ MessageFormatter.warning(message, { skipLogging: true });
739
+ logger.warn(`Selection dialog warning: ${message}`);
740
+ }
741
+
742
+ /**
743
+ * Shows a success message
744
+ */
745
+ static showSuccess(message: string): void {
746
+ MessageFormatter.success(message, { skipLogging: true });
747
+ logger.info(`Selection dialog success: ${message}`);
748
+ }
749
+ }