@njdamstra/appwrite-utils-cli 1.8.9
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 +19 -0
- package/README.md +1133 -0
- package/dist/adapters/AdapterFactory.d.ts +94 -0
- package/dist/adapters/AdapterFactory.js +405 -0
- package/dist/adapters/DatabaseAdapter.d.ts +233 -0
- package/dist/adapters/DatabaseAdapter.js +50 -0
- package/dist/adapters/LegacyAdapter.d.ts +50 -0
- package/dist/adapters/LegacyAdapter.js +612 -0
- package/dist/adapters/TablesDBAdapter.d.ts +45 -0
- package/dist/adapters/TablesDBAdapter.js +571 -0
- package/dist/adapters/index.d.ts +11 -0
- package/dist/adapters/index.js +12 -0
- package/dist/backups/operations/bucketBackup.d.ts +19 -0
- package/dist/backups/operations/bucketBackup.js +197 -0
- package/dist/backups/operations/collectionBackup.d.ts +30 -0
- package/dist/backups/operations/collectionBackup.js +201 -0
- package/dist/backups/operations/comprehensiveBackup.d.ts +25 -0
- package/dist/backups/operations/comprehensiveBackup.js +238 -0
- package/dist/backups/schemas/bucketManifest.d.ts +93 -0
- package/dist/backups/schemas/bucketManifest.js +33 -0
- package/dist/backups/schemas/comprehensiveManifest.d.ts +108 -0
- package/dist/backups/schemas/comprehensiveManifest.js +32 -0
- package/dist/backups/tracking/centralizedTracking.d.ts +34 -0
- package/dist/backups/tracking/centralizedTracking.js +274 -0
- package/dist/cli/commands/configCommands.d.ts +8 -0
- package/dist/cli/commands/configCommands.js +166 -0
- package/dist/cli/commands/databaseCommands.d.ts +13 -0
- package/dist/cli/commands/databaseCommands.js +554 -0
- package/dist/cli/commands/functionCommands.d.ts +7 -0
- package/dist/cli/commands/functionCommands.js +330 -0
- package/dist/cli/commands/schemaCommands.d.ts +7 -0
- package/dist/cli/commands/schemaCommands.js +169 -0
- package/dist/cli/commands/storageCommands.d.ts +5 -0
- package/dist/cli/commands/storageCommands.js +143 -0
- package/dist/cli/commands/transferCommands.d.ts +5 -0
- package/dist/cli/commands/transferCommands.js +384 -0
- package/dist/collections/attributes.d.ts +13 -0
- package/dist/collections/attributes.js +1364 -0
- package/dist/collections/indexes.d.ts +12 -0
- package/dist/collections/indexes.js +217 -0
- package/dist/collections/methods.d.ts +19 -0
- package/dist/collections/methods.js +682 -0
- package/dist/collections/tableOperations.d.ts +86 -0
- package/dist/collections/tableOperations.js +434 -0
- package/dist/collections/transferOperations.d.ts +8 -0
- package/dist/collections/transferOperations.js +412 -0
- package/dist/collections/wipeOperations.d.ts +16 -0
- package/dist/collections/wipeOperations.js +233 -0
- package/dist/config/ConfigManager.d.ts +445 -0
- package/dist/config/ConfigManager.js +625 -0
- package/dist/config/configMigration.d.ts +87 -0
- package/dist/config/configMigration.js +390 -0
- package/dist/config/configValidation.d.ts +66 -0
- package/dist/config/configValidation.js +358 -0
- package/dist/config/index.d.ts +8 -0
- package/dist/config/index.js +7 -0
- package/dist/config/services/ConfigDiscoveryService.d.ts +126 -0
- package/dist/config/services/ConfigDiscoveryService.js +374 -0
- package/dist/config/services/ConfigLoaderService.d.ts +129 -0
- package/dist/config/services/ConfigLoaderService.js +540 -0
- package/dist/config/services/ConfigMergeService.d.ts +208 -0
- package/dist/config/services/ConfigMergeService.js +308 -0
- package/dist/config/services/ConfigValidationService.d.ts +214 -0
- package/dist/config/services/ConfigValidationService.js +310 -0
- package/dist/config/services/SessionAuthService.d.ts +225 -0
- package/dist/config/services/SessionAuthService.js +456 -0
- package/dist/config/services/__tests__/ConfigMergeService.test.d.ts +1 -0
- package/dist/config/services/__tests__/ConfigMergeService.test.js +271 -0
- package/dist/config/services/index.d.ts +13 -0
- package/dist/config/services/index.js +10 -0
- package/dist/config/yamlConfig.d.ts +722 -0
- package/dist/config/yamlConfig.js +702 -0
- package/dist/databases/methods.d.ts +6 -0
- package/dist/databases/methods.js +35 -0
- package/dist/databases/setup.d.ts +5 -0
- package/dist/databases/setup.js +45 -0
- package/dist/examples/yamlTerminologyExample.d.ts +42 -0
- package/dist/examples/yamlTerminologyExample.js +272 -0
- package/dist/functions/deployments.d.ts +4 -0
- package/dist/functions/deployments.js +146 -0
- package/dist/functions/fnConfigDiscovery.d.ts +3 -0
- package/dist/functions/fnConfigDiscovery.js +108 -0
- package/dist/functions/methods.d.ts +16 -0
- package/dist/functions/methods.js +162 -0
- package/dist/functions/pathResolution.d.ts +37 -0
- package/dist/functions/pathResolution.js +185 -0
- package/dist/functions/templates/count-docs-in-collection/README.md +54 -0
- package/dist/functions/templates/count-docs-in-collection/src/main.ts +159 -0
- package/dist/functions/templates/count-docs-in-collection/src/request.ts +9 -0
- package/dist/functions/templates/hono-typescript/README.md +286 -0
- package/dist/functions/templates/hono-typescript/src/adapters/request.ts +74 -0
- package/dist/functions/templates/hono-typescript/src/adapters/response.ts +106 -0
- package/dist/functions/templates/hono-typescript/src/app.ts +180 -0
- package/dist/functions/templates/hono-typescript/src/context.ts +103 -0
- package/dist/functions/templates/hono-typescript/src/index.ts +54 -0
- package/dist/functions/templates/hono-typescript/src/middleware/appwrite.ts +119 -0
- package/dist/functions/templates/typescript-node/README.md +32 -0
- package/dist/functions/templates/typescript-node/src/context.ts +103 -0
- package/dist/functions/templates/typescript-node/src/index.ts +29 -0
- package/dist/functions/templates/uv/README.md +31 -0
- package/dist/functions/templates/uv/pyproject.toml +30 -0
- package/dist/functions/templates/uv/src/__init__.py +0 -0
- package/dist/functions/templates/uv/src/context.py +125 -0
- package/dist/functions/templates/uv/src/index.py +46 -0
- package/dist/init.d.ts +2 -0
- package/dist/init.js +57 -0
- package/dist/interactiveCLI.d.ts +31 -0
- package/dist/interactiveCLI.js +898 -0
- package/dist/main.d.ts +2 -0
- package/dist/main.js +1172 -0
- package/dist/migrations/afterImportActions.d.ts +17 -0
- package/dist/migrations/afterImportActions.js +306 -0
- package/dist/migrations/appwriteToX.d.ts +211 -0
- package/dist/migrations/appwriteToX.js +491 -0
- package/dist/migrations/comprehensiveTransfer.d.ts +147 -0
- package/dist/migrations/comprehensiveTransfer.js +1317 -0
- package/dist/migrations/dataLoader.d.ts +753 -0
- package/dist/migrations/dataLoader.js +1250 -0
- package/dist/migrations/importController.d.ts +23 -0
- package/dist/migrations/importController.js +268 -0
- package/dist/migrations/importDataActions.d.ts +50 -0
- package/dist/migrations/importDataActions.js +230 -0
- package/dist/migrations/relationships.d.ts +29 -0
- package/dist/migrations/relationships.js +204 -0
- package/dist/migrations/services/DataTransformationService.d.ts +55 -0
- package/dist/migrations/services/DataTransformationService.js +158 -0
- package/dist/migrations/services/FileHandlerService.d.ts +75 -0
- package/dist/migrations/services/FileHandlerService.js +236 -0
- package/dist/migrations/services/ImportOrchestrator.d.ts +97 -0
- package/dist/migrations/services/ImportOrchestrator.js +485 -0
- package/dist/migrations/services/RateLimitManager.d.ts +138 -0
- package/dist/migrations/services/RateLimitManager.js +279 -0
- package/dist/migrations/services/RelationshipResolver.d.ts +120 -0
- package/dist/migrations/services/RelationshipResolver.js +332 -0
- package/dist/migrations/services/UserMappingService.d.ts +109 -0
- package/dist/migrations/services/UserMappingService.js +277 -0
- package/dist/migrations/services/ValidationService.d.ts +74 -0
- package/dist/migrations/services/ValidationService.js +260 -0
- package/dist/migrations/transfer.d.ts +26 -0
- package/dist/migrations/transfer.js +608 -0
- package/dist/migrations/yaml/YamlImportConfigLoader.d.ts +131 -0
- package/dist/migrations/yaml/YamlImportConfigLoader.js +383 -0
- package/dist/migrations/yaml/YamlImportIntegration.d.ts +93 -0
- package/dist/migrations/yaml/YamlImportIntegration.js +341 -0
- package/dist/migrations/yaml/generateImportSchemas.d.ts +30 -0
- package/dist/migrations/yaml/generateImportSchemas.js +1327 -0
- package/dist/schemas/authUser.d.ts +24 -0
- package/dist/schemas/authUser.js +17 -0
- package/dist/setup.d.ts +2 -0
- package/dist/setup.js +5 -0
- package/dist/setupCommands.d.ts +58 -0
- package/dist/setupCommands.js +490 -0
- package/dist/setupController.d.ts +9 -0
- package/dist/setupController.js +34 -0
- package/dist/shared/attributeMapper.d.ts +20 -0
- package/dist/shared/attributeMapper.js +203 -0
- package/dist/shared/backupMetadataSchema.d.ts +94 -0
- package/dist/shared/backupMetadataSchema.js +38 -0
- package/dist/shared/backupTracking.d.ts +18 -0
- package/dist/shared/backupTracking.js +176 -0
- package/dist/shared/confirmationDialogs.d.ts +75 -0
- package/dist/shared/confirmationDialogs.js +236 -0
- package/dist/shared/errorUtils.d.ts +54 -0
- package/dist/shared/errorUtils.js +95 -0
- package/dist/shared/functionManager.d.ts +48 -0
- package/dist/shared/functionManager.js +336 -0
- package/dist/shared/indexManager.d.ts +24 -0
- package/dist/shared/indexManager.js +151 -0
- package/dist/shared/jsonSchemaGenerator.d.ts +50 -0
- package/dist/shared/jsonSchemaGenerator.js +290 -0
- package/dist/shared/logging.d.ts +61 -0
- package/dist/shared/logging.js +116 -0
- package/dist/shared/messageFormatter.d.ts +39 -0
- package/dist/shared/messageFormatter.js +162 -0
- package/dist/shared/migrationHelpers.d.ts +61 -0
- package/dist/shared/migrationHelpers.js +145 -0
- package/dist/shared/operationLogger.d.ts +10 -0
- package/dist/shared/operationLogger.js +12 -0
- package/dist/shared/operationQueue.d.ts +40 -0
- package/dist/shared/operationQueue.js +311 -0
- package/dist/shared/operationsTable.d.ts +26 -0
- package/dist/shared/operationsTable.js +286 -0
- package/dist/shared/operationsTableSchema.d.ts +48 -0
- package/dist/shared/operationsTableSchema.js +35 -0
- package/dist/shared/progressManager.d.ts +62 -0
- package/dist/shared/progressManager.js +215 -0
- package/dist/shared/pydanticModelGenerator.d.ts +17 -0
- package/dist/shared/pydanticModelGenerator.js +615 -0
- package/dist/shared/relationshipExtractor.d.ts +56 -0
- package/dist/shared/relationshipExtractor.js +138 -0
- package/dist/shared/schemaGenerator.d.ts +40 -0
- package/dist/shared/schemaGenerator.js +556 -0
- package/dist/shared/selectionDialogs.d.ts +214 -0
- package/dist/shared/selectionDialogs.js +544 -0
- package/dist/storage/backupCompression.d.ts +20 -0
- package/dist/storage/backupCompression.js +67 -0
- package/dist/storage/methods.d.ts +32 -0
- package/dist/storage/methods.js +472 -0
- package/dist/storage/schemas.d.ts +842 -0
- package/dist/storage/schemas.js +175 -0
- package/dist/types.d.ts +4 -0
- package/dist/types.js +3 -0
- package/dist/users/methods.d.ts +16 -0
- package/dist/users/methods.js +277 -0
- package/dist/utils/ClientFactory.d.ts +87 -0
- package/dist/utils/ClientFactory.js +212 -0
- package/dist/utils/configDiscovery.d.ts +78 -0
- package/dist/utils/configDiscovery.js +472 -0
- package/dist/utils/configMigration.d.ts +1 -0
- package/dist/utils/configMigration.js +261 -0
- package/dist/utils/constantsGenerator.d.ts +31 -0
- package/dist/utils/constantsGenerator.js +321 -0
- package/dist/utils/dataConverters.d.ts +46 -0
- package/dist/utils/dataConverters.js +139 -0
- package/dist/utils/directoryUtils.d.ts +22 -0
- package/dist/utils/directoryUtils.js +59 -0
- package/dist/utils/getClientFromConfig.d.ts +39 -0
- package/dist/utils/getClientFromConfig.js +199 -0
- package/dist/utils/helperFunctions.d.ts +63 -0
- package/dist/utils/helperFunctions.js +156 -0
- package/dist/utils/index.d.ts +2 -0
- package/dist/utils/index.js +2 -0
- package/dist/utils/loadConfigs.d.ts +50 -0
- package/dist/utils/loadConfigs.js +358 -0
- package/dist/utils/pathResolvers.d.ts +53 -0
- package/dist/utils/pathResolvers.js +72 -0
- package/dist/utils/projectConfig.d.ts +119 -0
- package/dist/utils/projectConfig.js +171 -0
- package/dist/utils/retryFailedPromises.d.ts +2 -0
- package/dist/utils/retryFailedPromises.js +23 -0
- package/dist/utils/sessionAuth.d.ts +48 -0
- package/dist/utils/sessionAuth.js +164 -0
- package/dist/utils/setupFiles.d.ts +4 -0
- package/dist/utils/setupFiles.js +1192 -0
- package/dist/utils/typeGuards.d.ts +35 -0
- package/dist/utils/typeGuards.js +57 -0
- package/dist/utils/validationRules.d.ts +43 -0
- package/dist/utils/validationRules.js +42 -0
- package/dist/utils/versionDetection.d.ts +58 -0
- package/dist/utils/versionDetection.js +251 -0
- package/dist/utils/yamlConverter.d.ts +100 -0
- package/dist/utils/yamlConverter.js +428 -0
- package/dist/utils/yamlLoader.d.ts +70 -0
- package/dist/utils/yamlLoader.js +267 -0
- package/dist/utilsController.d.ts +106 -0
- package/dist/utilsController.js +863 -0
- package/package.json +75 -0
- package/scripts/copy-templates.ts +23 -0
- package/src/adapters/AdapterFactory.ts +510 -0
- package/src/adapters/DatabaseAdapter.ts +306 -0
- package/src/adapters/LegacyAdapter.ts +841 -0
- package/src/adapters/TablesDBAdapter.ts +773 -0
- package/src/adapters/index.ts +37 -0
- package/src/backups/operations/bucketBackup.ts +277 -0
- package/src/backups/operations/collectionBackup.ts +310 -0
- package/src/backups/operations/comprehensiveBackup.ts +342 -0
- package/src/backups/schemas/bucketManifest.ts +78 -0
- package/src/backups/schemas/comprehensiveManifest.ts +76 -0
- package/src/backups/tracking/centralizedTracking.ts +352 -0
- package/src/cli/commands/configCommands.ts +201 -0
- package/src/cli/commands/databaseCommands.ts +749 -0
- package/src/cli/commands/functionCommands.ts +418 -0
- package/src/cli/commands/schemaCommands.ts +200 -0
- package/src/cli/commands/storageCommands.ts +152 -0
- package/src/cli/commands/transferCommands.ts +457 -0
- package/src/collections/attributes.ts +2054 -0
- package/src/collections/attributes.ts.backup +1555 -0
- package/src/collections/indexes.ts +352 -0
- package/src/collections/methods.ts +745 -0
- package/src/collections/tableOperations.ts +506 -0
- package/src/collections/transferOperations.ts +590 -0
- package/src/collections/wipeOperations.ts +346 -0
- package/src/config/ConfigManager.ts +808 -0
- package/src/config/README.md +274 -0
- package/src/config/configMigration.ts +575 -0
- package/src/config/configValidation.ts +445 -0
- package/src/config/index.ts +10 -0
- package/src/config/services/ConfigDiscoveryService.ts +463 -0
- package/src/config/services/ConfigLoaderService.ts +740 -0
- package/src/config/services/ConfigMergeService.ts +388 -0
- package/src/config/services/ConfigValidationService.ts +394 -0
- package/src/config/services/SessionAuthService.ts +565 -0
- package/src/config/services/__tests__/ConfigMergeService.test.ts +351 -0
- package/src/config/services/index.ts +29 -0
- package/src/config/yamlConfig.ts +761 -0
- package/src/databases/methods.ts +49 -0
- package/src/databases/setup.ts +77 -0
- package/src/examples/yamlTerminologyExample.ts +346 -0
- package/src/functions/deployments.ts +220 -0
- package/src/functions/fnConfigDiscovery.ts +103 -0
- package/src/functions/methods.ts +271 -0
- package/src/functions/pathResolution.ts +227 -0
- package/src/functions/templates/count-docs-in-collection/README.md +54 -0
- package/src/functions/templates/count-docs-in-collection/src/main.ts +159 -0
- package/src/functions/templates/count-docs-in-collection/src/request.ts +9 -0
- package/src/functions/templates/hono-typescript/README.md +286 -0
- package/src/functions/templates/hono-typescript/src/adapters/request.ts +74 -0
- package/src/functions/templates/hono-typescript/src/adapters/response.ts +106 -0
- package/src/functions/templates/hono-typescript/src/app.ts +180 -0
- package/src/functions/templates/hono-typescript/src/context.ts +103 -0
- package/src/functions/templates/hono-typescript/src/index.ts +54 -0
- package/src/functions/templates/hono-typescript/src/middleware/appwrite.ts +119 -0
- package/src/functions/templates/typescript-node/README.md +32 -0
- package/src/functions/templates/typescript-node/src/context.ts +103 -0
- package/src/functions/templates/typescript-node/src/index.ts +29 -0
- package/src/functions/templates/uv/README.md +31 -0
- package/src/functions/templates/uv/pyproject.toml +30 -0
- package/src/functions/templates/uv/src/__init__.py +0 -0
- package/src/functions/templates/uv/src/context.py +125 -0
- package/src/functions/templates/uv/src/index.py +46 -0
- package/src/init.ts +62 -0
- package/src/interactiveCLI.ts +1136 -0
- package/src/main.ts +1661 -0
- package/src/migrations/afterImportActions.ts +580 -0
- package/src/migrations/appwriteToX.ts +664 -0
- package/src/migrations/comprehensiveTransfer.ts +2285 -0
- package/src/migrations/dataLoader.ts +1702 -0
- package/src/migrations/importController.ts +428 -0
- package/src/migrations/importDataActions.ts +315 -0
- package/src/migrations/relationships.ts +334 -0
- package/src/migrations/services/DataTransformationService.ts +196 -0
- package/src/migrations/services/FileHandlerService.ts +311 -0
- package/src/migrations/services/ImportOrchestrator.ts +666 -0
- package/src/migrations/services/RateLimitManager.ts +363 -0
- package/src/migrations/services/RelationshipResolver.ts +461 -0
- package/src/migrations/services/UserMappingService.ts +345 -0
- package/src/migrations/services/ValidationService.ts +349 -0
- package/src/migrations/transfer.ts +1068 -0
- package/src/migrations/yaml/YamlImportConfigLoader.ts +439 -0
- package/src/migrations/yaml/YamlImportIntegration.ts +446 -0
- package/src/migrations/yaml/generateImportSchemas.ts +1354 -0
- package/src/schemas/authUser.ts +23 -0
- package/src/setup.ts +8 -0
- package/src/setupCommands.ts +603 -0
- package/src/setupController.ts +43 -0
- package/src/shared/attributeMapper.ts +229 -0
- package/src/shared/backupMetadataSchema.ts +93 -0
- package/src/shared/backupTracking.ts +211 -0
- package/src/shared/confirmationDialogs.ts +327 -0
- package/src/shared/errorUtils.ts +110 -0
- package/src/shared/functionManager.ts +525 -0
- package/src/shared/indexManager.ts +254 -0
- package/src/shared/jsonSchemaGenerator.ts +383 -0
- package/src/shared/logging.ts +149 -0
- package/src/shared/messageFormatter.ts +208 -0
- package/src/shared/migrationHelpers.ts +232 -0
- package/src/shared/operationLogger.ts +20 -0
- package/src/shared/operationQueue.ts +377 -0
- package/src/shared/operationsTable.ts +338 -0
- package/src/shared/operationsTableSchema.ts +60 -0
- package/src/shared/progressManager.ts +278 -0
- package/src/shared/pydanticModelGenerator.ts +618 -0
- package/src/shared/relationshipExtractor.ts +214 -0
- package/src/shared/schemaGenerator.ts +644 -0
- package/src/shared/selectionDialogs.ts +749 -0
- package/src/storage/backupCompression.ts +88 -0
- package/src/storage/methods.ts +698 -0
- package/src/storage/schemas.ts +205 -0
- package/src/types/node-appwrite-tablesdb.d.ts +44 -0
- package/src/types.ts +9 -0
- package/src/users/methods.ts +359 -0
- package/src/utils/ClientFactory.ts +240 -0
- package/src/utils/configDiscovery.ts +557 -0
- package/src/utils/configMigration.ts +348 -0
- package/src/utils/constantsGenerator.ts +369 -0
- package/src/utils/dataConverters.ts +159 -0
- package/src/utils/directoryUtils.ts +61 -0
- package/src/utils/getClientFromConfig.ts +257 -0
- package/src/utils/helperFunctions.ts +228 -0
- package/src/utils/index.ts +2 -0
- package/src/utils/loadConfigs.ts +449 -0
- package/src/utils/pathResolvers.ts +81 -0
- package/src/utils/projectConfig.ts +299 -0
- package/src/utils/retryFailedPromises.ts +29 -0
- package/src/utils/sessionAuth.ts +230 -0
- package/src/utils/setupFiles.ts +1238 -0
- package/src/utils/typeGuards.ts +65 -0
- package/src/utils/validationRules.ts +88 -0
- package/src/utils/versionDetection.ts +292 -0
- package/src/utils/yamlConverter.ts +542 -0
- package/src/utils/yamlLoader.ts +371 -0
- package/src/utilsController.ts +1203 -0
- package/tests/README.md +497 -0
- package/tests/adapters/AdapterFactory.test.ts +277 -0
- package/tests/integration/syncOperations.test.ts +463 -0
- package/tests/jest.config.js +25 -0
- package/tests/migration/configMigration.test.ts +546 -0
- package/tests/setup.ts +62 -0
- package/tests/testUtils.ts +340 -0
- package/tests/utils/loadConfigs.test.ts +350 -0
- package/tests/validation/configValidation.test.ts +412 -0
- package/tsconfig.json +44 -0
package/package.json
ADDED
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@njdamstra/appwrite-utils-cli",
|
|
3
|
+
"description": "Appwrite Utility Functions to help with database management, data conversion, data import, migrations, and much more. Meant to be used as a CLI tool, I do not recommend installing this in frontend environments.",
|
|
4
|
+
"version": "1.8.9",
|
|
5
|
+
"main": "src/main.ts",
|
|
6
|
+
"type": "module",
|
|
7
|
+
"repository": {
|
|
8
|
+
"type": "git",
|
|
9
|
+
"url": "https://github.com/njdamstra/AppwriteUtils"
|
|
10
|
+
},
|
|
11
|
+
"author": "Zach Handley <zach@blackleafdigital.com> (https://zachhandley.com)",
|
|
12
|
+
"keywords": [
|
|
13
|
+
"appwrite",
|
|
14
|
+
"cli",
|
|
15
|
+
"utils",
|
|
16
|
+
"migrations",
|
|
17
|
+
"data",
|
|
18
|
+
"database",
|
|
19
|
+
"import",
|
|
20
|
+
"migration",
|
|
21
|
+
"utility"
|
|
22
|
+
],
|
|
23
|
+
"bin": {
|
|
24
|
+
"appwrite-migrate": "./dist/main.js"
|
|
25
|
+
},
|
|
26
|
+
"scripts": {
|
|
27
|
+
"build": "bun run tsc && bun run copy-templates",
|
|
28
|
+
"prebuild": "rm -rf dist",
|
|
29
|
+
"copy-templates": "tsx scripts/copy-templates.ts",
|
|
30
|
+
"start": "tsx --no-cache src/main.ts",
|
|
31
|
+
"deploy": "bun run build && npm publish --access public",
|
|
32
|
+
"test": "jest",
|
|
33
|
+
"test:watch": "jest --watch",
|
|
34
|
+
"test:coverage": "jest --coverage",
|
|
35
|
+
"test:ci": "jest --ci --coverage --watchAll=false",
|
|
36
|
+
"postinstall": "echo 'This package is intended for CLI use only and should not be added as a dependency in other projects.'"
|
|
37
|
+
},
|
|
38
|
+
"dependencies": {
|
|
39
|
+
"@types/inquirer": "^9.0.8",
|
|
40
|
+
"@types/json-schema": "^7.0.15",
|
|
41
|
+
"@types/yargs": "^17.0.33",
|
|
42
|
+
"@njdamstra/appwrite-utils": "^1.6.5",
|
|
43
|
+
"chalk": "^5.4.1",
|
|
44
|
+
"cli-progress": "^3.12.0",
|
|
45
|
+
"commander": "^12.1.0",
|
|
46
|
+
"decimal.js": "^10.6.0",
|
|
47
|
+
"es-toolkit": "^1.39.4",
|
|
48
|
+
"ignore": "^6.0.2",
|
|
49
|
+
"inquirer": "^9.3.7",
|
|
50
|
+
"js-yaml": "^4.1.0",
|
|
51
|
+
"jszip": "^3.10.1",
|
|
52
|
+
"luxon": "^3.6.1",
|
|
53
|
+
"nanostores": "^0.10.3",
|
|
54
|
+
"node-appwrite": "^20.2.1",
|
|
55
|
+
"p-limit": "^6.2.0",
|
|
56
|
+
"tar": "^7.4.3",
|
|
57
|
+
"tsx": "^4.20.3",
|
|
58
|
+
"ulidx": "^2.4.1",
|
|
59
|
+
"winston": "^3.17.0",
|
|
60
|
+
"yargs": "^18.0.0",
|
|
61
|
+
"zod": "^4.0.0"
|
|
62
|
+
},
|
|
63
|
+
"devDependencies": {
|
|
64
|
+
"@jest/globals": "^29.7.0",
|
|
65
|
+
"@types/cli-progress": "^3.11.6",
|
|
66
|
+
"@types/jest": "^29.5.12",
|
|
67
|
+
"@types/js-yaml": "^4.0.9",
|
|
68
|
+
"@types/jszip": "^3.4.1",
|
|
69
|
+
"@types/lodash": "^4.17.18",
|
|
70
|
+
"@types/luxon": "^3.6.2",
|
|
71
|
+
"jest": "^29.7.0",
|
|
72
|
+
"ts-jest": "^29.1.2",
|
|
73
|
+
"typescript": "^5.8.3"
|
|
74
|
+
}
|
|
75
|
+
}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import { cpSync, mkdirSync, existsSync } from 'node:fs';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
|
|
4
|
+
const src = join(process.cwd(), 'src', 'functions', 'templates');
|
|
5
|
+
const dest = join(process.cwd(), 'dist', 'functions', 'templates');
|
|
6
|
+
|
|
7
|
+
// Verify source exists
|
|
8
|
+
if (!existsSync(src)) {
|
|
9
|
+
console.error('❌ Error: Template source directory not found:', src);
|
|
10
|
+
process.exit(1);
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
// Create destination directory
|
|
14
|
+
mkdirSync(dest, { recursive: true });
|
|
15
|
+
|
|
16
|
+
// Copy templates recursively
|
|
17
|
+
try {
|
|
18
|
+
cpSync(src, dest, { recursive: true });
|
|
19
|
+
console.log('✓ Templates copied to dist/functions/templates/');
|
|
20
|
+
} catch (error) {
|
|
21
|
+
console.error('❌ Failed to copy templates:', error);
|
|
22
|
+
process.exit(1);
|
|
23
|
+
}
|
|
@@ -0,0 +1,510 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AdapterFactory - Unified Client Creation with Automatic API Detection
|
|
3
|
+
*
|
|
4
|
+
* This factory creates the appropriate database adapter (TablesDB or Legacy)
|
|
5
|
+
* based on version detection and configuration. It handles dynamic SDK imports
|
|
6
|
+
* and provides a single entry point for all database operations.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import type { AppwriteConfig } from "@njdamstra/appwrite-utils";
|
|
10
|
+
import { detectAppwriteVersionCached, isVersionAtLeast, type ApiMode, type VersionDetectionResult } from "../utils/versionDetection.js";
|
|
11
|
+
import type { DatabaseAdapter } from './DatabaseAdapter.js';
|
|
12
|
+
import { TablesDBAdapter } from './TablesDBAdapter.js';
|
|
13
|
+
import { LegacyAdapter } from './LegacyAdapter.js';
|
|
14
|
+
import { logger } from '../shared/logging.js';
|
|
15
|
+
import { isValidSessionCookie } from '../utils/sessionAuth.js';
|
|
16
|
+
import { MessageFormatter } from '../shared/messageFormatter.js';
|
|
17
|
+
import { Client } from 'node-appwrite';
|
|
18
|
+
|
|
19
|
+
export interface AdapterFactoryConfig {
|
|
20
|
+
appwriteEndpoint: string;
|
|
21
|
+
appwriteProject: string;
|
|
22
|
+
appwriteKey?: string; // Made optional to support session-only auth
|
|
23
|
+
apiMode?: 'auto' | 'legacy' | 'tablesdb';
|
|
24
|
+
forceRefresh?: boolean; // Skip detection cache
|
|
25
|
+
sessionCookie?: string; // Session authentication support
|
|
26
|
+
authMethod?: 'session' | 'apikey' | 'auto'; // Authentication method preference
|
|
27
|
+
preConfiguredClient?: any; // Pre-configured authenticated client
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export interface AdapterFactoryResult {
|
|
31
|
+
adapter: DatabaseAdapter;
|
|
32
|
+
apiMode: ApiMode;
|
|
33
|
+
detectionResult?: VersionDetectionResult;
|
|
34
|
+
client: any;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* AdapterFactory - Main factory class for creating database adapters
|
|
39
|
+
*/
|
|
40
|
+
export class AdapterFactory {
|
|
41
|
+
private static cache = new Map<string, { adapter: DatabaseAdapter; timestamp: number }>();
|
|
42
|
+
private static readonly CACHE_DURATION = 5 * 60 * 1000; // 5 minutes
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Create a database adapter based on configuration and detection
|
|
46
|
+
*/
|
|
47
|
+
static async create(config: AdapterFactoryConfig): Promise<AdapterFactoryResult> {
|
|
48
|
+
const startTime = Date.now();
|
|
49
|
+
|
|
50
|
+
// Validate authentication configuration
|
|
51
|
+
this.validateAuthConfig(config);
|
|
52
|
+
|
|
53
|
+
const cacheKey = `${config.appwriteEndpoint}:${config.appwriteProject}:${config.apiMode || 'auto'}`;
|
|
54
|
+
|
|
55
|
+
logger.info('Creating database adapter', {
|
|
56
|
+
endpoint: config.appwriteEndpoint,
|
|
57
|
+
project: config.appwriteProject,
|
|
58
|
+
requestedApiMode: config.apiMode || 'auto',
|
|
59
|
+
authMethod: config.authMethod || 'auto',
|
|
60
|
+
hasSessionCookie: !!config.sessionCookie,
|
|
61
|
+
hasPreConfiguredClient: !!config.preConfiguredClient,
|
|
62
|
+
forceRefresh: config.forceRefresh,
|
|
63
|
+
cacheKey,
|
|
64
|
+
operation: 'AdapterFactory.create'
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
// Check cache first (unless force refresh)
|
|
68
|
+
if (!config.forceRefresh) {
|
|
69
|
+
const cached = this.getCachedAdapter(cacheKey);
|
|
70
|
+
if (cached) {
|
|
71
|
+
const cacheAge = Date.now() - cached.timestamp;
|
|
72
|
+
logger.info('Using cached adapter', {
|
|
73
|
+
cacheKey,
|
|
74
|
+
cacheAge,
|
|
75
|
+
apiMode: cached.adapter.getApiMode(),
|
|
76
|
+
operation: 'AdapterFactory.create'
|
|
77
|
+
});
|
|
78
|
+
return {
|
|
79
|
+
adapter: cached.adapter,
|
|
80
|
+
apiMode: cached.adapter.getApiMode(),
|
|
81
|
+
client: cached.adapter.getRawClient()
|
|
82
|
+
};
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Determine API mode
|
|
87
|
+
let apiMode: ApiMode;
|
|
88
|
+
let detectionResult: VersionDetectionResult | undefined;
|
|
89
|
+
|
|
90
|
+
if (config.apiMode && config.apiMode !== 'auto') {
|
|
91
|
+
// Use explicitly configured mode
|
|
92
|
+
apiMode = config.apiMode;
|
|
93
|
+
logger.info('Using explicitly configured API mode', {
|
|
94
|
+
apiMode,
|
|
95
|
+
endpoint: config.appwriteEndpoint,
|
|
96
|
+
operation: 'AdapterFactory.create'
|
|
97
|
+
});
|
|
98
|
+
} else {
|
|
99
|
+
// Auto-detect API mode
|
|
100
|
+
logger.info('Starting API mode auto-detection', {
|
|
101
|
+
endpoint: config.appwriteEndpoint,
|
|
102
|
+
project: config.appwriteProject,
|
|
103
|
+
forceRefresh: config.forceRefresh,
|
|
104
|
+
operation: 'AdapterFactory.create'
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
const detectionStartTime = Date.now();
|
|
108
|
+
// For version detection, we need some form of authentication
|
|
109
|
+
const authKey = config.appwriteKey || '';
|
|
110
|
+
detectionResult = await detectAppwriteVersionCached(
|
|
111
|
+
config.appwriteEndpoint,
|
|
112
|
+
config.appwriteProject,
|
|
113
|
+
authKey,
|
|
114
|
+
config.forceRefresh
|
|
115
|
+
);
|
|
116
|
+
const detectionDuration = Date.now() - detectionStartTime;
|
|
117
|
+
|
|
118
|
+
apiMode = detectionResult.apiMode;
|
|
119
|
+
|
|
120
|
+
logger.info('API mode detection completed', {
|
|
121
|
+
apiMode,
|
|
122
|
+
detectionMethod: detectionResult.detectionMethod,
|
|
123
|
+
confidence: detectionResult.confidence,
|
|
124
|
+
serverVersion: detectionResult.serverVersion,
|
|
125
|
+
detectionDuration,
|
|
126
|
+
endpoint: config.appwriteEndpoint,
|
|
127
|
+
operation: 'AdapterFactory.create'
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
// Add version-based safety check to prevent using TablesDB on old servers
|
|
131
|
+
if (detectionResult.serverVersion &&
|
|
132
|
+
!isVersionAtLeast(detectionResult.serverVersion, '1.8.0') &&
|
|
133
|
+
apiMode === 'tablesdb') {
|
|
134
|
+
logger.warn('Overriding TablesDB detection - server version too old', {
|
|
135
|
+
serverVersion: detectionResult.serverVersion,
|
|
136
|
+
detectedMode: apiMode,
|
|
137
|
+
overrideMode: 'legacy',
|
|
138
|
+
operation: 'AdapterFactory.create'
|
|
139
|
+
});
|
|
140
|
+
apiMode = 'legacy';
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// Create appropriate adapter
|
|
145
|
+
logger.info('Creating adapter instance', {
|
|
146
|
+
apiMode,
|
|
147
|
+
endpoint: config.appwriteEndpoint,
|
|
148
|
+
operation: 'AdapterFactory.create'
|
|
149
|
+
});
|
|
150
|
+
|
|
151
|
+
const adapterStartTime = Date.now();
|
|
152
|
+
const result = await this.createAdapter(config, apiMode);
|
|
153
|
+
const adapterDuration = Date.now() - adapterStartTime;
|
|
154
|
+
|
|
155
|
+
// Cache the result
|
|
156
|
+
this.setCachedAdapter(cacheKey, result.adapter);
|
|
157
|
+
|
|
158
|
+
const totalDuration = Date.now() - startTime;
|
|
159
|
+
logger.info('Adapter creation completed', {
|
|
160
|
+
apiMode,
|
|
161
|
+
adapterDuration,
|
|
162
|
+
totalDuration,
|
|
163
|
+
cached: true,
|
|
164
|
+
operation: 'AdapterFactory.create'
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
return {
|
|
168
|
+
...result,
|
|
169
|
+
apiMode,
|
|
170
|
+
detectionResult
|
|
171
|
+
};
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
/**
|
|
175
|
+
* Create adapter from AppwriteConfig (convenience method)
|
|
176
|
+
*/
|
|
177
|
+
static async createFromConfig(config: AppwriteConfig, forceRefresh?: boolean): Promise<AdapterFactoryResult> {
|
|
178
|
+
return this.create({
|
|
179
|
+
appwriteEndpoint: config.appwriteEndpoint,
|
|
180
|
+
appwriteProject: config.appwriteProject,
|
|
181
|
+
appwriteKey: config.appwriteKey,
|
|
182
|
+
apiMode: (config as any).apiMode || 'auto', // Cast to access new property
|
|
183
|
+
forceRefresh
|
|
184
|
+
});
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
/**
|
|
188
|
+
* Create specific adapter type (internal method)
|
|
189
|
+
*/
|
|
190
|
+
private static async createAdapter(
|
|
191
|
+
config: AdapterFactoryConfig,
|
|
192
|
+
apiMode: ApiMode
|
|
193
|
+
): Promise<{ adapter: DatabaseAdapter; client: any }> {
|
|
194
|
+
if (apiMode === 'tablesdb') {
|
|
195
|
+
return this.createTablesDBAdapter(config);
|
|
196
|
+
} else {
|
|
197
|
+
return this.createLegacyAdapter(config);
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
/**
|
|
202
|
+
* Create TablesDB adapter with dynamic import
|
|
203
|
+
*/
|
|
204
|
+
private static async createTablesDBAdapter(
|
|
205
|
+
config: AdapterFactoryConfig
|
|
206
|
+
): Promise<{ adapter: DatabaseAdapter; client: any }> {
|
|
207
|
+
const startTime = Date.now();
|
|
208
|
+
|
|
209
|
+
try {
|
|
210
|
+
logger.info('Creating TablesDB adapter (static SDK imports)', {
|
|
211
|
+
endpoint: config.appwriteEndpoint,
|
|
212
|
+
operation: 'createTablesDBAdapter'
|
|
213
|
+
});
|
|
214
|
+
|
|
215
|
+
// Use pre-configured client or create session-aware client with TablesDB Client
|
|
216
|
+
let client: any;
|
|
217
|
+
if (config.preConfiguredClient) {
|
|
218
|
+
client = config.preConfiguredClient;
|
|
219
|
+
} else {
|
|
220
|
+
client = new Client()
|
|
221
|
+
.setEndpoint(config.appwriteEndpoint)
|
|
222
|
+
.setProject(config.appwriteProject);
|
|
223
|
+
|
|
224
|
+
// Set authentication method with mode headers
|
|
225
|
+
// Prefer session with admin mode, fallback to API key with default mode
|
|
226
|
+
if (config.sessionCookie && isValidSessionCookie(config.sessionCookie)) {
|
|
227
|
+
client.setSession(config.sessionCookie);
|
|
228
|
+
client.headers['X-Appwrite-Mode'] = 'admin';
|
|
229
|
+
logger.debug('Using session authentication for TablesDB adapter', {
|
|
230
|
+
project: config.appwriteProject,
|
|
231
|
+
operation: 'createTablesDBAdapter'
|
|
232
|
+
});
|
|
233
|
+
} else if (config.appwriteKey) {
|
|
234
|
+
client.setKey(config.appwriteKey);
|
|
235
|
+
client.headers['X-Appwrite-Mode'] = 'default';
|
|
236
|
+
logger.debug('Using API key authentication for TablesDB adapter', {
|
|
237
|
+
project: config.appwriteProject,
|
|
238
|
+
operation: 'createTablesDBAdapter'
|
|
239
|
+
});
|
|
240
|
+
} else {
|
|
241
|
+
throw new Error("No authentication available for adapter");
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
const adapter = new TablesDBAdapter(client);
|
|
246
|
+
|
|
247
|
+
const totalDuration = Date.now() - startTime;
|
|
248
|
+
logger.info('TablesDB adapter created successfully', {
|
|
249
|
+
totalDuration,
|
|
250
|
+
endpoint: config.appwriteEndpoint,
|
|
251
|
+
operation: 'createTablesDBAdapter'
|
|
252
|
+
});
|
|
253
|
+
|
|
254
|
+
return { adapter, client };
|
|
255
|
+
|
|
256
|
+
} catch (error) {
|
|
257
|
+
const errorDuration = Date.now() - startTime;
|
|
258
|
+
const errorMessage = error instanceof Error ? error.message : 'Unknown error';
|
|
259
|
+
|
|
260
|
+
MessageFormatter.warning('Failed to create TablesDB adapter - falling back to legacy', { prefix: "Adapter" });
|
|
261
|
+
|
|
262
|
+
logger.warn('TablesDB adapter creation failed, falling back to legacy', {
|
|
263
|
+
error: errorMessage,
|
|
264
|
+
errorDuration,
|
|
265
|
+
endpoint: config.appwriteEndpoint,
|
|
266
|
+
operation: 'createTablesDBAdapter'
|
|
267
|
+
});
|
|
268
|
+
|
|
269
|
+
// Fallback to legacy adapter if TablesDB creation fails
|
|
270
|
+
return this.createLegacyAdapter(config);
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
/**
|
|
275
|
+
* Create Legacy adapter with dynamic import
|
|
276
|
+
*/
|
|
277
|
+
private static async createLegacyAdapter(
|
|
278
|
+
config: AdapterFactoryConfig
|
|
279
|
+
): Promise<{ adapter: DatabaseAdapter; client: any }> {
|
|
280
|
+
const startTime = Date.now();
|
|
281
|
+
|
|
282
|
+
try {
|
|
283
|
+
logger.info('Creating legacy adapter (static SDK imports)', {
|
|
284
|
+
endpoint: config.appwriteEndpoint,
|
|
285
|
+
operation: 'createLegacyAdapter'
|
|
286
|
+
});
|
|
287
|
+
|
|
288
|
+
// Use pre-configured client or create session-aware client with Legacy Client
|
|
289
|
+
let client: any;
|
|
290
|
+
if (config.preConfiguredClient) {
|
|
291
|
+
client = config.preConfiguredClient;
|
|
292
|
+
} else {
|
|
293
|
+
client = new Client()
|
|
294
|
+
.setEndpoint(config.appwriteEndpoint)
|
|
295
|
+
.setProject(config.appwriteProject);
|
|
296
|
+
|
|
297
|
+
// Set authentication method with mode headers
|
|
298
|
+
// Prefer session with admin mode, fallback to API key with default mode
|
|
299
|
+
if (config.sessionCookie && isValidSessionCookie(config.sessionCookie)) {
|
|
300
|
+
(client as any).setSession(config.sessionCookie);
|
|
301
|
+
client.headers['X-Appwrite-Mode'] = 'admin';
|
|
302
|
+
logger.debug('Using session authentication for Legacy adapter', {
|
|
303
|
+
project: config.appwriteProject,
|
|
304
|
+
operation: 'createLegacyAdapter'
|
|
305
|
+
});
|
|
306
|
+
} else if (config.appwriteKey) {
|
|
307
|
+
client.setKey(config.appwriteKey);
|
|
308
|
+
client.headers['X-Appwrite-Mode'] = 'default';
|
|
309
|
+
logger.debug('Using API key authentication for Legacy adapter', {
|
|
310
|
+
project: config.appwriteProject,
|
|
311
|
+
operation: 'createLegacyAdapter'
|
|
312
|
+
});
|
|
313
|
+
} else {
|
|
314
|
+
throw new Error("No authentication available for adapter");
|
|
315
|
+
}
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
const adapter = new LegacyAdapter(client);
|
|
319
|
+
|
|
320
|
+
const totalDuration = Date.now() - startTime;
|
|
321
|
+
logger.info('Legacy adapter created successfully', {
|
|
322
|
+
totalDuration,
|
|
323
|
+
endpoint: config.appwriteEndpoint,
|
|
324
|
+
operation: 'createLegacyAdapter'
|
|
325
|
+
});
|
|
326
|
+
|
|
327
|
+
return { adapter, client };
|
|
328
|
+
|
|
329
|
+
} catch (error) {
|
|
330
|
+
const errorDuration = Date.now() - startTime;
|
|
331
|
+
const errorMessage = error instanceof Error ? error.message : 'Unknown error';
|
|
332
|
+
|
|
333
|
+
logger.error('Failed to load legacy Appwrite SDK', {
|
|
334
|
+
error: errorMessage,
|
|
335
|
+
errorDuration,
|
|
336
|
+
endpoint: config.appwriteEndpoint,
|
|
337
|
+
operation: 'createLegacyAdapter'
|
|
338
|
+
});
|
|
339
|
+
|
|
340
|
+
throw new Error(`Failed to load legacy Appwrite SDK: ${errorMessage}`);
|
|
341
|
+
}
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
/**
|
|
345
|
+
* Get cached adapter if available and not expired
|
|
346
|
+
*/
|
|
347
|
+
private static getCachedAdapter(cacheKey: string): { adapter: DatabaseAdapter; timestamp: number } | null {
|
|
348
|
+
const cached = this.cache.get(cacheKey);
|
|
349
|
+
|
|
350
|
+
if (!cached) {
|
|
351
|
+
return null;
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
// Check if cache is expired
|
|
355
|
+
if (Date.now() - cached.timestamp > this.CACHE_DURATION) {
|
|
356
|
+
this.cache.delete(cacheKey);
|
|
357
|
+
return null;
|
|
358
|
+
}
|
|
359
|
+
|
|
360
|
+
return cached;
|
|
361
|
+
}
|
|
362
|
+
|
|
363
|
+
/**
|
|
364
|
+
* Cache adapter instance
|
|
365
|
+
*/
|
|
366
|
+
private static setCachedAdapter(cacheKey: string, adapter: DatabaseAdapter): void {
|
|
367
|
+
this.cache.set(cacheKey, {
|
|
368
|
+
adapter,
|
|
369
|
+
timestamp: Date.now()
|
|
370
|
+
});
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
/**
|
|
374
|
+
* Clear adapter cache (useful for testing)
|
|
375
|
+
*/
|
|
376
|
+
static clearCache(): void {
|
|
377
|
+
this.cache.clear();
|
|
378
|
+
}
|
|
379
|
+
|
|
380
|
+
/**
|
|
381
|
+
* Validate authentication configuration
|
|
382
|
+
*/
|
|
383
|
+
private static validateAuthConfig(config: AdapterFactoryConfig): void {
|
|
384
|
+
const hasApiKey = config.appwriteKey && config.appwriteKey.trim().length > 0;
|
|
385
|
+
const hasSessionCookie = config.sessionCookie && isValidSessionCookie(config.sessionCookie);
|
|
386
|
+
const hasPreConfiguredClient = !!config.preConfiguredClient;
|
|
387
|
+
|
|
388
|
+
// Must have at least one authentication method
|
|
389
|
+
if (!hasApiKey && !hasSessionCookie && !hasPreConfiguredClient) {
|
|
390
|
+
throw new Error(
|
|
391
|
+
`No valid authentication method provided for project ${config.appwriteProject}. ` +
|
|
392
|
+
`Please provide an API key, session cookie, or pre-configured client.`
|
|
393
|
+
);
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
// Validate session cookie format if provided
|
|
397
|
+
if (config.sessionCookie && !isValidSessionCookie(config.sessionCookie)) {
|
|
398
|
+
throw new Error(
|
|
399
|
+
`Invalid session cookie format provided for project ${config.appwriteProject}. ` +
|
|
400
|
+
`Session cookie must be a valid JWT token.`
|
|
401
|
+
);
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
logger.debug('Authentication configuration validated', {
|
|
405
|
+
hasApiKey,
|
|
406
|
+
hasSessionCookie,
|
|
407
|
+
hasPreConfiguredClient,
|
|
408
|
+
authMethod: config.authMethod || 'auto',
|
|
409
|
+
operation: 'validateAuthConfig'
|
|
410
|
+
});
|
|
411
|
+
}
|
|
412
|
+
|
|
413
|
+
/**
|
|
414
|
+
* Test connection and API capabilities
|
|
415
|
+
*/
|
|
416
|
+
static async testConnection(config: AdapterFactoryConfig): Promise<{
|
|
417
|
+
success: boolean;
|
|
418
|
+
apiMode: ApiMode;
|
|
419
|
+
capabilities: string[];
|
|
420
|
+
error?: string;
|
|
421
|
+
}> {
|
|
422
|
+
try {
|
|
423
|
+
const result = await this.create({ ...config, forceRefresh: true });
|
|
424
|
+
const metadata = result.adapter.getMetadata();
|
|
425
|
+
|
|
426
|
+
// Test basic operations
|
|
427
|
+
const capabilities = [];
|
|
428
|
+
|
|
429
|
+
if (metadata.capabilities.bulkOperations) {
|
|
430
|
+
capabilities.push('Bulk Operations');
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
if (metadata.capabilities.advancedQueries) {
|
|
434
|
+
capabilities.push('Advanced Queries');
|
|
435
|
+
}
|
|
436
|
+
|
|
437
|
+
if (metadata.capabilities.realtime) {
|
|
438
|
+
capabilities.push('Realtime');
|
|
439
|
+
}
|
|
440
|
+
|
|
441
|
+
if (metadata.capabilities.transactions) {
|
|
442
|
+
capabilities.push('Transactions');
|
|
443
|
+
}
|
|
444
|
+
|
|
445
|
+
return {
|
|
446
|
+
success: true,
|
|
447
|
+
apiMode: result.apiMode,
|
|
448
|
+
capabilities
|
|
449
|
+
};
|
|
450
|
+
|
|
451
|
+
} catch (error) {
|
|
452
|
+
return {
|
|
453
|
+
success: false,
|
|
454
|
+
apiMode: 'legacy', // Default fallback
|
|
455
|
+
capabilities: [],
|
|
456
|
+
error: error instanceof Error ? error.message : 'Unknown error'
|
|
457
|
+
};
|
|
458
|
+
}
|
|
459
|
+
}
|
|
460
|
+
}
|
|
461
|
+
|
|
462
|
+
/**
|
|
463
|
+
* Convenience function for quick adapter creation
|
|
464
|
+
*/
|
|
465
|
+
export async function createDatabaseAdapter(
|
|
466
|
+
endpoint: string,
|
|
467
|
+
project: string,
|
|
468
|
+
apiKey?: string,
|
|
469
|
+
mode: 'auto' | 'legacy' | 'tablesdb' = 'auto',
|
|
470
|
+
sessionCookie?: string
|
|
471
|
+
): Promise<DatabaseAdapter> {
|
|
472
|
+
const result = await AdapterFactory.create({
|
|
473
|
+
appwriteEndpoint: endpoint,
|
|
474
|
+
appwriteProject: project,
|
|
475
|
+
appwriteKey: apiKey,
|
|
476
|
+
apiMode: mode,
|
|
477
|
+
sessionCookie
|
|
478
|
+
});
|
|
479
|
+
|
|
480
|
+
return result.adapter;
|
|
481
|
+
}
|
|
482
|
+
|
|
483
|
+
/**
|
|
484
|
+
* Helper function to get adapter metadata without creating full adapter
|
|
485
|
+
*/
|
|
486
|
+
export async function getApiCapabilities(
|
|
487
|
+
endpoint: string,
|
|
488
|
+
project: string,
|
|
489
|
+
apiKey?: string,
|
|
490
|
+
sessionCookie?: string
|
|
491
|
+
): Promise<{
|
|
492
|
+
apiMode: ApiMode;
|
|
493
|
+
terminology: { container: string; item: string; service: string };
|
|
494
|
+
capabilities: string[];
|
|
495
|
+
}> {
|
|
496
|
+
const adapter = await createDatabaseAdapter(endpoint, project, apiKey, 'auto', sessionCookie);
|
|
497
|
+
const metadata = adapter.getMetadata();
|
|
498
|
+
|
|
499
|
+
const capabilities = [];
|
|
500
|
+
if (metadata.capabilities.bulkOperations) capabilities.push('Bulk Operations');
|
|
501
|
+
if (metadata.capabilities.advancedQueries) capabilities.push('Advanced Queries');
|
|
502
|
+
if (metadata.capabilities.realtime) capabilities.push('Realtime');
|
|
503
|
+
if (metadata.capabilities.transactions) capabilities.push('Transactions');
|
|
504
|
+
|
|
505
|
+
return {
|
|
506
|
+
apiMode: metadata.apiMode,
|
|
507
|
+
terminology: metadata.terminology,
|
|
508
|
+
capabilities
|
|
509
|
+
};
|
|
510
|
+
}
|