@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,254 +0,0 @@
1
- import { type Index, type CollectionCreate } from "@njdamstra/appwrite-utils";
2
- import { Databases, IndexType, Query, type Models } from "node-appwrite";
3
- import { delay, tryAwaitWithRetry } from "../utils/helperFunctions.js";
4
- import chalk from "chalk";
5
- import pLimit from "p-limit";
6
- import { MessageFormatter } from "./messageFormatter.js";
7
-
8
- // Concurrency limits for different operations
9
- const indexLimit = pLimit(3); // Low limit for index operations
10
- const queryLimit = pLimit(25); // Higher limit for read operations
11
-
12
- export const indexesSame = (
13
- databaseIndex: Models.Index,
14
- configIndex: Index
15
- ): boolean => {
16
- return (
17
- databaseIndex.key === configIndex.key &&
18
- databaseIndex.type === configIndex.type &&
19
- JSON.stringify(databaseIndex.attributes) === JSON.stringify(configIndex.attributes) &&
20
- JSON.stringify(databaseIndex.orders) === JSON.stringify(configIndex.orders)
21
- );
22
- };
23
-
24
- export const createOrUpdateIndex = async (
25
- dbId: string,
26
- db: Databases,
27
- collectionId: string,
28
- index: Index,
29
- options: {
30
- verbose?: boolean;
31
- forceRecreate?: boolean;
32
- } = {}
33
- ): Promise<Models.Index | null> => {
34
- const { verbose = false, forceRecreate = false } = options;
35
-
36
- return await indexLimit(async () => {
37
- // Check for existing index
38
- const existingIndexes = await queryLimit(() =>
39
- tryAwaitWithRetry(async () =>
40
- await db.listIndexes(dbId, collectionId, [Query.equal("key", index.key)])
41
- )
42
- );
43
-
44
- let shouldCreate = false;
45
- let existingIndex: Models.Index | undefined;
46
-
47
- if (existingIndexes.total > 0) {
48
- existingIndex = existingIndexes.indexes[0];
49
-
50
- if (forceRecreate || !indexesSame(existingIndex, index)) {
51
- if (verbose) {
52
- MessageFormatter.warning(`Updating index ${index.key} in collection ${collectionId}`, { prefix: "Index Manager" });
53
- }
54
-
55
- // Delete existing index
56
- await tryAwaitWithRetry(async () => {
57
- await db.deleteIndex(dbId, collectionId, existingIndex!.key);
58
- });
59
-
60
- await delay(500); // Wait for deletion to complete
61
- shouldCreate = true;
62
- } else {
63
- if (verbose) {
64
- MessageFormatter.success(`Index ${index.key} is up to date`, { prefix: "Index Manager" });
65
- }
66
- return existingIndex;
67
- }
68
- } else {
69
- shouldCreate = true;
70
- if (verbose) {
71
- MessageFormatter.info(`Creating index ${index.key} in collection ${collectionId}`, { prefix: "Index Manager" });
72
- }
73
- }
74
-
75
- if (shouldCreate) {
76
- const newIndex = await tryAwaitWithRetry(async () => {
77
- return await db.createIndex(
78
- dbId,
79
- collectionId,
80
- index.key,
81
- index.type as IndexType,
82
- index.attributes,
83
- index.orders
84
- );
85
- });
86
-
87
- if (verbose) {
88
- MessageFormatter.success(`Created index ${index.key}`, { prefix: "Index Manager" });
89
- }
90
-
91
- return newIndex;
92
- }
93
-
94
- return null;
95
- });
96
- };
97
-
98
- export const createOrUpdateIndexes = async (
99
- dbId: string,
100
- db: Databases,
101
- collectionId: string,
102
- indexes: Index[],
103
- options: {
104
- verbose?: boolean;
105
- forceRecreate?: boolean;
106
- } = {}
107
- ): Promise<void> => {
108
- const { verbose = false } = options;
109
-
110
- if (!indexes || indexes.length === 0) {
111
- return;
112
- }
113
-
114
- if (verbose) {
115
- MessageFormatter.info(`Processing ${indexes.length} indexes for collection ${collectionId}`, { prefix: "Index Manager" });
116
- }
117
-
118
- // Process indexes sequentially to avoid conflicts
119
- for (const index of indexes) {
120
- try {
121
- await createOrUpdateIndex(dbId, db, collectionId, index, options);
122
-
123
- // Add delay between index operations to prevent rate limiting
124
- await delay(250);
125
- } catch (error) {
126
- MessageFormatter.error(`Failed to process index ${index.key}`, error as Error, { prefix: "Index Manager" });
127
- throw error;
128
- }
129
- }
130
-
131
- if (verbose) {
132
- MessageFormatter.success(`Completed processing indexes for collection ${collectionId}`, { prefix: "Index Manager" });
133
- }
134
- };
135
-
136
- export const createUpdateCollectionIndexes = async (
137
- db: Databases,
138
- dbId: string,
139
- collection: Models.Collection,
140
- collectionConfig: CollectionCreate,
141
- options: {
142
- verbose?: boolean;
143
- forceRecreate?: boolean;
144
- } = {}
145
- ): Promise<void> => {
146
- if (!collectionConfig.indexes) return;
147
-
148
- await createOrUpdateIndexes(
149
- dbId,
150
- db,
151
- collection.$id,
152
- collectionConfig.indexes,
153
- options
154
- );
155
- };
156
-
157
- export const deleteObsoleteIndexes = async (
158
- db: Databases,
159
- dbId: string,
160
- collection: Models.Collection,
161
- collectionConfig: CollectionCreate,
162
- options: {
163
- verbose?: boolean;
164
- } = {}
165
- ): Promise<void> => {
166
- const { verbose = false } = options;
167
-
168
- const configIndexes = collectionConfig.indexes || [];
169
- const configIndexKeys = new Set(configIndexes.map(index => index.key));
170
-
171
- // Get all existing indexes
172
- const existingIndexes = await queryLimit(() =>
173
- tryAwaitWithRetry(async () =>
174
- await db.listIndexes(dbId, collection.$id)
175
- )
176
- );
177
-
178
- // Find indexes that exist in the database but not in the config
179
- const obsoleteIndexes = existingIndexes.indexes.filter(
180
- (index) => !configIndexKeys.has(index.key)
181
- );
182
-
183
- if (obsoleteIndexes.length === 0) {
184
- return;
185
- }
186
-
187
- if (verbose) {
188
- MessageFormatter.warning(`Removing ${obsoleteIndexes.length} obsolete indexes from collection ${collection.name}`, { prefix: "Index Manager" });
189
- }
190
-
191
- // Process deletions with rate limiting
192
- for (const index of obsoleteIndexes) {
193
- await indexLimit(async () => {
194
- await tryAwaitWithRetry(async () => {
195
- await db.deleteIndex(dbId, collection.$id, index.key);
196
- });
197
- });
198
-
199
- if (verbose) {
200
- MessageFormatter.info(`Deleted obsolete index ${index.key}`, { prefix: "Index Manager" });
201
- }
202
-
203
- await delay(250);
204
- }
205
- };
206
-
207
- export const validateIndexConfiguration = (
208
- indexes: Index[],
209
- options: {
210
- verbose?: boolean;
211
- } = {}
212
- ): { valid: boolean; errors: string[] } => {
213
- const { verbose = false } = options;
214
- const errors: string[] = [];
215
-
216
- for (const index of indexes) {
217
- // Validate required fields
218
- if (!index.key) {
219
- errors.push(`Index missing required 'key' field`);
220
- }
221
-
222
- if (!index.type) {
223
- errors.push(`Index '${index.key}' missing required 'type' field`);
224
- }
225
-
226
- if (!index.attributes || index.attributes.length === 0) {
227
- errors.push(`Index '${index.key}' missing required 'attributes' field`);
228
- }
229
-
230
- // Validate index type
231
- const validTypes = Object.values(IndexType);
232
- if (index.type && !validTypes.includes(index.type as IndexType)) {
233
- errors.push(`Index '${index.key}' has invalid type '${index.type}'. Valid types: ${validTypes.join(', ')}`);
234
- }
235
-
236
- // Validate orders array matches attributes length (if provided)
237
- if (index.orders && index.attributes && index.orders.length !== index.attributes.length) {
238
- errors.push(`Index '${index.key}' orders array length (${index.orders.length}) does not match attributes array length (${index.attributes.length})`);
239
- }
240
-
241
- // Check for duplicate keys within the same collection
242
- const duplicateKeys = indexes.filter(i => i.key === index.key);
243
- if (duplicateKeys.length > 1) {
244
- errors.push(`Duplicate index key '${index.key}' found`);
245
- }
246
- }
247
-
248
- if (verbose && errors.length > 0) {
249
- MessageFormatter.error("Index validation errors", undefined, { prefix: "Index Manager" });
250
- errors.forEach(error => MessageFormatter.error(` - ${error}`, undefined, { prefix: "Index Manager" }));
251
- }
252
-
253
- return { valid: errors.length === 0, errors };
254
- };
@@ -1,383 +0,0 @@
1
- import fs from "fs";
2
- import path from "path";
3
- import type { AppwriteConfig, Attribute, CollectionCreate } from "@njdamstra/appwrite-utils";
4
- import { toCamelCase, toPascalCase } from "../utils/index.js";
5
- import chalk from "chalk";
6
- import {
7
- extractSimpleRelationships,
8
- resolveCollectionName,
9
- type SimpleRelationship
10
- } from "./relationshipExtractor.js";
11
- import { MessageFormatter } from "./messageFormatter.js";
12
-
13
- export interface JsonSchemaProperty {
14
- type: string | string[];
15
- format?: string;
16
- minimum?: number;
17
- maximum?: number;
18
- minLength?: number;
19
- maxLength?: number;
20
- pattern?: string;
21
- enum?: any[];
22
- items?: JsonSchemaProperty;
23
- $ref?: string;
24
- properties?: Record<string, JsonSchemaProperty>;
25
- additionalProperties?: boolean;
26
- required?: string[];
27
- default?: any;
28
- oneOf?: JsonSchemaProperty[];
29
- }
30
-
31
- export interface JsonSchema {
32
- $schema: string;
33
- $id: string;
34
- title: string;
35
- type: "object";
36
- properties: Record<string, JsonSchemaProperty>;
37
- required: string[];
38
- additionalProperties: boolean;
39
- definitions?: Record<string, JsonSchemaProperty>;
40
- }
41
-
42
- export class JsonSchemaGenerator {
43
- private config: AppwriteConfig;
44
- private appwriteFolderPath: string;
45
- private relationshipMap = new Map<string, SimpleRelationship[]>();
46
-
47
- constructor(config: AppwriteConfig, appwriteFolderPath: string) {
48
- this.config = config;
49
- this.appwriteFolderPath = appwriteFolderPath;
50
- this.extractRelationships();
51
- }
52
-
53
- private resolveCollectionName = (idOrName: string): string => {
54
- return resolveCollectionName(this.config, idOrName);
55
- };
56
-
57
- private extractRelationships(): void {
58
- this.relationshipMap = extractSimpleRelationships(this.config);
59
- }
60
-
61
- private attributeToJsonSchemaProperty(attribute: Attribute): JsonSchemaProperty {
62
- const property: JsonSchemaProperty = {
63
- type: "string" // Default type
64
- };
65
-
66
-
67
- // Handle array attributes
68
- if (attribute.array) {
69
- property.type = "array";
70
- property.items = this.getBaseTypeSchema(attribute);
71
- } else {
72
- Object.assign(property, this.getBaseTypeSchema(attribute));
73
- }
74
-
75
- // Set default value (only for attributes that support it)
76
- if (attribute.type !== "relationship" && "xdefault" in attribute &&
77
- attribute.xdefault !== undefined && attribute.xdefault !== null) {
78
- property.default = attribute.xdefault;
79
- }
80
-
81
- return property;
82
- }
83
-
84
- private getBaseTypeSchema(attribute: Attribute): JsonSchemaProperty {
85
- const schema: JsonSchemaProperty = {
86
- type: "string" // Default type
87
- };
88
-
89
- switch (attribute.type) {
90
- case "string":
91
- schema.type = "string";
92
- if (attribute.size) {
93
- schema.maxLength = attribute.size;
94
- }
95
- break;
96
-
97
- case "integer":
98
- schema.type = "integer";
99
- if (attribute.min !== undefined) {
100
- schema.minimum = Number(attribute.min);
101
- }
102
- if (attribute.max !== undefined) {
103
- schema.maximum = Number(attribute.max);
104
- }
105
- break;
106
-
107
- case "double":
108
- case "float": // Backward compatibility
109
- schema.type = "number";
110
- if (attribute.min !== undefined) {
111
- schema.minimum = Number(attribute.min);
112
- }
113
- if (attribute.max !== undefined) {
114
- schema.maximum = Number(attribute.max);
115
- }
116
- break;
117
-
118
- case "boolean":
119
- schema.type = "boolean";
120
- break;
121
-
122
- case "datetime":
123
- schema.type = "string";
124
- schema.format = "date-time";
125
- break;
126
-
127
- case "email":
128
- schema.type = "string";
129
- schema.format = "email";
130
- break;
131
-
132
- case "ip":
133
- schema.type = "string";
134
- schema.format = "ipv4";
135
- break;
136
-
137
- case "url":
138
- schema.type = "string";
139
- schema.format = "uri";
140
- break;
141
-
142
- case "enum":
143
- schema.type = "string";
144
- if (attribute.elements) {
145
- schema.enum = attribute.elements;
146
- }
147
- break;
148
-
149
- case "relationship":
150
- if (attribute.relatedCollection) {
151
- // For relationships, reference the related collection schema
152
- schema.$ref = `#/definitions/${toPascalCase(this.resolveCollectionName(attribute.relatedCollection))}`;
153
- } else {
154
- schema.type = "string";
155
- }
156
- break;
157
-
158
- default:
159
- schema.type = "string";
160
- }
161
-
162
- return schema;
163
- }
164
-
165
- private createJsonSchema(collection: CollectionCreate): JsonSchema {
166
- const pascalName = toPascalCase(collection.name);
167
- const schema: JsonSchema = {
168
- $schema: "https://json-schema.org/draft/2020-12/schema",
169
- $id: `https://example.com/schemas/${toCamelCase(collection.name)}.json`,
170
- title: pascalName,
171
- type: "object",
172
- properties: {
173
- // Standard Appwrite document fields
174
- $id: {
175
- type: "string",
176
- pattern: "^[a-zA-Z0-9][a-zA-Z0-9._-]{0,35}$"
177
- },
178
- $createdAt: {
179
- type: "string",
180
- format: "date-time"
181
- },
182
- $updatedAt: {
183
- type: "string",
184
- format: "date-time"
185
- },
186
- $permissions: {
187
- type: "array",
188
- items: {
189
- type: "string"
190
- }
191
- }
192
- },
193
- required: ["$id", "$createdAt", "$updatedAt"],
194
- additionalProperties: false
195
- };
196
-
197
- // Add custom attributes
198
- const requiredFields: string[] = [...schema.required];
199
-
200
- if (collection.attributes) {
201
- collection.attributes.forEach((attribute) => {
202
- schema.properties[attribute.key] = this.attributeToJsonSchemaProperty(attribute);
203
-
204
- if (attribute.required) {
205
- requiredFields.push(attribute.key);
206
- }
207
- });
208
- }
209
-
210
- schema.required = requiredFields;
211
-
212
- // Add relationship definitions if any exist
213
- const relationships = this.relationshipMap.get(collection.name);
214
- if (relationships && relationships.length > 0) {
215
- schema.definitions = {};
216
-
217
- relationships.forEach((rel) => {
218
- const relatedPascalName = toPascalCase(rel.relatedCollection);
219
- schema.definitions![relatedPascalName] = {
220
- type: "object",
221
- properties: {
222
- $id: { type: "string" },
223
- $createdAt: { type: "string", format: "date-time" },
224
- $updatedAt: { type: "string", format: "date-time" }
225
- },
226
- additionalProperties: true
227
- };
228
- });
229
- }
230
-
231
- return schema;
232
- }
233
-
234
- public generateJsonSchemas(options: {
235
- outputFormat?: "json" | "typescript" | "both";
236
- outputDirectory?: string;
237
- verbose?: boolean;
238
- } = {}): void {
239
- const { outputFormat = "both", outputDirectory = "schemas", verbose = false } = options;
240
-
241
- if (!this.config.collections) {
242
- if (verbose) {
243
- MessageFormatter.warning("No collections found in config", { prefix: "Schema" });
244
- }
245
- return;
246
- }
247
-
248
- // Create JSON schemas directory using provided outputDirectory
249
- const jsonSchemasPath = path.isAbsolute(outputDirectory)
250
- ? outputDirectory
251
- : path.join(this.appwriteFolderPath, outputDirectory);
252
- if (!fs.existsSync(jsonSchemasPath)) {
253
- fs.mkdirSync(jsonSchemasPath, { recursive: true });
254
- }
255
-
256
- if (verbose) {
257
- MessageFormatter.processing(`Generating JSON schemas for ${this.config.collections.length} collections...`, { prefix: "Schema" });
258
- }
259
-
260
- this.config.collections.forEach((collection) => {
261
- const schema = this.createJsonSchema(collection);
262
- const camelCaseName = toCamelCase(collection.name);
263
-
264
- // Generate JSON file
265
- if (outputFormat === "json" || outputFormat === "both") {
266
- const jsonPath = path.join(jsonSchemasPath, `${camelCaseName}.json`);
267
- fs.writeFileSync(jsonPath, JSON.stringify(schema, null, 2), { encoding: "utf-8" });
268
-
269
- if (verbose) {
270
- MessageFormatter.success(`JSON schema written to ${jsonPath}`, { prefix: "Schema" });
271
- }
272
- }
273
-
274
- // Generate TypeScript file
275
- if (outputFormat === "typescript" || outputFormat === "both") {
276
- const tsContent = this.generateTypeScriptSchema(schema, collection.name);
277
- const tsPath = path.join(jsonSchemasPath, `${camelCaseName}.schema.ts`);
278
- fs.writeFileSync(tsPath, tsContent, { encoding: "utf-8" });
279
-
280
- if (verbose) {
281
- MessageFormatter.success(`TypeScript schema written to ${tsPath}`, { prefix: "Schema" });
282
- }
283
- }
284
- });
285
-
286
- // Generate index file only for TypeScript output
287
- if (outputFormat === "typescript" || outputFormat === "both") {
288
- this.generateIndexFile(outputFormat, jsonSchemasPath, verbose);
289
- }
290
-
291
- if (verbose) {
292
- MessageFormatter.success("JSON schema generation completed", { prefix: "Schema" });
293
- }
294
- }
295
-
296
- private generateTypeScriptSchema(schema: JsonSchema, collectionName: string): string {
297
- const camelName = toCamelCase(collectionName);
298
-
299
- return `// Auto-generated JSON schema for ${collectionName}
300
- import type { JSONSchema7 } from "json-schema";
301
-
302
- export const ${camelName}JsonSchema: JSONSchema7 = ${JSON.stringify(schema, null, 2)} as const;
303
-
304
- export type ${toPascalCase(collectionName)}JsonSchema = typeof ${camelName}JsonSchema;
305
-
306
- export default ${camelName}JsonSchema;
307
- `;
308
- }
309
-
310
- private generateIndexFile(outputFormat: string, jsonSchemasPath: string, verbose: boolean): void {
311
- if (!this.config.collections) return;
312
-
313
- const indexPath = path.join(jsonSchemasPath, "index.ts");
314
-
315
- const imports: string[] = [];
316
- const exports: string[] = [];
317
-
318
- this.config.collections.forEach((collection) => {
319
- const camelName = toCamelCase(collection.name);
320
- const pascalName = toPascalCase(collection.name);
321
-
322
- if (outputFormat === "typescript" || outputFormat === "both") {
323
- imports.push(`import { ${camelName}JsonSchema } from "./${camelName}.schema.js";`);
324
- exports.push(` ${camelName}: ${camelName}JsonSchema,`);
325
- }
326
- });
327
-
328
- const indexContent = `// Auto-generated index for JSON schemas
329
- ${imports.join('\n')}
330
-
331
- export const jsonSchemas = {
332
- ${exports.join('\n')}
333
- };
334
-
335
- export type JsonSchemas = typeof jsonSchemas;
336
-
337
- // Individual schema exports
338
- ${this.config.collections.map(collection => {
339
- const camelName = toCamelCase(collection.name);
340
- return `export { ${camelName}JsonSchema } from "./${camelName}.schema.js";`;
341
- }).join('\n')}
342
-
343
- export default jsonSchemas;
344
- `;
345
-
346
- fs.writeFileSync(indexPath, indexContent, { encoding: "utf-8" });
347
-
348
- if (verbose) {
349
- MessageFormatter.success(`Index file written to ${indexPath}`, { prefix: "Schema" });
350
- }
351
- }
352
-
353
- public validateSchema(schema: JsonSchema): { valid: boolean; errors: string[] } {
354
- const errors: string[] = [];
355
-
356
- // Basic validation
357
- if (!schema.$schema) {
358
- errors.push("Missing $schema property");
359
- }
360
-
361
- if (!schema.title) {
362
- errors.push("Missing title property");
363
- }
364
-
365
- if (!schema.properties) {
366
- errors.push("Missing properties");
367
- }
368
-
369
- if (!schema.required || !Array.isArray(schema.required)) {
370
- errors.push("Missing or invalid required array");
371
- }
372
-
373
- // Validate required Appwrite fields
374
- const requiredAppwriteFields = ["$id", "$createdAt", "$updatedAt"];
375
- requiredAppwriteFields.forEach((field) => {
376
- if (!schema.properties[field]) {
377
- errors.push(`Missing required Appwrite field: ${field}`);
378
- }
379
- });
380
-
381
- return { valid: errors.length === 0, errors };
382
- }
383
- }