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,87 @@
1
+ import type { Attribute } from "appwrite-utils";
2
+ interface ColumnPropertyChange {
3
+ property: string;
4
+ oldValue: any;
5
+ newValue: any;
6
+ requiresRecreate: boolean;
7
+ }
8
+ interface ColumnOperationPlan {
9
+ toCreate: Attribute[];
10
+ toUpdate: Array<{
11
+ attribute: Attribute;
12
+ changes: ColumnPropertyChange[];
13
+ }>;
14
+ toRecreate: Array<{
15
+ oldAttribute: any;
16
+ newAttribute: Attribute;
17
+ }>;
18
+ toDelete: Array<{
19
+ attribute: any;
20
+ }>;
21
+ unchanged: string[];
22
+ }
23
+ type ComparableColumn = {
24
+ key: string;
25
+ type: string;
26
+ required?: boolean;
27
+ array?: boolean;
28
+ default?: any;
29
+ size?: number;
30
+ min?: number;
31
+ max?: number;
32
+ elements?: string[];
33
+ encrypt?: boolean;
34
+ relatedCollection?: string;
35
+ relationType?: string;
36
+ twoWay?: boolean;
37
+ twoWayKey?: string;
38
+ onDelete?: string;
39
+ side?: string;
40
+ };
41
+ export declare function normalizeAttributeToComparable(attr: Attribute): ComparableColumn;
42
+ export declare function normalizeColumnToComparable(col: any): ComparableColumn;
43
+ export declare function isColumnEqualToColumn(a: any, b: any): boolean;
44
+ export declare function isIndexEqualToIndex(a: any, b: any): boolean;
45
+ /**
46
+ * Enhanced version of columns diff with detailed change analysis
47
+ * Order: desired first, then existing (matches internal usage here)
48
+ * Handles case-insensitive key matches as renames (recreates)
49
+ */
50
+ export declare function diffColumnsDetailed(desiredAttributes: Attribute[], existingColumns: any[]): ColumnOperationPlan;
51
+ /**
52
+ * Returns true if there is any difference between existing columns and desired attributes
53
+ */
54
+ export declare function areTableColumnsDiff(existingColumns: any[], desired: Attribute[]): boolean;
55
+ export declare function diffTableColumns(existingColumns: any[], desired: Attribute[]): {
56
+ toCreate: Attribute[];
57
+ toUpdate: Attribute[];
58
+ unchanged: string[];
59
+ };
60
+ /**
61
+ * Execute the column operation plan using the adapter
62
+ */
63
+ export declare function executeColumnOperations(adapter: any, databaseId: string, tableId: string, plan: ColumnOperationPlan): Promise<{
64
+ success: string[];
65
+ errors: Array<{
66
+ column: string;
67
+ error: string;
68
+ }>;
69
+ }>;
70
+ /**
71
+ * Integration function for methods.ts - processes columns using enhanced logic
72
+ */
73
+ export declare function processTableColumns(adapter: any, databaseId: string, tableId: string, desiredAttributes: Attribute[], existingColumns?: any[]): Promise<{
74
+ totalProcessed: number;
75
+ success: string[];
76
+ errors: Array<{
77
+ column: string;
78
+ error: string;
79
+ }>;
80
+ summary: {
81
+ created: number;
82
+ updated: number;
83
+ recreated: number;
84
+ unchanged: number;
85
+ };
86
+ }>;
87
+ export {};
@@ -0,0 +1,466 @@
1
+ import { mapToCreateAttributeParams, mapToUpdateAttributeParams } from "appwrite-utils-helpers";
2
+ import { Decimal } from "decimal.js";
3
+ const EXTREME_BOUND = new Decimal('1e12');
4
+ // Property configuration for different column types
5
+ const MUTABLE_PROPERTIES = {
6
+ string: ["required", "default", "size", "array"],
7
+ integer: ["required", "default", "min", "max", "array"],
8
+ float: ["required", "default", "min", "max", "array"],
9
+ double: ["required", "default", "min", "max", "array"],
10
+ boolean: ["required", "default", "array"],
11
+ datetime: ["required", "default", "array"],
12
+ email: ["required", "default", "array"],
13
+ ip: ["required", "default", "array"],
14
+ url: ["required", "default", "array"],
15
+ enum: ["required", "default", "elements", "array"],
16
+ relationship: ["required", "default"],
17
+ };
18
+ const IMMUTABLE_PROPERTIES = {
19
+ string: ["encrypt", "key"],
20
+ integer: ["encrypt", "key"],
21
+ float: ["encrypt", "key"],
22
+ double: ["encrypt", "key"],
23
+ boolean: ["key"],
24
+ datetime: ["key"],
25
+ email: ["key"],
26
+ ip: ["key"],
27
+ url: ["key"],
28
+ enum: ["key"],
29
+ relationship: ["key", "relatedCollection", "relationType", "twoWay", "twoWayKey", "onDelete"],
30
+ };
31
+ const TYPE_CHANGE_REQUIRES_RECREATE = [
32
+ "string",
33
+ "integer",
34
+ "float",
35
+ "double",
36
+ "boolean",
37
+ "datetime",
38
+ "email",
39
+ "ip",
40
+ "url",
41
+ "enum",
42
+ "relationship",
43
+ ];
44
+ function normDefault(val) {
45
+ // Treat undefined and null as equal unset default
46
+ return val === undefined ? null : val;
47
+ }
48
+ function toNumber(n) {
49
+ if (n === null || n === undefined)
50
+ return undefined;
51
+ const num = Number(n);
52
+ return Number.isFinite(num) ? num : undefined;
53
+ }
54
+ export function normalizeAttributeToComparable(attr) {
55
+ const t = String(attr.type || '').toLowerCase();
56
+ const base = {
57
+ key: attr.key,
58
+ type: t,
59
+ required: !!attr.required,
60
+ array: !!attr.array,
61
+ default: normDefault(attr.xdefault),
62
+ };
63
+ if (t === 'string') {
64
+ base.size = attr.size ?? 255;
65
+ base.encrypt = !!(attr.encrypt);
66
+ }
67
+ if (t === 'integer' || t === 'float' || t === 'double') {
68
+ const min = toNumber(attr.min);
69
+ const max = toNumber(attr.max);
70
+ if (min !== undefined && max !== undefined) {
71
+ base.min = Math.min(min, max);
72
+ base.max = Math.max(min, max);
73
+ }
74
+ else {
75
+ base.min = min;
76
+ base.max = max;
77
+ }
78
+ }
79
+ if (t === 'enum') {
80
+ base.elements = Array.isArray(attr.elements) ? attr.elements.slice().sort() : [];
81
+ }
82
+ if (t === 'relationship') {
83
+ base.relatedCollection = attr.relatedCollection;
84
+ base.relationType = attr.relationType;
85
+ base.twoWay = !!attr.twoWay;
86
+ base.twoWayKey = attr.twoWayKey;
87
+ base.onDelete = attr.onDelete;
88
+ base.side = attr.side;
89
+ }
90
+ return base;
91
+ }
92
+ export function normalizeColumnToComparable(col) {
93
+ // Detect enum surfaced as string+elements or string+format:enum from server and normalize to enum for comparison
94
+ let t = String((col?.type ?? col?.columnType ?? '')).toLowerCase();
95
+ const hasElements = Array.isArray(col?.elements) && col.elements.length > 0;
96
+ const hasEnumFormat = (col?.format === 'enum');
97
+ if (t === 'string' && (hasElements || hasEnumFormat)) {
98
+ t = 'enum';
99
+ }
100
+ const base = {
101
+ key: col?.key,
102
+ type: t,
103
+ required: !!col?.required,
104
+ array: !!col?.array,
105
+ default: normDefault(col?.default ?? col?.xdefault),
106
+ };
107
+ if (t === 'string') {
108
+ base.size = typeof col?.size === 'number' ? col.size : undefined;
109
+ base.encrypt = !!col?.encrypt;
110
+ }
111
+ if (t === 'integer' || t === 'float' || t === 'double') {
112
+ // Preserve raw min/max without forcing extremes; compare with Decimal in shallowEqual
113
+ const rawMin = col?.min;
114
+ const rawMax = col?.max;
115
+ base.min = rawMin;
116
+ base.max = rawMax;
117
+ }
118
+ if (t === 'enum') {
119
+ base.elements = Array.isArray(col?.elements) ? col.elements.slice().sort() : [];
120
+ }
121
+ if (t === 'relationship') {
122
+ base.relatedCollection = col?.relatedTableId || col?.relatedCollection;
123
+ base.relationType = col?.relationType || col?.typeName;
124
+ base.twoWay = !!col?.twoWay;
125
+ base.twoWayKey = col?.twoWayKey;
126
+ base.onDelete = col?.onDelete;
127
+ base.side = col?.side;
128
+ }
129
+ return base;
130
+ }
131
+ function shallowEqual(a, b) {
132
+ const keys = new Set([...Object.keys(a), ...Object.keys(b)]);
133
+ for (const k of keys) {
134
+ const va = a[k];
135
+ const vb = b[k];
136
+ if (Array.isArray(va) && Array.isArray(vb)) {
137
+ if (va.length !== vb.length)
138
+ return false;
139
+ for (let i = 0; i < va.length; i++)
140
+ if (va[i] !== vb[i])
141
+ return false;
142
+ }
143
+ else if (k === 'min' || k === 'max') {
144
+ // Compare numeric bounds with Decimal to avoid precision issues
145
+ if (va == null && vb == null)
146
+ continue;
147
+ if (va == null || vb == null) {
148
+ // Treat extreme bounds on one side as equivalent to undefined (unbounded)
149
+ const present = va == null ? vb : va;
150
+ try {
151
+ const dp = new Decimal(String(present));
152
+ if (dp.abs().greaterThanOrEqualTo(EXTREME_BOUND))
153
+ continue; // equal
154
+ }
155
+ catch { }
156
+ return false;
157
+ }
158
+ try {
159
+ const da = new Decimal(String(va));
160
+ const db = new Decimal(String(vb));
161
+ if (!da.equals(db))
162
+ return false;
163
+ }
164
+ catch {
165
+ if (va !== vb)
166
+ return false;
167
+ }
168
+ }
169
+ else if (va !== vb) {
170
+ // Treat null and undefined as equal for defaults
171
+ if (!(va == null && vb == null))
172
+ return false;
173
+ }
174
+ }
175
+ return true;
176
+ }
177
+ export function isColumnEqualToColumn(a, b) {
178
+ const na = normalizeColumnToComparable(a);
179
+ const nb = normalizeColumnToComparable(b);
180
+ return shallowEqual(na, nb);
181
+ }
182
+ export function isIndexEqualToIndex(a, b) {
183
+ if (!a || !b)
184
+ return false;
185
+ if (a.key !== b.key)
186
+ return false;
187
+ if (String(a.type).toLowerCase() !== String(b.type).toLowerCase())
188
+ return false;
189
+ // Compare attributes as sets (order-insensitive)
190
+ // Support TablesDB which returns 'columns' instead of 'attributes'
191
+ const attrsAraw = Array.isArray(a.attributes)
192
+ ? a.attributes
193
+ : (Array.isArray(a.columns) ? a.columns : []);
194
+ const attrsA = [...attrsAraw].sort();
195
+ const attrsB = Array.isArray(b.attributes)
196
+ ? [...b.attributes].sort()
197
+ : (Array.isArray(b.columns) ? [...b.columns].sort() : []);
198
+ if (attrsA.length !== attrsB.length)
199
+ return false;
200
+ for (let i = 0; i < attrsA.length; i++)
201
+ if (attrsA[i] !== attrsB[i])
202
+ return false;
203
+ // Orders are only considered if CONFIG (b) has orders defined
204
+ // This prevents false positives when Appwrite returns orders but user didn't specify them
205
+ const hasConfigOrders = Array.isArray(b.orders) && b.orders.length > 0;
206
+ if (hasConfigOrders) {
207
+ // Some APIs may expose 'directions' instead of 'orders'
208
+ const ordersA = Array.isArray(a.orders)
209
+ ? [...a.orders].sort()
210
+ : (Array.isArray(a.directions) ? [...a.directions].sort() : []);
211
+ const ordersB = [...b.orders].sort();
212
+ if (ordersA.length !== ordersB.length)
213
+ return false;
214
+ for (let i = 0; i < ordersA.length; i++)
215
+ if (ordersA[i] !== ordersB[i])
216
+ return false;
217
+ }
218
+ return true;
219
+ }
220
+ /**
221
+ * Compare individual properties between old and new columns
222
+ */
223
+ function compareColumnProperties(oldColumn, newAttribute, columnType) {
224
+ const changes = [];
225
+ const t = String(columnType || newAttribute.type || '').toLowerCase();
226
+ const key = newAttribute?.key || 'unknown';
227
+ const mutableProps = MUTABLE_PROPERTIES[t] || [];
228
+ const immutableProps = IMMUTABLE_PROPERTIES[t] || [];
229
+ const getNewVal = (prop) => {
230
+ const na = newAttribute;
231
+ if (prop === 'default')
232
+ return na.xdefault;
233
+ if (prop === 'encrypt')
234
+ return na.encrypt;
235
+ return na[prop];
236
+ };
237
+ const getOldVal = (prop) => {
238
+ if (prop === 'default')
239
+ return oldColumn?.default ?? oldColumn?.xdefault;
240
+ return oldColumn?.[prop];
241
+ };
242
+ for (const prop of mutableProps) {
243
+ const oldValue = getOldVal(prop);
244
+ let newValue = getNewVal(prop);
245
+ // Special-case: enum elements empty/missing should not trigger updates
246
+ if (t === 'enum' && prop === 'elements') {
247
+ if (!Array.isArray(newValue) || newValue.length === 0) {
248
+ newValue = oldValue;
249
+ }
250
+ }
251
+ if (Array.isArray(oldValue) && Array.isArray(newValue)) {
252
+ if (oldValue.length !== newValue.length || oldValue.some((v, i) => v !== newValue[i])) {
253
+ changes.push({ property: prop, oldValue, newValue, requiresRecreate: false });
254
+ }
255
+ }
256
+ else if (oldValue !== newValue) {
257
+ changes.push({ property: prop, oldValue, newValue, requiresRecreate: false });
258
+ }
259
+ }
260
+ for (const prop of immutableProps) {
261
+ const oldValue = getOldVal(prop);
262
+ const newValue = getNewVal(prop);
263
+ if (Array.isArray(oldValue) && Array.isArray(newValue)) {
264
+ if (oldValue.length !== newValue.length || oldValue.some((v, i) => v !== newValue[i])) {
265
+ changes.push({ property: prop, oldValue, newValue, requiresRecreate: true });
266
+ }
267
+ }
268
+ else if (oldValue !== newValue) {
269
+ changes.push({ property: prop, oldValue, newValue, requiresRecreate: true });
270
+ }
271
+ }
272
+ // Type change requires recreate (normalize string+elements to enum on old side)
273
+ const oldTypeRaw = String(oldColumn?.type || oldColumn?.columnType || '').toLowerCase();
274
+ const oldHasElements = Array.isArray(oldColumn?.elements) && oldColumn.elements.length > 0;
275
+ const oldHasEnumFormat = (oldColumn?.format === 'enum');
276
+ const oldType = oldTypeRaw === 'string' && (oldHasElements || oldHasEnumFormat) ? 'enum' : oldTypeRaw;
277
+ if (oldType && t && oldType !== t && TYPE_CHANGE_REQUIRES_RECREATE.includes(oldType)) {
278
+ changes.push({ property: 'type', oldValue: oldType, newValue: t, requiresRecreate: true });
279
+ }
280
+ return changes;
281
+ }
282
+ /**
283
+ * Analyze what changes are needed for a specific column
284
+ */
285
+ function analyzeColumnChanges(oldColumn, newAttribute) {
286
+ const columnType = String(newAttribute.type || 'string').toLowerCase();
287
+ const columnKey = newAttribute.key;
288
+ // Use normalized comparison to reduce false positives then property-wise details
289
+ const normalizedOld = normalizeColumnToComparable(oldColumn);
290
+ const normalizedNew = normalizeAttributeToComparable(newAttribute);
291
+ const hasAnyDiff = !shallowEqual(normalizedOld, normalizedNew);
292
+ const changes = hasAnyDiff ? compareColumnProperties(oldColumn, newAttribute, columnType) : [];
293
+ const requiresRecreate = changes.some((c) => c.requiresRecreate);
294
+ const mutableChanges = {};
295
+ const immutableChanges = {};
296
+ for (const c of changes) {
297
+ if (c.requiresRecreate)
298
+ immutableChanges[c.property] = { old: c.oldValue, new: c.newValue };
299
+ else
300
+ mutableChanges[c.property] = { old: c.oldValue, new: c.newValue };
301
+ }
302
+ return {
303
+ columnKey,
304
+ columnType,
305
+ hasChanges: changes.length > 0,
306
+ requiresRecreate,
307
+ changes,
308
+ mutableChanges,
309
+ immutableChanges,
310
+ };
311
+ }
312
+ /**
313
+ * Enhanced version of columns diff with detailed change analysis
314
+ * Order: desired first, then existing (matches internal usage here)
315
+ * Handles case-insensitive key matches as renames (recreates)
316
+ */
317
+ export function diffColumnsDetailed(desiredAttributes, existingColumns) {
318
+ // Exact key lookup (case-sensitive)
319
+ const byKey = new Map((existingColumns || []).map((col) => [col?.key, col]));
320
+ // Case-insensitive key lookup for detecting renames
321
+ const byKeyLower = new Map((existingColumns || []).map((col) => [col?.key?.toLowerCase(), col]));
322
+ const toCreate = [];
323
+ const toUpdate = [];
324
+ const toRecreate = [];
325
+ const unchanged = [];
326
+ const handledExistingKeys = new Set(); // Track which existing columns we've handled
327
+ for (const attr of desiredAttributes || []) {
328
+ const key = attr?.key;
329
+ if (!key)
330
+ continue;
331
+ // First try exact match
332
+ const exactMatch = byKey.get(key);
333
+ if (exactMatch) {
334
+ handledExistingKeys.add(key);
335
+ const analysis = analyzeColumnChanges(exactMatch, attr);
336
+ if (!analysis.hasChanges)
337
+ unchanged.push(analysis.columnKey);
338
+ else if (analysis.requiresRecreate)
339
+ toRecreate.push({ oldAttribute: exactMatch, newAttribute: attr });
340
+ else
341
+ toUpdate.push({ attribute: attr, changes: analysis.changes });
342
+ continue;
343
+ }
344
+ // Check for case-insensitive match (rename scenario like oAuthAccounts -> oauthAccounts)
345
+ const caseInsensitiveMatch = byKeyLower.get(key.toLowerCase());
346
+ if (caseInsensitiveMatch && caseInsensitiveMatch.key !== key) {
347
+ // This is a rename - treat as recreate (delete old, create new)
348
+ handledExistingKeys.add(caseInsensitiveMatch.key);
349
+ toRecreate.push({ oldAttribute: caseInsensitiveMatch, newAttribute: attr });
350
+ continue;
351
+ }
352
+ // No match - it's a new attribute
353
+ toCreate.push(attr);
354
+ }
355
+ // Note: we keep toDelete empty for now (conservative behavior)
356
+ // Deletions are handled separately in methods.ts
357
+ return { toCreate, toUpdate, toRecreate, toDelete: [], unchanged };
358
+ }
359
+ /**
360
+ * Returns true if there is any difference between existing columns and desired attributes
361
+ */
362
+ export function areTableColumnsDiff(existingColumns, desired) {
363
+ const byKey = new Map();
364
+ for (const c of existingColumns || []) {
365
+ if (c?.key)
366
+ byKey.set(c.key, c);
367
+ }
368
+ for (const attr of desired || []) {
369
+ const desiredNorm = normalizeAttributeToComparable(attr);
370
+ const existing = byKey.get(desiredNorm.key);
371
+ if (!existing)
372
+ return true;
373
+ const existingNorm = normalizeColumnToComparable(existing);
374
+ if (!shallowEqual(desiredNorm, existingNorm))
375
+ return true;
376
+ }
377
+ // Extra columns on remote also constitute a diff
378
+ const desiredKeys = new Set((desired || []).map((a) => a.key));
379
+ for (const k of byKey.keys())
380
+ if (!desiredKeys.has(k))
381
+ return true;
382
+ return false;
383
+ }
384
+ export function diffTableColumns(existingColumns, desired) {
385
+ // Use detailed plan but return legacy structure for compatibility
386
+ const plan = diffColumnsDetailed(desired, existingColumns);
387
+ const toUpdate = [
388
+ ...plan.toUpdate.map((u) => u.attribute),
389
+ ...plan.toRecreate.map((r) => r.newAttribute),
390
+ ];
391
+ return { toCreate: plan.toCreate, toUpdate, unchanged: plan.unchanged };
392
+ }
393
+ /**
394
+ * Execute the column operation plan using the adapter
395
+ */
396
+ export async function executeColumnOperations(adapter, databaseId, tableId, plan) {
397
+ if (!databaseId || !tableId)
398
+ throw new Error('Database ID and Table ID are required for column operations');
399
+ if (!adapter || typeof adapter.createAttribute !== 'function')
400
+ throw new Error('Valid adapter is required for column operations');
401
+ const results = { success: [], errors: [] };
402
+ const exec = async (fn, key, op) => {
403
+ try {
404
+ await fn();
405
+ results.success.push(`${op}: ${key}`);
406
+ }
407
+ catch (e) {
408
+ results.errors.push({ column: key, error: `${op} failed: ${e?.message || String(e)}` });
409
+ }
410
+ };
411
+ for (const attr of plan.toCreate) {
412
+ const params = mapToCreateAttributeParams(attr, { databaseId, tableId });
413
+ await exec(() => adapter.createAttribute(params), attr.key, 'CREATE');
414
+ }
415
+ for (const { attribute } of plan.toUpdate) {
416
+ const params = mapToUpdateAttributeParams(attribute, { databaseId, tableId });
417
+ await exec(() => adapter.updateAttribute(params), attribute.key, 'UPDATE');
418
+ }
419
+ for (const { oldAttribute, newAttribute } of plan.toRecreate) {
420
+ await exec(() => adapter.deleteAttribute({ databaseId, tableId, key: oldAttribute.key }), oldAttribute.key, 'DELETE (for recreate)');
421
+ // Wait until the attribute is actually removed (or no longer 'deleting') before recreating
422
+ try {
423
+ const start = Date.now();
424
+ const maxWaitMs = 60000; // 60s
425
+ while (Date.now() - start < maxWaitMs) {
426
+ try {
427
+ const tableRes = await adapter.getTable({ databaseId, tableId });
428
+ const cols = (tableRes?.data?.columns || tableRes?.data?.attributes || []);
429
+ const found = cols.find((c) => c.key === oldAttribute.key);
430
+ if (!found)
431
+ break; // fully removed
432
+ if (found.status && found.status !== 'deleting')
433
+ break; // no longer deleting (failed/stuck) -> stop waiting
434
+ }
435
+ catch { }
436
+ await new Promise((r) => setTimeout(r, 1500));
437
+ }
438
+ }
439
+ catch { }
440
+ const params = mapToCreateAttributeParams(newAttribute, { databaseId, tableId });
441
+ await exec(() => adapter.createAttribute(params), newAttribute.key, 'CREATE (after recreate)');
442
+ }
443
+ return results;
444
+ }
445
+ /**
446
+ * Integration function for methods.ts - processes columns using enhanced logic
447
+ */
448
+ export async function processTableColumns(adapter, databaseId, tableId, desiredAttributes, existingColumns = []) {
449
+ if (!existingColumns || existingColumns.length === 0) {
450
+ const tableInfo = await adapter.getTable({ databaseId, tableId });
451
+ existingColumns = (tableInfo?.data?.columns || tableInfo?.data?.attributes || []);
452
+ }
453
+ const plan = diffColumnsDetailed(desiredAttributes, existingColumns);
454
+ const results = await executeColumnOperations(adapter, databaseId, tableId, plan);
455
+ return {
456
+ totalProcessed: plan.toCreate.length + plan.toUpdate.length + plan.toRecreate.length,
457
+ success: results.success,
458
+ errors: results.errors,
459
+ summary: {
460
+ created: plan.toCreate.length,
461
+ updated: plan.toUpdate.length,
462
+ recreated: plan.toRecreate.length,
463
+ unchanged: plan.unchanged.length,
464
+ },
465
+ };
466
+ }
@@ -0,0 +1,8 @@
1
+ import { Databases } from "node-appwrite";
2
+ import type { DatabaseAdapter } from "appwrite-utils-helpers";
3
+ /**
4
+ * Transfers all documents from one collection to another in a different database
5
+ * within the same Appwrite Project
6
+ */
7
+ export declare const transferDocumentsBetweenDbsLocalToLocal: (db: Databases | DatabaseAdapter, fromDbId: string, toDbId: string, fromCollId: string, toCollId: string) => Promise<void>;
8
+ export declare const transferDocumentsBetweenDbsLocalToRemote: (localDb: Databases | DatabaseAdapter, endpoint: string, projectId: string, apiKey: string, fromDbId: string, toDbId: string, fromCollId: string, toCollId: string) => Promise<void>;