@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,205 @@
1
+ import { z } from "zod";
2
+ import {
3
+ attributeSchema,
4
+ type Attribute,
5
+ parseAttribute,
6
+ CollectionCreateSchema,
7
+ } from "@njdamstra/appwrite-utils";
8
+
9
+ export const BackupSchema = z.object({
10
+ $id: z.string(),
11
+ $createdAt: z.string(),
12
+ $updatedAt: z.string(),
13
+ database: z.string(),
14
+ collections: z.array(z.string()),
15
+ documents: z
16
+ .array(
17
+ z.object({
18
+ collectionId: z.string(),
19
+ data: z.string(),
20
+ })
21
+ )
22
+ .default([]),
23
+ });
24
+
25
+ export type Backup = z.infer<typeof BackupSchema>;
26
+
27
+ export const BackupCreateSchema = BackupSchema.omit({
28
+ $id: true,
29
+ $createdAt: true,
30
+ $updatedAt: true,
31
+ });
32
+
33
+ export type BackupCreate = z.infer<typeof BackupCreateSchema>;
34
+
35
+ export const BatchSchema = z.object({
36
+ $id: z.string(),
37
+ $createdAt: z.string(),
38
+ $updatedAt: z.string(),
39
+ data: z.string().describe("The serialized data for this batch"),
40
+ processed: z
41
+ .boolean()
42
+ .default(false)
43
+ .describe("Whether the batch has been processed"),
44
+ });
45
+
46
+ export type Batch = z.infer<typeof BatchSchema>;
47
+
48
+ export const BatchCreateSchema = BatchSchema.omit({
49
+ $id: true,
50
+ $createdAt: true,
51
+ $updatedAt: true,
52
+ });
53
+
54
+ export type BatchCreate = z.infer<typeof BatchCreateSchema>;
55
+
56
+ export const OperationSchema = z.object({
57
+ $id: z.string(),
58
+ $createdAt: z.string(),
59
+ $updatedAt: z.string(),
60
+ operationType: z.string(),
61
+ collectionId: z.string(),
62
+ data: z.any(),
63
+ batches: z.array(z.string()).default([]).optional(),
64
+ progress: z.number(),
65
+ total: z.number(),
66
+ error: z.string(),
67
+ status: z
68
+ .enum([
69
+ "pending",
70
+ "ready",
71
+ "in_progress",
72
+ "completed",
73
+ "error",
74
+ "cancelled",
75
+ ])
76
+ .default("pending"),
77
+ });
78
+
79
+ export type Operation = z.infer<typeof OperationSchema>;
80
+
81
+ export const OperationCreateSchema = OperationSchema.omit({
82
+ $id: true,
83
+ $createdAt: true,
84
+ $updatedAt: true,
85
+ });
86
+
87
+ export type OperationCreate = z.infer<typeof OperationCreateSchema>;
88
+
89
+ export const getMigrationCollectionSchemas = () => {
90
+ const currentOperationsAttributes: Attribute[] = [
91
+ parseAttribute({
92
+ key: "operationType",
93
+ type: "string",
94
+ error: "Invalid Operation Type",
95
+ size: 50,
96
+ required: true,
97
+ array: false,
98
+ xdefault: null,
99
+ }),
100
+ attributeSchema.parse({
101
+ key: "collectionId",
102
+ type: "string",
103
+ error: "Invalid Collection Id",
104
+ size: 50,
105
+ array: false,
106
+ xdefault: null,
107
+ }),
108
+ attributeSchema.parse({
109
+ key: "batches",
110
+ type: "string",
111
+ error: "Invalid Batches",
112
+ size: 1073741824,
113
+ array: true,
114
+ }),
115
+ attributeSchema.parse({
116
+ key: "data",
117
+ type: "string",
118
+ error: "Invalid Data",
119
+ size: 1073741824,
120
+ }),
121
+ attributeSchema.parse({
122
+ key: "progress",
123
+ type: "integer",
124
+ error: "Invalid Progress",
125
+ required: true,
126
+ array: false,
127
+ }),
128
+ attributeSchema.parse({
129
+ key: "total",
130
+ type: "integer",
131
+ error: "Invalid Total",
132
+ required: true,
133
+ array: false,
134
+ }),
135
+ attributeSchema.parse({
136
+ key: "error",
137
+ type: "string",
138
+ error: "Operation Error",
139
+ required: false,
140
+ array: false,
141
+ }),
142
+ attributeSchema.parse({
143
+ key: "status",
144
+ type: "enum",
145
+ elements: [
146
+ "pending",
147
+ "ready",
148
+ "in_progress",
149
+ "completed",
150
+ "error",
151
+ "cancelled",
152
+ ],
153
+ error: "Invalid Status",
154
+ array: false,
155
+ xdefault: "pending",
156
+ }),
157
+ ];
158
+
159
+ const currentOperationsConfig = CollectionCreateSchema.parse({
160
+ name: "CurrentOperations",
161
+ enabled: true,
162
+ documentSecurity: false,
163
+ attributes: [],
164
+ indexes: [],
165
+ });
166
+
167
+ const batchesAttributes: Attribute[] = [
168
+ attributeSchema.parse({
169
+ key: "data",
170
+ type: "string",
171
+ size: 1073741824,
172
+ error: "Invalid Data",
173
+ required: true,
174
+ array: false,
175
+ }),
176
+ attributeSchema.parse({
177
+ key: "processed",
178
+ type: "boolean",
179
+ error: "Invalid Processed",
180
+ required: true,
181
+ array: false,
182
+ xdefault: false,
183
+ }),
184
+ ];
185
+
186
+ const batchesConfig = CollectionCreateSchema.parse({
187
+ name: "Batches",
188
+ enabled: true,
189
+ documentSecurity: false,
190
+ attributes: [],
191
+ indexes: [],
192
+ });
193
+
194
+ const toReturn = {
195
+ CurrentOperations: {
196
+ collection: currentOperationsConfig,
197
+ attributes: currentOperationsAttributes,
198
+ },
199
+ Batches: {
200
+ collection: batchesConfig,
201
+ attributes: batchesAttributes,
202
+ },
203
+ };
204
+ return toReturn;
205
+ };
@@ -0,0 +1,44 @@
1
+ /**
2
+ * Type declarations for optional node-appwrite-tablesdb module
3
+ * This allows compilation when the module is not installed
4
+ */
5
+
6
+ declare module 'node-appwrite-tablesdb' {
7
+ export class Client {
8
+ setEndpoint(endpoint: string): this;
9
+ setProject(project: string): this;
10
+ setKey(key: string): this;
11
+ }
12
+
13
+ export class TablesDB {
14
+ constructor(client: Client);
15
+
16
+ // Core methods based on TablesDB API
17
+ listTables(databaseId: string, queries?: any[]): Promise<any>;
18
+ createTable(databaseId: string, id: string, name: string, permissions?: string[], documentSecurity?: boolean, enabled?: boolean): Promise<any>;
19
+ updateTable(databaseId: string, id: string, name: string, permissions?: string[], documentSecurity?: boolean, enabled?: boolean): Promise<any>;
20
+ deleteTable(databaseId: string, tableId: string): Promise<void>;
21
+ getTable(databaseId: string, tableId: string): Promise<any>;
22
+
23
+ listRows(databaseId: string, tableId: string, queries?: any[]): Promise<any>;
24
+ createRow(databaseId: string, tableId: string, id: string, data: any, permissions?: string[]): Promise<any>;
25
+ updateRow(databaseId: string, tableId: string, id: string, data?: any, permissions?: string[]): Promise<any>;
26
+ deleteRow(databaseId: string, tableId: string, id: string): Promise<void>;
27
+ getRow(databaseId: string, tableId: string, id: string): Promise<any>;
28
+
29
+ // Bulk operations (if supported)
30
+ bulkCreateRows?(databaseId: string, tableId: string, rows: any[]): Promise<any>;
31
+ bulkUpsertRows?(databaseId: string, tableId: string, rows: any[]): Promise<any>;
32
+ bulkDeleteRows?(databaseId: string, tableId: string, rowIds: string[]): Promise<any>;
33
+
34
+ // Index operations
35
+ listIndexes(databaseId: string, tableId: string, queries?: any[]): Promise<any>;
36
+ createIndex(databaseId: string, tableId: string, key: string, type: string, attributes: string[], orders?: string[]): Promise<any>;
37
+ deleteIndex(databaseId: string, tableId: string, key: string): Promise<void>;
38
+
39
+ // Attribute operations
40
+ createAttribute(databaseId: string, tableId: string, key: string, type: string, ...args: any[]): Promise<any>;
41
+ updateAttribute(databaseId: string, tableId: string, key: string, required?: boolean, defaultValue?: any): Promise<any>;
42
+ deleteAttribute(databaseId: string, tableId: string, key: string): Promise<void>;
43
+ }
44
+ }
package/src/types.ts ADDED
@@ -0,0 +1,9 @@
1
+ export type { ValidationRules } from "./utils/validationRules.js";
2
+ export {
3
+ type AuthUserCreate,
4
+ AuthUserCreateSchema,
5
+ type AuthUser,
6
+ AuthUserSchema,
7
+ } from "./schemas/authUser.js";
8
+ export { validationRules } from "./utils/validationRules.js";
9
+ export { afterImportActions } from "./migrations/afterImportActions.js";
@@ -0,0 +1,359 @@
1
+ import type { AppwriteConfig, ConfigCollection } from "@njdamstra/appwrite-utils";
2
+ import {
3
+ AppwriteException,
4
+ Databases,
5
+ ID,
6
+ Query,
7
+ Users,
8
+ type Models,
9
+ } from "node-appwrite";
10
+ import {
11
+ AuthUserSchema,
12
+ type AuthUser,
13
+ type AuthUserCreate,
14
+ } from "../schemas/authUser.js";
15
+ import { logger } from "../shared/logging.js";
16
+ import { splitIntoBatches } from "../shared/migrationHelpers.js";
17
+ import {
18
+ getAppwriteClient,
19
+ tryAwaitWithRetry,
20
+ } from "../utils/helperFunctions.js";
21
+ import { isUndefined } from "es-toolkit/compat";
22
+ import { isEmpty } from "es-toolkit/compat";
23
+ import { MessageFormatter } from "../shared/messageFormatter.js";
24
+
25
+ export class UsersController {
26
+ private config: AppwriteConfig;
27
+ private users: Users;
28
+ static userFields = [
29
+ "email",
30
+ "name",
31
+ "password",
32
+ "phone",
33
+ "labels",
34
+ "prefs",
35
+ "userId",
36
+ "$createdAt",
37
+ "$updatedAt",
38
+ ];
39
+
40
+ constructor(config: AppwriteConfig, db: Databases) {
41
+ this.config = config;
42
+ this.users = new Users(this.config.appwriteClient!);
43
+ }
44
+
45
+ async wipeUsers() {
46
+ const allUsers = await this.getAllUsers();
47
+ MessageFormatter.progress("Deleting all users...", { prefix: "Users" });
48
+
49
+ const createBatches = (finalData: any[], batchSize: number) => {
50
+ const finalBatches: any[][] = [];
51
+ for (let i = 0; i < finalData.length; i += batchSize) {
52
+ finalBatches.push(finalData.slice(i, i + batchSize));
53
+ }
54
+ return finalBatches;
55
+ };
56
+
57
+ let usersDeleted = 0;
58
+ if (allUsers.length > 0) {
59
+ const batchedUserPromises = createBatches(allUsers, 25); // Batch size of 25
60
+
61
+ for (const batch of batchedUserPromises) {
62
+ MessageFormatter.progress(`Deleting ${batch.length} users...`, { prefix: "Users" });
63
+ await Promise.all(
64
+ batch.map((user) =>
65
+ tryAwaitWithRetry(async () => await this.users.delete(user.$id))
66
+ )
67
+ );
68
+ usersDeleted += batch.length;
69
+ if (usersDeleted % 100 === 0) {
70
+ MessageFormatter.progress(`Deleted ${usersDeleted} users...`, { prefix: "Users" });
71
+ }
72
+ }
73
+ } else {
74
+ MessageFormatter.info("No users to delete", { prefix: "Users" });
75
+ }
76
+ }
77
+
78
+ async getAllUsers() {
79
+ const allUsers: Models.User<Models.Preferences>[] = [];
80
+ const users = await tryAwaitWithRetry(
81
+ async () => await this.users.list([Query.limit(200)])
82
+ );
83
+ if (users.users.length === 0) {
84
+ return [];
85
+ }
86
+ if (users.users.length === 200) {
87
+ let lastDocumentId = users.users[users.users.length - 1].$id;
88
+ allUsers.push(...users.users);
89
+ while (lastDocumentId) {
90
+ const moreUsers = await tryAwaitWithRetry(
91
+ async () =>
92
+ await this.users.list([
93
+ Query.limit(200),
94
+ Query.cursorAfter(lastDocumentId),
95
+ ])
96
+ );
97
+ allUsers.push(...moreUsers.users);
98
+ if (moreUsers.users.length < 200) {
99
+ break;
100
+ }
101
+ lastDocumentId = moreUsers.users[moreUsers.users.length - 1].$id;
102
+ }
103
+ } else {
104
+ allUsers.push(...users.users);
105
+ }
106
+ return allUsers;
107
+ }
108
+
109
+ async createUsersAndReturn(items: AuthUserCreate[]): Promise<any[]> {
110
+ const users = await Promise.all(
111
+ items.map((item) => this.createUserAndReturn(item))
112
+ );
113
+ return users;
114
+ }
115
+
116
+ async createUserAndReturn(item: AuthUserCreate): Promise<any> {
117
+ try {
118
+ const user = await tryAwaitWithRetry(async () => {
119
+ const createdUser = await this.users.create(
120
+ item.userId || ID.unique(),
121
+ item.email || undefined,
122
+ item.phone && item.phone.length < 15 && item.phone.startsWith("+")
123
+ ? item.phone
124
+ : undefined,
125
+ `changeMe${item.email?.toLowerCase()}` || `changeMePlease`,
126
+ item.name || undefined
127
+ );
128
+
129
+ if (item.labels) {
130
+ await this.users.updateLabels(createdUser.$id, item.labels);
131
+ }
132
+ if (item.prefs) {
133
+ await this.users.updatePrefs(createdUser.$id, item.prefs);
134
+ }
135
+
136
+ return createdUser;
137
+ }); // Set throwError to true since we want to handle errors
138
+
139
+ return user;
140
+ } catch (e) {
141
+ if (e instanceof Error) {
142
+ logger.error("FAILED CREATING USER: ", e.message, item);
143
+ }
144
+ }
145
+ }
146
+
147
+ async createAndCheckForUserAndReturn(item: AuthUserCreate) {
148
+ let userToReturn: Models.User<Models.Preferences> | undefined = undefined;
149
+ try {
150
+ // Attempt to find an existing user by email or phone.
151
+ let foundUsers: Models.User<Models.Preferences>[] = [];
152
+ if (item.email) {
153
+ const foundUsersByEmail = await this.users.list([
154
+ Query.equal("email", item.email),
155
+ ]);
156
+ foundUsers = foundUsersByEmail.users;
157
+ }
158
+ if (item.phone) {
159
+ const foundUsersByPhone = await this.users.list([
160
+ Query.equal("phone", item.phone),
161
+ ]);
162
+ foundUsers = foundUsers.length
163
+ ? foundUsers.concat(foundUsersByPhone.users)
164
+ : foundUsersByPhone.users;
165
+ }
166
+
167
+ userToReturn = foundUsers[0] || undefined;
168
+
169
+ if (!userToReturn) {
170
+ userToReturn = await this.users.create(
171
+ item.userId || ID.unique(),
172
+ item.email || undefined,
173
+ item.phone && item.phone.length < 15 && item.phone.startsWith("+")
174
+ ? item.phone
175
+ : undefined,
176
+ item.password?.toLowerCase() ||
177
+ `changeMe${item.email?.toLowerCase()}` ||
178
+ `changeMePlease`,
179
+ item.name || undefined
180
+ );
181
+ } else {
182
+ // Update user details as necessary, ensuring email uniqueness if attempting an update.
183
+ if (
184
+ item.email &&
185
+ item.email !== userToReturn.email &&
186
+ !isEmpty(item.email) &&
187
+ !isUndefined(item.email)
188
+ ) {
189
+ const emailExists = await this.users.list([
190
+ Query.equal("email", item.email),
191
+ ]);
192
+ if (emailExists.users.length === 0) {
193
+ userToReturn = await this.users.updateEmail(
194
+ userToReturn.$id,
195
+ item.email
196
+ );
197
+ } else {
198
+ MessageFormatter.warning("Email update skipped: Email already exists.", { prefix: "Users" });
199
+ }
200
+ }
201
+ if (item.password) {
202
+ userToReturn = await this.users.updatePassword(
203
+ userToReturn.$id,
204
+ item.password.toLowerCase()
205
+ );
206
+ }
207
+ if (item.name && item.name !== userToReturn.name) {
208
+ userToReturn = await this.users.updateName(
209
+ userToReturn.$id,
210
+ item.name
211
+ );
212
+ }
213
+ if (
214
+ item.phone &&
215
+ item.phone !== userToReturn.phone &&
216
+ item.phone.length < 15 &&
217
+ item.phone.startsWith("+") &&
218
+ (isUndefined(userToReturn.phone) || isEmpty(userToReturn.phone))
219
+ ) {
220
+ const userFoundWithPhone = await this.users.list([
221
+ Query.equal("phone", item.phone),
222
+ ]);
223
+ if (userFoundWithPhone.total === 0) {
224
+ userToReturn = await this.users.updatePhone(
225
+ userToReturn.$id,
226
+ item.phone
227
+ );
228
+ }
229
+ }
230
+ }
231
+ if (item.$createdAt && item.$updatedAt) {
232
+ MessageFormatter.warning(
233
+ "$createdAt and $updatedAt are not yet supported, sorry about that!",
234
+ { prefix: "Users" }
235
+ );
236
+ }
237
+ if (item.labels && item.labels.length) {
238
+ userToReturn = await this.users.updateLabels(
239
+ userToReturn.$id,
240
+ item.labels
241
+ );
242
+ }
243
+ if (item.prefs && Object.keys(item.prefs).length) {
244
+ await this.users.updatePrefs(userToReturn.$id, item.prefs);
245
+ userToReturn.prefs = item.prefs;
246
+ }
247
+ return userToReturn;
248
+ } catch (error) {
249
+ return userToReturn;
250
+ }
251
+ }
252
+
253
+ async getUserIdByEmailOrPhone(email?: string, phone?: string) {
254
+ if (!email && !phone) {
255
+ return undefined;
256
+ }
257
+ if (email && phone) {
258
+ const foundUsersByEmail = await this.users.list([
259
+ // @ts-ignore
260
+ Query.or([Query.equal("email", email), Query.equal("phone", phone)]),
261
+ ]);
262
+ if (foundUsersByEmail.users.length > 0) {
263
+ return foundUsersByEmail.users[0]?.$id;
264
+ }
265
+ } else if (email) {
266
+ const foundUsersByEmail = await this.users.list([
267
+ Query.equal("email", email),
268
+ ]);
269
+ if (foundUsersByEmail.users.length > 0) {
270
+ return foundUsersByEmail.users[0]?.$id;
271
+ } else {
272
+ if (!phone) {
273
+ return undefined;
274
+ } else {
275
+ const foundUsersByPhone = await this.users.list([
276
+ Query.equal("phone", phone),
277
+ ]);
278
+ if (foundUsersByPhone.users.length > 0) {
279
+ return foundUsersByPhone.users[0]?.$id;
280
+ } else {
281
+ return undefined;
282
+ }
283
+ }
284
+ }
285
+ }
286
+ if (phone) {
287
+ const foundUsersByPhone = await this.users.list([
288
+ Query.equal("phone", phone),
289
+ ]);
290
+ if (foundUsersByPhone.users.length > 0) {
291
+ return foundUsersByPhone.users[0]?.$id;
292
+ } else {
293
+ return undefined;
294
+ }
295
+ }
296
+ }
297
+
298
+ transferUsersBetweenDbsLocalToRemote = async (
299
+ endpoint: string,
300
+ projectId: string,
301
+ apiKey: string
302
+ ) => {
303
+ const localUsers = this.users;
304
+ const client = getAppwriteClient(endpoint, projectId, apiKey);
305
+ const remoteUsers = new Users(client);
306
+
307
+ let fromUsers = await localUsers.list([Query.limit(50)]);
308
+
309
+ if (fromUsers.users.length === 0) {
310
+ MessageFormatter.info("No users found", { prefix: "Users" });
311
+ return;
312
+ } else if (fromUsers.users.length < 50) {
313
+ MessageFormatter.progress(`Transferring ${fromUsers.users.length} users to remote`, { prefix: "Users" });
314
+ const batchedPromises = fromUsers.users.map((user) => {
315
+ return tryAwaitWithRetry(async () => {
316
+ const toCreateObject: Partial<typeof user> = {
317
+ ...user,
318
+ };
319
+ delete toCreateObject.$id;
320
+ delete toCreateObject.$createdAt;
321
+ delete toCreateObject.$updatedAt;
322
+ await remoteUsers.create(
323
+ user.$id,
324
+ user.email,
325
+ user.phone,
326
+ user.password,
327
+ user.name
328
+ );
329
+ });
330
+ });
331
+ await Promise.all(batchedPromises);
332
+ } else {
333
+ while (fromUsers.users.length === 50) {
334
+ fromUsers = await localUsers.list([
335
+ Query.limit(50),
336
+ Query.cursorAfter(fromUsers.users[fromUsers.users.length - 1].$id),
337
+ ]);
338
+ const batchedPromises = fromUsers.users.map((user) => {
339
+ return tryAwaitWithRetry(async () => {
340
+ const toCreateObject: Partial<typeof user> = {
341
+ ...user,
342
+ };
343
+ delete toCreateObject.$id;
344
+ delete toCreateObject.$createdAt;
345
+ delete toCreateObject.$updatedAt;
346
+ await remoteUsers.create(
347
+ user.$id,
348
+ user.email,
349
+ user.phone,
350
+ user.password,
351
+ user.name
352
+ );
353
+ });
354
+ });
355
+ await Promise.all(batchedPromises);
356
+ }
357
+ }
358
+ };
359
+ }