@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.
- package/CHANGELOG.md +16 -0
- package/CONFIG_TODO.md +1189 -0
- package/SELECTION_DIALOGS.md +146 -0
- package/SERVICE_IMPLEMENTATION_REPORT.md +462 -0
- package/dist/adapters/index.d.ts +7 -8
- package/dist/adapters/index.js +7 -9
- package/dist/backups/operations/bucketBackup.js +2 -2
- package/dist/backups/operations/collectionBackup.d.ts +1 -1
- package/dist/backups/operations/collectionBackup.js +3 -3
- package/dist/backups/operations/comprehensiveBackup.d.ts +1 -1
- package/dist/backups/operations/comprehensiveBackup.js +2 -2
- package/dist/backups/tracking/centralizedTracking.d.ts +1 -1
- package/dist/backups/tracking/centralizedTracking.js +2 -2
- package/dist/cli/commands/configCommands.js +51 -7
- package/dist/cli/commands/databaseCommands.d.ts +1 -0
- package/dist/cli/commands/databaseCommands.js +119 -9
- package/dist/cli/commands/functionCommands.js +3 -3
- package/dist/cli/commands/importFileCommands.d.ts +7 -0
- package/dist/cli/commands/importFileCommands.js +674 -0
- package/dist/cli/commands/schemaCommands.js +3 -3
- package/dist/cli/commands/storageCommands.js +2 -3
- package/dist/cli/commands/transferCommands.js +3 -5
- package/dist/collections/attributes.d.ts +1 -1
- package/dist/collections/attributes.js +85 -35
- package/dist/collections/indexes.js +2 -4
- package/dist/collections/methods.d.ts +1 -1
- package/dist/collections/methods.js +111 -192
- package/dist/collections/tableOperations.d.ts +1 -0
- package/dist/collections/tableOperations.js +90 -23
- package/dist/collections/transferOperations.d.ts +1 -1
- package/dist/collections/transferOperations.js +3 -4
- package/dist/collections/wipeOperations.d.ts +4 -3
- package/dist/collections/wipeOperations.js +112 -39
- package/dist/databases/methods.js +2 -2
- package/dist/databases/setup.js +2 -2
- package/dist/examples/yamlTerminologyExample.js +2 -2
- package/dist/functions/deployments.d.ts +1 -1
- package/dist/functions/deployments.js +5 -5
- package/dist/functions/fnConfigDiscovery.js +2 -2
- package/dist/functions/methods.js +16 -4
- package/dist/init.js +1 -1
- package/dist/interactiveCLI.d.ts +6 -1
- package/dist/interactiveCLI.js +64 -10
- package/dist/main.js +130 -177
- package/dist/migrations/afterImportActions.js +2 -3
- package/dist/migrations/appwriteToX.d.ts +97 -1
- package/dist/migrations/appwriteToX.js +9 -7
- package/dist/migrations/comprehensiveTransfer.js +3 -5
- package/dist/migrations/dataLoader.d.ts +194 -2
- package/dist/migrations/dataLoader.js +2 -5
- package/dist/migrations/importController.js +3 -4
- package/dist/migrations/importDataActions.js +3 -3
- package/dist/migrations/relationships.js +1 -2
- package/dist/migrations/services/DataTransformationService.js +2 -2
- package/dist/migrations/services/FileHandlerService.js +1 -1
- package/dist/migrations/services/ImportOrchestrator.js +4 -4
- package/dist/migrations/services/RateLimitManager.js +1 -1
- package/dist/migrations/services/RelationshipResolver.js +1 -1
- package/dist/migrations/services/UserMappingService.js +1 -1
- package/dist/migrations/services/ValidationService.js +1 -1
- package/dist/migrations/transfer.d.ts +8 -4
- package/dist/migrations/transfer.js +106 -55
- package/dist/migrations/yaml/YamlImportConfigLoader.js +1 -1
- package/dist/migrations/yaml/YamlImportIntegration.js +2 -2
- package/dist/migrations/yaml/generateImportSchemas.js +1 -1
- package/dist/setupCommands.d.ts +1 -1
- package/dist/setupCommands.js +5 -6
- package/dist/setupController.js +1 -1
- package/dist/shared/backupTracking.d.ts +1 -1
- package/dist/shared/backupTracking.js +2 -2
- package/dist/shared/confirmationDialogs.js +1 -1
- package/dist/shared/migrationHelpers.d.ts +1 -1
- package/dist/shared/migrationHelpers.js +3 -3
- package/dist/shared/operationQueue.d.ts +1 -1
- package/dist/shared/operationQueue.js +2 -3
- package/dist/shared/operationsTable.d.ts +1 -1
- package/dist/shared/operationsTable.js +2 -2
- package/dist/shared/progressManager.js +1 -1
- package/dist/shared/selectionDialogs.js +9 -8
- package/dist/storage/methods.js +4 -4
- package/dist/storage/schemas.d.ts +386 -2
- package/dist/tables/indexManager.d.ts +65 -0
- package/dist/tables/indexManager.js +294 -0
- package/dist/types.d.ts +2 -2
- package/dist/types.js +1 -1
- package/dist/users/methods.js +2 -3
- package/dist/utils/configMigration.js +1 -1
- package/dist/utils/index.d.ts +1 -1
- package/dist/utils/index.js +1 -1
- package/dist/utils/loadConfigs.d.ts +2 -2
- package/dist/utils/loadConfigs.js +6 -7
- package/dist/utils/setupFiles.js +5 -7
- package/dist/utilsController.d.ts +15 -8
- package/dist/utilsController.js +57 -28
- package/package.json +8 -4
- package/src/adapters/index.ts +8 -34
- package/src/backups/operations/bucketBackup.ts +2 -2
- package/src/backups/operations/collectionBackup.ts +4 -4
- package/src/backups/operations/comprehensiveBackup.ts +3 -3
- package/src/backups/tracking/centralizedTracking.ts +3 -3
- package/src/cli/commands/configCommands.ts +72 -8
- package/src/cli/commands/databaseCommands.ts +161 -9
- package/src/cli/commands/functionCommands.ts +4 -3
- package/src/cli/commands/importFileCommands.ts +815 -0
- package/src/cli/commands/schemaCommands.ts +3 -3
- package/src/cli/commands/storageCommands.ts +2 -3
- package/src/cli/commands/transferCommands.ts +3 -6
- package/src/collections/attributes.ts +155 -39
- package/src/collections/indexes.ts +5 -7
- package/src/collections/methods.ts +115 -150
- package/src/collections/tableOperations.ts +92 -21
- package/src/collections/transferOperations.ts +4 -5
- package/src/collections/wipeOperations.ts +154 -51
- package/src/databases/methods.ts +2 -2
- package/src/databases/setup.ts +2 -2
- package/src/examples/yamlTerminologyExample.ts +2 -2
- package/src/functions/deployments.ts +6 -5
- package/src/functions/fnConfigDiscovery.ts +2 -2
- package/src/functions/methods.ts +19 -6
- package/src/init.ts +1 -1
- package/src/interactiveCLI.ts +78 -13
- package/src/main.ts +143 -287
- package/src/migrations/afterImportActions.ts +2 -3
- package/src/migrations/appwriteToX.ts +12 -8
- package/src/migrations/comprehensiveTransfer.ts +6 -6
- package/src/migrations/dataLoader.ts +2 -5
- package/src/migrations/importController.ts +3 -4
- package/src/migrations/importDataActions.ts +3 -3
- package/src/migrations/relationships.ts +1 -2
- package/src/migrations/services/DataTransformationService.ts +2 -2
- package/src/migrations/services/FileHandlerService.ts +1 -1
- package/src/migrations/services/ImportOrchestrator.ts +4 -4
- package/src/migrations/services/RateLimitManager.ts +1 -1
- package/src/migrations/services/RelationshipResolver.ts +1 -1
- package/src/migrations/services/UserMappingService.ts +1 -1
- package/src/migrations/services/ValidationService.ts +1 -1
- package/src/migrations/transfer.ts +126 -83
- package/src/migrations/yaml/YamlImportConfigLoader.ts +1 -1
- package/src/migrations/yaml/YamlImportIntegration.ts +2 -2
- package/src/migrations/yaml/generateImportSchemas.ts +1 -1
- package/src/setupCommands.ts +5 -6
- package/src/setupController.ts +1 -1
- package/src/shared/backupTracking.ts +3 -3
- package/src/shared/confirmationDialogs.ts +1 -1
- package/src/shared/migrationHelpers.ts +4 -4
- package/src/shared/operationQueue.ts +3 -4
- package/src/shared/operationsTable.ts +3 -3
- package/src/shared/progressManager.ts +1 -1
- package/src/shared/selectionDialogs.ts +9 -8
- package/src/storage/methods.ts +4 -4
- package/src/tables/indexManager.ts +409 -0
- package/src/types.ts +2 -2
- package/src/users/methods.ts +2 -3
- package/src/utils/configMigration.ts +1 -1
- package/src/utils/index.ts +1 -1
- package/src/utils/loadConfigs.ts +15 -7
- package/src/utils/setupFiles.ts +5 -7
- package/src/utilsController.ts +86 -32
- package/dist/adapters/AdapterFactory.d.ts +0 -94
- package/dist/adapters/AdapterFactory.js +0 -405
- package/dist/adapters/DatabaseAdapter.d.ts +0 -233
- package/dist/adapters/DatabaseAdapter.js +0 -50
- package/dist/adapters/LegacyAdapter.d.ts +0 -50
- package/dist/adapters/LegacyAdapter.js +0 -612
- package/dist/adapters/TablesDBAdapter.d.ts +0 -45
- package/dist/adapters/TablesDBAdapter.js +0 -571
- package/dist/config/ConfigManager.d.ts +0 -445
- package/dist/config/ConfigManager.js +0 -625
- package/dist/config/configMigration.d.ts +0 -87
- package/dist/config/configMigration.js +0 -390
- package/dist/config/configValidation.d.ts +0 -66
- package/dist/config/configValidation.js +0 -358
- package/dist/config/index.d.ts +0 -8
- package/dist/config/index.js +0 -7
- package/dist/config/services/ConfigDiscoveryService.d.ts +0 -126
- package/dist/config/services/ConfigDiscoveryService.js +0 -374
- package/dist/config/services/ConfigLoaderService.d.ts +0 -129
- package/dist/config/services/ConfigLoaderService.js +0 -540
- package/dist/config/services/ConfigMergeService.d.ts +0 -208
- package/dist/config/services/ConfigMergeService.js +0 -308
- package/dist/config/services/ConfigValidationService.d.ts +0 -214
- package/dist/config/services/ConfigValidationService.js +0 -310
- package/dist/config/services/SessionAuthService.d.ts +0 -225
- package/dist/config/services/SessionAuthService.js +0 -456
- package/dist/config/services/__tests__/ConfigMergeService.test.d.ts +0 -1
- package/dist/config/services/__tests__/ConfigMergeService.test.js +0 -271
- package/dist/config/services/index.d.ts +0 -13
- package/dist/config/services/index.js +0 -10
- package/dist/config/yamlConfig.d.ts +0 -722
- package/dist/config/yamlConfig.js +0 -702
- package/dist/functions/pathResolution.d.ts +0 -37
- package/dist/functions/pathResolution.js +0 -185
- package/dist/shared/attributeMapper.d.ts +0 -20
- package/dist/shared/attributeMapper.js +0 -203
- package/dist/shared/errorUtils.d.ts +0 -54
- package/dist/shared/errorUtils.js +0 -95
- package/dist/shared/functionManager.d.ts +0 -48
- package/dist/shared/functionManager.js +0 -336
- package/dist/shared/indexManager.d.ts +0 -24
- package/dist/shared/indexManager.js +0 -151
- package/dist/shared/jsonSchemaGenerator.d.ts +0 -50
- package/dist/shared/jsonSchemaGenerator.js +0 -290
- package/dist/shared/logging.d.ts +0 -61
- package/dist/shared/logging.js +0 -116
- package/dist/shared/messageFormatter.d.ts +0 -39
- package/dist/shared/messageFormatter.js +0 -162
- package/dist/shared/pydanticModelGenerator.d.ts +0 -17
- package/dist/shared/pydanticModelGenerator.js +0 -615
- package/dist/shared/schemaGenerator.d.ts +0 -40
- package/dist/shared/schemaGenerator.js +0 -556
- package/dist/utils/ClientFactory.d.ts +0 -87
- package/dist/utils/ClientFactory.js +0 -212
- package/dist/utils/configDiscovery.d.ts +0 -78
- package/dist/utils/configDiscovery.js +0 -472
- package/dist/utils/constantsGenerator.d.ts +0 -31
- package/dist/utils/constantsGenerator.js +0 -321
- package/dist/utils/dataConverters.d.ts +0 -46
- package/dist/utils/dataConverters.js +0 -139
- package/dist/utils/directoryUtils.d.ts +0 -22
- package/dist/utils/directoryUtils.js +0 -59
- package/dist/utils/getClientFromConfig.d.ts +0 -39
- package/dist/utils/getClientFromConfig.js +0 -199
- package/dist/utils/helperFunctions.d.ts +0 -63
- package/dist/utils/helperFunctions.js +0 -156
- package/dist/utils/pathResolvers.d.ts +0 -53
- package/dist/utils/pathResolvers.js +0 -72
- package/dist/utils/projectConfig.d.ts +0 -119
- package/dist/utils/projectConfig.js +0 -171
- package/dist/utils/retryFailedPromises.d.ts +0 -2
- package/dist/utils/retryFailedPromises.js +0 -23
- package/dist/utils/sessionAuth.d.ts +0 -48
- package/dist/utils/sessionAuth.js +0 -164
- package/dist/utils/typeGuards.d.ts +0 -35
- package/dist/utils/typeGuards.js +0 -57
- package/dist/utils/validationRules.d.ts +0 -43
- package/dist/utils/validationRules.js +0 -42
- package/dist/utils/versionDetection.d.ts +0 -58
- package/dist/utils/versionDetection.js +0 -251
- package/dist/utils/yamlConverter.d.ts +0 -100
- package/dist/utils/yamlConverter.js +0 -428
- package/dist/utils/yamlLoader.d.ts +0 -70
- package/dist/utils/yamlLoader.js +0 -267
- package/src/adapters/AdapterFactory.ts +0 -510
- package/src/adapters/DatabaseAdapter.ts +0 -306
- package/src/adapters/LegacyAdapter.ts +0 -841
- package/src/adapters/TablesDBAdapter.ts +0 -773
- package/src/config/ConfigManager.ts +0 -808
- package/src/config/README.md +0 -274
- package/src/config/configMigration.ts +0 -575
- package/src/config/configValidation.ts +0 -445
- package/src/config/index.ts +0 -10
- package/src/config/services/ConfigDiscoveryService.ts +0 -463
- package/src/config/services/ConfigLoaderService.ts +0 -740
- package/src/config/services/ConfigMergeService.ts +0 -388
- package/src/config/services/ConfigValidationService.ts +0 -394
- package/src/config/services/SessionAuthService.ts +0 -565
- package/src/config/services/__tests__/ConfigMergeService.test.ts +0 -351
- package/src/config/services/index.ts +0 -29
- package/src/config/yamlConfig.ts +0 -761
- package/src/functions/pathResolution.ts +0 -227
- package/src/shared/attributeMapper.ts +0 -229
- package/src/shared/errorUtils.ts +0 -110
- package/src/shared/functionManager.ts +0 -525
- package/src/shared/indexManager.ts +0 -254
- package/src/shared/jsonSchemaGenerator.ts +0 -383
- package/src/shared/logging.ts +0 -149
- package/src/shared/messageFormatter.ts +0 -208
- package/src/shared/pydanticModelGenerator.ts +0 -618
- package/src/shared/schemaGenerator.ts +0 -644
- package/src/utils/ClientFactory.ts +0 -240
- package/src/utils/configDiscovery.ts +0 -557
- package/src/utils/constantsGenerator.ts +0 -369
- package/src/utils/dataConverters.ts +0 -159
- package/src/utils/directoryUtils.ts +0 -61
- package/src/utils/getClientFromConfig.ts +0 -257
- package/src/utils/helperFunctions.ts +0 -228
- package/src/utils/pathResolvers.ts +0 -81
- package/src/utils/projectConfig.ts +0 -299
- package/src/utils/retryFailedPromises.ts +0 -29
- package/src/utils/sessionAuth.ts +0 -230
- package/src/utils/typeGuards.ts +0 -65
- package/src/utils/validationRules.ts +0 -88
- package/src/utils/versionDetection.ts +0 -292
- package/src/utils/yamlConverter.ts +0 -542
- package/src/utils/yamlLoader.ts +0 -371
|
@@ -0,0 +1,294 @@
|
|
|
1
|
+
import { isIndexEqualToIndex } from "../collections/tableOperations.js";
|
|
2
|
+
import { MessageFormatter } from "@njdamstra/appwrite-utils-helpers";
|
|
3
|
+
import { delay, tryAwaitWithRetry } from "@njdamstra/appwrite-utils-helpers";
|
|
4
|
+
/**
|
|
5
|
+
* Plan index operations by comparing desired indexes with existing ones
|
|
6
|
+
* Uses the existing isIndexEqualToIndex function for consistent comparison
|
|
7
|
+
*/
|
|
8
|
+
export function planIndexOperations(desiredIndexes, existingIndexes) {
|
|
9
|
+
const plan = {
|
|
10
|
+
toCreate: [],
|
|
11
|
+
toUpdate: [],
|
|
12
|
+
toSkip: [],
|
|
13
|
+
toDelete: []
|
|
14
|
+
};
|
|
15
|
+
for (const desiredIndex of desiredIndexes) {
|
|
16
|
+
const existingIndex = existingIndexes.find(idx => idx.key === desiredIndex.key);
|
|
17
|
+
if (!existingIndex) {
|
|
18
|
+
// Index doesn't exist - create it
|
|
19
|
+
plan.toCreate.push({
|
|
20
|
+
type: 'create',
|
|
21
|
+
index: desiredIndex,
|
|
22
|
+
reason: 'New index'
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
else if (isIndexEqualToIndex(existingIndex, desiredIndex)) {
|
|
26
|
+
// Index exists and is identical - skip it
|
|
27
|
+
plan.toSkip.push({
|
|
28
|
+
type: 'skip',
|
|
29
|
+
index: desiredIndex,
|
|
30
|
+
existingIndex,
|
|
31
|
+
reason: 'Index unchanged'
|
|
32
|
+
});
|
|
33
|
+
}
|
|
34
|
+
else {
|
|
35
|
+
// Index exists but is different - update it
|
|
36
|
+
plan.toUpdate.push({
|
|
37
|
+
type: 'update',
|
|
38
|
+
index: desiredIndex,
|
|
39
|
+
existingIndex,
|
|
40
|
+
reason: 'Index configuration changed'
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
return plan;
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Plan index deletions for indexes that exist but aren't in the desired configuration
|
|
48
|
+
*/
|
|
49
|
+
export function planIndexDeletions(desiredIndexKeys, existingIndexes) {
|
|
50
|
+
const deletions = [];
|
|
51
|
+
for (const existingIndex of existingIndexes) {
|
|
52
|
+
if (!desiredIndexKeys.has(existingIndex.key)) {
|
|
53
|
+
deletions.push({
|
|
54
|
+
type: 'delete',
|
|
55
|
+
index: existingIndex, // Convert Models.Index to Index for compatibility
|
|
56
|
+
reason: 'Obsolete index'
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
return deletions;
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Execute index operations with proper error handling and status monitoring
|
|
64
|
+
*/
|
|
65
|
+
export async function executeIndexOperations(adapter, databaseId, tableId, plan) {
|
|
66
|
+
const result = {
|
|
67
|
+
created: [],
|
|
68
|
+
updated: [],
|
|
69
|
+
skipped: [],
|
|
70
|
+
deleted: [],
|
|
71
|
+
errors: [],
|
|
72
|
+
summary: {
|
|
73
|
+
total: 0,
|
|
74
|
+
created: 0,
|
|
75
|
+
updated: 0,
|
|
76
|
+
skipped: 0,
|
|
77
|
+
deleted: 0,
|
|
78
|
+
errors: 0
|
|
79
|
+
}
|
|
80
|
+
};
|
|
81
|
+
// Execute creates
|
|
82
|
+
for (const operation of plan.toCreate) {
|
|
83
|
+
try {
|
|
84
|
+
await adapter.createIndex({
|
|
85
|
+
databaseId,
|
|
86
|
+
tableId,
|
|
87
|
+
key: operation.index.key,
|
|
88
|
+
type: operation.index.type,
|
|
89
|
+
attributes: operation.index.attributes,
|
|
90
|
+
orders: operation.index.orders || []
|
|
91
|
+
});
|
|
92
|
+
result.created.push(operation.index.key);
|
|
93
|
+
MessageFormatter.success(`Created index ${operation.index.key}`, { prefix: 'Indexes' });
|
|
94
|
+
// Wait for index to become available
|
|
95
|
+
await waitForIndexAvailable(adapter, databaseId, tableId, operation.index.key);
|
|
96
|
+
await delay(150); // Brief delay between operations
|
|
97
|
+
}
|
|
98
|
+
catch (error) {
|
|
99
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
100
|
+
result.errors.push({ key: operation.index.key, error: errorMessage });
|
|
101
|
+
MessageFormatter.error(`Failed to create index ${operation.index.key}`, error instanceof Error ? error : new Error(String(error)), { prefix: 'Indexes' });
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
// Execute updates (delete + recreate)
|
|
105
|
+
for (const operation of plan.toUpdate) {
|
|
106
|
+
try {
|
|
107
|
+
// Delete existing index first
|
|
108
|
+
await adapter.deleteIndex({
|
|
109
|
+
databaseId,
|
|
110
|
+
tableId,
|
|
111
|
+
key: operation.index.key
|
|
112
|
+
});
|
|
113
|
+
await delay(100); // Brief delay for deletion to settle
|
|
114
|
+
// Create new index
|
|
115
|
+
await adapter.createIndex({
|
|
116
|
+
databaseId,
|
|
117
|
+
tableId,
|
|
118
|
+
key: operation.index.key,
|
|
119
|
+
type: operation.index.type,
|
|
120
|
+
attributes: operation.index.attributes,
|
|
121
|
+
orders: operation.index.orders || operation.existingIndex?.orders || []
|
|
122
|
+
});
|
|
123
|
+
result.updated.push(operation.index.key);
|
|
124
|
+
MessageFormatter.success(`Updated index ${operation.index.key}`, { prefix: 'Indexes' });
|
|
125
|
+
// Wait for index to become available
|
|
126
|
+
await waitForIndexAvailable(adapter, databaseId, tableId, operation.index.key);
|
|
127
|
+
await delay(150); // Brief delay between operations
|
|
128
|
+
}
|
|
129
|
+
catch (error) {
|
|
130
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
131
|
+
result.errors.push({ key: operation.index.key, error: errorMessage });
|
|
132
|
+
MessageFormatter.error(`Failed to update index ${operation.index.key}`, error instanceof Error ? error : new Error(String(error)), { prefix: 'Indexes' });
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
// Execute skips
|
|
136
|
+
for (const operation of plan.toSkip) {
|
|
137
|
+
result.skipped.push(operation.index.key);
|
|
138
|
+
MessageFormatter.info(`Index ${operation.index.key} unchanged`, { prefix: 'Indexes' });
|
|
139
|
+
}
|
|
140
|
+
// Calculate summary
|
|
141
|
+
result.summary.total = result.created.length + result.updated.length + result.skipped.length + result.deleted.length;
|
|
142
|
+
result.summary.created = result.created.length;
|
|
143
|
+
result.summary.updated = result.updated.length;
|
|
144
|
+
result.summary.skipped = result.skipped.length;
|
|
145
|
+
result.summary.deleted = result.deleted.length;
|
|
146
|
+
result.summary.errors = result.errors.length;
|
|
147
|
+
return result;
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* Execute index deletions with proper error handling
|
|
151
|
+
*/
|
|
152
|
+
export async function executeIndexDeletions(adapter, databaseId, tableId, deletions) {
|
|
153
|
+
const result = {
|
|
154
|
+
deleted: [],
|
|
155
|
+
errors: []
|
|
156
|
+
};
|
|
157
|
+
for (const operation of deletions) {
|
|
158
|
+
try {
|
|
159
|
+
await adapter.deleteIndex({
|
|
160
|
+
databaseId,
|
|
161
|
+
tableId,
|
|
162
|
+
key: operation.index.key
|
|
163
|
+
});
|
|
164
|
+
result.deleted.push(operation.index.key);
|
|
165
|
+
MessageFormatter.info(`Deleted obsolete index ${operation.index.key}`, { prefix: 'Indexes' });
|
|
166
|
+
// Wait briefly for deletion to settle
|
|
167
|
+
await delay(500);
|
|
168
|
+
}
|
|
169
|
+
catch (error) {
|
|
170
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
171
|
+
result.errors.push({ key: operation.index.key, error: errorMessage });
|
|
172
|
+
MessageFormatter.error(`Failed to delete index ${operation.index.key}`, error instanceof Error ? error : new Error(String(error)), { prefix: 'Indexes' });
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
return result;
|
|
176
|
+
}
|
|
177
|
+
/**
|
|
178
|
+
* Wait for an index to become available with timeout and retry logic
|
|
179
|
+
* This is an adapter-aware version of the logic from collections/indexes.ts
|
|
180
|
+
*/
|
|
181
|
+
async function waitForIndexAvailable(adapter, databaseId, tableId, indexKey, maxWaitTime = 60000, // 1 minute
|
|
182
|
+
checkInterval = 2000 // 2 seconds
|
|
183
|
+
) {
|
|
184
|
+
const startTime = Date.now();
|
|
185
|
+
while (Date.now() - startTime < maxWaitTime) {
|
|
186
|
+
try {
|
|
187
|
+
const indexList = await adapter.listIndexes({ databaseId, tableId });
|
|
188
|
+
const indexes = indexList.data || indexList.indexes || [];
|
|
189
|
+
const index = indexes.find((idx) => idx.key === indexKey);
|
|
190
|
+
if (!index) {
|
|
191
|
+
MessageFormatter.error(`Index '${indexKey}' not found after creation`, undefined, { prefix: 'Indexes' });
|
|
192
|
+
return false;
|
|
193
|
+
}
|
|
194
|
+
switch (index.status) {
|
|
195
|
+
case 'available':
|
|
196
|
+
return true;
|
|
197
|
+
case 'failed':
|
|
198
|
+
MessageFormatter.error(`Index '${indexKey}' failed: ${index.error || 'unknown error'}`, undefined, { prefix: 'Indexes' });
|
|
199
|
+
return false;
|
|
200
|
+
case 'stuck':
|
|
201
|
+
MessageFormatter.warning(`Index '${indexKey}' is stuck`, { prefix: 'Indexes' });
|
|
202
|
+
return false;
|
|
203
|
+
case 'processing':
|
|
204
|
+
case 'deleting':
|
|
205
|
+
// Continue waiting
|
|
206
|
+
break;
|
|
207
|
+
default:
|
|
208
|
+
MessageFormatter.warning(`Unknown status '${index.status}' for index '${indexKey}'`, { prefix: 'Indexes' });
|
|
209
|
+
break;
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
catch (error) {
|
|
213
|
+
MessageFormatter.error(`Error checking index '${indexKey}' status: ${error}`, undefined, { prefix: 'Indexes' });
|
|
214
|
+
}
|
|
215
|
+
await delay(checkInterval);
|
|
216
|
+
}
|
|
217
|
+
MessageFormatter.warning(`Timeout waiting for index '${indexKey}' to become available (${maxWaitTime}ms)`, { prefix: 'Indexes' });
|
|
218
|
+
return false;
|
|
219
|
+
}
|
|
220
|
+
/**
|
|
221
|
+
* Main function to create/update indexes via adapter
|
|
222
|
+
* This replaces the messy inline code in methods.ts
|
|
223
|
+
*/
|
|
224
|
+
export async function createOrUpdateIndexesViaAdapter(adapter, databaseId, tableId, desiredIndexes, configIndexes) {
|
|
225
|
+
if (!desiredIndexes || desiredIndexes.length === 0) {
|
|
226
|
+
MessageFormatter.info('No indexes to process', { prefix: 'Indexes' });
|
|
227
|
+
return;
|
|
228
|
+
}
|
|
229
|
+
MessageFormatter.info(`Processing ${desiredIndexes.length} indexes for table ${tableId}`, { prefix: 'Indexes' });
|
|
230
|
+
try {
|
|
231
|
+
// Get existing indexes
|
|
232
|
+
const existingIdxRes = await adapter.listIndexes({ databaseId, tableId });
|
|
233
|
+
const existingIndexes = existingIdxRes.data || existingIdxRes.indexes || [];
|
|
234
|
+
// Plan operations
|
|
235
|
+
const plan = planIndexOperations(desiredIndexes, existingIndexes);
|
|
236
|
+
// Show plan with icons (consistent with attribute handling)
|
|
237
|
+
const planParts = [];
|
|
238
|
+
if (plan.toCreate.length)
|
|
239
|
+
planParts.push(`➕ ${plan.toCreate.length} (${plan.toCreate.map(op => op.index.key).join(', ')})`);
|
|
240
|
+
if (plan.toUpdate.length)
|
|
241
|
+
planParts.push(`🔧 ${plan.toUpdate.length} (${plan.toUpdate.map(op => op.index.key).join(', ')})`);
|
|
242
|
+
if (plan.toSkip.length)
|
|
243
|
+
planParts.push(`⏭️ ${plan.toSkip.length}`);
|
|
244
|
+
MessageFormatter.info(`Plan → ${planParts.join(' | ') || 'no changes'}`, { prefix: 'Indexes' });
|
|
245
|
+
// Execute operations
|
|
246
|
+
const result = await executeIndexOperations(adapter, databaseId, tableId, plan);
|
|
247
|
+
// Show summary
|
|
248
|
+
MessageFormatter.info(`Summary → ➕ ${result.summary.created} | 🔧 ${result.summary.updated} | ⏭️ ${result.summary.skipped}`, { prefix: 'Indexes' });
|
|
249
|
+
// Handle errors if any
|
|
250
|
+
if (result.errors.length > 0) {
|
|
251
|
+
MessageFormatter.error(`${result.errors.length} index operations failed:`, undefined, { prefix: 'Indexes' });
|
|
252
|
+
for (const error of result.errors) {
|
|
253
|
+
MessageFormatter.error(` ${error.key}: ${error.error}`, undefined, { prefix: 'Indexes' });
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
}
|
|
257
|
+
catch (error) {
|
|
258
|
+
MessageFormatter.error('Failed to process indexes', error instanceof Error ? error : new Error(String(error)), { prefix: 'Indexes' });
|
|
259
|
+
throw error;
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
/**
|
|
263
|
+
* Handle index deletions for obsolete indexes
|
|
264
|
+
*/
|
|
265
|
+
export async function deleteObsoleteIndexesViaAdapter(adapter, databaseId, tableId, desiredIndexKeys) {
|
|
266
|
+
try {
|
|
267
|
+
// Get existing indexes
|
|
268
|
+
const existingIdxRes = await adapter.listIndexes({ databaseId, tableId });
|
|
269
|
+
const existingIndexes = existingIdxRes.data || existingIdxRes.indexes || [];
|
|
270
|
+
// Plan deletions
|
|
271
|
+
const deletions = planIndexDeletions(desiredIndexKeys, existingIndexes);
|
|
272
|
+
if (deletions.length === 0) {
|
|
273
|
+
MessageFormatter.info('Plan → 🗑️ 0 indexes', { prefix: 'Indexes' });
|
|
274
|
+
return;
|
|
275
|
+
}
|
|
276
|
+
// Show deletion plan
|
|
277
|
+
MessageFormatter.info(`Plan → 🗑️ ${deletions.length} (${deletions.map(op => op.index.key).join(', ')})`, { prefix: 'Indexes' });
|
|
278
|
+
// Execute deletions
|
|
279
|
+
const result = await executeIndexDeletions(adapter, databaseId, tableId, deletions);
|
|
280
|
+
// Show results
|
|
281
|
+
if (result.deleted.length > 0) {
|
|
282
|
+
MessageFormatter.success(`Deleted ${result.deleted.length} indexes: ${result.deleted.join(', ')}`, { prefix: 'Indexes' });
|
|
283
|
+
}
|
|
284
|
+
if (result.errors.length > 0) {
|
|
285
|
+
MessageFormatter.error(`${result.errors.length} index deletions failed:`, undefined, { prefix: 'Indexes' });
|
|
286
|
+
for (const error of result.errors) {
|
|
287
|
+
MessageFormatter.error(` ${error.key}: ${error.error}`, undefined, { prefix: 'Indexes' });
|
|
288
|
+
}
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
catch (error) {
|
|
292
|
+
MessageFormatter.warning(`Could not evaluate index deletions: ${error?.message || error}`, { prefix: 'Indexes' });
|
|
293
|
+
}
|
|
294
|
+
}
|
package/dist/types.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
export type { ValidationRules } from "
|
|
1
|
+
export type { ValidationRules } from "@njdamstra/appwrite-utils-helpers";
|
|
2
2
|
export { type AuthUserCreate, AuthUserCreateSchema, type AuthUser, AuthUserSchema, } from "./schemas/authUser.js";
|
|
3
|
-
export { validationRules } from "
|
|
3
|
+
export { validationRules } from "@njdamstra/appwrite-utils-helpers";
|
|
4
4
|
export { afterImportActions } from "./migrations/afterImportActions.js";
|
package/dist/types.js
CHANGED
|
@@ -1,3 +1,3 @@
|
|
|
1
1
|
export { AuthUserCreateSchema, AuthUserSchema, } from "./schemas/authUser.js";
|
|
2
|
-
export { validationRules } from "
|
|
2
|
+
export { validationRules } from "@njdamstra/appwrite-utils-helpers";
|
|
3
3
|
export { afterImportActions } from "./migrations/afterImportActions.js";
|
package/dist/users/methods.js
CHANGED
|
@@ -1,11 +1,10 @@
|
|
|
1
1
|
import { AppwriteException, Databases, ID, Query, Users, } from "node-appwrite";
|
|
2
2
|
import { AuthUserSchema, } from "../schemas/authUser.js";
|
|
3
|
-
import { logger } from "
|
|
3
|
+
import { logger, MessageFormatter } from "@njdamstra/appwrite-utils-helpers";
|
|
4
4
|
import { splitIntoBatches } from "../shared/migrationHelpers.js";
|
|
5
|
-
import { getAppwriteClient, tryAwaitWithRetry, } from "
|
|
5
|
+
import { getAppwriteClient, tryAwaitWithRetry, } from "@njdamstra/appwrite-utils-helpers";
|
|
6
6
|
import { isUndefined } from "es-toolkit/compat";
|
|
7
7
|
import { isEmpty } from "es-toolkit/compat";
|
|
8
|
-
import { MessageFormatter } from "../shared/messageFormatter.js";
|
|
9
8
|
export class UsersController {
|
|
10
9
|
config;
|
|
11
10
|
users;
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { promises as fs } from "fs";
|
|
2
2
|
import path from "path";
|
|
3
3
|
import { existsSync } from "fs";
|
|
4
|
-
import { MessageFormatter } from "
|
|
4
|
+
import { MessageFormatter } from "@njdamstra/appwrite-utils-helpers";
|
|
5
5
|
import { ConfirmationDialogs } from "../shared/confirmationDialogs.js";
|
|
6
6
|
import yaml from "js-yaml";
|
|
7
7
|
export async function migrateConfig(workingDir) {
|
package/dist/utils/index.d.ts
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export * from
|
|
1
|
+
export * from '@njdamstra/appwrite-utils-helpers';
|
|
2
2
|
export * from "./setupFiles.js";
|
package/dist/utils/index.js
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export * from
|
|
1
|
+
export * from '@njdamstra/appwrite-utils-helpers';
|
|
2
2
|
export * from "./setupFiles.js";
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { type AppwriteConfig } from "@njdamstra/appwrite-utils";
|
|
2
|
-
import { type ValidationResult } from "
|
|
2
|
+
import { type ValidationResult } from "@njdamstra/appwrite-utils-helpers";
|
|
3
3
|
/**
|
|
4
4
|
* Session authentication preservation options for config loading
|
|
5
5
|
*/
|
|
@@ -28,7 +28,7 @@ export interface ConfigLoadingOptions {
|
|
|
28
28
|
* @returns SessionPreservationOptions object
|
|
29
29
|
*/
|
|
30
30
|
export declare function createSessionPreservation(sessionCookie: string, email?: string, expiresAt?: string): SessionPreservationOptions;
|
|
31
|
-
export { findAppwriteConfig, findFunctionsDir } from
|
|
31
|
+
export { findAppwriteConfig, findFunctionsDir } from '@njdamstra/appwrite-utils-helpers';
|
|
32
32
|
/**
|
|
33
33
|
* Loads the Appwrite configuration and returns both config and the path where it was found.
|
|
34
34
|
* @param configDir The directory to search for config files.
|
|
@@ -4,12 +4,11 @@ import {} from "@njdamstra/appwrite-utils";
|
|
|
4
4
|
import { register } from "tsx/esm/api"; // Import the register function
|
|
5
5
|
import { pathToFileURL } from "node:url";
|
|
6
6
|
import chalk from "chalk";
|
|
7
|
-
import { findYamlConfig, loadYamlConfig, loadYamlConfigWithSession, extractSessionOptionsFromConfig } from "
|
|
8
|
-
import { detectAppwriteVersionCached, fetchServerVersion, isVersionAtLeast } from
|
|
9
|
-
import { MessageFormatter } from "
|
|
10
|
-
import {
|
|
11
|
-
import {
|
|
12
|
-
import { findAppwriteConfig, findAppwriteConfigTS, findFunctionsDir, discoverCollections, discoverTables, discoverLegacyDirectory } from "./configDiscovery.js";
|
|
7
|
+
import { findYamlConfig, loadYamlConfig, loadYamlConfigWithSession, extractSessionOptionsFromConfig, validateCollectionsTablesConfig, reportValidationResults } from "@njdamstra/appwrite-utils-helpers";
|
|
8
|
+
import { detectAppwriteVersionCached, fetchServerVersion, isVersionAtLeast } from '@njdamstra/appwrite-utils-helpers';
|
|
9
|
+
import { MessageFormatter } from "@njdamstra/appwrite-utils-helpers";
|
|
10
|
+
import { resolveCollectionsDir, resolveTablesDir } from '@njdamstra/appwrite-utils-helpers';
|
|
11
|
+
import { findAppwriteConfig, findAppwriteConfigTS, findFunctionsDir, discoverCollections, discoverTables, discoverLegacyDirectory } from '@njdamstra/appwrite-utils-helpers';
|
|
13
12
|
/**
|
|
14
13
|
* Helper function to create session preservation options from session data
|
|
15
14
|
* @param sessionCookie The session cookie string
|
|
@@ -28,7 +27,7 @@ export function createSessionPreservation(sessionCookie, email, expiresAt) {
|
|
|
28
27
|
};
|
|
29
28
|
}
|
|
30
29
|
// Re-export config discovery functions for backward compatibility
|
|
31
|
-
export { findAppwriteConfig, findFunctionsDir } from
|
|
30
|
+
export { findAppwriteConfig, findFunctionsDir } from '@njdamstra/appwrite-utils-helpers';
|
|
32
31
|
/**
|
|
33
32
|
* Loads the Appwrite configuration and returns both config and the path where it was found.
|
|
34
33
|
* @param configDir The directory to search for config files.
|
package/dist/utils/setupFiles.js
CHANGED
|
@@ -1,15 +1,13 @@
|
|
|
1
1
|
import { mkdirSync, writeFileSync, existsSync } from "node:fs";
|
|
2
2
|
import path from "node:path";
|
|
3
3
|
import { findAppwriteConfig } from "./loadConfigs.js";
|
|
4
|
-
import { loadYamlConfig } from "
|
|
5
|
-
import { fetchServerVersion, isVersionAtLeast } from
|
|
6
|
-
import { findYamlConfig } from "../config/yamlConfig.js";
|
|
4
|
+
import { loadYamlConfig, findYamlConfig, generateYamlConfigTemplate } from "@njdamstra/appwrite-utils-helpers";
|
|
5
|
+
import { fetchServerVersion, isVersionAtLeast } from '@njdamstra/appwrite-utils-helpers';
|
|
7
6
|
import { ID } from "node-appwrite";
|
|
8
7
|
import { ulid } from "ulidx";
|
|
9
|
-
import {
|
|
10
|
-
import {
|
|
11
|
-
import {
|
|
12
|
-
import { MessageFormatter } from "../shared/messageFormatter.js";
|
|
8
|
+
import { loadAppwriteProjectConfig, findAppwriteProjectConfig, getProjectDirectoryName, isTablesDBProject } from '@njdamstra/appwrite-utils-helpers';
|
|
9
|
+
import { hasSessionAuth, getSessionAuth } from '@njdamstra/appwrite-utils-helpers';
|
|
10
|
+
import { MessageFormatter } from "@njdamstra/appwrite-utils-helpers";
|
|
13
11
|
// Example base configuration using types from appwrite-utils
|
|
14
12
|
const baseConfig = {
|
|
15
13
|
appwriteEndpoint: "https://cloud.appwrite.io/v1",
|
|
@@ -2,8 +2,8 @@ import { Client, Databases, Storage, type Models } from "node-appwrite";
|
|
|
2
2
|
import { type AppwriteConfig, type AppwriteFunction, type Specification } from "@njdamstra/appwrite-utils";
|
|
3
3
|
import { type AfterImportActions, type ConverterFunctions, type ValidationRules } from "@njdamstra/appwrite-utils";
|
|
4
4
|
import { type TransferOptions } from "./migrations/transfer.js";
|
|
5
|
-
import type { DatabaseAdapter } from '
|
|
6
|
-
import { type ValidationResult } from "
|
|
5
|
+
import type { DatabaseAdapter } from '@njdamstra/appwrite-utils-helpers';
|
|
6
|
+
import { type ValidationResult } from "@njdamstra/appwrite-utils-helpers";
|
|
7
7
|
import type { DatabaseSelection, BucketSelection } from "./shared/selectionDialogs.js";
|
|
8
8
|
export interface SetupOptions {
|
|
9
9
|
databases?: Models.Database[];
|
|
@@ -19,6 +19,18 @@ export interface SetupOptions {
|
|
|
19
19
|
checkDuplicates?: boolean;
|
|
20
20
|
shouldWriteFile?: boolean;
|
|
21
21
|
}
|
|
22
|
+
export interface ControllerInitOptions {
|
|
23
|
+
validate?: boolean;
|
|
24
|
+
strictMode?: boolean;
|
|
25
|
+
useSession?: boolean;
|
|
26
|
+
sessionCookie?: string;
|
|
27
|
+
preferJson?: boolean;
|
|
28
|
+
overrides?: {
|
|
29
|
+
appwriteEndpoint?: string;
|
|
30
|
+
appwriteProject?: string;
|
|
31
|
+
appwriteKey?: string;
|
|
32
|
+
};
|
|
33
|
+
}
|
|
22
34
|
export declare class UtilsController {
|
|
23
35
|
private static instance;
|
|
24
36
|
private isInitialized;
|
|
@@ -50,12 +62,7 @@ export declare class UtilsController {
|
|
|
50
62
|
appwriteProject?: string;
|
|
51
63
|
appwriteKey?: string;
|
|
52
64
|
});
|
|
53
|
-
init(options?:
|
|
54
|
-
validate?: boolean;
|
|
55
|
-
strictMode?: boolean;
|
|
56
|
-
useSession?: boolean;
|
|
57
|
-
sessionCookie?: string;
|
|
58
|
-
}): Promise<void>;
|
|
65
|
+
init(options?: ControllerInitOptions): Promise<void>;
|
|
59
66
|
reloadConfig(): Promise<void>;
|
|
60
67
|
ensureDatabaseConfigBucketsExist(databases?: Models.Database[]): Promise<void>;
|
|
61
68
|
ensureDatabasesExist(databases?: Models.Database[]): Promise<void>;
|
package/dist/utilsController.js
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { Client, Databases, Query, Storage, Users, } from "node-appwrite";
|
|
2
2
|
import {} from "@njdamstra/appwrite-utils";
|
|
3
3
|
import { findAppwriteConfig, findFunctionsDir, } from "./utils/loadConfigs.js";
|
|
4
|
-
import { normalizeFunctionName, validateFunctionDirectory } from '
|
|
4
|
+
import { normalizeFunctionName, validateFunctionDirectory } from '@njdamstra/appwrite-utils-helpers';
|
|
5
5
|
import { UsersController } from "./users/methods.js";
|
|
6
6
|
import { AppwriteToX } from "./migrations/appwriteToX.js";
|
|
7
7
|
import { ImportController } from "./migrations/importController.js";
|
|
@@ -14,22 +14,18 @@ import path from "path";
|
|
|
14
14
|
import { converterFunctions, validationRules, } from "@njdamstra/appwrite-utils";
|
|
15
15
|
import { afterImportActions } from "./migrations/afterImportActions.js";
|
|
16
16
|
import { transferDatabaseLocalToLocal, transferDatabaseLocalToRemote, transferStorageLocalToLocal, transferStorageLocalToRemote, transferUsersLocalToRemote, } from "./migrations/transfer.js";
|
|
17
|
-
import { getClient, getClientWithAuth } from "
|
|
18
|
-
import { getAdapterFromConfig } from "
|
|
19
|
-
import { hasSessionAuth, findSessionByEndpointAndProject, isValidSessionCookie } from "
|
|
17
|
+
import { getClient, getClientWithAuth } from "@njdamstra/appwrite-utils-helpers";
|
|
18
|
+
import { getAdapterFromConfig } from "@njdamstra/appwrite-utils-helpers";
|
|
19
|
+
import { hasSessionAuth, findSessionByEndpointAndProject, isValidSessionCookie } from "@njdamstra/appwrite-utils-helpers";
|
|
20
20
|
import { fetchAllDatabases } from "./databases/methods.js";
|
|
21
21
|
import { listFunctions, updateFunctionSpecifications, } from "./functions/methods.js";
|
|
22
22
|
import chalk from "chalk";
|
|
23
23
|
import { deployLocalFunction } from "./functions/deployments.js";
|
|
24
24
|
import fs from "node:fs";
|
|
25
|
-
import { configureLogging, updateLogger, logger } from "
|
|
26
|
-
import { MessageFormatter, Messages } from "./shared/messageFormatter.js";
|
|
27
|
-
import { SchemaGenerator } from "./shared/schemaGenerator.js";
|
|
28
|
-
import { findYamlConfig } from "./config/yamlConfig.js";
|
|
25
|
+
import { configureLogging, updateLogger, logger, MessageFormatter, Messages, SchemaGenerator, findYamlConfig, validateCollectionsTablesConfig, reportValidationResults, validateWithStrictMode, ConfigManager } from "@njdamstra/appwrite-utils-helpers";
|
|
29
26
|
import { createImportSchemas } from "./migrations/yaml/generateImportSchemas.js";
|
|
30
|
-
import {
|
|
31
|
-
import {
|
|
32
|
-
import { ClientFactory } from "./utils/ClientFactory.js";
|
|
27
|
+
import { ClientFactory } from "@njdamstra/appwrite-utils-helpers";
|
|
28
|
+
import { clearProcessingState, processQueue } from "./shared/operationQueue.js";
|
|
33
29
|
export class UtilsController {
|
|
34
30
|
// ──────────────────────────────────────────────────
|
|
35
31
|
// SINGLETON PATTERN
|
|
@@ -151,7 +147,7 @@ export class UtilsController {
|
|
|
151
147
|
}
|
|
152
148
|
}
|
|
153
149
|
async init(options = {}) {
|
|
154
|
-
const { validate = false, strictMode = false } = options;
|
|
150
|
+
const { validate = false, strictMode = false, preferJson = false, useSession, sessionCookie, overrides } = options;
|
|
155
151
|
const configManager = ConfigManager.getInstance();
|
|
156
152
|
// Load config if not already loaded
|
|
157
153
|
if (!configManager.hasConfig()) {
|
|
@@ -159,6 +155,10 @@ export class UtilsController {
|
|
|
159
155
|
configDir: this.currentUserDir,
|
|
160
156
|
validate,
|
|
161
157
|
strictMode,
|
|
158
|
+
preferJson,
|
|
159
|
+
useSession,
|
|
160
|
+
explicitSessionCookie: sessionCookie,
|
|
161
|
+
overrides,
|
|
162
162
|
});
|
|
163
163
|
}
|
|
164
164
|
const config = configManager.getConfig();
|
|
@@ -351,7 +351,7 @@ export class UtilsController {
|
|
|
351
351
|
MessageFormatter.error(`Function ${functionName} not found in config`, undefined, { prefix: "Controller" });
|
|
352
352
|
return;
|
|
353
353
|
}
|
|
354
|
-
await deployLocalFunction(this.appwriteServer, functionName, functionConfig, functionPath);
|
|
354
|
+
await deployLocalFunction(this.appwriteServer, functionName, functionConfig, functionPath, this.appwriteFolderPath);
|
|
355
355
|
}
|
|
356
356
|
async syncFunctions() {
|
|
357
357
|
await this.init();
|
|
@@ -457,7 +457,6 @@ export class UtilsController {
|
|
|
457
457
|
// Ensure we don't carry state between databases in a multi-db push
|
|
458
458
|
// This resets processed sets and name->id mapping per database
|
|
459
459
|
try {
|
|
460
|
-
const { clearProcessingState } = await import('./shared/operationQueue.js');
|
|
461
460
|
clearProcessingState();
|
|
462
461
|
}
|
|
463
462
|
catch { }
|
|
@@ -474,6 +473,15 @@ export class UtilsController {
|
|
|
474
473
|
logger.debug("Adapter unavailable, falling back to legacy Databases path", { prefix: "UtilsController" });
|
|
475
474
|
await createOrUpdateCollections(this.database, database.$id, this.config, deletedCollections, collections);
|
|
476
475
|
}
|
|
476
|
+
// Safety net: Process any remaining queued operations to complete relationship sync
|
|
477
|
+
try {
|
|
478
|
+
MessageFormatter.info(`🔄 Processing final operation queue for database ${database.$id}`, { prefix: "UtilsController" });
|
|
479
|
+
await processQueue(this.adapter || this.database, database.$id);
|
|
480
|
+
MessageFormatter.info(`✅ Operation queue processing completed`, { prefix: "UtilsController" });
|
|
481
|
+
}
|
|
482
|
+
catch (error) {
|
|
483
|
+
MessageFormatter.error(`Failed to process operation queue`, error instanceof Error ? error : new Error(String(error)), { prefix: 'UtilsController' });
|
|
484
|
+
}
|
|
477
485
|
}
|
|
478
486
|
async generateSchemas() {
|
|
479
487
|
// Schema generation doesn't need Appwrite connection, just config
|
|
@@ -638,20 +646,38 @@ export class UtilsController {
|
|
|
638
646
|
MessageFormatter.progress("Starting selective push (local config → Appwrite)...", { prefix: "Controller" });
|
|
639
647
|
// Convert database selections to Models.Database format
|
|
640
648
|
const selectedDatabases = [];
|
|
649
|
+
const serverDatabases = await fetchAllDatabases(this.database);
|
|
650
|
+
const configuredDatabases = this.config?.databases || [];
|
|
641
651
|
for (const dbSelection of databaseSelections) {
|
|
642
|
-
//
|
|
643
|
-
const
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
|
|
647
|
-
MessageFormatter.info(`Selected database: ${database.name} (${database.$id})`, { prefix: "Controller" });
|
|
648
|
-
// Log selected tables for this database
|
|
649
|
-
if (dbSelection.tableIds && dbSelection.tableIds.length > 0) {
|
|
650
|
-
MessageFormatter.info(` Tables: ${dbSelection.tableIds.join(', ')}`, { prefix: "Controller" });
|
|
651
|
-
}
|
|
652
|
+
// First try to find on server
|
|
653
|
+
const serverDb = serverDatabases.find(db => db.$id === dbSelection.databaseId);
|
|
654
|
+
if (serverDb) {
|
|
655
|
+
selectedDatabases.push(serverDb);
|
|
656
|
+
MessageFormatter.info(`Selected database: ${serverDb.name} (${serverDb.$id})`, { prefix: "Controller" });
|
|
652
657
|
}
|
|
653
658
|
else {
|
|
654
|
-
|
|
659
|
+
// Database doesn't exist on server - check if it's in local config
|
|
660
|
+
const configDb = configuredDatabases.find((db) => db.$id === dbSelection.databaseId);
|
|
661
|
+
if (configDb) {
|
|
662
|
+
// Create a pseudo-database object that ensureDatabasesExist will create
|
|
663
|
+
const dbId = configDb.$id;
|
|
664
|
+
selectedDatabases.push({
|
|
665
|
+
$id: dbId,
|
|
666
|
+
name: configDb.name || dbId,
|
|
667
|
+
$createdAt: new Date().toISOString(),
|
|
668
|
+
$updatedAt: new Date().toISOString(),
|
|
669
|
+
enabled: true,
|
|
670
|
+
});
|
|
671
|
+
MessageFormatter.info(`Selected database: ${configDb.name || dbId} (${dbId}) [will be created]`, { prefix: "Controller" });
|
|
672
|
+
}
|
|
673
|
+
else {
|
|
674
|
+
MessageFormatter.warning(`Database with ID ${dbSelection.databaseId} not found in server or local config`, { prefix: "Controller" });
|
|
675
|
+
continue;
|
|
676
|
+
}
|
|
677
|
+
}
|
|
678
|
+
// Log selected tables for this database
|
|
679
|
+
if (dbSelection.tableIds && dbSelection.tableIds.length > 0) {
|
|
680
|
+
MessageFormatter.info(` Tables: ${dbSelection.tableIds.join(', ')}`, { prefix: "Controller" });
|
|
655
681
|
}
|
|
656
682
|
}
|
|
657
683
|
if (selectedDatabases.length === 0) {
|
|
@@ -670,7 +696,10 @@ export class UtilsController {
|
|
|
670
696
|
// Build database-specific collection mappings from databaseSelections
|
|
671
697
|
const databaseCollectionsMap = new Map();
|
|
672
698
|
// Get all collections/tables from config (they're at the root level, not nested in databases)
|
|
673
|
-
const allCollections =
|
|
699
|
+
const allCollections = [
|
|
700
|
+
...(this.config?.collections || []),
|
|
701
|
+
...(this.config?.tables || [])
|
|
702
|
+
];
|
|
674
703
|
// Create database-specific collection mapping to preserve relationships
|
|
675
704
|
for (const dbSelection of databaseSelections) {
|
|
676
705
|
const collectionsForDatabase = [];
|
|
@@ -760,10 +789,10 @@ export class UtilsController {
|
|
|
760
789
|
return;
|
|
761
790
|
}
|
|
762
791
|
if (options.isRemote && targetClient) {
|
|
763
|
-
await transferDatabaseLocalToRemote(sourceClient, options.transferEndpoint, options.transferProject, options.transferKey, fromDb.$id, targetDb.$id);
|
|
792
|
+
await transferDatabaseLocalToRemote(sourceClient, options.transferEndpoint, options.transferProject, options.transferKey, fromDb.$id, targetDb.$id, options.collections);
|
|
764
793
|
}
|
|
765
794
|
else {
|
|
766
|
-
await transferDatabaseLocalToLocal(sourceClient, fromDb.$id, targetDb.$id);
|
|
795
|
+
await transferDatabaseLocalToLocal(sourceClient, fromDb.$id, targetDb.$id, options.collections, this.adapter);
|
|
767
796
|
}
|
|
768
797
|
}
|
|
769
798
|
if (options.transferUsers) {
|