appwrite-utils-cli 0.10.86 → 1.0.2

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 (178) hide show
  1. package/.appwrite/.yaml_schemas/appwrite-config.schema.json +380 -0
  2. package/.appwrite/.yaml_schemas/collection.schema.json +255 -0
  3. package/.appwrite/collections/Categories.yaml +182 -0
  4. package/.appwrite/collections/ExampleCollection.yaml +36 -0
  5. package/.appwrite/collections/Posts.yaml +227 -0
  6. package/.appwrite/collections/Users.yaml +149 -0
  7. package/.appwrite/config.yaml +109 -0
  8. package/.appwrite/import/README.md +148 -0
  9. package/.appwrite/import/categories-import.yaml +129 -0
  10. package/.appwrite/import/posts-import.yaml +208 -0
  11. package/.appwrite/import/users-import.yaml +130 -0
  12. package/.appwrite/importData/categories.json +194 -0
  13. package/.appwrite/importData/posts.json +270 -0
  14. package/.appwrite/importData/users.json +220 -0
  15. package/.appwrite/schemas/categories.json +128 -0
  16. package/.appwrite/schemas/exampleCollection.json +52 -0
  17. package/.appwrite/schemas/posts.json +173 -0
  18. package/.appwrite/schemas/users.json +125 -0
  19. package/README.md +264 -33
  20. package/dist/collections/attributes.js +3 -2
  21. package/dist/collections/methods.js +56 -38
  22. package/dist/config/yamlConfig.d.ts +501 -0
  23. package/dist/config/yamlConfig.js +452 -0
  24. package/dist/databases/setup.d.ts +6 -0
  25. package/dist/databases/setup.js +119 -0
  26. package/dist/functions/methods.d.ts +1 -1
  27. package/dist/functions/methods.js +5 -2
  28. package/dist/functions/openapi.d.ts +4 -0
  29. package/dist/functions/openapi.js +60 -0
  30. package/dist/interactiveCLI.d.ts +5 -0
  31. package/dist/interactiveCLI.js +194 -49
  32. package/dist/main.js +91 -30
  33. package/dist/migrations/afterImportActions.js +2 -2
  34. package/dist/migrations/appwriteToX.d.ts +10 -0
  35. package/dist/migrations/appwriteToX.js +15 -4
  36. package/dist/migrations/backup.d.ts +16 -16
  37. package/dist/migrations/dataLoader.d.ts +83 -1
  38. package/dist/migrations/dataLoader.js +4 -4
  39. package/dist/migrations/importController.js +25 -18
  40. package/dist/migrations/importDataActions.js +2 -2
  41. package/dist/migrations/logging.d.ts +9 -1
  42. package/dist/migrations/logging.js +41 -22
  43. package/dist/migrations/migrationHelper.d.ts +4 -4
  44. package/dist/migrations/relationships.js +1 -1
  45. package/dist/migrations/services/DataTransformationService.d.ts +55 -0
  46. package/dist/migrations/services/DataTransformationService.js +158 -0
  47. package/dist/migrations/services/FileHandlerService.d.ts +75 -0
  48. package/dist/migrations/services/FileHandlerService.js +236 -0
  49. package/dist/migrations/services/ImportOrchestrator.d.ts +97 -0
  50. package/dist/migrations/services/ImportOrchestrator.js +488 -0
  51. package/dist/migrations/services/RateLimitManager.d.ts +138 -0
  52. package/dist/migrations/services/RateLimitManager.js +279 -0
  53. package/dist/migrations/services/RelationshipResolver.d.ts +120 -0
  54. package/dist/migrations/services/RelationshipResolver.js +332 -0
  55. package/dist/migrations/services/UserMappingService.d.ts +109 -0
  56. package/dist/migrations/services/UserMappingService.js +277 -0
  57. package/dist/migrations/services/ValidationService.d.ts +74 -0
  58. package/dist/migrations/services/ValidationService.js +260 -0
  59. package/dist/migrations/transfer.d.ts +0 -6
  60. package/dist/migrations/transfer.js +16 -132
  61. package/dist/migrations/yaml/YamlImportConfigLoader.d.ts +384 -0
  62. package/dist/migrations/yaml/YamlImportConfigLoader.js +375 -0
  63. package/dist/migrations/yaml/YamlImportIntegration.d.ts +87 -0
  64. package/dist/migrations/yaml/YamlImportIntegration.js +330 -0
  65. package/dist/migrations/yaml/generateImportSchemas.d.ts +17 -0
  66. package/dist/migrations/yaml/generateImportSchemas.js +575 -0
  67. package/dist/schemas/authUser.d.ts +9 -9
  68. package/dist/shared/attributeManager.d.ts +17 -0
  69. package/dist/shared/attributeManager.js +273 -0
  70. package/dist/shared/confirmationDialogs.d.ts +75 -0
  71. package/dist/shared/confirmationDialogs.js +236 -0
  72. package/dist/shared/functionManager.d.ts +48 -0
  73. package/dist/shared/functionManager.js +322 -0
  74. package/dist/shared/indexManager.d.ts +24 -0
  75. package/dist/shared/indexManager.js +150 -0
  76. package/dist/shared/jsonSchemaGenerator.d.ts +51 -0
  77. package/dist/shared/jsonSchemaGenerator.js +313 -0
  78. package/dist/shared/logging.d.ts +10 -0
  79. package/dist/shared/logging.js +46 -0
  80. package/dist/shared/messageFormatter.d.ts +37 -0
  81. package/dist/shared/messageFormatter.js +152 -0
  82. package/dist/shared/migrationHelpers.d.ts +173 -0
  83. package/dist/shared/migrationHelpers.js +142 -0
  84. package/dist/shared/operationLogger.d.ts +3 -0
  85. package/dist/shared/operationLogger.js +25 -0
  86. package/dist/shared/operationQueue.d.ts +13 -0
  87. package/dist/shared/operationQueue.js +79 -0
  88. package/dist/shared/progressManager.d.ts +62 -0
  89. package/dist/shared/progressManager.js +215 -0
  90. package/dist/shared/schemaGenerator.d.ts +18 -0
  91. package/dist/shared/schemaGenerator.js +523 -0
  92. package/dist/storage/methods.d.ts +3 -1
  93. package/dist/storage/methods.js +144 -55
  94. package/dist/storage/schemas.d.ts +56 -16
  95. package/dist/types.d.ts +2 -2
  96. package/dist/types.js +1 -1
  97. package/dist/users/methods.d.ts +16 -0
  98. package/dist/users/methods.js +276 -0
  99. package/dist/utils/configMigration.d.ts +1 -0
  100. package/dist/utils/configMigration.js +262 -0
  101. package/dist/utils/dataConverters.d.ts +46 -0
  102. package/dist/utils/dataConverters.js +139 -0
  103. package/dist/utils/loadConfigs.d.ts +15 -4
  104. package/dist/utils/loadConfigs.js +379 -51
  105. package/dist/utils/schemaStrings.js +2 -1
  106. package/dist/utils/setupFiles.d.ts +2 -1
  107. package/dist/utils/setupFiles.js +723 -28
  108. package/dist/utils/validationRules.d.ts +43 -0
  109. package/dist/utils/validationRules.js +42 -0
  110. package/dist/utils/yamlConverter.d.ts +48 -0
  111. package/dist/utils/yamlConverter.js +98 -0
  112. package/dist/utilsController.js +65 -43
  113. package/package.json +19 -15
  114. package/src/collections/attributes.ts +3 -2
  115. package/src/collections/methods.ts +85 -51
  116. package/src/config/yamlConfig.ts +488 -0
  117. package/src/{migrations/setupDatabase.ts → databases/setup.ts} +11 -5
  118. package/src/functions/methods.ts +8 -4
  119. package/src/functions/templates/count-docs-in-collection/package.json +25 -0
  120. package/src/functions/templates/count-docs-in-collection/tsconfig.json +28 -0
  121. package/src/functions/templates/typescript-node/package.json +24 -0
  122. package/src/functions/templates/typescript-node/tsconfig.json +28 -0
  123. package/src/functions/templates/uv/README.md +31 -0
  124. package/src/functions/templates/uv/pyproject.toml +29 -0
  125. package/src/interactiveCLI.ts +226 -61
  126. package/src/main.ts +111 -37
  127. package/src/migrations/afterImportActions.ts +2 -2
  128. package/src/migrations/appwriteToX.ts +17 -4
  129. package/src/migrations/dataLoader.ts +4 -4
  130. package/src/migrations/importController.ts +30 -22
  131. package/src/migrations/importDataActions.ts +2 -2
  132. package/src/migrations/relationships.ts +1 -1
  133. package/src/migrations/services/DataTransformationService.ts +196 -0
  134. package/src/migrations/services/FileHandlerService.ts +311 -0
  135. package/src/migrations/services/ImportOrchestrator.ts +669 -0
  136. package/src/migrations/services/RateLimitManager.ts +363 -0
  137. package/src/migrations/services/RelationshipResolver.ts +461 -0
  138. package/src/migrations/services/UserMappingService.ts +345 -0
  139. package/src/migrations/services/ValidationService.ts +349 -0
  140. package/src/migrations/transfer.ts +22 -228
  141. package/src/migrations/yaml/YamlImportConfigLoader.ts +427 -0
  142. package/src/migrations/yaml/YamlImportIntegration.ts +419 -0
  143. package/src/migrations/yaml/generateImportSchemas.ts +589 -0
  144. package/src/shared/attributeManager.ts +429 -0
  145. package/src/shared/confirmationDialogs.ts +327 -0
  146. package/src/shared/functionManager.ts +515 -0
  147. package/src/shared/indexManager.ts +253 -0
  148. package/src/shared/jsonSchemaGenerator.ts +403 -0
  149. package/src/shared/logging.ts +74 -0
  150. package/src/shared/messageFormatter.ts +195 -0
  151. package/src/{migrations/migrationHelper.ts → shared/migrationHelpers.ts} +22 -4
  152. package/src/{migrations/helper.ts → shared/operationLogger.ts} +7 -2
  153. package/src/{migrations/queue.ts → shared/operationQueue.ts} +1 -1
  154. package/src/shared/progressManager.ts +278 -0
  155. package/src/{migrations/schemaStrings.ts → shared/schemaGenerator.ts} +71 -17
  156. package/src/storage/methods.ts +199 -78
  157. package/src/types.ts +2 -2
  158. package/src/{migrations/users.ts → users/methods.ts} +2 -2
  159. package/src/utils/configMigration.ts +349 -0
  160. package/src/utils/loadConfigs.ts +416 -52
  161. package/src/utils/schemaStrings.ts +2 -1
  162. package/src/utils/setupFiles.ts +742 -40
  163. package/src/{migrations → utils}/validationRules.ts +1 -1
  164. package/src/utils/yamlConverter.ts +131 -0
  165. package/src/utilsController.ts +75 -54
  166. package/src/functions/templates/poetry/README.md +0 -30
  167. package/src/functions/templates/poetry/pyproject.toml +0 -16
  168. package/src/migrations/attributes.ts +0 -561
  169. package/src/migrations/backup.ts +0 -205
  170. package/src/migrations/databases.ts +0 -39
  171. package/src/migrations/dbHelpers.ts +0 -92
  172. package/src/migrations/indexes.ts +0 -40
  173. package/src/migrations/logging.ts +0 -29
  174. package/src/migrations/storage.ts +0 -538
  175. /package/src/{migrations → functions}/openapi.ts +0 -0
  176. /package/src/functions/templates/{poetry → uv}/src/__init__.py +0 -0
  177. /package/src/functions/templates/{poetry → uv}/src/index.py +0 -0
  178. /package/src/{migrations/converters.ts → utils/dataConverters.ts} +0 -0
@@ -0,0 +1,349 @@
1
+ import { promises as fs } from "fs";
2
+ import path from "path";
3
+ import { existsSync } from "fs";
4
+ import { MessageFormatter } from "../shared/messageFormatter.js";
5
+ import { ConfirmationDialogs } from "../shared/confirmationDialogs.js";
6
+ import yaml from "js-yaml";
7
+
8
+ interface AppwriteConfigTS {
9
+ appwriteEndpoint: string;
10
+ appwriteProject: string;
11
+ appwriteKey: string;
12
+ enableBackups?: boolean;
13
+ backupInterval?: number;
14
+ backupRetention?: number;
15
+ enableBackupCleanup?: boolean;
16
+ enableMockData?: boolean;
17
+ documentBucketId?: string;
18
+ usersCollectionName?: string;
19
+ databases: Array<{ $id: string; name: string }>;
20
+ buckets: Array<any>;
21
+ [key: string]: any;
22
+ }
23
+
24
+ interface AppwriteConfigYAML {
25
+ appwrite: {
26
+ endpoint: string;
27
+ project: string;
28
+ key: string;
29
+ };
30
+ logging: {
31
+ enabled: boolean;
32
+ level: string;
33
+ console: boolean;
34
+ logDirectory: string;
35
+ };
36
+ backups: {
37
+ enabled: boolean;
38
+ interval: number;
39
+ retention: number;
40
+ cleanup: boolean;
41
+ };
42
+ data: {
43
+ enableMockData: boolean;
44
+ documentBucketId: string;
45
+ usersCollectionName: string;
46
+ importDirectory: string;
47
+ };
48
+ schemas: {
49
+ outputDirectory: string;
50
+ yamlSchemaDirectory: string;
51
+ };
52
+ migrations: {
53
+ enabled: boolean;
54
+ };
55
+ databases: Array<{ id: string; name: string; collections?: string[] }>;
56
+ buckets: Array<any>;
57
+ functions: Array<any>;
58
+ }
59
+
60
+ export async function migrateConfig(workingDir: string): Promise<void> {
61
+ try {
62
+ // Look for appwriteConfig.ts files in the working directory and subdirectories
63
+ const configFiles = await findAppwriteConfigFiles(workingDir);
64
+
65
+ if (configFiles.length === 0) {
66
+ MessageFormatter.info("No appwriteConfig.ts files found to migrate", { prefix: "Migration" });
67
+ return;
68
+ }
69
+
70
+ MessageFormatter.info(`Found ${configFiles.length} appwriteConfig.ts file(s) to migrate`, { prefix: "Migration" });
71
+
72
+ for (const configFile of configFiles) {
73
+ await migrateConfigFile(configFile, workingDir);
74
+ }
75
+
76
+ MessageFormatter.success("Migration completed successfully", { prefix: "Migration" });
77
+ } catch (error) {
78
+ MessageFormatter.error("Migration failed", error instanceof Error ? error : new Error(String(error)), { prefix: "Migration" });
79
+ throw error;
80
+ }
81
+ }
82
+
83
+ async function findAppwriteConfigFiles(dir: string): Promise<string[]> {
84
+ const configFiles: string[] = [];
85
+
86
+ const checkDir = async (currentDir: string) => {
87
+ try {
88
+ const entries = await fs.readdir(currentDir, { withFileTypes: true });
89
+
90
+ for (const entry of entries) {
91
+ const fullPath = path.join(currentDir, entry.name);
92
+
93
+ if (entry.isDirectory() && !entry.name.startsWith('.') && entry.name !== 'node_modules') {
94
+ await checkDir(fullPath);
95
+ } else if (entry.isFile() && entry.name === 'appwriteConfig.ts') {
96
+ configFiles.push(fullPath);
97
+ }
98
+ }
99
+ } catch (error) {
100
+ // Ignore directory access errors
101
+ }
102
+ };
103
+
104
+ await checkDir(dir);
105
+ return configFiles;
106
+ }
107
+
108
+ async function migrateConfigFile(configFilePath: string, workingDir: string): Promise<void> {
109
+ const configDir = path.dirname(configFilePath);
110
+ const appwriteDir = path.join(path.dirname(configDir), '.appwrite');
111
+
112
+ MessageFormatter.info(`Migrating ${path.relative(workingDir, configFilePath)}`, { prefix: "Migration" });
113
+
114
+ // Check if .appwrite directory already exists
115
+ if (existsSync(appwriteDir)) {
116
+ const shouldOverwrite = await ConfirmationDialogs.confirmOverwrite(
117
+ `.appwrite directory already exists at ${path.relative(workingDir, appwriteDir)}`
118
+ );
119
+ if (!shouldOverwrite) {
120
+ MessageFormatter.info("Skipping migration for this config", { prefix: "Migration" });
121
+ return;
122
+ }
123
+ }
124
+
125
+ // Load and parse the TypeScript config
126
+ const config = await parseTypeScriptConfig(configFilePath);
127
+
128
+ // Create .appwrite directory
129
+ await fs.mkdir(appwriteDir, { recursive: true });
130
+
131
+ // Convert config to YAML and save
132
+ const yamlConfig = convertToYAMLConfig(config);
133
+ const yamlContent = yaml.dump(yamlConfig, {
134
+ indent: 2,
135
+ lineWidth: 120,
136
+ noRefs: true
137
+ });
138
+ await fs.writeFile(path.join(appwriteDir, 'appwriteConfig.yaml'), yamlContent);
139
+
140
+ // Copy all directories except collections and schemas (we handle collections separately, skip schemas entirely)
141
+ const entries = await fs.readdir(configDir, { withFileTypes: true });
142
+ for (const entry of entries) {
143
+ if (entry.isDirectory() && entry.name !== 'collections' && entry.name !== 'schemas') {
144
+ const sourcePath = path.join(configDir, entry.name);
145
+ const targetPath = path.join(appwriteDir, entry.name);
146
+
147
+ await fs.cp(sourcePath, targetPath, { recursive: true });
148
+ MessageFormatter.info(`Copied ${entry.name}/ to .appwrite/${entry.name}/`, { prefix: "Migration" });
149
+ }
150
+ }
151
+
152
+ // Convert TypeScript collections to YAML collections
153
+ const collectionsPath = path.join(configDir, 'collections');
154
+ if (existsSync(collectionsPath)) {
155
+ const targetCollectionsPath = path.join(appwriteDir, 'collections');
156
+ await fs.mkdir(targetCollectionsPath, { recursive: true });
157
+
158
+ const collectionFiles = await fs.readdir(collectionsPath);
159
+ for (const file of collectionFiles) {
160
+ if (file.endsWith('.ts')) {
161
+ await convertCollectionToYaml(path.join(collectionsPath, file), targetCollectionsPath);
162
+ }
163
+ }
164
+ MessageFormatter.info(`Converted TypeScript collections to YAML in .appwrite/collections/`, { prefix: "Migration" });
165
+ }
166
+
167
+ // Keep original config file in place (no backup needed since we're not deleting it)
168
+
169
+ MessageFormatter.success(`Migration completed for ${path.relative(workingDir, configFilePath)}`, { prefix: "Migration" });
170
+ }
171
+
172
+ async function parseTypeScriptConfig(configFilePath: string): Promise<AppwriteConfigTS> {
173
+ try {
174
+ // Use tsx to import the TypeScript config file directly
175
+ const { register } = await import("tsx/esm/api");
176
+ const { pathToFileURL } = await import("node:url");
177
+
178
+ const unregister = register();
179
+
180
+ try {
181
+ const configUrl = pathToFileURL(configFilePath).href;
182
+ const configModule = await import(configUrl);
183
+ const config = configModule.default?.default || configModule.default || configModule;
184
+
185
+ if (!config) {
186
+ throw new Error("Failed to load config from TypeScript file");
187
+ }
188
+
189
+ return config as AppwriteConfigTS;
190
+ } finally {
191
+ unregister();
192
+ }
193
+ } catch (error) {
194
+ MessageFormatter.error("Could not load TypeScript config", error instanceof Error ? error : new Error(String(error)), { prefix: "Migration" });
195
+ throw new Error('Failed to load TypeScript configuration file. Please ensure it exports a valid config object.');
196
+ }
197
+ }
198
+
199
+ function convertToYAMLConfig(config: AppwriteConfigTS): AppwriteConfigYAML {
200
+ // Convert the config to the nested YAML structure
201
+ const yamlConfig: AppwriteConfigYAML = {
202
+ appwrite: {
203
+ endpoint: config.appwriteEndpoint,
204
+ project: config.appwriteProject,
205
+ key: config.appwriteKey
206
+ },
207
+ logging: {
208
+ enabled: config.logging?.enabled ?? false,
209
+ level: config.logging?.level ?? "info",
210
+ console: config.logging?.console ?? false,
211
+ logDirectory: "./logs"
212
+ },
213
+ backups: {
214
+ enabled: config.enableBackups ?? false,
215
+ interval: config.backupInterval ?? 3600,
216
+ retention: config.backupRetention ?? 30,
217
+ cleanup: config.enableBackupCleanup ?? false
218
+ },
219
+ data: {
220
+ enableMockData: config.enableMockData ?? false,
221
+ documentBucketId: config.documentBucketId ?? "documents",
222
+ usersCollectionName: config.usersCollectionName ?? "Users",
223
+ importDirectory: "importData"
224
+ },
225
+ schemas: {
226
+ outputDirectory: "schemas",
227
+ yamlSchemaDirectory: ".yaml_schemas"
228
+ },
229
+ migrations: {
230
+ enabled: true
231
+ },
232
+ databases: (config.databases || []).map(db => ({
233
+ id: db.$id,
234
+ name: db.name,
235
+ collections: [] // Collections will be handled separately
236
+ })),
237
+ buckets: (config.buckets || []).map(bucket => ({
238
+ id: bucket.$id,
239
+ name: bucket.name,
240
+ permissions: bucket.$permissions?.map((p: any) => ({
241
+ permission: p.permission,
242
+ target: p.target
243
+ })) || [],
244
+ fileSecurity: bucket.fileSecurity ?? false,
245
+ enabled: bucket.enabled ?? true,
246
+ maximumFileSize: bucket.maximumFileSize ?? 30000000,
247
+ allowedFileExtensions: bucket.allowedFileExtensions || [],
248
+ compression: bucket.compression || "gzip",
249
+ encryption: bucket.encryption ?? false,
250
+ antivirus: bucket.antivirus ?? false
251
+ })),
252
+ functions: (config.functions || []).map((func: any) => ({
253
+ id: func.$id,
254
+ name: func.name,
255
+ runtime: func.runtime,
256
+ execute: func.execute || [],
257
+ events: func.events || [],
258
+ schedule: func.schedule || "",
259
+ timeout: func.timeout ?? 15,
260
+ enabled: func.enabled ?? true,
261
+ logging: func.logging ?? false,
262
+ entrypoint: func.entrypoint || "src/main.js",
263
+ commands: func.commands || "",
264
+ scopes: func.scopes || [],
265
+ specification: func.specification || "s-1vcpu-512mb"
266
+ }))
267
+ };
268
+
269
+ return yamlConfig;
270
+ }
271
+
272
+ async function convertCollectionToYaml(tsFilePath: string, targetDir: string): Promise<void> {
273
+ try {
274
+ // Load the TypeScript collection using tsx
275
+ const { register } = await import("tsx/esm/api");
276
+ const { pathToFileURL } = await import("node:url");
277
+
278
+ const unregister = register();
279
+
280
+ try {
281
+ const configUrl = pathToFileURL(tsFilePath).href;
282
+ const collectionModule = await import(configUrl);
283
+ const collection = collectionModule.default?.default || collectionModule.default || collectionModule;
284
+
285
+ if (!collection) {
286
+ throw new Error("Failed to load collection from TypeScript file");
287
+ }
288
+
289
+ // Convert collection to YAML format
290
+ const yamlCollection = {
291
+ name: collection.name,
292
+ id: collection.$id,
293
+ documentSecurity: collection.documentSecurity ?? false,
294
+ enabled: collection.enabled ?? true,
295
+ permissions: (collection.permissions || collection.$permissions || []).map((p: any) => ({
296
+ permission: p.permission,
297
+ target: p.target
298
+ })),
299
+ attributes: (collection.attributes || []).map((attr: any) => ({
300
+ key: attr.key,
301
+ type: attr.type,
302
+ size: attr.size,
303
+ required: attr.required ?? false,
304
+ array: attr.array,
305
+ default: attr.xdefault || attr.default,
306
+ description: attr.description,
307
+ min: attr.min,
308
+ max: attr.max,
309
+ elements: attr.elements,
310
+ relatedCollection: attr.relatedCollection,
311
+ relationType: attr.relationType,
312
+ twoWay: attr.twoWay,
313
+ twoWayKey: attr.twoWayKey,
314
+ onDelete: attr.onDelete,
315
+ side: attr.side
316
+ })),
317
+ indexes: (collection.indexes || []).map((idx: any) => ({
318
+ key: idx.key,
319
+ type: idx.type,
320
+ attributes: idx.attributes,
321
+ orders: idx.orders
322
+ })),
323
+ importDefs: collection.importDefs || []
324
+ };
325
+
326
+ // Remove undefined values
327
+ const cleanYamlCollection = JSON.parse(JSON.stringify(yamlCollection, (key, value) =>
328
+ value === undefined ? undefined : value
329
+ ));
330
+
331
+ // Write YAML file
332
+ const fileName = path.basename(tsFilePath, '.ts') + '.yaml';
333
+ const targetPath = path.join(targetDir, fileName);
334
+ const yamlContent = yaml.dump(cleanYamlCollection, {
335
+ indent: 2,
336
+ lineWidth: 120,
337
+ noRefs: true
338
+ });
339
+
340
+ await fs.writeFile(targetPath, yamlContent);
341
+ MessageFormatter.info(`Converted ${path.basename(tsFilePath)} to ${fileName}`, { prefix: "Migration" });
342
+
343
+ } finally {
344
+ unregister();
345
+ }
346
+ } catch (error) {
347
+ MessageFormatter.error(`Failed to convert collection ${path.basename(tsFilePath)}`, error instanceof Error ? error : new Error(String(error)), { prefix: "Migration" });
348
+ }
349
+ }