appwrite-utils-cli 1.11.0 → 1.12.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 (250) hide show
  1. package/{src/adapters/index.ts → dist/adapters/index.d.ts} +0 -1
  2. package/dist/adapters/index.js +10 -0
  3. package/dist/backups/operations/bucketBackup.d.ts +19 -0
  4. package/dist/backups/operations/bucketBackup.js +197 -0
  5. package/dist/backups/operations/collectionBackup.d.ts +30 -0
  6. package/dist/backups/operations/collectionBackup.js +201 -0
  7. package/dist/backups/operations/comprehensiveBackup.d.ts +25 -0
  8. package/dist/backups/operations/comprehensiveBackup.js +238 -0
  9. package/dist/backups/schemas/bucketManifest.d.ts +93 -0
  10. package/dist/backups/schemas/bucketManifest.js +33 -0
  11. package/dist/backups/schemas/comprehensiveManifest.d.ts +108 -0
  12. package/dist/backups/schemas/comprehensiveManifest.js +32 -0
  13. package/dist/backups/tracking/centralizedTracking.d.ts +34 -0
  14. package/dist/backups/tracking/centralizedTracking.js +274 -0
  15. package/dist/cli/commands/configCommands.d.ts +8 -0
  16. package/dist/cli/commands/configCommands.js +210 -0
  17. package/dist/cli/commands/databaseCommands.d.ts +14 -0
  18. package/dist/cli/commands/databaseCommands.js +696 -0
  19. package/dist/cli/commands/functionCommands.d.ts +7 -0
  20. package/dist/cli/commands/functionCommands.js +330 -0
  21. package/dist/cli/commands/importFileCommands.d.ts +7 -0
  22. package/dist/cli/commands/importFileCommands.js +674 -0
  23. package/dist/cli/commands/schemaCommands.d.ts +7 -0
  24. package/dist/cli/commands/schemaCommands.js +169 -0
  25. package/dist/cli/commands/storageCommands.d.ts +5 -0
  26. package/dist/cli/commands/storageCommands.js +142 -0
  27. package/dist/cli/commands/transferCommands.d.ts +5 -0
  28. package/dist/cli/commands/transferCommands.js +382 -0
  29. package/dist/collections/columns.d.ts +13 -0
  30. package/dist/collections/columns.js +1339 -0
  31. package/dist/collections/indexes.d.ts +12 -0
  32. package/dist/collections/indexes.js +215 -0
  33. package/dist/collections/methods.d.ts +19 -0
  34. package/dist/collections/methods.js +605 -0
  35. package/dist/collections/tableOperations.d.ts +87 -0
  36. package/dist/collections/tableOperations.js +466 -0
  37. package/dist/collections/transferOperations.d.ts +8 -0
  38. package/dist/collections/transferOperations.js +411 -0
  39. package/dist/collections/wipeOperations.d.ts +17 -0
  40. package/dist/collections/wipeOperations.js +306 -0
  41. package/dist/databases/methods.d.ts +6 -0
  42. package/dist/databases/methods.js +35 -0
  43. package/dist/databases/setup.d.ts +5 -0
  44. package/dist/databases/setup.js +45 -0
  45. package/dist/examples/yamlTerminologyExample.d.ts +42 -0
  46. package/dist/examples/yamlTerminologyExample.js +272 -0
  47. package/dist/functions/deployments.d.ts +4 -0
  48. package/dist/functions/deployments.js +146 -0
  49. package/dist/functions/fnConfigDiscovery.d.ts +3 -0
  50. package/dist/functions/fnConfigDiscovery.js +108 -0
  51. package/dist/functions/methods.d.ts +16 -0
  52. package/dist/functions/methods.js +174 -0
  53. package/dist/init.d.ts +2 -0
  54. package/dist/init.js +57 -0
  55. package/dist/interactiveCLI.d.ts +36 -0
  56. package/dist/interactiveCLI.js +952 -0
  57. package/dist/main.d.ts +2 -0
  58. package/dist/main.js +1125 -0
  59. package/dist/migrations/afterImportActions.d.ts +17 -0
  60. package/dist/migrations/afterImportActions.js +305 -0
  61. package/dist/migrations/appwriteToX.d.ts +211 -0
  62. package/dist/migrations/appwriteToX.js +493 -0
  63. package/dist/migrations/comprehensiveTransfer.d.ts +147 -0
  64. package/dist/migrations/comprehensiveTransfer.js +1315 -0
  65. package/dist/migrations/dataLoader.d.ts +755 -0
  66. package/dist/migrations/dataLoader.js +1272 -0
  67. package/dist/migrations/importController.d.ts +25 -0
  68. package/dist/migrations/importController.js +283 -0
  69. package/dist/migrations/importDataActions.d.ts +50 -0
  70. package/dist/migrations/importDataActions.js +230 -0
  71. package/dist/migrations/relationships.d.ts +29 -0
  72. package/dist/migrations/relationships.js +203 -0
  73. package/dist/migrations/services/DataTransformationService.d.ts +55 -0
  74. package/dist/migrations/services/DataTransformationService.js +158 -0
  75. package/dist/migrations/services/FileHandlerService.d.ts +75 -0
  76. package/dist/migrations/services/FileHandlerService.js +236 -0
  77. package/dist/migrations/services/ImportOrchestrator.d.ts +99 -0
  78. package/dist/migrations/services/ImportOrchestrator.js +493 -0
  79. package/dist/migrations/services/RateLimitManager.d.ts +138 -0
  80. package/dist/migrations/services/RateLimitManager.js +279 -0
  81. package/dist/migrations/services/RelationshipResolver.d.ts +120 -0
  82. package/dist/migrations/services/RelationshipResolver.js +332 -0
  83. package/dist/migrations/services/UserMappingService.d.ts +109 -0
  84. package/dist/migrations/services/UserMappingService.js +277 -0
  85. package/dist/migrations/services/ValidationService.d.ts +74 -0
  86. package/dist/migrations/services/ValidationService.js +260 -0
  87. package/dist/migrations/transfer.d.ts +30 -0
  88. package/dist/migrations/transfer.js +661 -0
  89. package/dist/migrations/yaml/YamlImportConfigLoader.d.ts +131 -0
  90. package/dist/migrations/yaml/YamlImportConfigLoader.js +383 -0
  91. package/dist/migrations/yaml/YamlImportIntegration.d.ts +93 -0
  92. package/dist/migrations/yaml/YamlImportIntegration.js +341 -0
  93. package/dist/migrations/yaml/generateImportSchemas.d.ts +30 -0
  94. package/dist/migrations/yaml/generateImportSchemas.js +1327 -0
  95. package/dist/schemas/authUser.d.ts +24 -0
  96. package/dist/schemas/authUser.js +17 -0
  97. package/dist/setup.d.ts +2 -0
  98. package/{src/setup.ts → dist/setup.js} +0 -3
  99. package/dist/setupCommands.d.ts +58 -0
  100. package/dist/setupCommands.js +489 -0
  101. package/dist/setupController.d.ts +9 -0
  102. package/dist/setupController.js +34 -0
  103. package/dist/shared/backupMetadataSchema.d.ts +94 -0
  104. package/dist/shared/backupMetadataSchema.js +38 -0
  105. package/dist/shared/backupTracking.d.ts +18 -0
  106. package/dist/shared/backupTracking.js +176 -0
  107. package/dist/shared/confirmationDialogs.d.ts +75 -0
  108. package/dist/shared/confirmationDialogs.js +236 -0
  109. package/dist/shared/migrationHelpers.d.ts +61 -0
  110. package/dist/shared/migrationHelpers.js +145 -0
  111. package/{src/shared/operationLogger.ts → dist/shared/operationLogger.d.ts} +1 -11
  112. package/dist/shared/operationLogger.js +12 -0
  113. package/dist/shared/operationQueue.d.ts +40 -0
  114. package/dist/shared/operationQueue.js +310 -0
  115. package/dist/shared/operationsTable.d.ts +26 -0
  116. package/dist/shared/operationsTable.js +287 -0
  117. package/dist/shared/operationsTableSchema.d.ts +48 -0
  118. package/dist/shared/operationsTableSchema.js +35 -0
  119. package/dist/shared/progressManager.d.ts +62 -0
  120. package/dist/shared/progressManager.js +215 -0
  121. package/dist/shared/relationshipExtractor.d.ts +56 -0
  122. package/dist/shared/relationshipExtractor.js +138 -0
  123. package/dist/shared/selectionDialogs.d.ts +220 -0
  124. package/dist/shared/selectionDialogs.js +588 -0
  125. package/dist/storage/backupCompression.d.ts +20 -0
  126. package/dist/storage/backupCompression.js +67 -0
  127. package/dist/storage/methods.d.ts +44 -0
  128. package/dist/storage/methods.js +475 -0
  129. package/dist/storage/schemas.d.ts +842 -0
  130. package/dist/storage/schemas.js +175 -0
  131. package/dist/tables/indexManager.d.ts +65 -0
  132. package/dist/tables/indexManager.js +294 -0
  133. package/{src/types.ts → dist/types.d.ts} +1 -6
  134. package/dist/types.js +3 -0
  135. package/dist/users/methods.d.ts +16 -0
  136. package/dist/users/methods.js +276 -0
  137. package/dist/utils/configMigration.d.ts +1 -0
  138. package/dist/utils/configMigration.js +261 -0
  139. package/dist/utils/index.js +2 -0
  140. package/dist/utils/loadConfigs.d.ts +50 -0
  141. package/dist/utils/loadConfigs.js +357 -0
  142. package/dist/utils/setupFiles.d.ts +4 -0
  143. package/dist/utils/setupFiles.js +1190 -0
  144. package/dist/utilsController.d.ts +114 -0
  145. package/dist/utilsController.js +898 -0
  146. package/package.json +6 -3
  147. package/CHANGELOG.md +0 -35
  148. package/CONFIG_TODO.md +0 -1189
  149. package/SELECTION_DIALOGS.md +0 -146
  150. package/SERVICE_IMPLEMENTATION_REPORT.md +0 -462
  151. package/scripts/copy-templates.ts +0 -23
  152. package/src/backups/operations/bucketBackup.ts +0 -277
  153. package/src/backups/operations/collectionBackup.ts +0 -310
  154. package/src/backups/operations/comprehensiveBackup.ts +0 -342
  155. package/src/backups/schemas/bucketManifest.ts +0 -78
  156. package/src/backups/schemas/comprehensiveManifest.ts +0 -76
  157. package/src/backups/tracking/centralizedTracking.ts +0 -352
  158. package/src/cli/commands/configCommands.ts +0 -265
  159. package/src/cli/commands/databaseCommands.ts +0 -931
  160. package/src/cli/commands/functionCommands.ts +0 -419
  161. package/src/cli/commands/importFileCommands.ts +0 -815
  162. package/src/cli/commands/schemaCommands.ts +0 -200
  163. package/src/cli/commands/storageCommands.ts +0 -151
  164. package/src/cli/commands/transferCommands.ts +0 -454
  165. package/src/collections/attributes.ts.backup +0 -1555
  166. package/src/collections/columns.ts +0 -2025
  167. package/src/collections/indexes.ts +0 -350
  168. package/src/collections/methods.ts +0 -714
  169. package/src/collections/tableOperations.ts +0 -542
  170. package/src/collections/transferOperations.ts +0 -589
  171. package/src/collections/wipeOperations.ts +0 -449
  172. package/src/databases/methods.ts +0 -49
  173. package/src/databases/setup.ts +0 -77
  174. package/src/examples/yamlTerminologyExample.ts +0 -346
  175. package/src/functions/deployments.ts +0 -221
  176. package/src/functions/fnConfigDiscovery.ts +0 -103
  177. package/src/functions/methods.ts +0 -284
  178. package/src/init.ts +0 -62
  179. package/src/interactiveCLI.ts +0 -1201
  180. package/src/main.ts +0 -1517
  181. package/src/migrations/afterImportActions.ts +0 -579
  182. package/src/migrations/appwriteToX.ts +0 -668
  183. package/src/migrations/comprehensiveTransfer.ts +0 -2285
  184. package/src/migrations/dataLoader.ts +0 -1729
  185. package/src/migrations/importController.ts +0 -440
  186. package/src/migrations/importDataActions.ts +0 -315
  187. package/src/migrations/relationships.ts +0 -333
  188. package/src/migrations/services/DataTransformationService.ts +0 -196
  189. package/src/migrations/services/FileHandlerService.ts +0 -311
  190. package/src/migrations/services/ImportOrchestrator.ts +0 -675
  191. package/src/migrations/services/RateLimitManager.ts +0 -363
  192. package/src/migrations/services/RelationshipResolver.ts +0 -461
  193. package/src/migrations/services/UserMappingService.ts +0 -345
  194. package/src/migrations/services/ValidationService.ts +0 -349
  195. package/src/migrations/transfer.ts +0 -1113
  196. package/src/migrations/yaml/YamlImportConfigLoader.ts +0 -439
  197. package/src/migrations/yaml/YamlImportIntegration.ts +0 -446
  198. package/src/migrations/yaml/generateImportSchemas.ts +0 -1354
  199. package/src/schemas/authUser.ts +0 -23
  200. package/src/setupCommands.ts +0 -602
  201. package/src/setupController.ts +0 -43
  202. package/src/shared/backupMetadataSchema.ts +0 -93
  203. package/src/shared/backupTracking.ts +0 -211
  204. package/src/shared/confirmationDialogs.ts +0 -327
  205. package/src/shared/migrationHelpers.ts +0 -232
  206. package/src/shared/operationQueue.ts +0 -376
  207. package/src/shared/operationsTable.ts +0 -338
  208. package/src/shared/operationsTableSchema.ts +0 -60
  209. package/src/shared/progressManager.ts +0 -278
  210. package/src/shared/relationshipExtractor.ts +0 -214
  211. package/src/shared/selectionDialogs.ts +0 -802
  212. package/src/storage/backupCompression.ts +0 -88
  213. package/src/storage/methods.ts +0 -711
  214. package/src/storage/schemas.ts +0 -205
  215. package/src/tables/indexManager.ts +0 -409
  216. package/src/types/node-appwrite-tablesdb.d.ts +0 -44
  217. package/src/users/methods.ts +0 -358
  218. package/src/utils/configMigration.ts +0 -348
  219. package/src/utils/loadConfigs.ts +0 -457
  220. package/src/utils/setupFiles.ts +0 -1236
  221. package/src/utilsController.ts +0 -1263
  222. package/tests/README.md +0 -497
  223. package/tests/adapters/AdapterFactory.test.ts +0 -277
  224. package/tests/integration/syncOperations.test.ts +0 -463
  225. package/tests/jest.config.js +0 -25
  226. package/tests/migration/configMigration.test.ts +0 -546
  227. package/tests/setup.ts +0 -62
  228. package/tests/testUtils.ts +0 -340
  229. package/tests/utils/loadConfigs.test.ts +0 -350
  230. package/tests/validation/configValidation.test.ts +0 -412
  231. package/tsconfig.json +0 -44
  232. /package/{src → dist}/functions/templates/count-docs-in-collection/README.md +0 -0
  233. /package/{src → dist}/functions/templates/count-docs-in-collection/src/main.ts +0 -0
  234. /package/{src → dist}/functions/templates/count-docs-in-collection/src/request.ts +0 -0
  235. /package/{src → dist}/functions/templates/hono-typescript/README.md +0 -0
  236. /package/{src → dist}/functions/templates/hono-typescript/src/adapters/request.ts +0 -0
  237. /package/{src → dist}/functions/templates/hono-typescript/src/adapters/response.ts +0 -0
  238. /package/{src → dist}/functions/templates/hono-typescript/src/app.ts +0 -0
  239. /package/{src → dist}/functions/templates/hono-typescript/src/context.ts +0 -0
  240. /package/{src → dist}/functions/templates/hono-typescript/src/main.ts +0 -0
  241. /package/{src → dist}/functions/templates/hono-typescript/src/middleware/appwrite.ts +0 -0
  242. /package/{src → dist}/functions/templates/typescript-node/README.md +0 -0
  243. /package/{src → dist}/functions/templates/typescript-node/src/context.ts +0 -0
  244. /package/{src → dist}/functions/templates/typescript-node/src/main.ts +0 -0
  245. /package/{src → dist}/functions/templates/uv/README.md +0 -0
  246. /package/{src → dist}/functions/templates/uv/pyproject.toml +0 -0
  247. /package/{src → dist}/functions/templates/uv/src/__init__.py +0 -0
  248. /package/{src → dist}/functions/templates/uv/src/context.py +0 -0
  249. /package/{src → dist}/functions/templates/uv/src/main.py +0 -0
  250. /package/{src/utils/index.ts → dist/utils/index.d.ts} +0 -0
@@ -0,0 +1,287 @@
1
+ import { logger } from 'appwrite-utils-helpers';
2
+ import { tryAwaitWithRetry } from "appwrite-utils-helpers";
3
+ import { Query, ID } from "node-appwrite";
4
+ import { OPERATIONS_TABLE_ID, OPERATIONS_TABLE_NAME, OperationRecordSchema, } from "./operationsTableSchema.js";
5
+ /**
6
+ * Operations Table Manager
7
+ *
8
+ * Manages dynamic operations tracking tables in each database.
9
+ * Creates _appwrite_operations table on-demand for tracking imports, exports, transfers, etc.
10
+ */
11
+ /**
12
+ * Checks if operations table exists in database
13
+ */
14
+ async function tableExists(db, databaseId) {
15
+ try {
16
+ await db.getTable({ databaseId, tableId: OPERATIONS_TABLE_ID });
17
+ return true;
18
+ }
19
+ catch (error) {
20
+ return false;
21
+ }
22
+ }
23
+ /**
24
+ * Creates the operations tracking table in the specified database
25
+ * Table is created with underscore prefix to indicate it's a system table
26
+ */
27
+ export async function createOperationsTable(db, databaseId) {
28
+ const exists = await tableExists(db, databaseId);
29
+ if (!exists) {
30
+ logger.info("Creating operations tracking table", {
31
+ databaseId,
32
+ tableId: OPERATIONS_TABLE_ID,
33
+ });
34
+ await tryAwaitWithRetry(async () => {
35
+ await db.createTable({
36
+ databaseId,
37
+ id: OPERATIONS_TABLE_ID,
38
+ name: OPERATIONS_TABLE_NAME,
39
+ });
40
+ });
41
+ }
42
+ // Always ensure attributes exist (handles partial creation from previous runs)
43
+ const attributes = [
44
+ {
45
+ key: "operationType",
46
+ type: "string",
47
+ size: 255,
48
+ required: true,
49
+ },
50
+ {
51
+ key: "targetTable",
52
+ type: "string",
53
+ size: 255,
54
+ required: false,
55
+ },
56
+ {
57
+ key: "status",
58
+ type: "enum",
59
+ elements: ["pending", "in_progress", "completed", "failed", "cancelled"],
60
+ required: false,
61
+ default: "pending",
62
+ },
63
+ {
64
+ key: "progress",
65
+ type: "integer",
66
+ required: false,
67
+ default: 0,
68
+ },
69
+ {
70
+ key: "total",
71
+ type: "integer",
72
+ required: false,
73
+ default: 0,
74
+ },
75
+ {
76
+ key: "data",
77
+ type: "string",
78
+ size: 1048576, // 1MB for serialized data
79
+ required: false,
80
+ },
81
+ {
82
+ key: "error",
83
+ type: "string",
84
+ size: 65535,
85
+ required: false,
86
+ },
87
+ ];
88
+ for (const attr of attributes) {
89
+ try {
90
+ await db.createAttribute({
91
+ databaseId,
92
+ tableId: OPERATIONS_TABLE_ID,
93
+ ...attr,
94
+ });
95
+ }
96
+ catch (error) {
97
+ // Attribute may already exist from a previous run — that's fine, skip it
98
+ const msg = error instanceof Error ? error.message : String(error);
99
+ if (msg.includes("already exists") || msg.includes("column_already_exists")) {
100
+ continue;
101
+ }
102
+ throw error;
103
+ }
104
+ }
105
+ logger.info("Operations tracking table ready", {
106
+ databaseId,
107
+ tableId: OPERATIONS_TABLE_ID,
108
+ attributes: attributes.length,
109
+ });
110
+ }
111
+ /**
112
+ * Finds an existing operation or creates a new one
113
+ * Useful for resuming interrupted operations
114
+ */
115
+ export async function findOrCreateOperation(db, databaseId, operationType, params) {
116
+ // Ensure operations table exists
117
+ await createOperationsTable(db, databaseId);
118
+ // Try to find existing pending operation
119
+ try {
120
+ const queries = [
121
+ Query.equal("operationType", operationType),
122
+ Query.equal("status", "pending"),
123
+ ];
124
+ if (params?.targetTable) {
125
+ queries.push(Query.equal("targetTable", params.targetTable));
126
+ }
127
+ const response = await db.listRows({
128
+ databaseId,
129
+ tableId: OPERATIONS_TABLE_ID,
130
+ queries,
131
+ });
132
+ if (response.rows && response.rows.length > 0) {
133
+ logger.debug("Found existing operation", {
134
+ operationId: response.rows[0].$id,
135
+ operationType,
136
+ });
137
+ return response.rows[0];
138
+ }
139
+ }
140
+ catch (error) {
141
+ logger.debug("No existing operation found, creating new one", {
142
+ operationType,
143
+ error: error instanceof Error ? error.message : String(error),
144
+ });
145
+ }
146
+ // Create new operation
147
+ const newOperation = {
148
+ operationType,
149
+ targetTable: params?.targetTable,
150
+ status: "pending",
151
+ progress: params?.progress ?? 0,
152
+ total: params?.total ?? 0,
153
+ data: params?.data ? JSON.stringify(params.data) : undefined,
154
+ error: params?.error,
155
+ };
156
+ const result = await db.createRow({
157
+ databaseId,
158
+ tableId: OPERATIONS_TABLE_ID,
159
+ id: ID.unique(),
160
+ data: newOperation,
161
+ });
162
+ logger.info("Created new operation", {
163
+ operationId: result.data?.$id,
164
+ operationType,
165
+ });
166
+ return result.data;
167
+ }
168
+ /**
169
+ * Updates an existing operation record
170
+ */
171
+ export async function updateOperation(db, databaseId, operationId, updates) {
172
+ // Prepare update data (exclude system fields like $id, $createdAt, $updatedAt)
173
+ const updateData = {};
174
+ if (updates.operationType !== undefined)
175
+ updateData.operationType = updates.operationType;
176
+ if (updates.targetTable !== undefined)
177
+ updateData.targetTable = updates.targetTable;
178
+ if (updates.status !== undefined)
179
+ updateData.status = updates.status;
180
+ if (updates.progress !== undefined)
181
+ updateData.progress = updates.progress;
182
+ if (updates.total !== undefined)
183
+ updateData.total = updates.total;
184
+ if (updates.error !== undefined)
185
+ updateData.error = updates.error;
186
+ if (updates.data !== undefined) {
187
+ updateData.data =
188
+ typeof updates.data === "string"
189
+ ? updates.data
190
+ : JSON.stringify(updates.data);
191
+ }
192
+ try {
193
+ const result = await db.updateRow({
194
+ databaseId,
195
+ tableId: OPERATIONS_TABLE_ID,
196
+ id: operationId,
197
+ data: updateData,
198
+ });
199
+ logger.debug("Updated operation", {
200
+ operationId,
201
+ updates: Object.keys(updateData),
202
+ });
203
+ return result.data;
204
+ }
205
+ catch (error) {
206
+ logger.error("Failed to update operation", {
207
+ operationId,
208
+ error: error instanceof Error ? error.message : String(error),
209
+ });
210
+ throw error;
211
+ }
212
+ }
213
+ /**
214
+ * Gets a single operation by ID
215
+ */
216
+ export async function getOperation(db, databaseId, operationId) {
217
+ try {
218
+ const result = await db.getRow({
219
+ databaseId,
220
+ tableId: OPERATIONS_TABLE_ID,
221
+ id: operationId,
222
+ });
223
+ return result.data;
224
+ }
225
+ catch (error) {
226
+ logger.debug("Operation not found", {
227
+ operationId,
228
+ error: error instanceof Error ? error.message : String(error),
229
+ });
230
+ return null;
231
+ }
232
+ }
233
+ /**
234
+ * Cleans up old completed operations
235
+ * @param olderThan - Optional date, defaults to operations older than 7 days
236
+ * @returns Number of operations deleted
237
+ */
238
+ export async function cleanupOperations(db, databaseId, olderThan) {
239
+ const cutoffDate = olderThan || new Date(Date.now() - 7 * 24 * 60 * 60 * 1000); // 7 days ago
240
+ const cutoffIso = cutoffDate.toISOString();
241
+ try {
242
+ // Query for completed operations older than cutoff
243
+ const response = await db.listRows({
244
+ databaseId,
245
+ tableId: OPERATIONS_TABLE_ID,
246
+ queries: [
247
+ Query.equal("status", ["completed", "failed", "cancelled"]),
248
+ Query.lessThan("$createdAt", cutoffIso),
249
+ ],
250
+ });
251
+ if (!response.rows || response.rows.length === 0) {
252
+ logger.debug("No old operations to clean up", { databaseId });
253
+ return 0;
254
+ }
255
+ // Delete in batches
256
+ let deletedCount = 0;
257
+ for (const operation of response.rows) {
258
+ try {
259
+ await db.deleteRow({
260
+ databaseId,
261
+ tableId: OPERATIONS_TABLE_ID,
262
+ id: operation.$id,
263
+ });
264
+ deletedCount++;
265
+ }
266
+ catch (error) {
267
+ logger.warn("Failed to delete operation", {
268
+ operationId: operation.$id,
269
+ error: error instanceof Error ? error.message : String(error),
270
+ });
271
+ }
272
+ }
273
+ logger.info("Cleaned up old operations", {
274
+ databaseId,
275
+ deletedCount,
276
+ cutoffDate: cutoffIso,
277
+ });
278
+ return deletedCount;
279
+ }
280
+ catch (error) {
281
+ logger.error("Failed to cleanup operations", {
282
+ databaseId,
283
+ error: error instanceof Error ? error.message : String(error),
284
+ });
285
+ return 0;
286
+ }
287
+ }
@@ -0,0 +1,48 @@
1
+ /**
2
+ * Schema for operation tracking table (_appwrite_operations)
3
+ *
4
+ * This table is created dynamically in each database to track long-running operations
5
+ * like imports, exports, transfers, and backups.
6
+ */
7
+ import { z } from "zod";
8
+ export interface OperationRecord {
9
+ $id: string;
10
+ $createdAt: string;
11
+ $updatedAt: string;
12
+ operationType: string;
13
+ targetTable?: string;
14
+ status: OperationStatus;
15
+ progress: number;
16
+ total: number;
17
+ data?: any;
18
+ error?: string;
19
+ }
20
+ export type OperationStatus = 'pending' | 'in_progress' | 'completed' | 'failed' | 'cancelled';
21
+ export declare const OPERATION_STATUSES: readonly ["pending", "in_progress", "completed", "failed", "cancelled"];
22
+ export declare const OperationStatusSchema: z.ZodEnum<{
23
+ pending: "pending";
24
+ in_progress: "in_progress";
25
+ completed: "completed";
26
+ cancelled: "cancelled";
27
+ failed: "failed";
28
+ }>;
29
+ export declare const OperationRecordSchema: z.ZodObject<{
30
+ $id: z.ZodString;
31
+ $createdAt: z.ZodString;
32
+ $updatedAt: z.ZodString;
33
+ operationType: z.ZodString;
34
+ targetTable: z.ZodOptional<z.ZodString>;
35
+ status: z.ZodEnum<{
36
+ pending: "pending";
37
+ in_progress: "in_progress";
38
+ completed: "completed";
39
+ cancelled: "cancelled";
40
+ failed: "failed";
41
+ }>;
42
+ progress: z.ZodNumber;
43
+ total: z.ZodNumber;
44
+ data: z.ZodOptional<z.ZodAny>;
45
+ error: z.ZodOptional<z.ZodString>;
46
+ }, z.core.$strip>;
47
+ export declare const OPERATIONS_TABLE_ID = "appwrite_operations";
48
+ export declare const OPERATIONS_TABLE_NAME = "Operations Tracking";
@@ -0,0 +1,35 @@
1
+ /**
2
+ * Schema for operation tracking table (_appwrite_operations)
3
+ *
4
+ * This table is created dynamically in each database to track long-running operations
5
+ * like imports, exports, transfers, and backups.
6
+ */
7
+ import { z } from "zod";
8
+ export const OPERATION_STATUSES = [
9
+ 'pending',
10
+ 'in_progress',
11
+ 'completed',
12
+ 'failed',
13
+ 'cancelled'
14
+ ];
15
+ export const OperationStatusSchema = z.enum([
16
+ 'pending',
17
+ 'in_progress',
18
+ 'completed',
19
+ 'failed',
20
+ 'cancelled'
21
+ ]);
22
+ export const OperationRecordSchema = z.object({
23
+ $id: z.string(),
24
+ $createdAt: z.string(),
25
+ $updatedAt: z.string(),
26
+ operationType: z.string(),
27
+ targetTable: z.string().optional(),
28
+ status: OperationStatusSchema,
29
+ progress: z.number(),
30
+ total: z.number(),
31
+ data: z.any().optional(),
32
+ error: z.string().optional()
33
+ });
34
+ export const OPERATIONS_TABLE_ID = "appwrite_operations";
35
+ export const OPERATIONS_TABLE_NAME = "Operations Tracking";
@@ -0,0 +1,62 @@
1
+ export interface ProgressOptions {
2
+ title?: string;
3
+ showETA?: boolean;
4
+ showSpeed?: boolean;
5
+ showPercentage?: boolean;
6
+ width?: number;
7
+ format?: string;
8
+ }
9
+ export declare class ProgressManager {
10
+ private static instances;
11
+ private bar;
12
+ private startTime;
13
+ private totalItems;
14
+ private completed;
15
+ private id;
16
+ private title;
17
+ private constructor();
18
+ static create(id: string, total: number, options?: ProgressOptions): ProgressManager;
19
+ static get(id: string): ProgressManager | undefined;
20
+ update(current: number, detail?: string): void;
21
+ increment(amount?: number, detail?: string): void;
22
+ setTotal(total: number): void;
23
+ stop(showSummary?: boolean): void;
24
+ fail(error: string): void;
25
+ getStats(): {
26
+ completed: number;
27
+ total: number;
28
+ percentage: number;
29
+ duration: number;
30
+ rate: number;
31
+ remaining: number;
32
+ eta: number;
33
+ };
34
+ static stopAll(): void;
35
+ }
36
+ export declare class MultiProgressManager {
37
+ private multiBar;
38
+ private bars;
39
+ private startTime;
40
+ constructor(options?: ProgressOptions);
41
+ addTask(id: string, total: number, title: string): void;
42
+ updateTask(id: string, current: number, detail?: string): void;
43
+ incrementTask(id: string, amount?: number, detail?: string): void;
44
+ completeTask(id: string): void;
45
+ failTask(id: string, error: string): void;
46
+ stop(showSummary?: boolean): void;
47
+ getTaskStats(id: string): {
48
+ completed: number;
49
+ total: number;
50
+ percentage: number;
51
+ remaining: number;
52
+ } | null;
53
+ getAllStats(): Map<string, any>;
54
+ }
55
+ export declare const ProgressUtils: {
56
+ withProgress<T>(id: string, total: number, title: string, operation: (progress: ProgressManager) => Promise<T>): Promise<T>;
57
+ processArrayWithProgress<T, R>(items: T[], processor: (item: T, index: number) => Promise<R>, options?: {
58
+ title?: string;
59
+ batchSize?: number;
60
+ showDetail?: boolean;
61
+ }): Promise<R[]>;
62
+ };
@@ -0,0 +1,215 @@
1
+ import cliProgress from "cli-progress";
2
+ import chalk from "chalk";
3
+ import { MessageFormatter } from 'appwrite-utils-helpers';
4
+ export class ProgressManager {
5
+ static instances = new Map();
6
+ bar;
7
+ startTime;
8
+ totalItems;
9
+ completed = 0;
10
+ id;
11
+ title;
12
+ constructor(id, total, options = {}) {
13
+ this.id = id;
14
+ this.totalItems = total;
15
+ this.startTime = Date.now();
16
+ this.title = options.title || "Processing";
17
+ const format = options.format ||
18
+ `${chalk.cyan(this.title)} ${chalk.yellow("[{bar}]")} {percentage}% | {value}/{total} | ETA: {eta}s | Speed: {speed}/s`;
19
+ this.bar = new cliProgress.SingleBar({
20
+ format,
21
+ barCompleteChar: '█',
22
+ barIncompleteChar: '░',
23
+ hideCursor: true,
24
+ clearOnComplete: false,
25
+ stopOnComplete: true,
26
+ ...options,
27
+ }, cliProgress.Presets.shades_classic);
28
+ this.bar.start(total, 0);
29
+ }
30
+ static create(id, total, options = {}) {
31
+ if (ProgressManager.instances.has(id)) {
32
+ const existing = ProgressManager.instances.get(id);
33
+ existing.stop();
34
+ }
35
+ const instance = new ProgressManager(id, total, options);
36
+ ProgressManager.instances.set(id, instance);
37
+ return instance;
38
+ }
39
+ static get(id) {
40
+ return ProgressManager.instances.get(id);
41
+ }
42
+ update(current, detail) {
43
+ this.completed = current;
44
+ // Calculate speed
45
+ const elapsed = (Date.now() - this.startTime) / 1000;
46
+ const speed = elapsed > 0 ? Math.round(current / elapsed) : 0;
47
+ this.bar.update(current, {
48
+ speed,
49
+ detail: detail || '',
50
+ });
51
+ if (detail) {
52
+ // Update the payload for custom formatting
53
+ this.bar.update(current, { detail });
54
+ }
55
+ }
56
+ increment(amount = 1, detail) {
57
+ this.update(this.completed + amount, detail);
58
+ }
59
+ setTotal(total) {
60
+ this.totalItems = total;
61
+ this.bar.setTotal(total);
62
+ }
63
+ stop(showSummary = true) {
64
+ this.bar.stop();
65
+ if (showSummary) {
66
+ const duration = Date.now() - this.startTime;
67
+ const rate = this.completed / (duration / 1000);
68
+ MessageFormatter.success(`${this.title} completed`, {
69
+ prefix: `${this.completed}/${this.totalItems} items in ${MessageFormatter.formatDuration(duration)} (${rate.toFixed(1)}/s)`
70
+ });
71
+ }
72
+ ProgressManager.instances.delete(this.id);
73
+ }
74
+ fail(error) {
75
+ this.bar.stop();
76
+ MessageFormatter.error(`${this.title} failed: ${error}`);
77
+ ProgressManager.instances.delete(this.id);
78
+ }
79
+ getStats() {
80
+ const duration = Date.now() - this.startTime;
81
+ const rate = this.completed / (duration / 1000);
82
+ return {
83
+ completed: this.completed,
84
+ total: this.totalItems,
85
+ percentage: (this.completed / this.totalItems) * 100,
86
+ duration,
87
+ rate,
88
+ remaining: this.totalItems - this.completed,
89
+ eta: this.completed > 0 ? ((this.totalItems - this.completed) / rate) * 1000 : 0,
90
+ };
91
+ }
92
+ static stopAll() {
93
+ for (const [id, instance] of ProgressManager.instances) {
94
+ instance.stop(false);
95
+ }
96
+ ProgressManager.instances.clear();
97
+ }
98
+ }
99
+ export class MultiProgressManager {
100
+ multiBar;
101
+ bars = new Map();
102
+ startTime;
103
+ constructor(options = {}) {
104
+ this.startTime = Date.now();
105
+ this.multiBar = new cliProgress.MultiBar({
106
+ clearOnComplete: false,
107
+ hideCursor: true,
108
+ format: options.format || `${chalk.cyan("{title}")} ${chalk.yellow("[{bar}]")} {percentage}% | {value}/{total} | {detail}`,
109
+ barCompleteChar: '█',
110
+ barIncompleteChar: '░',
111
+ }, cliProgress.Presets.shades_classic);
112
+ }
113
+ addTask(id, total, title) {
114
+ const bar = this.multiBar.create(total, 0, {
115
+ title: title.padEnd(20),
116
+ detail: '',
117
+ });
118
+ this.bars.set(id, bar);
119
+ }
120
+ updateTask(id, current, detail) {
121
+ const bar = this.bars.get(id);
122
+ if (bar) {
123
+ bar.update(current, {
124
+ detail: detail || '',
125
+ });
126
+ }
127
+ }
128
+ incrementTask(id, amount = 1, detail) {
129
+ const bar = this.bars.get(id);
130
+ if (bar) {
131
+ const currentValue = bar.getProgress() * bar.getTotal();
132
+ this.updateTask(id, currentValue + amount, detail);
133
+ }
134
+ }
135
+ completeTask(id) {
136
+ const bar = this.bars.get(id);
137
+ if (bar) {
138
+ bar.update(bar.getTotal());
139
+ }
140
+ }
141
+ failTask(id, error) {
142
+ const bar = this.bars.get(id);
143
+ if (bar) {
144
+ bar.update(bar.getProgress() * bar.getTotal(), {
145
+ detail: chalk.red(`Failed: ${error}`),
146
+ });
147
+ }
148
+ }
149
+ stop(showSummary = true) {
150
+ this.multiBar.stop();
151
+ if (showSummary) {
152
+ const duration = Date.now() - this.startTime;
153
+ MessageFormatter.success(`All tasks completed in ${MessageFormatter.formatDuration(duration)}`);
154
+ }
155
+ }
156
+ getTaskStats(id) {
157
+ const bar = this.bars.get(id);
158
+ if (!bar)
159
+ return null;
160
+ const progress = bar.getProgress();
161
+ const total = bar.getTotal();
162
+ const completed = Math.floor(progress * total);
163
+ return {
164
+ completed,
165
+ total,
166
+ percentage: progress * 100,
167
+ remaining: total - completed,
168
+ };
169
+ }
170
+ getAllStats() {
171
+ const stats = new Map();
172
+ for (const [id, bar] of this.bars) {
173
+ stats.set(id, this.getTaskStats(id));
174
+ }
175
+ return stats;
176
+ }
177
+ }
178
+ // Utility functions for common progress scenarios
179
+ export const ProgressUtils = {
180
+ async withProgress(id, total, title, operation) {
181
+ const progress = ProgressManager.create(id, total, { title });
182
+ try {
183
+ const result = await operation(progress);
184
+ progress.stop();
185
+ return result;
186
+ }
187
+ catch (error) {
188
+ progress.fail(error instanceof Error ? error.message : String(error));
189
+ throw error;
190
+ }
191
+ },
192
+ async processArrayWithProgress(items, processor, options = {}) {
193
+ const { title = "Processing items", batchSize = 1, showDetail = true } = options;
194
+ const progress = ProgressManager.create(`process-${Date.now()}`, items.length, { title });
195
+ const results = [];
196
+ try {
197
+ for (let i = 0; i < items.length; i += batchSize) {
198
+ const batch = items.slice(i, i + batchSize);
199
+ const batchResults = await Promise.all(batch.map(async (item, batchIndex) => {
200
+ const result = await processor(item, i + batchIndex);
201
+ const detail = showDetail ? `Item ${i + batchIndex + 1}: ${String(item).slice(0, 30)}...` : undefined;
202
+ progress.update(i + batchIndex + 1, detail);
203
+ return result;
204
+ }));
205
+ results.push(...batchResults);
206
+ }
207
+ progress.stop();
208
+ return results;
209
+ }
210
+ catch (error) {
211
+ progress.fail(error instanceof Error ? error.message : String(error));
212
+ throw error;
213
+ }
214
+ },
215
+ };
@@ -0,0 +1,56 @@
1
+ import type { AppwriteConfig, Attribute, RelationshipAttribute } from "appwrite-utils";
2
+ /**
3
+ * Represents detailed information about a two-way relationship between collections
4
+ */
5
+ export interface RelationshipDetail {
6
+ parentCollection: string;
7
+ childCollection: string;
8
+ parentKey: string;
9
+ childKey: string;
10
+ isArray: boolean;
11
+ isChild: boolean;
12
+ }
13
+ /**
14
+ * Represents basic relationship information for JSON schema generation
15
+ */
16
+ export interface SimpleRelationship {
17
+ attributeKey: string;
18
+ relatedCollection: string;
19
+ relationType: string;
20
+ isArray: boolean;
21
+ }
22
+ /**
23
+ * Helper function to resolve collection name from ID or name
24
+ * @param config - Appwrite configuration containing collections
25
+ * @param idOrName - Collection ID or name to resolve
26
+ * @returns Resolved collection name, or the original input if not found
27
+ */
28
+ export declare function resolveCollectionName(config: AppwriteConfig, idOrName: string): string;
29
+ /**
30
+ * Determines if a relationship type results in an array
31
+ * @param relationType - The type of relationship (oneToOne, oneToMany, manyToOne, manyToMany)
32
+ * @returns true if the relationship results in an array
33
+ */
34
+ export declare function isArrayRelationship(relationType: string): boolean;
35
+ /**
36
+ * Extracts two-way relationship details from collections for Zod schema generation
37
+ * This handles complex bidirectional relationships with parent-child tracking
38
+ *
39
+ * @param config - Appwrite configuration containing collections
40
+ * @returns Map of collection names to their relationship details
41
+ */
42
+ export declare function extractTwoWayRelationships(config: AppwriteConfig): Map<string, RelationshipDetail[]>;
43
+ /**
44
+ * Extracts simple relationship information from collections for JSON schema generation
45
+ * This handles one-way and basic relationship tracking
46
+ *
47
+ * @param config - Appwrite configuration containing collections
48
+ * @returns Map of collection names to their simple relationships
49
+ */
50
+ export declare function extractSimpleRelationships(config: AppwriteConfig): Map<string, SimpleRelationship[]>;
51
+ /**
52
+ * Extracts all relationship attributes from a collection's attributes
53
+ * @param attributes - Array of collection attributes
54
+ * @returns Array of relationship attributes only
55
+ */
56
+ export declare function filterRelationshipAttributes(attributes: Attribute[]): RelationshipAttribute[];