@njdamstra/appwrite-utils-cli 1.8.9 → 1.10.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (284) hide show
  1. package/CHANGELOG.md +16 -0
  2. package/CONFIG_TODO.md +1189 -0
  3. package/SELECTION_DIALOGS.md +146 -0
  4. package/SERVICE_IMPLEMENTATION_REPORT.md +462 -0
  5. package/dist/adapters/index.d.ts +7 -8
  6. package/dist/adapters/index.js +7 -9
  7. package/dist/backups/operations/bucketBackup.js +2 -2
  8. package/dist/backups/operations/collectionBackup.d.ts +1 -1
  9. package/dist/backups/operations/collectionBackup.js +3 -3
  10. package/dist/backups/operations/comprehensiveBackup.d.ts +1 -1
  11. package/dist/backups/operations/comprehensiveBackup.js +2 -2
  12. package/dist/backups/tracking/centralizedTracking.d.ts +1 -1
  13. package/dist/backups/tracking/centralizedTracking.js +2 -2
  14. package/dist/cli/commands/configCommands.js +51 -7
  15. package/dist/cli/commands/databaseCommands.d.ts +1 -0
  16. package/dist/cli/commands/databaseCommands.js +119 -9
  17. package/dist/cli/commands/functionCommands.js +3 -3
  18. package/dist/cli/commands/importFileCommands.d.ts +7 -0
  19. package/dist/cli/commands/importFileCommands.js +674 -0
  20. package/dist/cli/commands/schemaCommands.js +3 -3
  21. package/dist/cli/commands/storageCommands.js +2 -3
  22. package/dist/cli/commands/transferCommands.js +3 -5
  23. package/dist/collections/attributes.d.ts +1 -1
  24. package/dist/collections/attributes.js +2 -35
  25. package/dist/collections/indexes.js +1 -3
  26. package/dist/collections/methods.d.ts +1 -1
  27. package/dist/collections/methods.js +111 -192
  28. package/dist/collections/tableOperations.d.ts +1 -0
  29. package/dist/collections/tableOperations.js +55 -23
  30. package/dist/collections/transferOperations.d.ts +1 -1
  31. package/dist/collections/transferOperations.js +3 -4
  32. package/dist/collections/wipeOperations.d.ts +4 -3
  33. package/dist/collections/wipeOperations.js +112 -39
  34. package/dist/databases/methods.js +2 -2
  35. package/dist/databases/setup.js +2 -2
  36. package/dist/examples/yamlTerminologyExample.js +2 -2
  37. package/dist/functions/deployments.d.ts +1 -1
  38. package/dist/functions/deployments.js +5 -5
  39. package/dist/functions/fnConfigDiscovery.js +2 -2
  40. package/dist/functions/methods.js +16 -4
  41. package/dist/init.js +1 -1
  42. package/dist/interactiveCLI.d.ts +6 -1
  43. package/dist/interactiveCLI.js +63 -9
  44. package/dist/main.js +130 -177
  45. package/dist/migrations/afterImportActions.js +2 -3
  46. package/dist/migrations/appwriteToX.d.ts +1 -1
  47. package/dist/migrations/appwriteToX.js +9 -7
  48. package/dist/migrations/comprehensiveTransfer.js +3 -5
  49. package/dist/migrations/dataLoader.js +2 -5
  50. package/dist/migrations/importController.js +3 -4
  51. package/dist/migrations/importDataActions.js +3 -3
  52. package/dist/migrations/relationships.js +1 -2
  53. package/dist/migrations/services/DataTransformationService.js +2 -2
  54. package/dist/migrations/services/FileHandlerService.js +1 -1
  55. package/dist/migrations/services/ImportOrchestrator.js +4 -4
  56. package/dist/migrations/services/RateLimitManager.js +1 -1
  57. package/dist/migrations/services/RelationshipResolver.js +1 -1
  58. package/dist/migrations/services/UserMappingService.js +1 -1
  59. package/dist/migrations/services/ValidationService.js +1 -1
  60. package/dist/migrations/transfer.d.ts +8 -4
  61. package/dist/migrations/transfer.js +106 -55
  62. package/dist/migrations/yaml/YamlImportConfigLoader.js +1 -1
  63. package/dist/migrations/yaml/YamlImportIntegration.js +2 -2
  64. package/dist/migrations/yaml/generateImportSchemas.js +1 -1
  65. package/dist/setupCommands.d.ts +1 -1
  66. package/dist/setupCommands.js +5 -6
  67. package/dist/setupController.js +1 -1
  68. package/dist/shared/backupTracking.d.ts +1 -1
  69. package/dist/shared/backupTracking.js +2 -2
  70. package/dist/shared/confirmationDialogs.js +1 -1
  71. package/dist/shared/migrationHelpers.d.ts +1 -1
  72. package/dist/shared/migrationHelpers.js +3 -3
  73. package/dist/shared/operationQueue.d.ts +1 -1
  74. package/dist/shared/operationQueue.js +2 -3
  75. package/dist/shared/operationsTable.d.ts +1 -1
  76. package/dist/shared/operationsTable.js +2 -2
  77. package/dist/shared/progressManager.js +1 -1
  78. package/dist/shared/selectionDialogs.js +9 -8
  79. package/dist/storage/methods.js +4 -4
  80. package/dist/storage/schemas.d.ts +2 -2
  81. package/dist/tables/indexManager.d.ts +65 -0
  82. package/dist/tables/indexManager.js +294 -0
  83. package/dist/types.d.ts +2 -2
  84. package/dist/types.js +1 -1
  85. package/dist/users/methods.js +2 -3
  86. package/dist/utils/configMigration.js +1 -1
  87. package/dist/utils/index.d.ts +1 -1
  88. package/dist/utils/index.js +1 -1
  89. package/dist/utils/loadConfigs.d.ts +2 -2
  90. package/dist/utils/loadConfigs.js +6 -7
  91. package/dist/utils/setupFiles.js +5 -7
  92. package/dist/utilsController.d.ts +15 -8
  93. package/dist/utilsController.js +57 -28
  94. package/package.json +7 -3
  95. package/src/adapters/index.ts +8 -34
  96. package/src/backups/operations/bucketBackup.ts +2 -2
  97. package/src/backups/operations/collectionBackup.ts +4 -4
  98. package/src/backups/operations/comprehensiveBackup.ts +3 -3
  99. package/src/backups/tracking/centralizedTracking.ts +3 -3
  100. package/src/cli/commands/configCommands.ts +72 -8
  101. package/src/cli/commands/databaseCommands.ts +161 -9
  102. package/src/cli/commands/functionCommands.ts +4 -3
  103. package/src/cli/commands/importFileCommands.ts +815 -0
  104. package/src/cli/commands/schemaCommands.ts +3 -3
  105. package/src/cli/commands/storageCommands.ts +2 -3
  106. package/src/cli/commands/transferCommands.ts +3 -6
  107. package/src/collections/attributes.ts +3 -39
  108. package/src/collections/indexes.ts +2 -4
  109. package/src/collections/methods.ts +115 -150
  110. package/src/collections/tableOperations.ts +57 -21
  111. package/src/collections/transferOperations.ts +4 -5
  112. package/src/collections/wipeOperations.ts +154 -51
  113. package/src/databases/methods.ts +2 -2
  114. package/src/databases/setup.ts +2 -2
  115. package/src/examples/yamlTerminologyExample.ts +2 -2
  116. package/src/functions/deployments.ts +6 -5
  117. package/src/functions/fnConfigDiscovery.ts +2 -2
  118. package/src/functions/methods.ts +17 -4
  119. package/src/init.ts +1 -1
  120. package/src/interactiveCLI.ts +75 -10
  121. package/src/main.ts +143 -287
  122. package/src/migrations/afterImportActions.ts +2 -3
  123. package/src/migrations/appwriteToX.ts +12 -8
  124. package/src/migrations/comprehensiveTransfer.ts +6 -6
  125. package/src/migrations/dataLoader.ts +2 -5
  126. package/src/migrations/importController.ts +3 -4
  127. package/src/migrations/importDataActions.ts +3 -3
  128. package/src/migrations/relationships.ts +1 -2
  129. package/src/migrations/services/DataTransformationService.ts +2 -2
  130. package/src/migrations/services/FileHandlerService.ts +1 -1
  131. package/src/migrations/services/ImportOrchestrator.ts +4 -4
  132. package/src/migrations/services/RateLimitManager.ts +1 -1
  133. package/src/migrations/services/RelationshipResolver.ts +1 -1
  134. package/src/migrations/services/UserMappingService.ts +1 -1
  135. package/src/migrations/services/ValidationService.ts +1 -1
  136. package/src/migrations/transfer.ts +126 -83
  137. package/src/migrations/yaml/YamlImportConfigLoader.ts +1 -1
  138. package/src/migrations/yaml/YamlImportIntegration.ts +2 -2
  139. package/src/migrations/yaml/generateImportSchemas.ts +1 -1
  140. package/src/setupCommands.ts +5 -6
  141. package/src/setupController.ts +1 -1
  142. package/src/shared/backupTracking.ts +3 -3
  143. package/src/shared/confirmationDialogs.ts +1 -1
  144. package/src/shared/migrationHelpers.ts +4 -4
  145. package/src/shared/operationQueue.ts +3 -4
  146. package/src/shared/operationsTable.ts +3 -3
  147. package/src/shared/progressManager.ts +1 -1
  148. package/src/shared/selectionDialogs.ts +9 -8
  149. package/src/storage/methods.ts +4 -4
  150. package/src/tables/indexManager.ts +409 -0
  151. package/src/types.ts +2 -2
  152. package/src/users/methods.ts +2 -3
  153. package/src/utils/configMigration.ts +1 -1
  154. package/src/utils/index.ts +1 -1
  155. package/src/utils/loadConfigs.ts +15 -7
  156. package/src/utils/setupFiles.ts +5 -7
  157. package/src/utilsController.ts +86 -32
  158. package/dist/adapters/AdapterFactory.d.ts +0 -94
  159. package/dist/adapters/AdapterFactory.js +0 -405
  160. package/dist/adapters/DatabaseAdapter.d.ts +0 -233
  161. package/dist/adapters/DatabaseAdapter.js +0 -50
  162. package/dist/adapters/LegacyAdapter.d.ts +0 -50
  163. package/dist/adapters/LegacyAdapter.js +0 -612
  164. package/dist/adapters/TablesDBAdapter.d.ts +0 -45
  165. package/dist/adapters/TablesDBAdapter.js +0 -571
  166. package/dist/config/ConfigManager.d.ts +0 -445
  167. package/dist/config/ConfigManager.js +0 -625
  168. package/dist/config/configMigration.d.ts +0 -87
  169. package/dist/config/configMigration.js +0 -390
  170. package/dist/config/configValidation.d.ts +0 -66
  171. package/dist/config/configValidation.js +0 -358
  172. package/dist/config/index.d.ts +0 -8
  173. package/dist/config/index.js +0 -7
  174. package/dist/config/services/ConfigDiscoveryService.d.ts +0 -126
  175. package/dist/config/services/ConfigDiscoveryService.js +0 -374
  176. package/dist/config/services/ConfigLoaderService.d.ts +0 -129
  177. package/dist/config/services/ConfigLoaderService.js +0 -540
  178. package/dist/config/services/ConfigMergeService.d.ts +0 -208
  179. package/dist/config/services/ConfigMergeService.js +0 -308
  180. package/dist/config/services/ConfigValidationService.d.ts +0 -214
  181. package/dist/config/services/ConfigValidationService.js +0 -310
  182. package/dist/config/services/SessionAuthService.d.ts +0 -225
  183. package/dist/config/services/SessionAuthService.js +0 -456
  184. package/dist/config/services/__tests__/ConfigMergeService.test.d.ts +0 -1
  185. package/dist/config/services/__tests__/ConfigMergeService.test.js +0 -271
  186. package/dist/config/services/index.d.ts +0 -13
  187. package/dist/config/services/index.js +0 -10
  188. package/dist/config/yamlConfig.d.ts +0 -722
  189. package/dist/config/yamlConfig.js +0 -702
  190. package/dist/functions/pathResolution.d.ts +0 -37
  191. package/dist/functions/pathResolution.js +0 -185
  192. package/dist/shared/attributeMapper.d.ts +0 -20
  193. package/dist/shared/attributeMapper.js +0 -203
  194. package/dist/shared/errorUtils.d.ts +0 -54
  195. package/dist/shared/errorUtils.js +0 -95
  196. package/dist/shared/functionManager.d.ts +0 -48
  197. package/dist/shared/functionManager.js +0 -336
  198. package/dist/shared/indexManager.d.ts +0 -24
  199. package/dist/shared/indexManager.js +0 -151
  200. package/dist/shared/jsonSchemaGenerator.d.ts +0 -50
  201. package/dist/shared/jsonSchemaGenerator.js +0 -290
  202. package/dist/shared/logging.d.ts +0 -61
  203. package/dist/shared/logging.js +0 -116
  204. package/dist/shared/messageFormatter.d.ts +0 -39
  205. package/dist/shared/messageFormatter.js +0 -162
  206. package/dist/shared/pydanticModelGenerator.d.ts +0 -17
  207. package/dist/shared/pydanticModelGenerator.js +0 -615
  208. package/dist/shared/schemaGenerator.d.ts +0 -40
  209. package/dist/shared/schemaGenerator.js +0 -556
  210. package/dist/utils/ClientFactory.d.ts +0 -87
  211. package/dist/utils/ClientFactory.js +0 -212
  212. package/dist/utils/configDiscovery.d.ts +0 -78
  213. package/dist/utils/configDiscovery.js +0 -472
  214. package/dist/utils/constantsGenerator.d.ts +0 -31
  215. package/dist/utils/constantsGenerator.js +0 -321
  216. package/dist/utils/dataConverters.d.ts +0 -46
  217. package/dist/utils/dataConverters.js +0 -139
  218. package/dist/utils/directoryUtils.d.ts +0 -22
  219. package/dist/utils/directoryUtils.js +0 -59
  220. package/dist/utils/getClientFromConfig.d.ts +0 -39
  221. package/dist/utils/getClientFromConfig.js +0 -199
  222. package/dist/utils/helperFunctions.d.ts +0 -63
  223. package/dist/utils/helperFunctions.js +0 -156
  224. package/dist/utils/pathResolvers.d.ts +0 -53
  225. package/dist/utils/pathResolvers.js +0 -72
  226. package/dist/utils/projectConfig.d.ts +0 -119
  227. package/dist/utils/projectConfig.js +0 -171
  228. package/dist/utils/retryFailedPromises.d.ts +0 -2
  229. package/dist/utils/retryFailedPromises.js +0 -23
  230. package/dist/utils/sessionAuth.d.ts +0 -48
  231. package/dist/utils/sessionAuth.js +0 -164
  232. package/dist/utils/typeGuards.d.ts +0 -35
  233. package/dist/utils/typeGuards.js +0 -57
  234. package/dist/utils/validationRules.d.ts +0 -43
  235. package/dist/utils/validationRules.js +0 -42
  236. package/dist/utils/versionDetection.d.ts +0 -58
  237. package/dist/utils/versionDetection.js +0 -251
  238. package/dist/utils/yamlConverter.d.ts +0 -100
  239. package/dist/utils/yamlConverter.js +0 -428
  240. package/dist/utils/yamlLoader.d.ts +0 -70
  241. package/dist/utils/yamlLoader.js +0 -267
  242. package/src/adapters/AdapterFactory.ts +0 -510
  243. package/src/adapters/DatabaseAdapter.ts +0 -306
  244. package/src/adapters/LegacyAdapter.ts +0 -841
  245. package/src/adapters/TablesDBAdapter.ts +0 -773
  246. package/src/config/ConfigManager.ts +0 -808
  247. package/src/config/README.md +0 -274
  248. package/src/config/configMigration.ts +0 -575
  249. package/src/config/configValidation.ts +0 -445
  250. package/src/config/index.ts +0 -10
  251. package/src/config/services/ConfigDiscoveryService.ts +0 -463
  252. package/src/config/services/ConfigLoaderService.ts +0 -740
  253. package/src/config/services/ConfigMergeService.ts +0 -388
  254. package/src/config/services/ConfigValidationService.ts +0 -394
  255. package/src/config/services/SessionAuthService.ts +0 -565
  256. package/src/config/services/__tests__/ConfigMergeService.test.ts +0 -351
  257. package/src/config/services/index.ts +0 -29
  258. package/src/config/yamlConfig.ts +0 -761
  259. package/src/functions/pathResolution.ts +0 -227
  260. package/src/shared/attributeMapper.ts +0 -229
  261. package/src/shared/errorUtils.ts +0 -110
  262. package/src/shared/functionManager.ts +0 -525
  263. package/src/shared/indexManager.ts +0 -254
  264. package/src/shared/jsonSchemaGenerator.ts +0 -383
  265. package/src/shared/logging.ts +0 -149
  266. package/src/shared/messageFormatter.ts +0 -208
  267. package/src/shared/pydanticModelGenerator.ts +0 -618
  268. package/src/shared/schemaGenerator.ts +0 -644
  269. package/src/utils/ClientFactory.ts +0 -240
  270. package/src/utils/configDiscovery.ts +0 -557
  271. package/src/utils/constantsGenerator.ts +0 -369
  272. package/src/utils/dataConverters.ts +0 -159
  273. package/src/utils/directoryUtils.ts +0 -61
  274. package/src/utils/getClientFromConfig.ts +0 -257
  275. package/src/utils/helperFunctions.ts +0 -228
  276. package/src/utils/pathResolvers.ts +0 -81
  277. package/src/utils/projectConfig.ts +0 -299
  278. package/src/utils/retryFailedPromises.ts +0 -29
  279. package/src/utils/sessionAuth.ts +0 -230
  280. package/src/utils/typeGuards.ts +0 -65
  281. package/src/utils/validationRules.ts +0 -88
  282. package/src/utils/versionDetection.ts +0 -292
  283. package/src/utils/yamlConverter.ts +0 -542
  284. package/src/utils/yamlLoader.ts +0 -371
@@ -1,8 +1,8 @@
1
1
  import inquirer from "inquirer";
2
2
  import chalk from "chalk";
3
3
  import type { Models } from "node-appwrite";
4
- import { MessageFormatter } from "./messageFormatter.js";
5
- import { logger } from "./logging.js";
4
+ import { MessageFormatter } from '@njdamstra/appwrite-utils-helpers';
5
+ import { logger } from '@njdamstra/appwrite-utils-helpers';
6
6
 
7
7
  /**
8
8
  * Interface for sync selection summary
@@ -307,7 +307,7 @@ export class SelectionDialogs {
307
307
  MessageFormatter.section(`Table Selection for ${databaseName}`);
308
308
  }
309
309
 
310
- const configuredIds = new Set(configuredTables.map(table => table.$id || table.id));
310
+ const configuredIds = new Set(configuredTables.map(table => table.$id || table.id || (table as any).tableId || table.name));
311
311
 
312
312
  let choices: any[] = [];
313
313
 
@@ -320,9 +320,10 @@ export class SelectionDialogs {
320
320
  }
321
321
 
322
322
  availableTables.forEach(table => {
323
- const isConfigured = configuredIds.has(table.$id);
323
+ const tableId = table.$id || table.id || (table as any).tableId || table.name;
324
+ const isConfigured = configuredIds.has(tableId);
324
325
  const status = isConfigured ? chalk.green('✅') : chalk.blue('○');
325
- const name = `${status} ${table.name} (${table.$id})`;
326
+ const name = `${status} ${table.name} (${tableId})`;
326
327
 
327
328
  if (allowNewOnly && isConfigured) {
328
329
  return; // Skip configured tables if only allowing new ones
@@ -330,10 +331,10 @@ export class SelectionDialogs {
330
331
 
331
332
  choices.push({
332
333
  name,
333
- value: table.$id,
334
+ value: tableId,
334
335
  short: table.name,
335
336
  // Do not preselect anything unless explicitly provided
336
- checked: defaultSelected.includes(table.$id)
337
+ checked: defaultSelected.includes(tableId)
337
338
  });
338
339
  });
339
340
 
@@ -360,7 +361,7 @@ export class SelectionDialogs {
360
361
 
361
362
  // Handle select all
362
363
  if (selectedTableIds.includes('__SELECT_ALL__')) {
363
- const allIds = availableTables.map(table => table.$id);
364
+ const allIds = availableTables.map(table => table.$id || table.id || (table as any).tableId || table.name);
364
365
  if (allowNewOnly) {
365
366
  return allIds.filter(id => !configuredIds.has(id));
366
367
  }
@@ -8,17 +8,17 @@ import {
8
8
  type Models,
9
9
  } from "node-appwrite";
10
10
  import { tryAwaitWithRetry, type AppwriteConfig } from "@njdamstra/appwrite-utils";
11
- import { getClientFromConfig } from "../utils/getClientFromConfig.js";
11
+ import { getClientFromConfig } from "@njdamstra/appwrite-utils-helpers";
12
12
  import { ulid } from "ulidx";
13
13
  import type { BackupCreate } from "./schemas.js";
14
14
  import { logOperation } from "../shared/operationLogger.js";
15
15
  import { splitIntoBatches } from "../shared/migrationHelpers.js";
16
- import { retryFailedPromises } from "../utils/retryFailedPromises.js";
16
+ import { retryFailedPromises } from "@njdamstra/appwrite-utils-helpers";
17
17
  import { InputFile } from "node-appwrite/file";
18
- import { MessageFormatter, Messages } from "../shared/messageFormatter.js";
18
+ import { MessageFormatter, Messages } from "@njdamstra/appwrite-utils-helpers";
19
19
  import { ProgressManager } from "../shared/progressManager.js";
20
20
  import { recordBackup } from "../shared/backupTracking.js";
21
- import { AdapterFactory } from "../adapters/AdapterFactory.js";
21
+ import { AdapterFactory } from "@njdamstra/appwrite-utils-helpers";
22
22
  import { createBackupZip } from "./backupCompression.js";
23
23
 
24
24
  export const getStorage = (config: AppwriteConfig) => {
@@ -0,0 +1,409 @@
1
+ import type { Index } from "@njdamstra/appwrite-utils";
2
+ import type { DatabaseAdapter } from "@njdamstra/appwrite-utils-helpers";
3
+ import type { Models } from "node-appwrite";
4
+ import { isIndexEqualToIndex } from "../collections/tableOperations.js";
5
+ import { MessageFormatter } from "@njdamstra/appwrite-utils-helpers";
6
+ import { delay, tryAwaitWithRetry } from "@njdamstra/appwrite-utils-helpers";
7
+
8
+ // Enhanced index operation interfaces
9
+ export interface IndexOperation {
10
+ type: 'create' | 'update' | 'skip' | 'delete';
11
+ index: Index;
12
+ existingIndex?: Models.Index;
13
+ reason?: string;
14
+ }
15
+
16
+ export interface IndexOperationPlan {
17
+ toCreate: IndexOperation[];
18
+ toUpdate: IndexOperation[];
19
+ toSkip: IndexOperation[];
20
+ toDelete: IndexOperation[];
21
+ }
22
+
23
+ export interface IndexExecutionResult {
24
+ created: string[];
25
+ updated: string[];
26
+ skipped: string[];
27
+ deleted: string[];
28
+ errors: Array<{ key: string; error: string }>;
29
+ summary: {
30
+ total: number;
31
+ created: number;
32
+ updated: number;
33
+ skipped: number;
34
+ deleted: number;
35
+ errors: number;
36
+ };
37
+ }
38
+
39
+ /**
40
+ * Plan index operations by comparing desired indexes with existing ones
41
+ * Uses the existing isIndexEqualToIndex function for consistent comparison
42
+ */
43
+ export function planIndexOperations(
44
+ desiredIndexes: Index[],
45
+ existingIndexes: Models.Index[]
46
+ ): IndexOperationPlan {
47
+ const plan: IndexOperationPlan = {
48
+ toCreate: [],
49
+ toUpdate: [],
50
+ toSkip: [],
51
+ toDelete: []
52
+ };
53
+
54
+ for (const desiredIndex of desiredIndexes) {
55
+ const existingIndex = existingIndexes.find(idx => idx.key === desiredIndex.key);
56
+
57
+ if (!existingIndex) {
58
+ // Index doesn't exist - create it
59
+ plan.toCreate.push({
60
+ type: 'create',
61
+ index: desiredIndex,
62
+ reason: 'New index'
63
+ });
64
+ } else if (isIndexEqualToIndex(existingIndex, desiredIndex)) {
65
+ // Index exists and is identical - skip it
66
+ plan.toSkip.push({
67
+ type: 'skip',
68
+ index: desiredIndex,
69
+ existingIndex,
70
+ reason: 'Index unchanged'
71
+ });
72
+ } else {
73
+ // Index exists but is different - update it
74
+ plan.toUpdate.push({
75
+ type: 'update',
76
+ index: desiredIndex,
77
+ existingIndex,
78
+ reason: 'Index configuration changed'
79
+ });
80
+ }
81
+ }
82
+
83
+ return plan;
84
+ }
85
+
86
+ /**
87
+ * Plan index deletions for indexes that exist but aren't in the desired configuration
88
+ */
89
+ export function planIndexDeletions(
90
+ desiredIndexKeys: Set<string>,
91
+ existingIndexes: Models.Index[]
92
+ ): IndexOperation[] {
93
+ const deletions: IndexOperation[] = [];
94
+
95
+ for (const existingIndex of existingIndexes) {
96
+ if (!desiredIndexKeys.has(existingIndex.key)) {
97
+ deletions.push({
98
+ type: 'delete',
99
+ index: existingIndex as Index, // Convert Models.Index to Index for compatibility
100
+ reason: 'Obsolete index'
101
+ });
102
+ }
103
+ }
104
+
105
+ return deletions;
106
+ }
107
+
108
+ /**
109
+ * Execute index operations with proper error handling and status monitoring
110
+ */
111
+ export async function executeIndexOperations(
112
+ adapter: DatabaseAdapter,
113
+ databaseId: string,
114
+ tableId: string,
115
+ plan: IndexOperationPlan
116
+ ): Promise<IndexExecutionResult> {
117
+ const result: IndexExecutionResult = {
118
+ created: [],
119
+ updated: [],
120
+ skipped: [],
121
+ deleted: [],
122
+ errors: [],
123
+ summary: {
124
+ total: 0,
125
+ created: 0,
126
+ updated: 0,
127
+ skipped: 0,
128
+ deleted: 0,
129
+ errors: 0
130
+ }
131
+ };
132
+
133
+ // Execute creates
134
+ for (const operation of plan.toCreate) {
135
+ try {
136
+ await adapter.createIndex({
137
+ databaseId,
138
+ tableId,
139
+ key: operation.index.key,
140
+ type: operation.index.type,
141
+ attributes: operation.index.attributes,
142
+ orders: operation.index.orders || []
143
+ });
144
+
145
+ result.created.push(operation.index.key);
146
+ MessageFormatter.success(`Created index ${operation.index.key}`, { prefix: 'Indexes' });
147
+
148
+ // Wait for index to become available
149
+ await waitForIndexAvailable(adapter, databaseId, tableId, operation.index.key);
150
+
151
+ await delay(150); // Brief delay between operations
152
+ } catch (error) {
153
+ const errorMessage = error instanceof Error ? error.message : String(error);
154
+ result.errors.push({ key: operation.index.key, error: errorMessage });
155
+ MessageFormatter.error(`Failed to create index ${operation.index.key}`, error instanceof Error ? error : new Error(String(error)), { prefix: 'Indexes' });
156
+ }
157
+ }
158
+
159
+ // Execute updates (delete + recreate)
160
+ for (const operation of plan.toUpdate) {
161
+ try {
162
+ // Delete existing index first
163
+ await adapter.deleteIndex({
164
+ databaseId,
165
+ tableId,
166
+ key: operation.index.key
167
+ });
168
+
169
+ await delay(100); // Brief delay for deletion to settle
170
+
171
+ // Create new index
172
+ await adapter.createIndex({
173
+ databaseId,
174
+ tableId,
175
+ key: operation.index.key,
176
+ type: operation.index.type,
177
+ attributes: operation.index.attributes,
178
+ orders: operation.index.orders || operation.existingIndex?.orders || []
179
+ });
180
+
181
+ result.updated.push(operation.index.key);
182
+ MessageFormatter.success(`Updated index ${operation.index.key}`, { prefix: 'Indexes' });
183
+
184
+ // Wait for index to become available
185
+ await waitForIndexAvailable(adapter, databaseId, tableId, operation.index.key);
186
+
187
+ await delay(150); // Brief delay between operations
188
+ } catch (error) {
189
+ const errorMessage = error instanceof Error ? error.message : String(error);
190
+ result.errors.push({ key: operation.index.key, error: errorMessage });
191
+ MessageFormatter.error(`Failed to update index ${operation.index.key}`, error instanceof Error ? error : new Error(String(error)), { prefix: 'Indexes' });
192
+ }
193
+ }
194
+
195
+ // Execute skips
196
+ for (const operation of plan.toSkip) {
197
+ result.skipped.push(operation.index.key);
198
+ MessageFormatter.info(`Index ${operation.index.key} unchanged`, { prefix: 'Indexes' });
199
+ }
200
+
201
+ // Calculate summary
202
+ result.summary.total = result.created.length + result.updated.length + result.skipped.length + result.deleted.length;
203
+ result.summary.created = result.created.length;
204
+ result.summary.updated = result.updated.length;
205
+ result.summary.skipped = result.skipped.length;
206
+ result.summary.deleted = result.deleted.length;
207
+ result.summary.errors = result.errors.length;
208
+
209
+ return result;
210
+ }
211
+
212
+ /**
213
+ * Execute index deletions with proper error handling
214
+ */
215
+ export async function executeIndexDeletions(
216
+ adapter: DatabaseAdapter,
217
+ databaseId: string,
218
+ tableId: string,
219
+ deletions: IndexOperation[]
220
+ ): Promise<{ deleted: string[]; errors: Array<{ key: string; error: string }> }> {
221
+ const result = {
222
+ deleted: [] as string[],
223
+ errors: [] as Array<{ key: string; error: string }>
224
+ };
225
+
226
+ for (const operation of deletions) {
227
+ try {
228
+ await adapter.deleteIndex({
229
+ databaseId,
230
+ tableId,
231
+ key: operation.index.key
232
+ });
233
+
234
+ result.deleted.push(operation.index.key);
235
+ MessageFormatter.info(`Deleted obsolete index ${operation.index.key}`, { prefix: 'Indexes' });
236
+
237
+ // Wait briefly for deletion to settle
238
+ await delay(500);
239
+ } catch (error) {
240
+ const errorMessage = error instanceof Error ? error.message : String(error);
241
+ result.errors.push({ key: operation.index.key, error: errorMessage });
242
+ MessageFormatter.error(`Failed to delete index ${operation.index.key}`, error instanceof Error ? error : new Error(String(error)), { prefix: 'Indexes' });
243
+ }
244
+ }
245
+
246
+ return result;
247
+ }
248
+
249
+ /**
250
+ * Wait for an index to become available with timeout and retry logic
251
+ * This is an adapter-aware version of the logic from collections/indexes.ts
252
+ */
253
+ async function waitForIndexAvailable(
254
+ adapter: DatabaseAdapter,
255
+ databaseId: string,
256
+ tableId: string,
257
+ indexKey: string,
258
+ maxWaitTime: number = 60000, // 1 minute
259
+ checkInterval: number = 2000 // 2 seconds
260
+ ): Promise<boolean> {
261
+ const startTime = Date.now();
262
+
263
+ while (Date.now() - startTime < maxWaitTime) {
264
+ try {
265
+ const indexList = await adapter.listIndexes({ databaseId, tableId });
266
+ const indexes: any[] = (indexList as any).data || (indexList as any).indexes || [];
267
+ const index = indexes.find((idx: any) => idx.key === indexKey);
268
+
269
+ if (!index) {
270
+ MessageFormatter.error(`Index '${indexKey}' not found after creation`, undefined, { prefix: 'Indexes' });
271
+ return false;
272
+ }
273
+
274
+ switch (index.status) {
275
+ case 'available':
276
+ return true;
277
+
278
+ case 'failed':
279
+ MessageFormatter.error(`Index '${indexKey}' failed: ${index.error || 'unknown error'}`, undefined, { prefix: 'Indexes' });
280
+ return false;
281
+
282
+ case 'stuck':
283
+ MessageFormatter.warning(`Index '${indexKey}' is stuck`, { prefix: 'Indexes' });
284
+ return false;
285
+
286
+ case 'processing':
287
+ case 'deleting':
288
+ // Continue waiting
289
+ break;
290
+
291
+ default:
292
+ MessageFormatter.warning(`Unknown status '${index.status}' for index '${indexKey}'`, { prefix: 'Indexes' });
293
+ break;
294
+ }
295
+ } catch (error) {
296
+ MessageFormatter.error(`Error checking index '${indexKey}' status: ${error}`, undefined, { prefix: 'Indexes' });
297
+ }
298
+
299
+ await delay(checkInterval);
300
+ }
301
+
302
+ MessageFormatter.warning(`Timeout waiting for index '${indexKey}' to become available (${maxWaitTime}ms)`, { prefix: 'Indexes' });
303
+ return false;
304
+ }
305
+
306
+ /**
307
+ * Main function to create/update indexes via adapter
308
+ * This replaces the messy inline code in methods.ts
309
+ */
310
+ export async function createOrUpdateIndexesViaAdapter(
311
+ adapter: DatabaseAdapter,
312
+ databaseId: string,
313
+ tableId: string,
314
+ desiredIndexes: Index[],
315
+ configIndexes?: Index[]
316
+ ): Promise<void> {
317
+ if (!desiredIndexes || desiredIndexes.length === 0) {
318
+ MessageFormatter.info('No indexes to process', { prefix: 'Indexes' });
319
+ return;
320
+ }
321
+
322
+ MessageFormatter.info(`Processing ${desiredIndexes.length} indexes for table ${tableId}`, { prefix: 'Indexes' });
323
+
324
+ try {
325
+ // Get existing indexes
326
+ const existingIdxRes = await adapter.listIndexes({ databaseId, tableId });
327
+ const existingIndexes: Models.Index[] = (existingIdxRes as any).data || (existingIdxRes as any).indexes || [];
328
+
329
+ // Plan operations
330
+ const plan = planIndexOperations(desiredIndexes, existingIndexes);
331
+
332
+ // Show plan with icons (consistent with attribute handling)
333
+ const planParts: string[] = [];
334
+ if (plan.toCreate.length) planParts.push(`➕ ${plan.toCreate.length} (${plan.toCreate.map(op => op.index.key).join(', ')})`);
335
+ if (plan.toUpdate.length) planParts.push(`🔧 ${plan.toUpdate.length} (${plan.toUpdate.map(op => op.index.key).join(', ')})`);
336
+ if (plan.toSkip.length) planParts.push(`⏭️ ${plan.toSkip.length}`);
337
+
338
+ MessageFormatter.info(`Plan → ${planParts.join(' | ') || 'no changes'}`, { prefix: 'Indexes' });
339
+
340
+ // Execute operations
341
+ const result = await executeIndexOperations(adapter, databaseId, tableId, plan);
342
+
343
+ // Show summary
344
+ MessageFormatter.info(
345
+ `Summary → ➕ ${result.summary.created} | 🔧 ${result.summary.updated} | ⏭️ ${result.summary.skipped}`,
346
+ { prefix: 'Indexes' }
347
+ );
348
+
349
+ // Handle errors if any
350
+ if (result.errors.length > 0) {
351
+ MessageFormatter.error(`${result.errors.length} index operations failed:`, undefined, { prefix: 'Indexes' });
352
+ for (const error of result.errors) {
353
+ MessageFormatter.error(` ${error.key}: ${error.error}`, undefined, { prefix: 'Indexes' });
354
+ }
355
+ }
356
+
357
+ } catch (error) {
358
+ MessageFormatter.error('Failed to process indexes', error instanceof Error ? error : new Error(String(error)), { prefix: 'Indexes' });
359
+ throw error;
360
+ }
361
+ }
362
+
363
+ /**
364
+ * Handle index deletions for obsolete indexes
365
+ */
366
+ export async function deleteObsoleteIndexesViaAdapter(
367
+ adapter: DatabaseAdapter,
368
+ databaseId: string,
369
+ tableId: string,
370
+ desiredIndexKeys: Set<string>
371
+ ): Promise<void> {
372
+ try {
373
+ // Get existing indexes
374
+ const existingIdxRes = await adapter.listIndexes({ databaseId, tableId });
375
+ const existingIndexes: Models.Index[] = (existingIdxRes as any).data || (existingIdxRes as any).indexes || [];
376
+
377
+ // Plan deletions
378
+ const deletions = planIndexDeletions(desiredIndexKeys, existingIndexes);
379
+
380
+ if (deletions.length === 0) {
381
+ MessageFormatter.info('Plan → 🗑️ 0 indexes', { prefix: 'Indexes' });
382
+ return;
383
+ }
384
+
385
+ // Show deletion plan
386
+ MessageFormatter.info(
387
+ `Plan → 🗑️ ${deletions.length} (${deletions.map(op => op.index.key).join(', ')})`,
388
+ { prefix: 'Indexes' }
389
+ );
390
+
391
+ // Execute deletions
392
+ const result = await executeIndexDeletions(adapter, databaseId, tableId, deletions);
393
+
394
+ // Show results
395
+ if (result.deleted.length > 0) {
396
+ MessageFormatter.success(`Deleted ${result.deleted.length} indexes: ${result.deleted.join(', ')}`, { prefix: 'Indexes' });
397
+ }
398
+
399
+ if (result.errors.length > 0) {
400
+ MessageFormatter.error(`${result.errors.length} index deletions failed:`, undefined, { prefix: 'Indexes' });
401
+ for (const error of result.errors) {
402
+ MessageFormatter.error(` ${error.key}: ${error.error}`, undefined, { prefix: 'Indexes' });
403
+ }
404
+ }
405
+
406
+ } catch (error) {
407
+ MessageFormatter.warning(`Could not evaluate index deletions: ${(error as Error)?.message || error}`, { prefix: 'Indexes' });
408
+ }
409
+ }
package/src/types.ts CHANGED
@@ -1,9 +1,9 @@
1
- export type { ValidationRules } from "./utils/validationRules.js";
1
+ export type { ValidationRules } from "@njdamstra/appwrite-utils-helpers";
2
2
  export {
3
3
  type AuthUserCreate,
4
4
  AuthUserCreateSchema,
5
5
  type AuthUser,
6
6
  AuthUserSchema,
7
7
  } from "./schemas/authUser.js";
8
- export { validationRules } from "./utils/validationRules.js";
8
+ export { validationRules } from "@njdamstra/appwrite-utils-helpers";
9
9
  export { afterImportActions } from "./migrations/afterImportActions.js";
@@ -12,15 +12,14 @@ import {
12
12
  type AuthUser,
13
13
  type AuthUserCreate,
14
14
  } from "../schemas/authUser.js";
15
- import { logger } from "../shared/logging.js";
15
+ import { logger, MessageFormatter } from "@njdamstra/appwrite-utils-helpers";
16
16
  import { splitIntoBatches } from "../shared/migrationHelpers.js";
17
17
  import {
18
18
  getAppwriteClient,
19
19
  tryAwaitWithRetry,
20
- } from "../utils/helperFunctions.js";
20
+ } from "@njdamstra/appwrite-utils-helpers";
21
21
  import { isUndefined } from "es-toolkit/compat";
22
22
  import { isEmpty } from "es-toolkit/compat";
23
- import { MessageFormatter } from "../shared/messageFormatter.js";
24
23
 
25
24
  export class UsersController {
26
25
  private config: AppwriteConfig;
@@ -1,7 +1,7 @@
1
1
  import { promises as fs } from "fs";
2
2
  import path from "path";
3
3
  import { existsSync } from "fs";
4
- import { MessageFormatter } from "../shared/messageFormatter.js";
4
+ import { MessageFormatter } from "@njdamstra/appwrite-utils-helpers";
5
5
  import { ConfirmationDialogs } from "../shared/confirmationDialogs.js";
6
6
  import yaml from "js-yaml";
7
7
 
@@ -1,2 +1,2 @@
1
- export * from "./helperFunctions.js";
1
+ export * from '@njdamstra/appwrite-utils-helpers';
2
2
  export * from "./setupFiles.js";
@@ -4,11 +4,19 @@ import { type AppwriteConfig, type Collection, type CollectionCreate, type Table
4
4
  import { register } from "tsx/esm/api"; // Import the register function
5
5
  import { pathToFileURL } from "node:url";
6
6
  import chalk from "chalk";
7
- import { findYamlConfig, loadYamlConfig, loadYamlConfigWithSession, extractSessionOptionsFromConfig, type YamlSessionOptions } from "../config/yamlConfig.js";
8
- import { detectAppwriteVersionCached, fetchServerVersion, isVersionAtLeast } from "./versionDetection.js";
9
- import { MessageFormatter } from "../shared/messageFormatter.js";
10
- import { validateCollectionsTablesConfig, reportValidationResults, type ValidationResult } from "../config/configValidation.js";
11
- import { resolveCollectionsDir, resolveTablesDir } from "./pathResolvers.js";
7
+ import {
8
+ findYamlConfig,
9
+ loadYamlConfig,
10
+ loadYamlConfigWithSession,
11
+ extractSessionOptionsFromConfig,
12
+ validateCollectionsTablesConfig,
13
+ reportValidationResults,
14
+ type YamlSessionOptions,
15
+ type ValidationResult
16
+ } from "@njdamstra/appwrite-utils-helpers";
17
+ import { detectAppwriteVersionCached, fetchServerVersion, isVersionAtLeast } from '@njdamstra/appwrite-utils-helpers';
18
+ import { MessageFormatter } from "@njdamstra/appwrite-utils-helpers";
19
+ import { resolveCollectionsDir, resolveTablesDir } from '@njdamstra/appwrite-utils-helpers';
12
20
  import {
13
21
  findAppwriteConfig,
14
22
  findAppwriteConfigTS,
@@ -16,7 +24,7 @@ import {
16
24
  discoverCollections,
17
25
  discoverTables,
18
26
  discoverLegacyDirectory
19
- } from "./configDiscovery.js";
27
+ } from '@njdamstra/appwrite-utils-helpers';
20
28
 
21
29
  /**
22
30
  * Session authentication preservation options for config loading
@@ -63,7 +71,7 @@ export function createSessionPreservation(
63
71
  }
64
72
 
65
73
  // Re-export config discovery functions for backward compatibility
66
- export { findAppwriteConfig, findFunctionsDir } from "./configDiscovery.js";
74
+ export { findAppwriteConfig, findFunctionsDir } from '@njdamstra/appwrite-utils-helpers';
67
75
 
68
76
  /**
69
77
  * Loads the Appwrite configuration and returns both config and the path where it was found.
@@ -2,15 +2,13 @@ import { mkdirSync, writeFileSync, existsSync } from "node:fs";
2
2
  import path from "node:path";
3
3
  import type { AppwriteConfig } from "@njdamstra/appwrite-utils";
4
4
  import { findAppwriteConfig } from "./loadConfigs.js";
5
- import { loadYamlConfig } from "../config/yamlConfig.js";
6
- import { fetchServerVersion, isVersionAtLeast } from "./versionDetection.js";
7
- import { findYamlConfig } from "../config/yamlConfig.js";
5
+ import { loadYamlConfig, findYamlConfig, generateYamlConfigTemplate } from "@njdamstra/appwrite-utils-helpers";
6
+ import { fetchServerVersion, isVersionAtLeast } from '@njdamstra/appwrite-utils-helpers';
8
7
  import { ID } from "node-appwrite";
9
8
  import { ulid } from "ulidx";
10
- import { generateYamlConfigTemplate } from "../config/yamlConfig.js";
11
- import { loadAppwriteProjectConfig, findAppwriteProjectConfig, getProjectDirectoryName, isTablesDBProject } from "./projectConfig.js";
12
- import { hasSessionAuth, getSessionAuth } from "./sessionAuth.js";
13
- import { MessageFormatter } from "../shared/messageFormatter.js";
9
+ import { loadAppwriteProjectConfig, findAppwriteProjectConfig, getProjectDirectoryName, isTablesDBProject } from '@njdamstra/appwrite-utils-helpers';
10
+ import { hasSessionAuth, getSessionAuth } from '@njdamstra/appwrite-utils-helpers';
11
+ import { MessageFormatter } from "@njdamstra/appwrite-utils-helpers";
14
12
 
15
13
  // Example base configuration using types from appwrite-utils
16
14
  const baseConfig: AppwriteConfig = {