appwrite-utils-cli 1.9.6 → 1.11.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.
- package/CONFIG_TODO.md +1189 -1189
- package/README.md +1004 -1004
- package/SELECTION_DIALOGS.md +145 -145
- package/SERVICE_IMPLEMENTATION_REPORT.md +462 -462
- package/package.json +6 -3
- package/scripts/copy-templates.ts +23 -23
- package/src/adapters/index.ts +11 -37
- package/src/backups/operations/bucketBackup.ts +277 -277
- package/src/backups/operations/collectionBackup.ts +310 -310
- package/src/backups/operations/comprehensiveBackup.ts +342 -342
- package/src/backups/schemas/bucketManifest.ts +78 -78
- package/src/backups/schemas/comprehensiveManifest.ts +76 -76
- package/src/backups/tracking/centralizedTracking.ts +352 -352
- package/src/cli/commands/configCommands.ts +265 -201
- package/src/cli/commands/databaseCommands.ts +931 -879
- package/src/cli/commands/functionCommands.ts +333 -332
- package/src/cli/commands/importFileCommands.ts +815 -0
- package/src/cli/commands/schemaCommands.ts +141 -141
- package/src/cli/commands/storageCommands.ts +2 -3
- package/src/cli/commands/transferCommands.ts +454 -457
- package/src/collections/attributes.ts.backup +1555 -1555
- package/src/collections/{attributes.ts → columns.ts} +15 -44
- package/src/collections/indexes.ts +350 -352
- package/src/collections/methods.ts +714 -815
- package/src/collections/tableOperations.ts +57 -21
- package/src/collections/transferOperations.ts +376 -377
- package/src/collections/wipeOperations.ts +449 -346
- package/src/databases/methods.ts +49 -49
- package/src/databases/setup.ts +77 -77
- package/src/examples/yamlTerminologyExample.ts +346 -346
- package/src/functions/deployments.ts +221 -220
- package/src/functions/fnConfigDiscovery.ts +2 -2
- package/src/functions/methods.ts +284 -284
- package/src/functions/templates/count-docs-in-collection/README.md +53 -53
- package/src/functions/templates/count-docs-in-collection/src/main.ts +159 -159
- package/src/functions/templates/count-docs-in-collection/src/request.ts +8 -8
- package/src/functions/templates/hono-typescript/README.md +285 -285
- package/src/functions/templates/hono-typescript/src/adapters/request.ts +73 -73
- package/src/functions/templates/hono-typescript/src/adapters/response.ts +105 -105
- package/src/functions/templates/hono-typescript/src/app.ts +179 -179
- package/src/functions/templates/hono-typescript/src/context.ts +102 -102
- package/src/functions/templates/hono-typescript/src/{index.ts → main.ts} +53 -53
- package/src/functions/templates/hono-typescript/src/middleware/appwrite.ts +118 -118
- package/src/functions/templates/typescript-node/README.md +31 -31
- package/src/functions/templates/typescript-node/src/context.ts +102 -102
- package/src/functions/templates/typescript-node/src/{index.ts → main.ts} +29 -29
- package/src/functions/templates/uv/README.md +30 -30
- package/src/functions/templates/uv/pyproject.toml +29 -29
- package/src/functions/templates/uv/src/context.py +124 -124
- package/src/functions/templates/uv/src/{index.py → main.py} +45 -45
- package/src/init.ts +62 -62
- package/src/interactiveCLI.ts +1095 -1030
- package/src/main.ts +1517 -1670
- package/src/migrations/afterImportActions.ts +579 -580
- package/src/migrations/appwriteToX.ts +634 -630
- package/src/migrations/comprehensiveTransfer.ts +2149 -2149
- package/src/migrations/dataLoader.ts +1729 -1702
- package/src/migrations/importController.ts +440 -428
- package/src/migrations/importDataActions.ts +315 -315
- package/src/migrations/relationships.ts +333 -334
- package/src/migrations/services/DataTransformationService.ts +195 -195
- package/src/migrations/services/FileHandlerService.ts +310 -310
- package/src/migrations/services/ImportOrchestrator.ts +674 -665
- package/src/migrations/services/RateLimitManager.ts +362 -362
- package/src/migrations/services/RelationshipResolver.ts +460 -460
- package/src/migrations/services/UserMappingService.ts +344 -344
- package/src/migrations/services/ValidationService.ts +333 -333
- package/src/migrations/transfer.ts +987 -942
- package/src/migrations/yaml/YamlImportConfigLoader.ts +438 -438
- package/src/migrations/yaml/YamlImportIntegration.ts +438 -438
- package/src/migrations/yaml/generateImportSchemas.ts +1347 -1347
- package/src/schemas/authUser.ts +23 -23
- package/src/setup.ts +8 -8
- package/src/setupCommands.ts +5 -6
- package/src/setupController.ts +42 -42
- package/src/shared/backupMetadataSchema.ts +93 -93
- package/src/shared/backupTracking.ts +211 -211
- package/src/shared/confirmationDialogs.ts +326 -326
- package/src/shared/migrationHelpers.ts +232 -232
- package/src/shared/operationLogger.ts +20 -20
- package/src/shared/operationQueue.ts +326 -327
- package/src/shared/operationsTable.ts +338 -338
- package/src/shared/operationsTableSchema.ts +60 -60
- package/src/shared/progressManager.ts +277 -277
- package/src/shared/relationshipExtractor.ts +214 -214
- package/src/shared/selectionDialogs.ts +775 -722
- package/src/storage/backupCompression.ts +88 -88
- package/src/storage/methods.ts +695 -682
- package/src/storage/schemas.ts +205 -205
- package/src/tables/indexManager.ts +409 -0
- package/src/types/node-appwrite-tablesdb.d.ts +43 -43
- package/src/types.ts +9 -9
- package/src/users/methods.ts +358 -359
- package/src/utils/configMigration.ts +347 -347
- package/src/utils/index.ts +2 -2
- package/src/utils/loadConfigs.ts +457 -449
- package/src/utils/setupFiles.ts +1236 -1238
- package/src/utilsController.ts +1263 -1213
- package/tests/README.md +496 -496
- package/tests/adapters/AdapterFactory.test.ts +276 -276
- package/tests/integration/syncOperations.test.ts +462 -462
- package/tests/jest.config.js +24 -24
- package/tests/migration/configMigration.test.ts +545 -545
- package/tests/setup.ts +61 -61
- package/tests/testUtils.ts +339 -339
- package/tests/utils/loadConfigs.test.ts +349 -349
- package/tests/validation/configValidation.test.ts +411 -411
- package/tsconfig.json +44 -44
- package/.appwrite/.yaml_schemas/appwrite-config.schema.json +0 -380
- package/.appwrite/.yaml_schemas/collection.schema.json +0 -255
- package/.appwrite/collections/Categories.yaml +0 -182
- package/.appwrite/collections/ExampleCollection.yaml +0 -36
- package/.appwrite/collections/Posts.yaml +0 -227
- package/.appwrite/collections/Users.yaml +0 -149
- package/.appwrite/config.yaml +0 -109
- package/.appwrite/import/README.md +0 -148
- package/.appwrite/import/categories-import.yaml +0 -129
- package/.appwrite/import/posts-import.yaml +0 -208
- package/.appwrite/import/users-import.yaml +0 -130
- package/.appwrite/importData/categories.json +0 -194
- package/.appwrite/importData/posts.json +0 -270
- package/.appwrite/importData/users.json +0 -220
- package/.appwrite/schemas/categories.json +0 -128
- package/.appwrite/schemas/exampleCollection.json +0 -52
- package/.appwrite/schemas/posts.json +0 -173
- package/.appwrite/schemas/users.json +0 -125
- package/dist/adapters/AdapterFactory.d.ts +0 -94
- package/dist/adapters/AdapterFactory.js +0 -405
- package/dist/adapters/DatabaseAdapter.d.ts +0 -242
- 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 -596
- package/dist/adapters/index.d.ts +0 -11
- package/dist/adapters/index.js +0 -12
- package/dist/backups/operations/bucketBackup.d.ts +0 -19
- package/dist/backups/operations/bucketBackup.js +0 -197
- package/dist/backups/operations/collectionBackup.d.ts +0 -30
- package/dist/backups/operations/collectionBackup.js +0 -201
- package/dist/backups/operations/comprehensiveBackup.d.ts +0 -25
- package/dist/backups/operations/comprehensiveBackup.js +0 -238
- package/dist/backups/schemas/bucketManifest.d.ts +0 -93
- package/dist/backups/schemas/bucketManifest.js +0 -33
- package/dist/backups/schemas/comprehensiveManifest.d.ts +0 -108
- package/dist/backups/schemas/comprehensiveManifest.js +0 -32
- package/dist/backups/tracking/centralizedTracking.d.ts +0 -34
- package/dist/backups/tracking/centralizedTracking.js +0 -274
- package/dist/cli/commands/configCommands.d.ts +0 -8
- package/dist/cli/commands/configCommands.js +0 -166
- package/dist/cli/commands/databaseCommands.d.ts +0 -14
- package/dist/cli/commands/databaseCommands.js +0 -644
- package/dist/cli/commands/functionCommands.d.ts +0 -7
- package/dist/cli/commands/functionCommands.js +0 -330
- package/dist/cli/commands/schemaCommands.d.ts +0 -7
- package/dist/cli/commands/schemaCommands.js +0 -169
- package/dist/cli/commands/storageCommands.d.ts +0 -5
- package/dist/cli/commands/storageCommands.js +0 -143
- package/dist/cli/commands/transferCommands.d.ts +0 -5
- package/dist/cli/commands/transferCommands.js +0 -384
- package/dist/collections/attributes.d.ts +0 -13
- package/dist/collections/attributes.js +0 -1364
- package/dist/collections/indexes.d.ts +0 -12
- package/dist/collections/indexes.js +0 -217
- package/dist/collections/methods.d.ts +0 -19
- package/dist/collections/methods.js +0 -734
- package/dist/collections/tableOperations.d.ts +0 -86
- package/dist/collections/tableOperations.js +0 -434
- package/dist/collections/transferOperations.d.ts +0 -8
- package/dist/collections/transferOperations.js +0 -412
- package/dist/collections/wipeOperations.d.ts +0 -16
- package/dist/collections/wipeOperations.js +0 -233
- package/dist/config/ConfigManager.d.ts +0 -450
- 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 -122
- package/dist/config/services/ConfigDiscoveryService.js +0 -322
- package/dist/config/services/ConfigLoaderService.d.ts +0 -129
- package/dist/config/services/ConfigLoaderService.js +0 -535
- 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/databases/methods.d.ts +0 -6
- package/dist/databases/methods.js +0 -35
- package/dist/databases/setup.d.ts +0 -5
- package/dist/databases/setup.js +0 -45
- package/dist/examples/yamlTerminologyExample.d.ts +0 -42
- package/dist/examples/yamlTerminologyExample.js +0 -272
- package/dist/functions/deployments.d.ts +0 -4
- package/dist/functions/deployments.js +0 -146
- package/dist/functions/fnConfigDiscovery.d.ts +0 -3
- package/dist/functions/fnConfigDiscovery.js +0 -108
- package/dist/functions/methods.d.ts +0 -16
- package/dist/functions/methods.js +0 -174
- package/dist/functions/pathResolution.d.ts +0 -37
- package/dist/functions/pathResolution.js +0 -185
- package/dist/functions/templates/count-docs-in-collection/README.md +0 -54
- package/dist/functions/templates/count-docs-in-collection/package.json +0 -25
- package/dist/functions/templates/count-docs-in-collection/src/main.ts +0 -159
- package/dist/functions/templates/count-docs-in-collection/src/request.ts +0 -9
- package/dist/functions/templates/count-docs-in-collection/tsconfig.json +0 -28
- package/dist/functions/templates/hono-typescript/README.md +0 -286
- package/dist/functions/templates/hono-typescript/package.json +0 -26
- package/dist/functions/templates/hono-typescript/src/adapters/request.ts +0 -74
- package/dist/functions/templates/hono-typescript/src/adapters/response.ts +0 -106
- package/dist/functions/templates/hono-typescript/src/app.ts +0 -180
- package/dist/functions/templates/hono-typescript/src/context.ts +0 -103
- package/dist/functions/templates/hono-typescript/src/index.ts +0 -54
- package/dist/functions/templates/hono-typescript/src/middleware/appwrite.ts +0 -119
- package/dist/functions/templates/hono-typescript/tsconfig.json +0 -20
- package/dist/functions/templates/typescript-node/README.md +0 -32
- package/dist/functions/templates/typescript-node/package.json +0 -25
- package/dist/functions/templates/typescript-node/src/context.ts +0 -103
- package/dist/functions/templates/typescript-node/src/index.ts +0 -29
- package/dist/functions/templates/typescript-node/tsconfig.json +0 -28
- package/dist/functions/templates/uv/README.md +0 -31
- package/dist/functions/templates/uv/pyproject.toml +0 -30
- package/dist/functions/templates/uv/src/__init__.py +0 -0
- package/dist/functions/templates/uv/src/context.py +0 -125
- package/dist/functions/templates/uv/src/index.py +0 -46
- package/dist/init.d.ts +0 -2
- package/dist/init.js +0 -57
- package/dist/interactiveCLI.d.ts +0 -31
- package/dist/interactiveCLI.js +0 -898
- package/dist/main.d.ts +0 -2
- package/dist/main.js +0 -1180
- package/dist/migrations/afterImportActions.d.ts +0 -17
- package/dist/migrations/afterImportActions.js +0 -306
- package/dist/migrations/appwriteToX.d.ts +0 -211
- package/dist/migrations/appwriteToX.js +0 -491
- package/dist/migrations/comprehensiveTransfer.d.ts +0 -147
- package/dist/migrations/comprehensiveTransfer.js +0 -1317
- package/dist/migrations/dataLoader.d.ts +0 -753
- package/dist/migrations/dataLoader.js +0 -1250
- package/dist/migrations/importController.d.ts +0 -23
- package/dist/migrations/importController.js +0 -268
- package/dist/migrations/importDataActions.d.ts +0 -50
- package/dist/migrations/importDataActions.js +0 -230
- package/dist/migrations/relationships.d.ts +0 -29
- package/dist/migrations/relationships.js +0 -204
- package/dist/migrations/services/DataTransformationService.d.ts +0 -55
- package/dist/migrations/services/DataTransformationService.js +0 -158
- package/dist/migrations/services/FileHandlerService.d.ts +0 -75
- package/dist/migrations/services/FileHandlerService.js +0 -236
- package/dist/migrations/services/ImportOrchestrator.d.ts +0 -97
- package/dist/migrations/services/ImportOrchestrator.js +0 -485
- package/dist/migrations/services/RateLimitManager.d.ts +0 -138
- package/dist/migrations/services/RateLimitManager.js +0 -279
- package/dist/migrations/services/RelationshipResolver.d.ts +0 -120
- package/dist/migrations/services/RelationshipResolver.js +0 -332
- package/dist/migrations/services/UserMappingService.d.ts +0 -109
- package/dist/migrations/services/UserMappingService.js +0 -277
- package/dist/migrations/services/ValidationService.d.ts +0 -74
- package/dist/migrations/services/ValidationService.js +0 -260
- package/dist/migrations/transfer.d.ts +0 -26
- package/dist/migrations/transfer.js +0 -608
- package/dist/migrations/yaml/YamlImportConfigLoader.d.ts +0 -131
- package/dist/migrations/yaml/YamlImportConfigLoader.js +0 -383
- package/dist/migrations/yaml/YamlImportIntegration.d.ts +0 -93
- package/dist/migrations/yaml/YamlImportIntegration.js +0 -341
- package/dist/migrations/yaml/generateImportSchemas.d.ts +0 -30
- package/dist/migrations/yaml/generateImportSchemas.js +0 -1327
- package/dist/schemas/authUser.d.ts +0 -24
- package/dist/schemas/authUser.js +0 -17
- package/dist/setup.d.ts +0 -2
- package/dist/setup.js +0 -5
- package/dist/setupCommands.d.ts +0 -58
- package/dist/setupCommands.js +0 -490
- package/dist/setupController.d.ts +0 -9
- package/dist/setupController.js +0 -34
- package/dist/shared/attributeMapper.d.ts +0 -20
- package/dist/shared/attributeMapper.js +0 -203
- package/dist/shared/backupMetadataSchema.d.ts +0 -94
- package/dist/shared/backupMetadataSchema.js +0 -38
- package/dist/shared/backupTracking.d.ts +0 -18
- package/dist/shared/backupTracking.js +0 -176
- package/dist/shared/confirmationDialogs.d.ts +0 -75
- package/dist/shared/confirmationDialogs.js +0 -236
- 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 -348
- 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/migrationHelpers.d.ts +0 -61
- package/dist/shared/migrationHelpers.js +0 -145
- package/dist/shared/operationLogger.d.ts +0 -10
- package/dist/shared/operationLogger.js +0 -12
- package/dist/shared/operationQueue.d.ts +0 -40
- package/dist/shared/operationQueue.js +0 -311
- package/dist/shared/operationsTable.d.ts +0 -26
- package/dist/shared/operationsTable.js +0 -286
- package/dist/shared/operationsTableSchema.d.ts +0 -48
- package/dist/shared/operationsTableSchema.js +0 -35
- package/dist/shared/progressManager.d.ts +0 -62
- package/dist/shared/progressManager.js +0 -215
- package/dist/shared/pydanticModelGenerator.d.ts +0 -17
- package/dist/shared/pydanticModelGenerator.js +0 -615
- package/dist/shared/relationshipExtractor.d.ts +0 -56
- package/dist/shared/relationshipExtractor.js +0 -138
- package/dist/shared/schemaGenerator.d.ts +0 -40
- package/dist/shared/schemaGenerator.js +0 -556
- package/dist/shared/selectionDialogs.d.ts +0 -214
- package/dist/shared/selectionDialogs.js +0 -544
- package/dist/storage/backupCompression.d.ts +0 -20
- package/dist/storage/backupCompression.js +0 -67
- package/dist/storage/methods.d.ts +0 -32
- package/dist/storage/methods.js +0 -472
- package/dist/storage/schemas.d.ts +0 -842
- package/dist/storage/schemas.js +0 -175
- package/dist/types.d.ts +0 -4
- package/dist/types.js +0 -3
- package/dist/users/methods.d.ts +0 -16
- package/dist/users/methods.js +0 -277
- 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/configMigration.d.ts +0 -1
- package/dist/utils/configMigration.js +0 -261
- 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/index.d.ts +0 -2
- package/dist/utils/index.js +0 -2
- package/dist/utils/loadConfigs.d.ts +0 -50
- package/dist/utils/loadConfigs.js +0 -358
- package/dist/utils/pathResolvers.d.ts +0 -53
- package/dist/utils/pathResolvers.js +0 -72
- package/dist/utils/projectConfig.d.ts +0 -122
- package/dist/utils/projectConfig.js +0 -206
- 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/setupFiles.d.ts +0 -4
- package/dist/utils/setupFiles.js +0 -1192
- 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/dist/utilsController.d.ts +0 -107
- package/dist/utilsController.js +0 -873
- package/src/adapters/AdapterFactory.ts +0 -510
- package/src/adapters/DatabaseAdapter.ts +0 -318
- package/src/adapters/LegacyAdapter.ts +0 -841
- package/src/adapters/TablesDBAdapter.ts +0 -815
- package/src/config/ConfigManager.ts +0 -817
- 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 -410
- package/src/config/services/ConfigLoaderService.ts +0 -732
- 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/functions/templates/count-docs-in-collection/package.json +0 -25
- package/src/functions/templates/count-docs-in-collection/tsconfig.json +0 -28
- package/src/functions/templates/hono-typescript/package.json +0 -26
- package/src/functions/templates/hono-typescript/tsconfig.json +0 -20
- package/src/functions/templates/typescript-node/package.json +0 -25
- package/src/functions/templates/typescript-node/tsconfig.json +0 -28
- package/src/shared/attributeMapper.ts +0 -229
- package/src/shared/errorUtils.ts +0 -110
- package/src/shared/functionManager.ts +0 -537
- 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 -340
- 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
- package/tmp-sync-test/.appwrite/collections/TestCollection.yaml +0 -7
|
@@ -1,21 +1,20 @@
|
|
|
1
|
-
import { Query, type Databases, type Models } from "node-appwrite";
|
|
2
|
-
import type { Attribute } from "appwrite-utils";
|
|
3
|
-
import { createOrUpdateAttributeWithStatusCheck } from "../collections/
|
|
4
|
-
import { fetchAndCacheCollectionByName } from "../collections/methods.js";
|
|
5
|
-
import { tryAwaitWithRetry } from "
|
|
6
|
-
import type { DatabaseAdapter } from "
|
|
7
|
-
import { logger } from "
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
// Global state management
|
|
1
|
+
import { Query, type Databases, type Models } from "node-appwrite";
|
|
2
|
+
import type { Attribute } from "appwrite-utils";
|
|
3
|
+
import { createOrUpdateAttributeWithStatusCheck } from "../collections/columns.js";
|
|
4
|
+
import { fetchAndCacheCollectionByName } from "../collections/methods.js";
|
|
5
|
+
import { tryAwaitWithRetry } from "appwrite-utils-helpers";
|
|
6
|
+
import type { DatabaseAdapter } from "appwrite-utils-helpers";
|
|
7
|
+
import { logger, MessageFormatter } from "appwrite-utils-helpers";
|
|
8
|
+
|
|
9
|
+
export interface QueuedOperation {
|
|
10
|
+
type: "attribute";
|
|
11
|
+
collectionId?: string;
|
|
12
|
+
attribute?: Attribute;
|
|
13
|
+
collection?: Models.Collection;
|
|
14
|
+
dependencies?: string[];
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
// Global state management
|
|
19
18
|
export const queuedOperations: QueuedOperation[] = [];
|
|
20
19
|
export const nameToIdMapping: Map<string, string> = new Map();
|
|
21
20
|
// Keys are scoped per database to avoid cross-database collisions
|
|
@@ -27,79 +26,79 @@ export const processedAttributes: Set<string> = new Set();
|
|
|
27
26
|
// Helpers to build scoped keys
|
|
28
27
|
const collectionKey = (databaseId: string, collectionId: string) => `${databaseId}::${collectionId}`;
|
|
29
28
|
const attributeKeyScoped = (databaseId: string, collectionId: string, key: string) => `${databaseId}::${collectionId}::${key}`;
|
|
30
|
-
|
|
31
|
-
export const enqueueOperation = (operation: QueuedOperation) => {
|
|
32
|
-
// Avoid duplicate queue entries for same attribute
|
|
33
|
-
const attributeKey = operation.attribute?.key;
|
|
34
|
-
const collectionId = operation.collectionId;
|
|
35
|
-
|
|
36
|
-
logger.info('Enqueueing operation', {
|
|
37
|
-
type: operation.type,
|
|
38
|
-
attributeKey,
|
|
39
|
-
collectionId,
|
|
40
|
-
dependencies: operation.dependencies,
|
|
41
|
-
queueSizeBefore: queuedOperations.length,
|
|
42
|
-
operation: 'enqueueOperation'
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
if (attributeKey && collectionId) {
|
|
46
|
-
const duplicateIndex = queuedOperations.findIndex(
|
|
47
|
-
(op) => op.collectionId === collectionId && op.attribute?.key === attributeKey
|
|
48
|
-
);
|
|
49
|
-
|
|
50
|
-
if (duplicateIndex !== -1) {
|
|
51
|
-
MessageFormatter.info(`Replacing existing queue entry for attribute: ${attributeKey}`);
|
|
52
|
-
logger.info('Replacing duplicate queue entry', {
|
|
53
|
-
attributeKey,
|
|
54
|
-
collectionId,
|
|
55
|
-
duplicateIndex,
|
|
56
|
-
operation: 'enqueueOperation'
|
|
57
|
-
});
|
|
58
|
-
queuedOperations[duplicateIndex] = operation;
|
|
59
|
-
return;
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
queuedOperations.push(operation);
|
|
64
|
-
logger.debug('Operation enqueued successfully', {
|
|
65
|
-
attributeKey,
|
|
66
|
-
collectionId,
|
|
67
|
-
queueSizeAfter: queuedOperations.length,
|
|
68
|
-
operation: 'enqueueOperation'
|
|
69
|
-
});
|
|
70
|
-
};
|
|
71
|
-
|
|
72
|
-
/**
|
|
73
|
-
* Clear all caches and processing state - use between operations
|
|
74
|
-
*/
|
|
75
|
-
export const clearProcessingState = () => {
|
|
76
|
-
const sizeBefore = {
|
|
77
|
-
collections: processedCollections.size,
|
|
78
|
-
attributes: processedAttributes.size,
|
|
79
|
-
nameMapping: nameToIdMapping.size
|
|
80
|
-
};
|
|
81
|
-
|
|
82
|
-
processedCollections.clear();
|
|
83
|
-
processedAttributes.clear();
|
|
84
|
-
nameToIdMapping.clear();
|
|
85
|
-
|
|
86
|
-
logger.debug("Cleared processing state caches", { operation: "clearProcessingState", sizeBefore });
|
|
87
|
-
logger.info('Processing state cleared', {
|
|
88
|
-
sizeBefore,
|
|
89
|
-
operation: 'clearProcessingState'
|
|
90
|
-
});
|
|
91
|
-
};
|
|
92
|
-
|
|
93
|
-
/**
|
|
94
|
-
* Check if a collection has already been fully processed
|
|
95
|
-
*/
|
|
29
|
+
|
|
30
|
+
export const enqueueOperation = (operation: QueuedOperation) => {
|
|
31
|
+
// Avoid duplicate queue entries for same attribute
|
|
32
|
+
const attributeKey = operation.attribute?.key;
|
|
33
|
+
const collectionId = operation.collectionId;
|
|
34
|
+
|
|
35
|
+
logger.info('Enqueueing operation', {
|
|
36
|
+
type: operation.type,
|
|
37
|
+
attributeKey,
|
|
38
|
+
collectionId,
|
|
39
|
+
dependencies: operation.dependencies,
|
|
40
|
+
queueSizeBefore: queuedOperations.length,
|
|
41
|
+
operation: 'enqueueOperation'
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
if (attributeKey && collectionId) {
|
|
45
|
+
const duplicateIndex = queuedOperations.findIndex(
|
|
46
|
+
(op) => op.collectionId === collectionId && op.attribute?.key === attributeKey
|
|
47
|
+
);
|
|
48
|
+
|
|
49
|
+
if (duplicateIndex !== -1) {
|
|
50
|
+
MessageFormatter.info(`Replacing existing queue entry for attribute: ${attributeKey}`);
|
|
51
|
+
logger.info('Replacing duplicate queue entry', {
|
|
52
|
+
attributeKey,
|
|
53
|
+
collectionId,
|
|
54
|
+
duplicateIndex,
|
|
55
|
+
operation: 'enqueueOperation'
|
|
56
|
+
});
|
|
57
|
+
queuedOperations[duplicateIndex] = operation;
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
queuedOperations.push(operation);
|
|
63
|
+
logger.debug('Operation enqueued successfully', {
|
|
64
|
+
attributeKey,
|
|
65
|
+
collectionId,
|
|
66
|
+
queueSizeAfter: queuedOperations.length,
|
|
67
|
+
operation: 'enqueueOperation'
|
|
68
|
+
});
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
/**
|
|
72
|
+
* Clear all caches and processing state - use between operations
|
|
73
|
+
*/
|
|
74
|
+
export const clearProcessingState = () => {
|
|
75
|
+
const sizeBefore = {
|
|
76
|
+
collections: processedCollections.size,
|
|
77
|
+
attributes: processedAttributes.size,
|
|
78
|
+
nameMapping: nameToIdMapping.size
|
|
79
|
+
};
|
|
80
|
+
|
|
81
|
+
processedCollections.clear();
|
|
82
|
+
processedAttributes.clear();
|
|
83
|
+
nameToIdMapping.clear();
|
|
84
|
+
|
|
85
|
+
logger.debug("Cleared processing state caches", { operation: "clearProcessingState", sizeBefore });
|
|
86
|
+
logger.info('Processing state cleared', {
|
|
87
|
+
sizeBefore,
|
|
88
|
+
operation: 'clearProcessingState'
|
|
89
|
+
});
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
/**
|
|
93
|
+
* Check if a collection has already been fully processed
|
|
94
|
+
*/
|
|
96
95
|
export const isCollectionProcessed = (collectionId: string, databaseId: string): boolean => {
|
|
97
96
|
return processedCollections.has(collectionKey(databaseId, collectionId));
|
|
98
97
|
};
|
|
99
|
-
|
|
100
|
-
/**
|
|
101
|
-
* Mark a collection as fully processed
|
|
102
|
-
*/
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* Mark a collection as fully processed
|
|
101
|
+
*/
|
|
103
102
|
export const markCollectionProcessed = (collectionId: string, collectionName: string | undefined, databaseId: string) => {
|
|
104
103
|
processedCollections.add(collectionKey(databaseId, collectionId));
|
|
105
104
|
|
|
@@ -117,17 +116,17 @@ export const markCollectionProcessed = (collectionId: string, collectionName: st
|
|
|
117
116
|
|
|
118
117
|
logger.info('Collection marked as processed', logData);
|
|
119
118
|
};
|
|
120
|
-
|
|
121
|
-
/**
|
|
122
|
-
* Check if a specific attribute has been processed
|
|
123
|
-
*/
|
|
119
|
+
|
|
120
|
+
/**
|
|
121
|
+
* Check if a specific attribute has been processed
|
|
122
|
+
*/
|
|
124
123
|
export const isAttributeProcessed = (databaseId: string, collectionId: string, attributeKey: string): boolean => {
|
|
125
124
|
return processedAttributes.has(attributeKeyScoped(databaseId, collectionId, attributeKey));
|
|
126
125
|
};
|
|
127
|
-
|
|
128
|
-
/**
|
|
129
|
-
* Mark a specific attribute as processed
|
|
130
|
-
*/
|
|
126
|
+
|
|
127
|
+
/**
|
|
128
|
+
* Mark a specific attribute as processed
|
|
129
|
+
*/
|
|
131
130
|
export const markAttributeProcessed = (databaseId: string, collectionId: string, attributeKey: string) => {
|
|
132
131
|
const identifier = attributeKeyScoped(databaseId, collectionId, attributeKey);
|
|
133
132
|
processedAttributes.add(identifier);
|
|
@@ -141,237 +140,237 @@ export const markAttributeProcessed = (databaseId: string, collectionId: string,
|
|
|
141
140
|
operation: 'markAttributeProcessed'
|
|
142
141
|
});
|
|
143
142
|
};
|
|
144
|
-
|
|
145
|
-
/**
|
|
146
|
-
* Process only specific attributes in the queue, not entire collections
|
|
147
|
-
* This prevents triggering full collection re-processing cycles
|
|
148
|
-
*/
|
|
149
|
-
export const processQueue = async (db: Databases | DatabaseAdapter, dbId: string) => {
|
|
150
|
-
const startTime = Date.now();
|
|
151
|
-
|
|
152
|
-
if (queuedOperations.length === 0) {
|
|
153
|
-
MessageFormatter.info("No queued operations to process");
|
|
154
|
-
logger.info('Queue processing skipped - no operations', {
|
|
155
|
-
dbId,
|
|
156
|
-
operation: 'processQueue'
|
|
157
|
-
});
|
|
158
|
-
return;
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
MessageFormatter.section(`Starting surgical queue processing of ${queuedOperations.length} operations for ${dbId}`);
|
|
162
|
-
|
|
163
|
-
logger.info('Starting queue processing', {
|
|
164
|
-
dbId,
|
|
165
|
-
queueSize: queuedOperations.length,
|
|
166
|
-
operations: queuedOperations.map(op => ({
|
|
167
|
-
type: op.type,
|
|
168
|
-
attributeKey: op.attribute?.key,
|
|
169
|
-
collectionId: op.collectionId,
|
|
170
|
-
dependencies: op.dependencies
|
|
171
|
-
})),
|
|
172
|
-
operation: 'processQueue'
|
|
173
|
-
});
|
|
174
|
-
|
|
175
|
-
let progress = true;
|
|
176
|
-
let attempts = 0;
|
|
177
|
-
const maxAttempts = 3; // Prevent infinite loops
|
|
178
|
-
|
|
179
|
-
while (progress && attempts < maxAttempts) {
|
|
180
|
-
progress = false;
|
|
181
|
-
attempts++;
|
|
182
|
-
|
|
183
|
-
MessageFormatter.info(`Queue processing attempt ${attempts}/${maxAttempts}`);
|
|
184
|
-
|
|
185
|
-
logger.info('Queue processing attempt started', {
|
|
186
|
-
attempt: attempts,
|
|
187
|
-
maxAttempts,
|
|
188
|
-
remainingOperations: queuedOperations.length,
|
|
189
|
-
dbId,
|
|
190
|
-
operation: 'processQueue'
|
|
191
|
-
});
|
|
192
|
-
|
|
193
|
-
for (let i = queuedOperations.length - 1; i >= 0; i--) {
|
|
194
|
-
const operation = queuedOperations[i];
|
|
195
|
-
|
|
196
|
-
if (!operation.attribute || !operation.collectionId) {
|
|
197
|
-
MessageFormatter.warning("Invalid operation, removing from queue");
|
|
198
|
-
queuedOperations.splice(i, 1);
|
|
199
|
-
continue;
|
|
200
|
-
}
|
|
201
|
-
|
|
202
|
-
const attributeKey = operation.attribute.key;
|
|
203
|
-
const collectionId = operation.collectionId;
|
|
204
|
-
|
|
143
|
+
|
|
144
|
+
/**
|
|
145
|
+
* Process only specific attributes in the queue, not entire collections
|
|
146
|
+
* This prevents triggering full collection re-processing cycles
|
|
147
|
+
*/
|
|
148
|
+
export const processQueue = async (db: Databases | DatabaseAdapter, dbId: string) => {
|
|
149
|
+
const startTime = Date.now();
|
|
150
|
+
|
|
151
|
+
if (queuedOperations.length === 0) {
|
|
152
|
+
MessageFormatter.info("No queued operations to process");
|
|
153
|
+
logger.info('Queue processing skipped - no operations', {
|
|
154
|
+
dbId,
|
|
155
|
+
operation: 'processQueue'
|
|
156
|
+
});
|
|
157
|
+
return;
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
MessageFormatter.section(`Starting surgical queue processing of ${queuedOperations.length} operations for ${dbId}`);
|
|
161
|
+
|
|
162
|
+
logger.info('Starting queue processing', {
|
|
163
|
+
dbId,
|
|
164
|
+
queueSize: queuedOperations.length,
|
|
165
|
+
operations: queuedOperations.map(op => ({
|
|
166
|
+
type: op.type,
|
|
167
|
+
attributeKey: op.attribute?.key,
|
|
168
|
+
collectionId: op.collectionId,
|
|
169
|
+
dependencies: op.dependencies
|
|
170
|
+
})),
|
|
171
|
+
operation: 'processQueue'
|
|
172
|
+
});
|
|
173
|
+
|
|
174
|
+
let progress = true;
|
|
175
|
+
let attempts = 0;
|
|
176
|
+
const maxAttempts = 3; // Prevent infinite loops
|
|
177
|
+
|
|
178
|
+
while (progress && attempts < maxAttempts) {
|
|
179
|
+
progress = false;
|
|
180
|
+
attempts++;
|
|
181
|
+
|
|
182
|
+
MessageFormatter.info(`Queue processing attempt ${attempts}/${maxAttempts}`);
|
|
183
|
+
|
|
184
|
+
logger.info('Queue processing attempt started', {
|
|
185
|
+
attempt: attempts,
|
|
186
|
+
maxAttempts,
|
|
187
|
+
remainingOperations: queuedOperations.length,
|
|
188
|
+
dbId,
|
|
189
|
+
operation: 'processQueue'
|
|
190
|
+
});
|
|
191
|
+
|
|
192
|
+
for (let i = queuedOperations.length - 1; i >= 0; i--) {
|
|
193
|
+
const operation = queuedOperations[i];
|
|
194
|
+
|
|
195
|
+
if (!operation.attribute || !operation.collectionId) {
|
|
196
|
+
MessageFormatter.warning("Invalid operation, removing from queue");
|
|
197
|
+
queuedOperations.splice(i, 1);
|
|
198
|
+
continue;
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
const attributeKey = operation.attribute.key;
|
|
202
|
+
const collectionId = operation.collectionId;
|
|
203
|
+
|
|
205
204
|
// Skip if this specific attribute was already processed (per database)
|
|
206
205
|
if (isAttributeProcessed(dbId, collectionId, attributeKey)) {
|
|
207
|
-
MessageFormatter.debug(`Attribute '${attributeKey}' already processed, removing from queue`);
|
|
208
|
-
logger.debug('Removing already processed attribute from queue', {
|
|
209
|
-
attributeKey,
|
|
210
|
-
collectionId,
|
|
211
|
-
queueIndex: i,
|
|
212
|
-
operation: 'processQueue'
|
|
213
|
-
});
|
|
214
|
-
queuedOperations.splice(i, 1);
|
|
215
|
-
continue;
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
let targetCollection: Models.Collection | undefined;
|
|
219
|
-
|
|
220
|
-
// Resolve the target collection (where the attribute will be created)
|
|
221
|
-
try {
|
|
222
|
-
targetCollection = await tryAwaitWithRetry(
|
|
223
|
-
async () => {
|
|
224
|
-
if ('getMetadata' in db && typeof db.getMetadata === 'function') {
|
|
225
|
-
// DatabaseAdapter
|
|
226
|
-
return (await (db as DatabaseAdapter).getTable({ databaseId: dbId, tableId: collectionId })).data;
|
|
227
|
-
} else {
|
|
228
|
-
// Legacy Databases
|
|
229
|
-
return await (db as Databases).getCollection(dbId, collectionId);
|
|
230
|
-
}
|
|
231
|
-
}
|
|
232
|
-
);
|
|
233
|
-
} catch (e) {
|
|
234
|
-
const errorMessage = e instanceof Error ? e.message : String(e);
|
|
235
|
-
MessageFormatter.error(`Target collection ${collectionId} not found, removing from queue`);
|
|
236
|
-
logger.error('Target collection not found during queue processing', {
|
|
237
|
-
collectionId,
|
|
238
|
-
attributeKey,
|
|
239
|
-
error: errorMessage,
|
|
240
|
-
operation: 'processQueue'
|
|
241
|
-
});
|
|
242
|
-
queuedOperations.splice(i, 1);
|
|
243
|
-
continue;
|
|
244
|
-
}
|
|
245
|
-
|
|
246
|
-
// For relationship attributes, ensure the related collection exists
|
|
247
|
-
let canProcess = true;
|
|
248
|
-
if (operation.attribute.type === "relationship") {
|
|
249
|
-
const relatedCollection = operation.attribute.relatedCollection;
|
|
250
|
-
if (relatedCollection) {
|
|
251
|
-
// Try to resolve related collection by ID first, then by name
|
|
252
|
-
let relatedFound = false;
|
|
253
|
-
|
|
254
|
-
try {
|
|
255
|
-
await tryAwaitWithRetry(
|
|
256
|
-
async () => {
|
|
257
|
-
if ('getMetadata' in db && typeof db.getMetadata === 'function') {
|
|
258
|
-
// DatabaseAdapter
|
|
259
|
-
return (await (db as DatabaseAdapter).getTable({ databaseId: dbId, tableId: relatedCollection })).data;
|
|
260
|
-
} else {
|
|
261
|
-
// Legacy Databases
|
|
262
|
-
return await (db as Databases).getCollection(dbId, relatedCollection);
|
|
263
|
-
}
|
|
264
|
-
}
|
|
265
|
-
);
|
|
266
|
-
relatedFound = true;
|
|
267
|
-
nameToIdMapping.set(relatedCollection, relatedCollection); // Cache the ID mapping
|
|
268
|
-
} catch (_) {
|
|
269
|
-
// Try by name lookup
|
|
270
|
-
const cachedId = nameToIdMapping.get(relatedCollection);
|
|
271
|
-
if (cachedId) {
|
|
272
|
-
try {
|
|
273
|
-
await tryAwaitWithRetry(
|
|
274
|
-
async () => {
|
|
275
|
-
if ('getMetadata' in db && typeof db.getMetadata === 'function') {
|
|
276
|
-
// DatabaseAdapter
|
|
277
|
-
return (await (db as DatabaseAdapter).getTable({ databaseId: dbId, tableId: cachedId })).data;
|
|
278
|
-
} else {
|
|
279
|
-
// Legacy Databases
|
|
280
|
-
return await (db as Databases).getCollection(dbId, cachedId);
|
|
281
|
-
}
|
|
282
|
-
}
|
|
283
|
-
);
|
|
284
|
-
relatedFound = true;
|
|
285
|
-
} catch (_) {
|
|
286
|
-
nameToIdMapping.delete(relatedCollection); // Remove stale cache
|
|
287
|
-
}
|
|
288
|
-
}
|
|
289
|
-
|
|
290
|
-
if (!relatedFound) {
|
|
291
|
-
// Final attempt: search by name
|
|
292
|
-
try {
|
|
293
|
-
const collections = 'getMetadata' in db && typeof db.getMetadata === 'function'
|
|
294
|
-
? await (db as DatabaseAdapter).listTables({ databaseId: dbId, queries: [Query.equal("name", relatedCollection)] })
|
|
295
|
-
: await (db as Databases).listCollections(dbId, [Query.equal("name", relatedCollection)]);
|
|
296
|
-
|
|
297
|
-
if (collections.total && collections.total > 0) {
|
|
298
|
-
const firstCollection = 'getMetadata' in db && typeof db.getMetadata === 'function'
|
|
299
|
-
? (collections as any).tables?.[0]
|
|
300
|
-
: (collections as any).collections?.[0];
|
|
301
|
-
nameToIdMapping.set(relatedCollection, firstCollection.$id);
|
|
302
|
-
relatedFound = true;
|
|
303
|
-
}
|
|
304
|
-
} catch (_) {
|
|
305
|
-
// Related collection truly doesn't exist yet
|
|
306
|
-
}
|
|
307
|
-
}
|
|
308
|
-
}
|
|
309
|
-
|
|
310
|
-
if (!relatedFound) {
|
|
311
|
-
MessageFormatter.warning(
|
|
312
|
-
`Related collection '${relatedCollection}' not ready for attribute '${attributeKey}', keeping in queue`
|
|
313
|
-
);
|
|
314
|
-
canProcess = false;
|
|
315
|
-
}
|
|
316
|
-
}
|
|
317
|
-
}
|
|
318
|
-
|
|
319
|
-
if (canProcess && targetCollection) {
|
|
320
|
-
MessageFormatter.progress(
|
|
321
|
-
`Processing queued ${operation.attribute.type} attribute: '${attributeKey}' for collection: '${targetCollection.name}'`
|
|
322
|
-
);
|
|
323
|
-
|
|
324
|
-
const success = await createOrUpdateAttributeWithStatusCheck(
|
|
325
|
-
db,
|
|
326
|
-
dbId,
|
|
327
|
-
targetCollection,
|
|
328
|
-
operation.attribute
|
|
329
|
-
);
|
|
330
|
-
|
|
331
|
-
if (success) {
|
|
332
|
-
MessageFormatter.success(`Successfully processed queued attribute: '${attributeKey}'`);
|
|
333
|
-
logger.info('Queued attribute processed successfully', {
|
|
334
|
-
attributeKey,
|
|
335
|
-
collectionId,
|
|
336
|
-
targetCollectionName: targetCollection.name,
|
|
337
|
-
operation: 'processQueue'
|
|
338
|
-
});
|
|
206
|
+
MessageFormatter.debug(`Attribute '${attributeKey}' already processed, removing from queue`);
|
|
207
|
+
logger.debug('Removing already processed attribute from queue', {
|
|
208
|
+
attributeKey,
|
|
209
|
+
collectionId,
|
|
210
|
+
queueIndex: i,
|
|
211
|
+
operation: 'processQueue'
|
|
212
|
+
});
|
|
213
|
+
queuedOperations.splice(i, 1);
|
|
214
|
+
continue;
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
let targetCollection: Models.Collection | undefined;
|
|
218
|
+
|
|
219
|
+
// Resolve the target collection (where the attribute will be created)
|
|
220
|
+
try {
|
|
221
|
+
targetCollection = await tryAwaitWithRetry(
|
|
222
|
+
async () => {
|
|
223
|
+
if ('getMetadata' in db && typeof db.getMetadata === 'function') {
|
|
224
|
+
// DatabaseAdapter
|
|
225
|
+
return (await (db as DatabaseAdapter).getTable({ databaseId: dbId, tableId: collectionId })).data;
|
|
226
|
+
} else {
|
|
227
|
+
// Legacy Databases
|
|
228
|
+
return await (db as Databases).getCollection(dbId, collectionId);
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
);
|
|
232
|
+
} catch (e) {
|
|
233
|
+
const errorMessage = e instanceof Error ? e.message : String(e);
|
|
234
|
+
MessageFormatter.error(`Target collection ${collectionId} not found, removing from queue`);
|
|
235
|
+
logger.error('Target collection not found during queue processing', {
|
|
236
|
+
collectionId,
|
|
237
|
+
attributeKey,
|
|
238
|
+
error: errorMessage,
|
|
239
|
+
operation: 'processQueue'
|
|
240
|
+
});
|
|
241
|
+
queuedOperations.splice(i, 1);
|
|
242
|
+
continue;
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
// For relationship attributes, ensure the related collection exists
|
|
246
|
+
let canProcess = true;
|
|
247
|
+
if (operation.attribute.type === "relationship") {
|
|
248
|
+
const relatedCollection = operation.attribute.relatedCollection;
|
|
249
|
+
if (relatedCollection) {
|
|
250
|
+
// Try to resolve related collection by ID first, then by name
|
|
251
|
+
let relatedFound = false;
|
|
252
|
+
|
|
253
|
+
try {
|
|
254
|
+
await tryAwaitWithRetry(
|
|
255
|
+
async () => {
|
|
256
|
+
if ('getMetadata' in db && typeof db.getMetadata === 'function') {
|
|
257
|
+
// DatabaseAdapter
|
|
258
|
+
return (await (db as DatabaseAdapter).getTable({ databaseId: dbId, tableId: relatedCollection })).data;
|
|
259
|
+
} else {
|
|
260
|
+
// Legacy Databases
|
|
261
|
+
return await (db as Databases).getCollection(dbId, relatedCollection);
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
);
|
|
265
|
+
relatedFound = true;
|
|
266
|
+
nameToIdMapping.set(relatedCollection, relatedCollection); // Cache the ID mapping
|
|
267
|
+
} catch (_) {
|
|
268
|
+
// Try by name lookup
|
|
269
|
+
const cachedId = nameToIdMapping.get(relatedCollection);
|
|
270
|
+
if (cachedId) {
|
|
271
|
+
try {
|
|
272
|
+
await tryAwaitWithRetry(
|
|
273
|
+
async () => {
|
|
274
|
+
if ('getMetadata' in db && typeof db.getMetadata === 'function') {
|
|
275
|
+
// DatabaseAdapter
|
|
276
|
+
return (await (db as DatabaseAdapter).getTable({ databaseId: dbId, tableId: cachedId })).data;
|
|
277
|
+
} else {
|
|
278
|
+
// Legacy Databases
|
|
279
|
+
return await (db as Databases).getCollection(dbId, cachedId);
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
);
|
|
283
|
+
relatedFound = true;
|
|
284
|
+
} catch (_) {
|
|
285
|
+
nameToIdMapping.delete(relatedCollection); // Remove stale cache
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
if (!relatedFound) {
|
|
290
|
+
// Final attempt: search by name
|
|
291
|
+
try {
|
|
292
|
+
const collections = 'getMetadata' in db && typeof db.getMetadata === 'function'
|
|
293
|
+
? await (db as DatabaseAdapter).listTables({ databaseId: dbId, queries: [Query.equal("name", relatedCollection)] })
|
|
294
|
+
: await (db as Databases).listCollections(dbId, [Query.equal("name", relatedCollection)]);
|
|
295
|
+
|
|
296
|
+
if (collections.total && collections.total > 0) {
|
|
297
|
+
const firstCollection = 'getMetadata' in db && typeof db.getMetadata === 'function'
|
|
298
|
+
? (collections as any).tables?.[0]
|
|
299
|
+
: (collections as any).collections?.[0];
|
|
300
|
+
nameToIdMapping.set(relatedCollection, firstCollection.$id);
|
|
301
|
+
relatedFound = true;
|
|
302
|
+
}
|
|
303
|
+
} catch (_) {
|
|
304
|
+
// Related collection truly doesn't exist yet
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
if (!relatedFound) {
|
|
310
|
+
MessageFormatter.warning(
|
|
311
|
+
`Related collection '${relatedCollection}' not ready for attribute '${attributeKey}', keeping in queue`
|
|
312
|
+
);
|
|
313
|
+
canProcess = false;
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
if (canProcess && targetCollection) {
|
|
319
|
+
MessageFormatter.progress(
|
|
320
|
+
`Processing queued ${operation.attribute.type} attribute: '${attributeKey}' for collection: '${targetCollection.name}'`
|
|
321
|
+
);
|
|
322
|
+
|
|
323
|
+
const success = await createOrUpdateAttributeWithStatusCheck(
|
|
324
|
+
db,
|
|
325
|
+
dbId,
|
|
326
|
+
targetCollection,
|
|
327
|
+
operation.attribute
|
|
328
|
+
);
|
|
329
|
+
|
|
330
|
+
if (success) {
|
|
331
|
+
MessageFormatter.success(`Successfully processed queued attribute: '${attributeKey}'`);
|
|
332
|
+
logger.info('Queued attribute processed successfully', {
|
|
333
|
+
attributeKey,
|
|
334
|
+
collectionId,
|
|
335
|
+
targetCollectionName: targetCollection.name,
|
|
336
|
+
operation: 'processQueue'
|
|
337
|
+
});
|
|
339
338
|
markAttributeProcessed(dbId, collectionId, attributeKey);
|
|
340
|
-
queuedOperations.splice(i, 1);
|
|
341
|
-
progress = true;
|
|
342
|
-
} else {
|
|
343
|
-
MessageFormatter.error(`Failed to process queued attribute: '${attributeKey}', removing from queue`);
|
|
344
|
-
logger.error('Failed to process queued attribute', {
|
|
345
|
-
attributeKey,
|
|
346
|
-
collectionId,
|
|
347
|
-
targetCollectionName: targetCollection.name,
|
|
348
|
-
operation: 'processQueue'
|
|
349
|
-
});
|
|
350
|
-
queuedOperations.splice(i, 1);
|
|
351
|
-
}
|
|
352
|
-
}
|
|
353
|
-
}
|
|
354
|
-
|
|
355
|
-
if (queuedOperations.length === 0) {
|
|
356
|
-
break;
|
|
357
|
-
}
|
|
358
|
-
|
|
359
|
-
MessageFormatter.info(`Remaining operations after attempt ${attempts}: ${queuedOperations.length}`);
|
|
360
|
-
}
|
|
361
|
-
|
|
362
|
-
if (queuedOperations.length > 0) {
|
|
363
|
-
MessageFormatter.warning(
|
|
364
|
-
`${queuedOperations.length} operations remain unresolved after ${maxAttempts} attempts:`
|
|
365
|
-
);
|
|
366
|
-
queuedOperations.forEach((op, index) => {
|
|
367
|
-
MessageFormatter.warning(
|
|
368
|
-
` ${index + 1}. ${op.attribute?.type} attribute '${op.attribute?.key}' for collection ${op.collectionId}`
|
|
369
|
-
);
|
|
370
|
-
});
|
|
371
|
-
MessageFormatter.warning("These may have unmet dependencies or require manual intervention");
|
|
372
|
-
} else {
|
|
373
|
-
MessageFormatter.success("All queued operations processed successfully");
|
|
374
|
-
}
|
|
375
|
-
|
|
376
|
-
MessageFormatter.section(`Surgical queue processing complete for ${dbId}`);
|
|
377
|
-
};
|
|
339
|
+
queuedOperations.splice(i, 1);
|
|
340
|
+
progress = true;
|
|
341
|
+
} else {
|
|
342
|
+
MessageFormatter.error(`Failed to process queued attribute: '${attributeKey}', removing from queue`);
|
|
343
|
+
logger.error('Failed to process queued attribute', {
|
|
344
|
+
attributeKey,
|
|
345
|
+
collectionId,
|
|
346
|
+
targetCollectionName: targetCollection.name,
|
|
347
|
+
operation: 'processQueue'
|
|
348
|
+
});
|
|
349
|
+
queuedOperations.splice(i, 1);
|
|
350
|
+
}
|
|
351
|
+
}
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
if (queuedOperations.length === 0) {
|
|
355
|
+
break;
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
MessageFormatter.info(`Remaining operations after attempt ${attempts}: ${queuedOperations.length}`);
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
if (queuedOperations.length > 0) {
|
|
362
|
+
MessageFormatter.warning(
|
|
363
|
+
`${queuedOperations.length} operations remain unresolved after ${maxAttempts} attempts:`
|
|
364
|
+
);
|
|
365
|
+
queuedOperations.forEach((op, index) => {
|
|
366
|
+
MessageFormatter.warning(
|
|
367
|
+
` ${index + 1}. ${op.attribute?.type} attribute '${op.attribute?.key}' for collection ${op.collectionId}`
|
|
368
|
+
);
|
|
369
|
+
});
|
|
370
|
+
MessageFormatter.warning("These may have unmet dependencies or require manual intervention");
|
|
371
|
+
} else {
|
|
372
|
+
MessageFormatter.success("All queued operations processed successfully");
|
|
373
|
+
}
|
|
374
|
+
|
|
375
|
+
MessageFormatter.section(`Surgical queue processing complete for ${dbId}`);
|
|
376
|
+
};
|