appwrite-utils-cli 1.5.1 → 1.6.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 (233) hide show
  1. package/CHANGELOG.md +199 -0
  2. package/README.md +251 -29
  3. package/dist/adapters/AdapterFactory.d.ts +10 -3
  4. package/dist/adapters/AdapterFactory.js +213 -17
  5. package/dist/adapters/TablesDBAdapter.js +60 -17
  6. package/dist/backups/operations/bucketBackup.d.ts +19 -0
  7. package/dist/backups/operations/bucketBackup.js +197 -0
  8. package/dist/backups/operations/collectionBackup.d.ts +30 -0
  9. package/dist/backups/operations/collectionBackup.js +201 -0
  10. package/dist/backups/operations/comprehensiveBackup.d.ts +25 -0
  11. package/dist/backups/operations/comprehensiveBackup.js +238 -0
  12. package/dist/backups/schemas/bucketManifest.d.ts +93 -0
  13. package/dist/backups/schemas/bucketManifest.js +33 -0
  14. package/dist/backups/schemas/comprehensiveManifest.d.ts +108 -0
  15. package/dist/backups/schemas/comprehensiveManifest.js +32 -0
  16. package/dist/backups/tracking/centralizedTracking.d.ts +34 -0
  17. package/dist/backups/tracking/centralizedTracking.js +274 -0
  18. package/dist/cli/commands/configCommands.d.ts +8 -0
  19. package/dist/cli/commands/configCommands.js +160 -0
  20. package/dist/cli/commands/databaseCommands.d.ts +13 -0
  21. package/dist/cli/commands/databaseCommands.js +478 -0
  22. package/dist/cli/commands/functionCommands.d.ts +7 -0
  23. package/dist/cli/commands/functionCommands.js +289 -0
  24. package/dist/cli/commands/schemaCommands.d.ts +7 -0
  25. package/dist/cli/commands/schemaCommands.js +134 -0
  26. package/dist/cli/commands/transferCommands.d.ts +5 -0
  27. package/dist/cli/commands/transferCommands.js +384 -0
  28. package/dist/collections/attributes.d.ts +5 -4
  29. package/dist/collections/attributes.js +539 -246
  30. package/dist/collections/indexes.js +39 -37
  31. package/dist/collections/methods.d.ts +2 -16
  32. package/dist/collections/methods.js +90 -538
  33. package/dist/collections/transferOperations.d.ts +7 -0
  34. package/dist/collections/transferOperations.js +331 -0
  35. package/dist/collections/wipeOperations.d.ts +16 -0
  36. package/dist/collections/wipeOperations.js +328 -0
  37. package/dist/config/configMigration.d.ts +87 -0
  38. package/dist/config/configMigration.js +390 -0
  39. package/dist/config/configValidation.d.ts +66 -0
  40. package/dist/config/configValidation.js +358 -0
  41. package/dist/config/yamlConfig.d.ts +455 -1
  42. package/dist/config/yamlConfig.js +145 -52
  43. package/dist/databases/methods.js +3 -2
  44. package/dist/databases/setup.d.ts +1 -2
  45. package/dist/databases/setup.js +9 -87
  46. package/dist/examples/yamlTerminologyExample.d.ts +42 -0
  47. package/dist/examples/yamlTerminologyExample.js +269 -0
  48. package/dist/functions/deployments.js +11 -10
  49. package/dist/functions/methods.d.ts +1 -1
  50. package/dist/functions/methods.js +5 -4
  51. package/dist/init.js +9 -9
  52. package/dist/interactiveCLI.d.ts +8 -17
  53. package/dist/interactiveCLI.js +186 -1171
  54. package/dist/main.js +364 -21
  55. package/dist/migrations/afterImportActions.js +22 -30
  56. package/dist/migrations/appwriteToX.js +71 -25
  57. package/dist/migrations/dataLoader.js +35 -26
  58. package/dist/migrations/importController.js +29 -30
  59. package/dist/migrations/relationships.js +13 -12
  60. package/dist/migrations/services/ImportOrchestrator.js +16 -19
  61. package/dist/migrations/transfer.js +46 -46
  62. package/dist/migrations/yaml/YamlImportConfigLoader.d.ts +3 -1
  63. package/dist/migrations/yaml/YamlImportConfigLoader.js +6 -3
  64. package/dist/migrations/yaml/YamlImportIntegration.d.ts +9 -3
  65. package/dist/migrations/yaml/YamlImportIntegration.js +22 -11
  66. package/dist/migrations/yaml/generateImportSchemas.d.ts +14 -1
  67. package/dist/migrations/yaml/generateImportSchemas.js +736 -7
  68. package/dist/schemas/authUser.d.ts +1 -1
  69. package/dist/setupController.js +3 -2
  70. package/dist/shared/backupMetadataSchema.d.ts +94 -0
  71. package/dist/shared/backupMetadataSchema.js +38 -0
  72. package/dist/shared/backupTracking.d.ts +18 -0
  73. package/dist/shared/backupTracking.js +176 -0
  74. package/dist/shared/confirmationDialogs.js +15 -15
  75. package/dist/shared/errorUtils.d.ts +54 -0
  76. package/dist/shared/errorUtils.js +95 -0
  77. package/dist/shared/functionManager.js +20 -19
  78. package/dist/shared/indexManager.js +12 -11
  79. package/dist/shared/jsonSchemaGenerator.js +10 -26
  80. package/dist/shared/logging.d.ts +51 -0
  81. package/dist/shared/logging.js +70 -0
  82. package/dist/shared/messageFormatter.d.ts +2 -0
  83. package/dist/shared/messageFormatter.js +10 -0
  84. package/dist/shared/migrationHelpers.d.ts +6 -16
  85. package/dist/shared/migrationHelpers.js +24 -21
  86. package/dist/shared/operationLogger.d.ts +8 -1
  87. package/dist/shared/operationLogger.js +11 -24
  88. package/dist/shared/operationQueue.d.ts +28 -1
  89. package/dist/shared/operationQueue.js +268 -66
  90. package/dist/shared/operationsTable.d.ts +26 -0
  91. package/dist/shared/operationsTable.js +286 -0
  92. package/dist/shared/operationsTableSchema.d.ts +48 -0
  93. package/dist/shared/operationsTableSchema.js +35 -0
  94. package/dist/shared/relationshipExtractor.d.ts +56 -0
  95. package/dist/shared/relationshipExtractor.js +138 -0
  96. package/dist/shared/schemaGenerator.d.ts +19 -1
  97. package/dist/shared/schemaGenerator.js +56 -75
  98. package/dist/storage/backupCompression.d.ts +20 -0
  99. package/dist/storage/backupCompression.js +67 -0
  100. package/dist/storage/methods.d.ts +16 -2
  101. package/dist/storage/methods.js +98 -14
  102. package/dist/users/methods.js +9 -8
  103. package/dist/utils/configDiscovery.d.ts +78 -0
  104. package/dist/utils/configDiscovery.js +430 -0
  105. package/dist/utils/directoryUtils.d.ts +22 -0
  106. package/dist/utils/directoryUtils.js +59 -0
  107. package/dist/utils/getClientFromConfig.d.ts +17 -8
  108. package/dist/utils/getClientFromConfig.js +162 -17
  109. package/dist/utils/helperFunctions.d.ts +16 -2
  110. package/dist/utils/helperFunctions.js +19 -5
  111. package/dist/utils/loadConfigs.d.ts +34 -9
  112. package/dist/utils/loadConfigs.js +236 -316
  113. package/dist/utils/pathResolvers.d.ts +53 -0
  114. package/dist/utils/pathResolvers.js +72 -0
  115. package/dist/utils/projectConfig.d.ts +119 -0
  116. package/dist/utils/projectConfig.js +171 -0
  117. package/dist/utils/retryFailedPromises.js +4 -2
  118. package/dist/utils/sessionAuth.d.ts +48 -0
  119. package/dist/utils/sessionAuth.js +164 -0
  120. package/dist/utils/sessionPreservationExample.d.ts +1666 -0
  121. package/dist/utils/sessionPreservationExample.js +101 -0
  122. package/dist/utils/setupFiles.js +301 -41
  123. package/dist/utils/typeGuards.d.ts +35 -0
  124. package/dist/utils/typeGuards.js +57 -0
  125. package/dist/utils/versionDetection.js +145 -9
  126. package/dist/utils/yamlConverter.d.ts +53 -3
  127. package/dist/utils/yamlConverter.js +232 -13
  128. package/dist/utils/yamlLoader.d.ts +70 -0
  129. package/dist/utils/yamlLoader.js +263 -0
  130. package/dist/utilsController.d.ts +36 -3
  131. package/dist/utilsController.js +186 -56
  132. package/package.json +12 -2
  133. package/src/adapters/AdapterFactory.ts +263 -35
  134. package/src/adapters/TablesDBAdapter.ts +225 -36
  135. package/src/backups/operations/bucketBackup.ts +277 -0
  136. package/src/backups/operations/collectionBackup.ts +310 -0
  137. package/src/backups/operations/comprehensiveBackup.ts +342 -0
  138. package/src/backups/schemas/bucketManifest.ts +78 -0
  139. package/src/backups/schemas/comprehensiveManifest.ts +76 -0
  140. package/src/backups/tracking/centralizedTracking.ts +352 -0
  141. package/src/cli/commands/configCommands.ts +194 -0
  142. package/src/cli/commands/databaseCommands.ts +635 -0
  143. package/src/cli/commands/functionCommands.ts +379 -0
  144. package/src/cli/commands/schemaCommands.ts +163 -0
  145. package/src/cli/commands/transferCommands.ts +457 -0
  146. package/src/collections/attributes.ts +900 -621
  147. package/src/collections/attributes.ts.backup +1555 -0
  148. package/src/collections/indexes.ts +116 -114
  149. package/src/collections/methods.ts +295 -968
  150. package/src/collections/transferOperations.ts +516 -0
  151. package/src/collections/wipeOperations.ts +501 -0
  152. package/src/config/README.md +274 -0
  153. package/src/config/configMigration.ts +575 -0
  154. package/src/config/configValidation.ts +445 -0
  155. package/src/config/yamlConfig.ts +168 -55
  156. package/src/databases/methods.ts +3 -2
  157. package/src/databases/setup.ts +11 -138
  158. package/src/examples/yamlTerminologyExample.ts +341 -0
  159. package/src/functions/deployments.ts +14 -12
  160. package/src/functions/methods.ts +11 -11
  161. package/src/functions/templates/hono-typescript/README.md +286 -0
  162. package/src/functions/templates/hono-typescript/package.json +26 -0
  163. package/src/functions/templates/hono-typescript/src/adapters/request.ts +74 -0
  164. package/src/functions/templates/hono-typescript/src/adapters/response.ts +106 -0
  165. package/src/functions/templates/hono-typescript/src/app.ts +180 -0
  166. package/src/functions/templates/hono-typescript/src/context.ts +103 -0
  167. package/src/functions/templates/hono-typescript/src/index.ts +54 -0
  168. package/src/functions/templates/hono-typescript/src/middleware/appwrite.ts +119 -0
  169. package/src/functions/templates/hono-typescript/tsconfig.json +20 -0
  170. package/src/functions/templates/typescript-node/package.json +2 -1
  171. package/src/functions/templates/typescript-node/src/context.ts +103 -0
  172. package/src/functions/templates/typescript-node/src/index.ts +18 -12
  173. package/src/functions/templates/uv/pyproject.toml +1 -0
  174. package/src/functions/templates/uv/src/context.py +125 -0
  175. package/src/functions/templates/uv/src/index.py +35 -5
  176. package/src/init.ts +9 -11
  177. package/src/interactiveCLI.ts +276 -1591
  178. package/src/main.ts +418 -24
  179. package/src/migrations/afterImportActions.ts +71 -44
  180. package/src/migrations/appwriteToX.ts +100 -34
  181. package/src/migrations/dataLoader.ts +48 -34
  182. package/src/migrations/importController.ts +44 -39
  183. package/src/migrations/relationships.ts +28 -18
  184. package/src/migrations/services/ImportOrchestrator.ts +24 -27
  185. package/src/migrations/transfer.ts +159 -121
  186. package/src/migrations/yaml/YamlImportConfigLoader.ts +11 -4
  187. package/src/migrations/yaml/YamlImportIntegration.ts +47 -20
  188. package/src/migrations/yaml/generateImportSchemas.ts +751 -12
  189. package/src/setupController.ts +3 -2
  190. package/src/shared/backupMetadataSchema.ts +93 -0
  191. package/src/shared/backupTracking.ts +211 -0
  192. package/src/shared/confirmationDialogs.ts +19 -19
  193. package/src/shared/errorUtils.ts +110 -0
  194. package/src/shared/functionManager.ts +21 -20
  195. package/src/shared/indexManager.ts +12 -11
  196. package/src/shared/jsonSchemaGenerator.ts +38 -52
  197. package/src/shared/logging.ts +75 -0
  198. package/src/shared/messageFormatter.ts +14 -1
  199. package/src/shared/migrationHelpers.ts +45 -38
  200. package/src/shared/operationLogger.ts +11 -36
  201. package/src/shared/operationQueue.ts +322 -93
  202. package/src/shared/operationsTable.ts +338 -0
  203. package/src/shared/operationsTableSchema.ts +60 -0
  204. package/src/shared/relationshipExtractor.ts +214 -0
  205. package/src/shared/schemaGenerator.ts +179 -219
  206. package/src/storage/backupCompression.ts +88 -0
  207. package/src/storage/methods.ts +131 -34
  208. package/src/users/methods.ts +11 -9
  209. package/src/utils/configDiscovery.ts +502 -0
  210. package/src/utils/directoryUtils.ts +61 -0
  211. package/src/utils/getClientFromConfig.ts +205 -22
  212. package/src/utils/helperFunctions.ts +23 -5
  213. package/src/utils/loadConfigs.ts +313 -345
  214. package/src/utils/pathResolvers.ts +81 -0
  215. package/src/utils/projectConfig.ts +299 -0
  216. package/src/utils/retryFailedPromises.ts +4 -2
  217. package/src/utils/sessionAuth.ts +230 -0
  218. package/src/utils/setupFiles.ts +322 -54
  219. package/src/utils/typeGuards.ts +65 -0
  220. package/src/utils/versionDetection.ts +218 -64
  221. package/src/utils/yamlConverter.ts +296 -13
  222. package/src/utils/yamlLoader.ts +364 -0
  223. package/src/utilsController.ts +314 -110
  224. package/tests/README.md +497 -0
  225. package/tests/adapters/AdapterFactory.test.ts +277 -0
  226. package/tests/integration/syncOperations.test.ts +463 -0
  227. package/tests/jest.config.js +25 -0
  228. package/tests/migration/configMigration.test.ts +546 -0
  229. package/tests/setup.ts +62 -0
  230. package/tests/testUtils.ts +340 -0
  231. package/tests/utils/loadConfigs.test.ts +350 -0
  232. package/tests/validation/configValidation.test.ts +412 -0
  233. package/src/utils/schemaStrings.ts +0 -517
@@ -0,0 +1,575 @@
1
+ import { type AppwriteConfig, type CollectionCreate, type TableCreate, type TableCreateInput, TableCreateSchema } from "appwrite-utils";
2
+ import { MessageFormatter } from "../shared/messageFormatter.js";
3
+ import { validateCollectionsTablesConfig, type ValidationResult } from "./configValidation.js";
4
+ import path from "path";
5
+ import fs from "fs";
6
+ import chalk from "chalk";
7
+
8
+ /**
9
+ * Migration strategy types
10
+ */
11
+ export type MigrationStrategy =
12
+ | "full_migration" // Convert all collections to tables, remove collections array
13
+ | "dual_format" // Keep collections and add equivalent tables
14
+ | "incremental" // Migrate specific collections to tables
15
+ | "tables_only"; // Create tables-only configuration
16
+
17
+ /**
18
+ * Migration plan detailing what will be changed
19
+ */
20
+ export interface MigrationPlan {
21
+ strategy: MigrationStrategy;
22
+ collectionsToMigrate: CollectionMigrationItem[];
23
+ collectionsToKeep: CollectionCreate[];
24
+ tablesToCreate: TableCreate[];
25
+ expectedChanges: MigrationChange[];
26
+ estimatedComplexity: "low" | "medium" | "high";
27
+ warnings: string[];
28
+ recommendations: string[];
29
+ }
30
+
31
+ /**
32
+ * Individual collection migration details
33
+ */
34
+ export interface CollectionMigrationItem {
35
+ collection: CollectionCreate;
36
+ index: number;
37
+ newTable: TableCreate;
38
+ changes: string[];
39
+ warnings: string[];
40
+ }
41
+
42
+ /**
43
+ * Migration change description
44
+ */
45
+ export interface MigrationChange {
46
+ type: "add" | "remove" | "modify" | "rename";
47
+ description: string;
48
+ impact: "low" | "medium" | "high";
49
+ location: string;
50
+ }
51
+
52
+ /**
53
+ * Migration execution result
54
+ */
55
+ export interface MigrationResult {
56
+ success: boolean;
57
+ newConfig: AppwriteConfig;
58
+ changes: MigrationChange[];
59
+ validation: ValidationResult;
60
+ warnings: string[];
61
+ errors: string[];
62
+ }
63
+
64
+ /**
65
+ * Migration options
66
+ */
67
+ export interface MigrationOptions {
68
+ preserveOriginal?: boolean;
69
+ validateResult?: boolean;
70
+ dryRun?: boolean;
71
+ backupConfig?: boolean;
72
+ targetDirectory?: string;
73
+ }
74
+
75
+ /**
76
+ * Creates a migration plan for converting collections to tables
77
+ */
78
+ export function createMigrationPlan(
79
+ config: AppwriteConfig,
80
+ strategy: MigrationStrategy = "full_migration",
81
+ specificCollections?: string[]
82
+ ): MigrationPlan {
83
+ const collections = config.collections || [];
84
+ const existingTables = config.tables || [];
85
+
86
+ const collectionsToMigrate: CollectionMigrationItem[] = [];
87
+ const collectionsToKeep: CollectionCreate[] = [];
88
+ const tablesToCreate: TableCreate[] = [];
89
+ const expectedChanges: MigrationChange[] = [];
90
+ const warnings: string[] = [];
91
+ const recommendations: string[] = [];
92
+
93
+ // Determine which collections to migrate based on strategy
94
+ collections.forEach((collection, index) => {
95
+ const shouldMigrate = determineShouldMigrate(collection, strategy, specificCollections);
96
+
97
+ if (shouldMigrate) {
98
+ const migrationItem = createCollectionMigrationItem(collection, index);
99
+ collectionsToMigrate.push(migrationItem);
100
+ tablesToCreate.push(migrationItem.newTable);
101
+
102
+ // Check for potential issues
103
+ if (migrationItem.warnings.length > 0) {
104
+ warnings.push(...migrationItem.warnings);
105
+ }
106
+ } else {
107
+ collectionsToKeep.push(collection);
108
+ }
109
+ });
110
+
111
+ // Create expected changes
112
+ expectedChanges.push(...createExpectedChanges(strategy, collectionsToMigrate, collectionsToKeep));
113
+
114
+ // Assess complexity
115
+ const estimatedComplexity = assessMigrationComplexity(collectionsToMigrate, strategy);
116
+
117
+ // Generate recommendations
118
+ recommendations.push(...generateMigrationRecommendations(config, strategy, collectionsToMigrate));
119
+
120
+ // Add strategy-specific warnings
121
+ if (strategy === "dual_format") {
122
+ warnings.push("Dual format increases configuration file size and maintenance overhead");
123
+ recommendations.push("Consider migrating to tables_only after testing dual format");
124
+ }
125
+
126
+ if (strategy === "full_migration" && collections.length > 10) {
127
+ warnings.push("Large migration may require careful testing and staged deployment");
128
+ recommendations.push("Consider incremental migration for large projects");
129
+ }
130
+
131
+ return {
132
+ strategy,
133
+ collectionsToMigrate,
134
+ collectionsToKeep,
135
+ tablesToCreate,
136
+ expectedChanges,
137
+ estimatedComplexity,
138
+ warnings,
139
+ recommendations
140
+ };
141
+ }
142
+
143
+ /**
144
+ * Executes a migration plan
145
+ */
146
+ export function executeMigrationPlan(
147
+ config: AppwriteConfig,
148
+ plan: MigrationPlan,
149
+ options: MigrationOptions = {}
150
+ ): MigrationResult {
151
+ const {
152
+ validateResult = true,
153
+ dryRun = false,
154
+ preserveOriginal = false
155
+ } = options;
156
+
157
+ try {
158
+ // Create new configuration based on strategy
159
+ const newConfig = applyMigrationPlan(config, plan, preserveOriginal);
160
+
161
+ // Validate the result if requested
162
+ let validation: ValidationResult = { isValid: true, errors: [], warnings: [], suggestions: [] };
163
+ if (validateResult) {
164
+ validation = validateCollectionsTablesConfig(newConfig);
165
+ }
166
+
167
+ const changes = plan.expectedChanges;
168
+ const warnings: string[] = [...plan.warnings];
169
+ const errors: string[] = [];
170
+
171
+ // Add validation errors to the result
172
+ if (validation.errors.length > 0) {
173
+ errors.push(...validation.errors.map(e => e.message));
174
+ }
175
+
176
+ if (validation.warnings.length > 0) {
177
+ warnings.push(...validation.warnings.map(w => w.message));
178
+ }
179
+
180
+ const success = errors.length === 0;
181
+
182
+ if (dryRun) {
183
+ MessageFormatter.info("Dry run completed - no changes were made", { prefix: "Migration" });
184
+ }
185
+
186
+ return {
187
+ success,
188
+ newConfig,
189
+ changes,
190
+ validation,
191
+ warnings,
192
+ errors
193
+ };
194
+ } catch (error) {
195
+ return {
196
+ success: false,
197
+ newConfig: config,
198
+ changes: [],
199
+ validation: { isValid: false, errors: [], warnings: [], suggestions: [] },
200
+ warnings: [],
201
+ errors: [error instanceof Error ? error.message : "Unknown migration error"]
202
+ };
203
+ }
204
+ }
205
+
206
+ /**
207
+ * Converts a single collection to table format
208
+ */
209
+ export function convertCollectionToTable(collection: CollectionCreate): TableCreateInput {
210
+ const table: TableCreateInput = {
211
+ name: collection.name,
212
+ tableId: collection.$id || collection.name.toLowerCase().replace(/\s+/g, '_'),
213
+ enabled: collection.enabled,
214
+ documentSecurity: collection.documentSecurity,
215
+ $permissions: collection.$permissions,
216
+ attributes: collection.attributes,
217
+ indexes: collection.indexes,
218
+ importDefs: collection.importDefs,
219
+ databaseId: collection.databaseId
220
+ };
221
+
222
+ // Add $id for backward compatibility if it exists
223
+ if (collection.$id) {
224
+ table.$id = collection.$id;
225
+ }
226
+
227
+ return table;
228
+ }
229
+
230
+ /**
231
+ * Creates a migration item for a single collection
232
+ */
233
+ function createCollectionMigrationItem(collection: CollectionCreate, index: number): CollectionMigrationItem {
234
+ const tableInput = convertCollectionToTable(collection);
235
+ const newTable = TableCreateSchema.parse(tableInput);
236
+ const changes: string[] = [];
237
+ const warnings: string[] = [];
238
+
239
+ // Document changes
240
+ changes.push(`Convert collection '${collection.name}' to table format`);
241
+
242
+ if (collection.$id && collection.$id !== newTable.tableId) {
243
+ changes.push(`Change ID from '$id: ${collection.$id}' to 'tableId: ${newTable.tableId}'`);
244
+ }
245
+
246
+ // Check for potential issues
247
+ if (collection.attributes.some(attr => attr.type === 'relationship')) {
248
+ warnings.push(`Collection '${collection.name}' has relationship attributes that may need manual review`);
249
+ }
250
+
251
+ if (collection.indexes && collection.indexes.length > 5) {
252
+ warnings.push(`Collection '${collection.name}' has many indexes (${collection.indexes.length}) - verify compatibility`);
253
+ }
254
+
255
+ return {
256
+ collection,
257
+ index,
258
+ newTable,
259
+ changes,
260
+ warnings
261
+ };
262
+ }
263
+
264
+ /**
265
+ * Determines if a collection should be migrated based on strategy
266
+ */
267
+ function determineShouldMigrate(
268
+ collection: CollectionCreate,
269
+ strategy: MigrationStrategy,
270
+ specificCollections?: string[]
271
+ ): boolean {
272
+ switch (strategy) {
273
+ case "full_migration":
274
+ case "tables_only":
275
+ return true;
276
+
277
+ case "dual_format":
278
+ return true;
279
+
280
+ case "incremental":
281
+ if (!specificCollections || specificCollections.length === 0) {
282
+ return false;
283
+ }
284
+ return specificCollections.includes(collection.name) ||
285
+ specificCollections.includes(collection.$id || "");
286
+
287
+ default:
288
+ return false;
289
+ }
290
+ }
291
+
292
+ /**
293
+ * Creates expected changes list based on migration plan
294
+ */
295
+ function createExpectedChanges(
296
+ strategy: MigrationStrategy,
297
+ collectionsToMigrate: CollectionMigrationItem[],
298
+ collectionsToKeep: CollectionCreate[]
299
+ ): MigrationChange[] {
300
+ const changes: MigrationChange[] = [];
301
+
302
+ // Add table creation changes
303
+ collectionsToMigrate.forEach(item => {
304
+ changes.push({
305
+ type: "add",
306
+ description: `Add table '${item.newTable.name}' converted from collection`,
307
+ impact: "medium",
308
+ location: `tables[${item.newTable.name}]`
309
+ });
310
+ });
311
+
312
+ // Add collection removal changes (if not preserving)
313
+ if (strategy === "full_migration" || strategy === "tables_only") {
314
+ collectionsToMigrate.forEach(item => {
315
+ changes.push({
316
+ type: "remove",
317
+ description: `Remove collection '${item.collection.name}' (converted to table)`,
318
+ impact: "high",
319
+ location: `collections[${item.index}]`
320
+ });
321
+ });
322
+ }
323
+
324
+ // Add array structure changes
325
+ if (collectionsToMigrate.length > 0) {
326
+ changes.push({
327
+ type: "modify",
328
+ description: `${strategy === "dual_format" ? "Add" : "Create"} tables array with ${collectionsToMigrate.length} items`,
329
+ impact: "medium",
330
+ location: "config.tables"
331
+ });
332
+ }
333
+
334
+ if (strategy === "tables_only" && collectionsToKeep.length === 0) {
335
+ changes.push({
336
+ type: "remove",
337
+ description: "Remove collections array (fully migrated to tables)",
338
+ impact: "high",
339
+ location: "config.collections"
340
+ });
341
+ }
342
+
343
+ return changes;
344
+ }
345
+
346
+ /**
347
+ * Assesses migration complexity
348
+ */
349
+ function assessMigrationComplexity(
350
+ collectionsToMigrate: CollectionMigrationItem[],
351
+ strategy: MigrationStrategy
352
+ ): "low" | "medium" | "high" {
353
+ const collectionCount = collectionsToMigrate.length;
354
+ const hasRelationships = collectionsToMigrate.some(item =>
355
+ item.collection.attributes.some(attr => attr.type === 'relationship')
356
+ );
357
+ const hasComplexIndexes = collectionsToMigrate.some(item =>
358
+ (item.collection.indexes?.length || 0) > 5
359
+ );
360
+
361
+ if (collectionCount === 0) return "low";
362
+ if (collectionCount <= 3 && !hasRelationships && !hasComplexIndexes) return "low";
363
+ if (collectionCount <= 10 && !hasComplexIndexes && strategy !== "full_migration") return "medium";
364
+
365
+ return "high";
366
+ }
367
+
368
+ /**
369
+ * Generates migration recommendations
370
+ */
371
+ function generateMigrationRecommendations(
372
+ config: AppwriteConfig,
373
+ strategy: MigrationStrategy,
374
+ collectionsToMigrate: CollectionMigrationItem[]
375
+ ): string[] {
376
+ const recommendations: string[] = [];
377
+
378
+ if (collectionsToMigrate.length > 5) {
379
+ recommendations.push("Consider migrating in smaller batches for easier rollback");
380
+ }
381
+
382
+ if (collectionsToMigrate.some(item => item.warnings.length > 0)) {
383
+ recommendations.push("Review relationship attributes and complex indexes after migration");
384
+ }
385
+
386
+ if (!config.apiMode || config.apiMode === "auto") {
387
+ recommendations.push("Set apiMode to 'tablesdb' after migration for explicit API selection");
388
+ }
389
+
390
+ if (strategy === "dual_format") {
391
+ recommendations.push("Test with dual format before removing collections array");
392
+ recommendations.push("Monitor for any breaking changes with both APIs");
393
+ }
394
+
395
+ return recommendations;
396
+ }
397
+
398
+ /**
399
+ * Applies migration plan to create new configuration
400
+ */
401
+ function applyMigrationPlan(
402
+ config: AppwriteConfig,
403
+ plan: MigrationPlan,
404
+ preserveOriginal: boolean
405
+ ): AppwriteConfig {
406
+ const newConfig: AppwriteConfig = { ...config };
407
+
408
+ // Initialize tables array if it doesn't exist
409
+ if (!newConfig.tables) {
410
+ newConfig.tables = [];
411
+ }
412
+
413
+ // Add migrated tables
414
+ newConfig.tables.push(...plan.tablesToCreate);
415
+
416
+ // Handle collections based on strategy
417
+ if (plan.strategy === "full_migration" || plan.strategy === "tables_only") {
418
+ if (preserveOriginal) {
419
+ // Keep original collections but mark them as migrated
420
+ newConfig.collections = (newConfig.collections || []).map(collection => ({
421
+ ...collection,
422
+ // Add a comment or marker to indicate it's been migrated
423
+ _migrated: true
424
+ } as any));
425
+ } else {
426
+ // Remove migrated collections
427
+ const migratedNames = new Set(plan.collectionsToMigrate.map(item => item.collection.name));
428
+ newConfig.collections = (newConfig.collections || []).filter(
429
+ collection => !migratedNames.has(collection.name)
430
+ );
431
+
432
+ // If no collections remain, remove the array for tables_only strategy
433
+ if (newConfig.collections.length === 0 && plan.strategy === "tables_only") {
434
+ delete newConfig.collections;
435
+ }
436
+ }
437
+ }
438
+ // For dual_format and incremental, keep existing collections unchanged
439
+
440
+ return newConfig;
441
+ }
442
+
443
+ /**
444
+ * Utility function to migrate collections to tables with simple interface
445
+ */
446
+ export function migrateCollectionsToTables(
447
+ config: AppwriteConfig,
448
+ options: {
449
+ strategy?: MigrationStrategy;
450
+ specificCollections?: string[];
451
+ validateResult?: boolean;
452
+ dryRun?: boolean;
453
+ } = {}
454
+ ): MigrationResult {
455
+ const {
456
+ strategy = "full_migration",
457
+ specificCollections,
458
+ validateResult = true,
459
+ dryRun = false
460
+ } = options;
461
+
462
+ // Create migration plan
463
+ const plan = createMigrationPlan(config, strategy, specificCollections);
464
+
465
+ // Execute migration
466
+ const result = executeMigrationPlan(config, plan, {
467
+ validateResult,
468
+ dryRun
469
+ });
470
+
471
+ return result;
472
+ }
473
+
474
+ /**
475
+ * Saves migration results to files
476
+ */
477
+ export async function saveMigrationResult(
478
+ result: MigrationResult,
479
+ outputPath: string,
480
+ options: {
481
+ createBackup?: boolean;
482
+ originalConfigPath?: string;
483
+ } = {}
484
+ ): Promise<void> {
485
+ const { createBackup = true, originalConfigPath } = options;
486
+
487
+ try {
488
+ // Create backup if requested
489
+ if (createBackup && originalConfigPath && fs.existsSync(originalConfigPath)) {
490
+ const backupPath = `${originalConfigPath}.backup.${Date.now()}`;
491
+ fs.copyFileSync(originalConfigPath, backupPath);
492
+ MessageFormatter.success(`Backup created: ${backupPath}`, { prefix: "Migration" });
493
+ }
494
+
495
+ // Write new configuration
496
+ const configContent = `import { type AppwriteConfig } from "appwrite-utils";
497
+
498
+ const appwriteConfig: AppwriteConfig = ${JSON.stringify(result.newConfig, null, 2)};
499
+
500
+ export default appwriteConfig;
501
+ `;
502
+
503
+ fs.writeFileSync(outputPath, configContent, 'utf8');
504
+ MessageFormatter.success(`Migration result saved: ${outputPath}`, { prefix: "Migration" });
505
+
506
+ // Create migration report
507
+ const reportPath = outputPath.replace(/\.(ts|js)$/, '.migration-report.md');
508
+ const report = generateMigrationReport(result);
509
+ fs.writeFileSync(reportPath, report, 'utf8');
510
+ MessageFormatter.info(`Migration report saved: ${reportPath}`, { prefix: "Migration" });
511
+
512
+ } catch (error) {
513
+ throw new Error(`Failed to save migration result: ${error instanceof Error ? error.message : 'Unknown error'}`);
514
+ }
515
+ }
516
+
517
+ /**
518
+ * Generates a detailed migration report
519
+ */
520
+ function generateMigrationReport(result: MigrationResult): string {
521
+ const timestamp = new Date().toISOString();
522
+
523
+ return `# Migration Report
524
+
525
+ **Generated:** ${timestamp}
526
+ **Status:** ${result.success ? "✅ Success" : "❌ Failed"}
527
+
528
+ ## Changes Applied
529
+
530
+ ${result.changes.map(change =>
531
+ `- **${change.type.toUpperCase()}**: ${change.description} (Impact: ${change.impact})`
532
+ ).join('\n')}
533
+
534
+ ## Validation Results
535
+
536
+ **Valid Configuration:** ${result.validation.isValid ? "✅ Yes" : "❌ No"}
537
+
538
+ ${result.validation.errors.length > 0 ? `
539
+ ### Errors
540
+ ${result.validation.errors.map(error => `- ${error.message}`).join('\n')}
541
+ ` : ''}
542
+
543
+ ${result.validation.warnings.length > 0 ? `
544
+ ### Warnings
545
+ ${result.validation.warnings.map(warning => `- ${warning.message}`).join('\n')}
546
+ ` : ''}
547
+
548
+ ${result.validation.suggestions.length > 0 ? `
549
+ ### Suggestions
550
+ ${result.validation.suggestions.map(suggestion => `- ${suggestion.message}`).join('\n')}
551
+ ` : ''}
552
+
553
+ ${result.warnings.length > 0 ? `
554
+ ## Migration Warnings
555
+
556
+ ${result.warnings.map(warning => `- ${warning}`).join('\n')}
557
+ ` : ''}
558
+
559
+ ${result.errors.length > 0 ? `
560
+ ## Migration Errors
561
+
562
+ ${result.errors.map(error => `- ${error}`).join('\n')}
563
+ ` : ''}
564
+
565
+ ## Next Steps
566
+
567
+ 1. Review the migrated configuration
568
+ 2. Test with your Appwrite instance
569
+ 3. Update your deployment scripts if necessary
570
+ 4. Consider setting \`apiMode\` to 'tablesdb' for explicit API selection
571
+
572
+ ---
573
+ *Generated by appwrite-utils-cli migration tools*
574
+ `;
575
+ }