@njdamstra/appwrite-utils-cli 1.8.9 → 1.10.1

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 (285) 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 +85 -35
  25. package/dist/collections/indexes.js +2 -4
  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 +90 -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 +64 -10
  44. package/dist/main.js +130 -177
  45. package/dist/migrations/afterImportActions.js +2 -3
  46. package/dist/migrations/appwriteToX.d.ts +97 -1
  47. package/dist/migrations/appwriteToX.js +9 -7
  48. package/dist/migrations/comprehensiveTransfer.js +3 -5
  49. package/dist/migrations/dataLoader.d.ts +194 -2
  50. package/dist/migrations/dataLoader.js +2 -5
  51. package/dist/migrations/importController.js +3 -4
  52. package/dist/migrations/importDataActions.js +3 -3
  53. package/dist/migrations/relationships.js +1 -2
  54. package/dist/migrations/services/DataTransformationService.js +2 -2
  55. package/dist/migrations/services/FileHandlerService.js +1 -1
  56. package/dist/migrations/services/ImportOrchestrator.js +4 -4
  57. package/dist/migrations/services/RateLimitManager.js +1 -1
  58. package/dist/migrations/services/RelationshipResolver.js +1 -1
  59. package/dist/migrations/services/UserMappingService.js +1 -1
  60. package/dist/migrations/services/ValidationService.js +1 -1
  61. package/dist/migrations/transfer.d.ts +8 -4
  62. package/dist/migrations/transfer.js +106 -55
  63. package/dist/migrations/yaml/YamlImportConfigLoader.js +1 -1
  64. package/dist/migrations/yaml/YamlImportIntegration.js +2 -2
  65. package/dist/migrations/yaml/generateImportSchemas.js +1 -1
  66. package/dist/setupCommands.d.ts +1 -1
  67. package/dist/setupCommands.js +5 -6
  68. package/dist/setupController.js +1 -1
  69. package/dist/shared/backupTracking.d.ts +1 -1
  70. package/dist/shared/backupTracking.js +2 -2
  71. package/dist/shared/confirmationDialogs.js +1 -1
  72. package/dist/shared/migrationHelpers.d.ts +1 -1
  73. package/dist/shared/migrationHelpers.js +3 -3
  74. package/dist/shared/operationQueue.d.ts +1 -1
  75. package/dist/shared/operationQueue.js +2 -3
  76. package/dist/shared/operationsTable.d.ts +1 -1
  77. package/dist/shared/operationsTable.js +2 -2
  78. package/dist/shared/progressManager.js +1 -1
  79. package/dist/shared/selectionDialogs.js +9 -8
  80. package/dist/storage/methods.js +4 -4
  81. package/dist/storage/schemas.d.ts +386 -2
  82. package/dist/tables/indexManager.d.ts +65 -0
  83. package/dist/tables/indexManager.js +294 -0
  84. package/dist/types.d.ts +2 -2
  85. package/dist/types.js +1 -1
  86. package/dist/users/methods.js +2 -3
  87. package/dist/utils/configMigration.js +1 -1
  88. package/dist/utils/index.d.ts +1 -1
  89. package/dist/utils/index.js +1 -1
  90. package/dist/utils/loadConfigs.d.ts +2 -2
  91. package/dist/utils/loadConfigs.js +6 -7
  92. package/dist/utils/setupFiles.js +5 -7
  93. package/dist/utilsController.d.ts +15 -8
  94. package/dist/utilsController.js +57 -28
  95. package/package.json +8 -4
  96. package/src/adapters/index.ts +8 -34
  97. package/src/backups/operations/bucketBackup.ts +2 -2
  98. package/src/backups/operations/collectionBackup.ts +4 -4
  99. package/src/backups/operations/comprehensiveBackup.ts +3 -3
  100. package/src/backups/tracking/centralizedTracking.ts +3 -3
  101. package/src/cli/commands/configCommands.ts +72 -8
  102. package/src/cli/commands/databaseCommands.ts +161 -9
  103. package/src/cli/commands/functionCommands.ts +4 -3
  104. package/src/cli/commands/importFileCommands.ts +815 -0
  105. package/src/cli/commands/schemaCommands.ts +3 -3
  106. package/src/cli/commands/storageCommands.ts +2 -3
  107. package/src/cli/commands/transferCommands.ts +3 -6
  108. package/src/collections/attributes.ts +155 -39
  109. package/src/collections/indexes.ts +5 -7
  110. package/src/collections/methods.ts +115 -150
  111. package/src/collections/tableOperations.ts +92 -21
  112. package/src/collections/transferOperations.ts +4 -5
  113. package/src/collections/wipeOperations.ts +154 -51
  114. package/src/databases/methods.ts +2 -2
  115. package/src/databases/setup.ts +2 -2
  116. package/src/examples/yamlTerminologyExample.ts +2 -2
  117. package/src/functions/deployments.ts +6 -5
  118. package/src/functions/fnConfigDiscovery.ts +2 -2
  119. package/src/functions/methods.ts +19 -6
  120. package/src/init.ts +1 -1
  121. package/src/interactiveCLI.ts +78 -13
  122. package/src/main.ts +143 -287
  123. package/src/migrations/afterImportActions.ts +2 -3
  124. package/src/migrations/appwriteToX.ts +12 -8
  125. package/src/migrations/comprehensiveTransfer.ts +6 -6
  126. package/src/migrations/dataLoader.ts +2 -5
  127. package/src/migrations/importController.ts +3 -4
  128. package/src/migrations/importDataActions.ts +3 -3
  129. package/src/migrations/relationships.ts +1 -2
  130. package/src/migrations/services/DataTransformationService.ts +2 -2
  131. package/src/migrations/services/FileHandlerService.ts +1 -1
  132. package/src/migrations/services/ImportOrchestrator.ts +4 -4
  133. package/src/migrations/services/RateLimitManager.ts +1 -1
  134. package/src/migrations/services/RelationshipResolver.ts +1 -1
  135. package/src/migrations/services/UserMappingService.ts +1 -1
  136. package/src/migrations/services/ValidationService.ts +1 -1
  137. package/src/migrations/transfer.ts +126 -83
  138. package/src/migrations/yaml/YamlImportConfigLoader.ts +1 -1
  139. package/src/migrations/yaml/YamlImportIntegration.ts +2 -2
  140. package/src/migrations/yaml/generateImportSchemas.ts +1 -1
  141. package/src/setupCommands.ts +5 -6
  142. package/src/setupController.ts +1 -1
  143. package/src/shared/backupTracking.ts +3 -3
  144. package/src/shared/confirmationDialogs.ts +1 -1
  145. package/src/shared/migrationHelpers.ts +4 -4
  146. package/src/shared/operationQueue.ts +3 -4
  147. package/src/shared/operationsTable.ts +3 -3
  148. package/src/shared/progressManager.ts +1 -1
  149. package/src/shared/selectionDialogs.ts +9 -8
  150. package/src/storage/methods.ts +4 -4
  151. package/src/tables/indexManager.ts +409 -0
  152. package/src/types.ts +2 -2
  153. package/src/users/methods.ts +2 -3
  154. package/src/utils/configMigration.ts +1 -1
  155. package/src/utils/index.ts +1 -1
  156. package/src/utils/loadConfigs.ts +15 -7
  157. package/src/utils/setupFiles.ts +5 -7
  158. package/src/utilsController.ts +86 -32
  159. package/dist/adapters/AdapterFactory.d.ts +0 -94
  160. package/dist/adapters/AdapterFactory.js +0 -405
  161. package/dist/adapters/DatabaseAdapter.d.ts +0 -233
  162. package/dist/adapters/DatabaseAdapter.js +0 -50
  163. package/dist/adapters/LegacyAdapter.d.ts +0 -50
  164. package/dist/adapters/LegacyAdapter.js +0 -612
  165. package/dist/adapters/TablesDBAdapter.d.ts +0 -45
  166. package/dist/adapters/TablesDBAdapter.js +0 -571
  167. package/dist/config/ConfigManager.d.ts +0 -445
  168. package/dist/config/ConfigManager.js +0 -625
  169. package/dist/config/configMigration.d.ts +0 -87
  170. package/dist/config/configMigration.js +0 -390
  171. package/dist/config/configValidation.d.ts +0 -66
  172. package/dist/config/configValidation.js +0 -358
  173. package/dist/config/index.d.ts +0 -8
  174. package/dist/config/index.js +0 -7
  175. package/dist/config/services/ConfigDiscoveryService.d.ts +0 -126
  176. package/dist/config/services/ConfigDiscoveryService.js +0 -374
  177. package/dist/config/services/ConfigLoaderService.d.ts +0 -129
  178. package/dist/config/services/ConfigLoaderService.js +0 -540
  179. package/dist/config/services/ConfigMergeService.d.ts +0 -208
  180. package/dist/config/services/ConfigMergeService.js +0 -308
  181. package/dist/config/services/ConfigValidationService.d.ts +0 -214
  182. package/dist/config/services/ConfigValidationService.js +0 -310
  183. package/dist/config/services/SessionAuthService.d.ts +0 -225
  184. package/dist/config/services/SessionAuthService.js +0 -456
  185. package/dist/config/services/__tests__/ConfigMergeService.test.d.ts +0 -1
  186. package/dist/config/services/__tests__/ConfigMergeService.test.js +0 -271
  187. package/dist/config/services/index.d.ts +0 -13
  188. package/dist/config/services/index.js +0 -10
  189. package/dist/config/yamlConfig.d.ts +0 -722
  190. package/dist/config/yamlConfig.js +0 -702
  191. package/dist/functions/pathResolution.d.ts +0 -37
  192. package/dist/functions/pathResolution.js +0 -185
  193. package/dist/shared/attributeMapper.d.ts +0 -20
  194. package/dist/shared/attributeMapper.js +0 -203
  195. package/dist/shared/errorUtils.d.ts +0 -54
  196. package/dist/shared/errorUtils.js +0 -95
  197. package/dist/shared/functionManager.d.ts +0 -48
  198. package/dist/shared/functionManager.js +0 -336
  199. package/dist/shared/indexManager.d.ts +0 -24
  200. package/dist/shared/indexManager.js +0 -151
  201. package/dist/shared/jsonSchemaGenerator.d.ts +0 -50
  202. package/dist/shared/jsonSchemaGenerator.js +0 -290
  203. package/dist/shared/logging.d.ts +0 -61
  204. package/dist/shared/logging.js +0 -116
  205. package/dist/shared/messageFormatter.d.ts +0 -39
  206. package/dist/shared/messageFormatter.js +0 -162
  207. package/dist/shared/pydanticModelGenerator.d.ts +0 -17
  208. package/dist/shared/pydanticModelGenerator.js +0 -615
  209. package/dist/shared/schemaGenerator.d.ts +0 -40
  210. package/dist/shared/schemaGenerator.js +0 -556
  211. package/dist/utils/ClientFactory.d.ts +0 -87
  212. package/dist/utils/ClientFactory.js +0 -212
  213. package/dist/utils/configDiscovery.d.ts +0 -78
  214. package/dist/utils/configDiscovery.js +0 -472
  215. package/dist/utils/constantsGenerator.d.ts +0 -31
  216. package/dist/utils/constantsGenerator.js +0 -321
  217. package/dist/utils/dataConverters.d.ts +0 -46
  218. package/dist/utils/dataConverters.js +0 -139
  219. package/dist/utils/directoryUtils.d.ts +0 -22
  220. package/dist/utils/directoryUtils.js +0 -59
  221. package/dist/utils/getClientFromConfig.d.ts +0 -39
  222. package/dist/utils/getClientFromConfig.js +0 -199
  223. package/dist/utils/helperFunctions.d.ts +0 -63
  224. package/dist/utils/helperFunctions.js +0 -156
  225. package/dist/utils/pathResolvers.d.ts +0 -53
  226. package/dist/utils/pathResolvers.js +0 -72
  227. package/dist/utils/projectConfig.d.ts +0 -119
  228. package/dist/utils/projectConfig.js +0 -171
  229. package/dist/utils/retryFailedPromises.d.ts +0 -2
  230. package/dist/utils/retryFailedPromises.js +0 -23
  231. package/dist/utils/sessionAuth.d.ts +0 -48
  232. package/dist/utils/sessionAuth.js +0 -164
  233. package/dist/utils/typeGuards.d.ts +0 -35
  234. package/dist/utils/typeGuards.js +0 -57
  235. package/dist/utils/validationRules.d.ts +0 -43
  236. package/dist/utils/validationRules.js +0 -42
  237. package/dist/utils/versionDetection.d.ts +0 -58
  238. package/dist/utils/versionDetection.js +0 -251
  239. package/dist/utils/yamlConverter.d.ts +0 -100
  240. package/dist/utils/yamlConverter.js +0 -428
  241. package/dist/utils/yamlLoader.d.ts +0 -70
  242. package/dist/utils/yamlLoader.js +0 -267
  243. package/src/adapters/AdapterFactory.ts +0 -510
  244. package/src/adapters/DatabaseAdapter.ts +0 -306
  245. package/src/adapters/LegacyAdapter.ts +0 -841
  246. package/src/adapters/TablesDBAdapter.ts +0 -773
  247. package/src/config/ConfigManager.ts +0 -808
  248. package/src/config/README.md +0 -274
  249. package/src/config/configMigration.ts +0 -575
  250. package/src/config/configValidation.ts +0 -445
  251. package/src/config/index.ts +0 -10
  252. package/src/config/services/ConfigDiscoveryService.ts +0 -463
  253. package/src/config/services/ConfigLoaderService.ts +0 -740
  254. package/src/config/services/ConfigMergeService.ts +0 -388
  255. package/src/config/services/ConfigValidationService.ts +0 -394
  256. package/src/config/services/SessionAuthService.ts +0 -565
  257. package/src/config/services/__tests__/ConfigMergeService.test.ts +0 -351
  258. package/src/config/services/index.ts +0 -29
  259. package/src/config/yamlConfig.ts +0 -761
  260. package/src/functions/pathResolution.ts +0 -227
  261. package/src/shared/attributeMapper.ts +0 -229
  262. package/src/shared/errorUtils.ts +0 -110
  263. package/src/shared/functionManager.ts +0 -525
  264. package/src/shared/indexManager.ts +0 -254
  265. package/src/shared/jsonSchemaGenerator.ts +0 -383
  266. package/src/shared/logging.ts +0 -149
  267. package/src/shared/messageFormatter.ts +0 -208
  268. package/src/shared/pydanticModelGenerator.ts +0 -618
  269. package/src/shared/schemaGenerator.ts +0 -644
  270. package/src/utils/ClientFactory.ts +0 -240
  271. package/src/utils/configDiscovery.ts +0 -557
  272. package/src/utils/constantsGenerator.ts +0 -369
  273. package/src/utils/dataConverters.ts +0 -159
  274. package/src/utils/directoryUtils.ts +0 -61
  275. package/src/utils/getClientFromConfig.ts +0 -257
  276. package/src/utils/helperFunctions.ts +0 -228
  277. package/src/utils/pathResolvers.ts +0 -81
  278. package/src/utils/projectConfig.ts +0 -299
  279. package/src/utils/retryFailedPromises.ts +0 -29
  280. package/src/utils/sessionAuth.ts +0 -230
  281. package/src/utils/typeGuards.ts +0 -65
  282. package/src/utils/validationRules.ts +0 -88
  283. package/src/utils/versionDetection.ts +0 -292
  284. package/src/utils/yamlConverter.ts +0 -542
  285. 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 = {