@friggframework/devtools 2.0.0-next.62 → 2.0.0-next.63
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/infrastructure/ARCHITECTURE.md +487 -0
- package/infrastructure/CLAUDE.md +481 -0
- package/infrastructure/HEALTH.md +468 -0
- package/infrastructure/README.md +522 -0
- package/infrastructure/__tests__/fixtures/mock-aws-resources.js +391 -0
- package/infrastructure/__tests__/helpers/test-utils.js +277 -0
- package/infrastructure/__tests__/postgres-config.test.js +914 -0
- package/infrastructure/__tests__/template-generation.test.js +687 -0
- package/infrastructure/create-frigg-infrastructure.js +147 -0
- package/infrastructure/docs/POSTGRES-CONFIGURATION.md +630 -0
- package/infrastructure/docs/PRE-DEPLOYMENT-HEALTH-CHECK-SPEC.md +1317 -0
- package/infrastructure/docs/WEBSOCKET-CONFIGURATION.md +105 -0
- package/infrastructure/docs/deployment-instructions.md +268 -0
- package/infrastructure/docs/generate-iam-command.md +278 -0
- package/infrastructure/docs/iam-policy-templates.md +193 -0
- package/infrastructure/domains/database/aurora-builder.js +809 -0
- package/infrastructure/domains/database/aurora-builder.test.js +950 -0
- package/infrastructure/domains/database/aurora-discovery.js +87 -0
- package/infrastructure/domains/database/aurora-discovery.test.js +188 -0
- package/infrastructure/domains/database/aurora-resolver.js +210 -0
- package/infrastructure/domains/database/aurora-resolver.test.js +347 -0
- package/infrastructure/domains/database/migration-builder.js +701 -0
- package/infrastructure/domains/database/migration-builder.test.js +321 -0
- package/infrastructure/domains/database/migration-resolver.js +163 -0
- package/infrastructure/domains/database/migration-resolver.test.js +337 -0
- package/infrastructure/domains/health/application/ports/IPropertyReconciler.js +164 -0
- package/infrastructure/domains/health/application/ports/IResourceDetector.js +129 -0
- package/infrastructure/domains/health/application/ports/IResourceImporter.js +142 -0
- package/infrastructure/domains/health/application/ports/IStackRepository.js +131 -0
- package/infrastructure/domains/health/application/ports/index.js +26 -0
- package/infrastructure/domains/health/application/use-cases/__tests__/execute-resource-import-use-case.test.js +679 -0
- package/infrastructure/domains/health/application/use-cases/__tests__/mismatch-analyzer-method-name.test.js +167 -0
- package/infrastructure/domains/health/application/use-cases/__tests__/repair-via-import-use-case.test.js +1130 -0
- package/infrastructure/domains/health/application/use-cases/execute-resource-import-use-case.js +221 -0
- package/infrastructure/domains/health/application/use-cases/reconcile-properties-use-case.js +152 -0
- package/infrastructure/domains/health/application/use-cases/reconcile-properties-use-case.test.js +343 -0
- package/infrastructure/domains/health/application/use-cases/repair-via-import-use-case.js +535 -0
- package/infrastructure/domains/health/application/use-cases/repair-via-import-use-case.test.js +376 -0
- package/infrastructure/domains/health/application/use-cases/run-health-check-use-case.js +213 -0
- package/infrastructure/domains/health/application/use-cases/run-health-check-use-case.test.js +441 -0
- package/infrastructure/domains/health/docs/ACME-DEV-DRIFT-ANALYSIS.md +267 -0
- package/infrastructure/domains/health/docs/BUILD-VS-DEPLOYED-TEMPLATE-ANALYSIS.md +324 -0
- package/infrastructure/domains/health/docs/ORPHAN-DETECTION-ANALYSIS.md +386 -0
- package/infrastructure/domains/health/docs/SPEC-CLEANUP-COMMAND.md +1419 -0
- package/infrastructure/domains/health/docs/TDD-IMPLEMENTATION-SUMMARY.md +391 -0
- package/infrastructure/domains/health/docs/TEMPLATE-COMPARISON-IMPLEMENTATION.md +551 -0
- package/infrastructure/domains/health/domain/entities/issue.js +299 -0
- package/infrastructure/domains/health/domain/entities/issue.test.js +528 -0
- package/infrastructure/domains/health/domain/entities/property-mismatch.js +108 -0
- package/infrastructure/domains/health/domain/entities/property-mismatch.test.js +275 -0
- package/infrastructure/domains/health/domain/entities/resource.js +159 -0
- package/infrastructure/domains/health/domain/entities/resource.test.js +432 -0
- package/infrastructure/domains/health/domain/entities/stack-health-report.js +306 -0
- package/infrastructure/domains/health/domain/entities/stack-health-report.test.js +601 -0
- package/infrastructure/domains/health/domain/services/__tests__/health-score-percentage-based.test.js +380 -0
- package/infrastructure/domains/health/domain/services/__tests__/import-progress-monitor.test.js +971 -0
- package/infrastructure/domains/health/domain/services/__tests__/import-template-generator.test.js +1150 -0
- package/infrastructure/domains/health/domain/services/__tests__/logical-id-mapper.test.js +672 -0
- package/infrastructure/domains/health/domain/services/__tests__/template-parser.test.js +496 -0
- package/infrastructure/domains/health/domain/services/__tests__/update-progress-monitor.test.js +419 -0
- package/infrastructure/domains/health/domain/services/health-score-calculator.js +248 -0
- package/infrastructure/domains/health/domain/services/health-score-calculator.test.js +504 -0
- package/infrastructure/domains/health/domain/services/import-progress-monitor.js +195 -0
- package/infrastructure/domains/health/domain/services/import-template-generator.js +435 -0
- package/infrastructure/domains/health/domain/services/logical-id-mapper.js +345 -0
- package/infrastructure/domains/health/domain/services/mismatch-analyzer.js +234 -0
- package/infrastructure/domains/health/domain/services/mismatch-analyzer.test.js +431 -0
- package/infrastructure/domains/health/domain/services/property-mutability-config.js +382 -0
- package/infrastructure/domains/health/domain/services/template-parser.js +245 -0
- package/infrastructure/domains/health/domain/services/update-progress-monitor.js +192 -0
- package/infrastructure/domains/health/domain/value-objects/health-score.js +138 -0
- package/infrastructure/domains/health/domain/value-objects/health-score.test.js +267 -0
- package/infrastructure/domains/health/domain/value-objects/property-mutability.js +161 -0
- package/infrastructure/domains/health/domain/value-objects/property-mutability.test.js +198 -0
- package/infrastructure/domains/health/domain/value-objects/resource-state.js +167 -0
- package/infrastructure/domains/health/domain/value-objects/resource-state.test.js +196 -0
- package/infrastructure/domains/health/domain/value-objects/stack-identifier.js +192 -0
- package/infrastructure/domains/health/domain/value-objects/stack-identifier.test.js +262 -0
- package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-cfn-tagged.test.js +312 -0
- package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-multi-stack.test.js +367 -0
- package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-relationship-analysis.test.js +432 -0
- package/infrastructure/domains/health/infrastructure/adapters/aws-property-reconciler.js +784 -0
- package/infrastructure/domains/health/infrastructure/adapters/aws-property-reconciler.test.js +1133 -0
- package/infrastructure/domains/health/infrastructure/adapters/aws-resource-detector.js +565 -0
- package/infrastructure/domains/health/infrastructure/adapters/aws-resource-detector.test.js +554 -0
- package/infrastructure/domains/health/infrastructure/adapters/aws-resource-importer.js +318 -0
- package/infrastructure/domains/health/infrastructure/adapters/aws-resource-importer.test.js +398 -0
- package/infrastructure/domains/health/infrastructure/adapters/aws-stack-repository.js +777 -0
- package/infrastructure/domains/health/infrastructure/adapters/aws-stack-repository.test.js +580 -0
- package/infrastructure/domains/integration/integration-builder.js +404 -0
- package/infrastructure/domains/integration/integration-builder.test.js +690 -0
- package/infrastructure/domains/integration/integration-resolver.js +170 -0
- package/infrastructure/domains/integration/integration-resolver.test.js +369 -0
- package/infrastructure/domains/integration/websocket-builder.js +69 -0
- package/infrastructure/domains/integration/websocket-builder.test.js +195 -0
- package/infrastructure/domains/networking/vpc-builder.js +2051 -0
- package/infrastructure/domains/networking/vpc-builder.test.js +1960 -0
- package/infrastructure/domains/networking/vpc-discovery.js +177 -0
- package/infrastructure/domains/networking/vpc-discovery.test.js +350 -0
- package/infrastructure/domains/networking/vpc-resolver.js +505 -0
- package/infrastructure/domains/networking/vpc-resolver.test.js +801 -0
- package/infrastructure/domains/parameters/ssm-builder.js +79 -0
- package/infrastructure/domains/parameters/ssm-builder.test.js +189 -0
- package/infrastructure/domains/parameters/ssm-discovery.js +84 -0
- package/infrastructure/domains/parameters/ssm-discovery.test.js +210 -0
- package/infrastructure/domains/security/iam-generator.js +816 -0
- package/infrastructure/domains/security/iam-generator.test.js +204 -0
- package/infrastructure/domains/security/kms-builder.js +415 -0
- package/infrastructure/domains/security/kms-builder.test.js +392 -0
- package/infrastructure/domains/security/kms-discovery.js +80 -0
- package/infrastructure/domains/security/kms-discovery.test.js +177 -0
- package/infrastructure/domains/security/kms-resolver.js +96 -0
- package/infrastructure/domains/security/kms-resolver.test.js +216 -0
- package/infrastructure/domains/security/templates/frigg-deployment-iam-stack.yaml +401 -0
- package/infrastructure/domains/security/templates/iam-policy-basic.json +218 -0
- package/infrastructure/domains/security/templates/iam-policy-full.json +288 -0
- package/infrastructure/domains/shared/base-builder.js +112 -0
- package/infrastructure/domains/shared/base-resolver.js +186 -0
- package/infrastructure/domains/shared/base-resolver.test.js +305 -0
- package/infrastructure/domains/shared/builder-orchestrator.js +212 -0
- package/infrastructure/domains/shared/builder-orchestrator.test.js +213 -0
- package/infrastructure/domains/shared/cloudformation-discovery-v2.js +334 -0
- package/infrastructure/domains/shared/cloudformation-discovery.js +672 -0
- package/infrastructure/domains/shared/cloudformation-discovery.test.js +985 -0
- package/infrastructure/domains/shared/environment-builder.js +119 -0
- package/infrastructure/domains/shared/environment-builder.test.js +247 -0
- package/infrastructure/domains/shared/providers/aws-provider-adapter.js +579 -0
- package/infrastructure/domains/shared/providers/aws-provider-adapter.test.js +416 -0
- package/infrastructure/domains/shared/providers/azure-provider-adapter.stub.js +93 -0
- package/infrastructure/domains/shared/providers/cloud-provider-adapter.js +136 -0
- package/infrastructure/domains/shared/providers/gcp-provider-adapter.stub.js +82 -0
- package/infrastructure/domains/shared/providers/provider-factory.js +108 -0
- package/infrastructure/domains/shared/providers/provider-factory.test.js +170 -0
- package/infrastructure/domains/shared/resource-discovery.enhanced.test.js +306 -0
- package/infrastructure/domains/shared/resource-discovery.js +233 -0
- package/infrastructure/domains/shared/resource-discovery.test.js +588 -0
- package/infrastructure/domains/shared/types/app-definition.js +205 -0
- package/infrastructure/domains/shared/types/discovery-result.js +106 -0
- package/infrastructure/domains/shared/types/discovery-result.test.js +258 -0
- package/infrastructure/domains/shared/types/index.js +46 -0
- package/infrastructure/domains/shared/types/resource-ownership.js +108 -0
- package/infrastructure/domains/shared/types/resource-ownership.test.js +101 -0
- package/infrastructure/domains/shared/utilities/base-definition-factory.js +408 -0
- package/infrastructure/domains/shared/utilities/base-definition-factory.js.bak +338 -0
- package/infrastructure/domains/shared/utilities/base-definition-factory.test.js +291 -0
- package/infrastructure/domains/shared/utilities/handler-path-resolver.js +134 -0
- package/infrastructure/domains/shared/utilities/handler-path-resolver.test.js +268 -0
- package/infrastructure/domains/shared/utilities/prisma-layer-manager.js +159 -0
- package/infrastructure/domains/shared/utilities/prisma-layer-manager.test.js +444 -0
- package/infrastructure/domains/shared/validation/env-validator.js +78 -0
- package/infrastructure/domains/shared/validation/env-validator.test.js +173 -0
- package/infrastructure/domains/shared/validation/plugin-validator.js +187 -0
- package/infrastructure/domains/shared/validation/plugin-validator.test.js +323 -0
- package/infrastructure/esbuild.config.js +53 -0
- package/infrastructure/index.js +4 -0
- package/infrastructure/infrastructure-composer.js +117 -0
- package/infrastructure/infrastructure-composer.test.js +1895 -0
- package/infrastructure/integration.test.js +383 -0
- package/infrastructure/scripts/build-prisma-layer.js +701 -0
- package/infrastructure/scripts/build-prisma-layer.test.js +170 -0
- package/infrastructure/scripts/build-time-discovery.js +238 -0
- package/infrastructure/scripts/build-time-discovery.test.js +379 -0
- package/infrastructure/scripts/run-discovery.js +110 -0
- package/infrastructure/scripts/verify-prisma-layer.js +72 -0
- package/package.json +8 -7
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* FUTURE: Google Cloud Platform Provider Adapter
|
|
3
|
+
*
|
|
4
|
+
* This file serves as a placeholder for future GCP support.
|
|
5
|
+
*
|
|
6
|
+
* Implementation will use:
|
|
7
|
+
* - @google-cloud/compute for VPC/network discovery
|
|
8
|
+
* - @google-cloud/kms for encryption key management
|
|
9
|
+
* - @google-cloud/sql for Cloud SQL database discovery
|
|
10
|
+
* - @google-cloud/secret-manager for secrets management
|
|
11
|
+
*
|
|
12
|
+
* Resources:
|
|
13
|
+
* - GCP Node.js SDK: https://cloud.google.com/nodejs/docs/reference
|
|
14
|
+
* - Compute Engine API: https://cloud.google.com/compute/docs/reference/rest/v1
|
|
15
|
+
* - Cloud KMS API: https://cloud.google.com/kms/docs/reference/rest
|
|
16
|
+
* - Cloud SQL API: https://cloud.google.com/sql/docs/mysql/admin-api
|
|
17
|
+
*
|
|
18
|
+
* Architecture mapping:
|
|
19
|
+
* - AWS VPC → GCP VPC Network
|
|
20
|
+
* - AWS KMS → GCP Cloud KMS
|
|
21
|
+
* - AWS RDS Aurora → GCP Cloud SQL
|
|
22
|
+
* - AWS SSM Parameter Store → GCP Secret Manager
|
|
23
|
+
* - AWS Lambda → GCP Cloud Functions / Cloud Run
|
|
24
|
+
*
|
|
25
|
+
* Example structure:
|
|
26
|
+
*
|
|
27
|
+
* const { Compute } = require('@google-cloud/compute');
|
|
28
|
+
* const { KeyManagementServiceClient } = require('@google-cloud/kms');
|
|
29
|
+
* const { CloudProviderAdapter } = require('./cloud-provider-adapter');
|
|
30
|
+
*
|
|
31
|
+
* class GCPProviderAdapter extends CloudProviderAdapter {
|
|
32
|
+
* constructor(region, credentials = {}) {
|
|
33
|
+
* super();
|
|
34
|
+
* this.region = region || 'us-central1';
|
|
35
|
+
* this.projectId = credentials.projectId || process.env.GCP_PROJECT_ID;
|
|
36
|
+
*
|
|
37
|
+
* this.compute = new Compute({ projectId: this.projectId });
|
|
38
|
+
* this.kms = new KeyManagementServiceClient();
|
|
39
|
+
* }
|
|
40
|
+
*
|
|
41
|
+
* getName() {
|
|
42
|
+
* return 'gcp';
|
|
43
|
+
* }
|
|
44
|
+
*
|
|
45
|
+
* getSupportedRegions() {
|
|
46
|
+
* return [
|
|
47
|
+
* 'us-central1', 'us-east1', 'us-west1',
|
|
48
|
+
* 'europe-west1', 'asia-east1', 'asia-northeast1'
|
|
49
|
+
* ];
|
|
50
|
+
* }
|
|
51
|
+
*
|
|
52
|
+
* async discoverVpc(config) {
|
|
53
|
+
* // Discover GCP VPC networks
|
|
54
|
+
* const [networks] = await this.compute.getNetworks();
|
|
55
|
+
* // ... implementation
|
|
56
|
+
* }
|
|
57
|
+
*
|
|
58
|
+
* async discoverKmsKeys(config) {
|
|
59
|
+
* // Discover GCP Cloud KMS keys
|
|
60
|
+
* const [keyRings] = await this.kms.listKeyRings({
|
|
61
|
+
* parent: `projects/${this.projectId}/locations/${this.region}`
|
|
62
|
+
* });
|
|
63
|
+
* // ... implementation
|
|
64
|
+
* }
|
|
65
|
+
*
|
|
66
|
+
* async discoverDatabase(config) {
|
|
67
|
+
* // Discover GCP Cloud SQL instances
|
|
68
|
+
* // ... implementation
|
|
69
|
+
* }
|
|
70
|
+
*
|
|
71
|
+
* async discoverParameters(config) {
|
|
72
|
+
* // Discover GCP Secret Manager secrets
|
|
73
|
+
* // ... implementation
|
|
74
|
+
* }
|
|
75
|
+
* }
|
|
76
|
+
*
|
|
77
|
+
* module.exports = { GCPProviderAdapter };
|
|
78
|
+
*/
|
|
79
|
+
|
|
80
|
+
// Placeholder export to prevent import errors
|
|
81
|
+
module.exports = {};
|
|
82
|
+
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Cloud Provider Factory
|
|
3
|
+
*
|
|
4
|
+
* Factory Pattern - Hexagonal Architecture
|
|
5
|
+
*
|
|
6
|
+
* Creates appropriate cloud provider adapter instances based on configuration.
|
|
7
|
+
* This enables runtime provider selection and makes it easy to add new providers.
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
const { CloudProviderAdapter } = require('./cloud-provider-adapter');
|
|
11
|
+
const { AWSProviderAdapter } = require('./aws-provider-adapter');
|
|
12
|
+
|
|
13
|
+
class CloudProviderFactory {
|
|
14
|
+
/**
|
|
15
|
+
* Create cloud provider adapter instance
|
|
16
|
+
*
|
|
17
|
+
* @param {string} providerName - Provider name ('aws', 'gcp', 'azure')
|
|
18
|
+
* @param {string} region - Provider region
|
|
19
|
+
* @param {Object} [credentials] - Optional credential configuration
|
|
20
|
+
* @returns {CloudProviderAdapter} Provider adapter instance
|
|
21
|
+
* @throws {Error} If provider is not supported
|
|
22
|
+
*/
|
|
23
|
+
static create(providerName, region, credentials = {}) {
|
|
24
|
+
const normalizedProvider = (providerName || 'aws').toLowerCase();
|
|
25
|
+
|
|
26
|
+
switch (normalizedProvider) {
|
|
27
|
+
case 'aws':
|
|
28
|
+
return new AWSProviderAdapter(region, credentials);
|
|
29
|
+
|
|
30
|
+
case 'gcp':
|
|
31
|
+
case 'google':
|
|
32
|
+
throw new Error(
|
|
33
|
+
'GCP provider not yet implemented. ' +
|
|
34
|
+
'AWS is currently the only supported cloud provider. ' +
|
|
35
|
+
'GCP support is planned for future releases.'
|
|
36
|
+
);
|
|
37
|
+
|
|
38
|
+
case 'azure':
|
|
39
|
+
case 'microsoft':
|
|
40
|
+
throw new Error(
|
|
41
|
+
'Azure provider not yet implemented. ' +
|
|
42
|
+
'AWS is currently the only supported cloud provider. ' +
|
|
43
|
+
'Azure support is planned for future releases.'
|
|
44
|
+
);
|
|
45
|
+
|
|
46
|
+
default:
|
|
47
|
+
throw new Error(
|
|
48
|
+
`Unknown cloud provider: "${providerName}". ` +
|
|
49
|
+
`Supported providers: aws (gcp and azure coming soon)`
|
|
50
|
+
);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Get list of supported providers
|
|
56
|
+
*
|
|
57
|
+
* @returns {Array<Object>} List of provider metadata
|
|
58
|
+
*/
|
|
59
|
+
static getSupportedProviders() {
|
|
60
|
+
return [
|
|
61
|
+
{
|
|
62
|
+
name: 'aws',
|
|
63
|
+
displayName: 'Amazon Web Services',
|
|
64
|
+
status: 'available',
|
|
65
|
+
description: 'AWS cloud provider with support for Lambda, VPC, RDS, KMS, etc.',
|
|
66
|
+
},
|
|
67
|
+
{
|
|
68
|
+
name: 'gcp',
|
|
69
|
+
displayName: 'Google Cloud Platform',
|
|
70
|
+
status: 'planned',
|
|
71
|
+
description: 'GCP cloud provider support coming soon',
|
|
72
|
+
},
|
|
73
|
+
{
|
|
74
|
+
name: 'azure',
|
|
75
|
+
displayName: 'Microsoft Azure',
|
|
76
|
+
status: 'planned',
|
|
77
|
+
description: 'Azure cloud provider support coming soon',
|
|
78
|
+
},
|
|
79
|
+
];
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Check if a provider is supported
|
|
84
|
+
*
|
|
85
|
+
* @param {string} providerName - Provider name to check
|
|
86
|
+
* @returns {boolean} True if provider is supported
|
|
87
|
+
*/
|
|
88
|
+
static isSupported(providerName) {
|
|
89
|
+
const normalized = (providerName || '').toLowerCase();
|
|
90
|
+
return ['aws', 'gcp', 'google', 'azure', 'microsoft'].includes(normalized);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
/**
|
|
94
|
+
* Check if a provider is available (implemented)
|
|
95
|
+
*
|
|
96
|
+
* @param {string} providerName - Provider name to check
|
|
97
|
+
* @returns {boolean} True if provider is implemented
|
|
98
|
+
*/
|
|
99
|
+
static isAvailable(providerName) {
|
|
100
|
+
const normalized = (providerName || '').toLowerCase();
|
|
101
|
+
return normalized === 'aws';
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
module.exports = {
|
|
106
|
+
CloudProviderFactory,
|
|
107
|
+
};
|
|
108
|
+
|
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tests for CloudProviderFactory
|
|
3
|
+
*
|
|
4
|
+
* Verifies provider instantiation, error handling, and factory patterns
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
const { CloudProviderFactory } = require('./provider-factory');
|
|
8
|
+
const { AWSProviderAdapter } = require('./aws-provider-adapter');
|
|
9
|
+
|
|
10
|
+
describe('CloudProviderFactory', () => {
|
|
11
|
+
describe('create()', () => {
|
|
12
|
+
it('should create AWS provider when provider is "aws"', () => {
|
|
13
|
+
const provider = CloudProviderFactory.create('aws', 'us-east-1');
|
|
14
|
+
|
|
15
|
+
expect(provider).toBeInstanceOf(AWSProviderAdapter);
|
|
16
|
+
expect(provider.getName()).toBe('aws');
|
|
17
|
+
expect(provider.region).toBe('us-east-1');
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
it('should default to AWS provider when no provider specified', () => {
|
|
21
|
+
const provider = CloudProviderFactory.create(null, 'us-west-2');
|
|
22
|
+
|
|
23
|
+
expect(provider).toBeInstanceOf(AWSProviderAdapter);
|
|
24
|
+
expect(provider.region).toBe('us-west-2');
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
it('should be case-insensitive for provider names', () => {
|
|
28
|
+
const provider = CloudProviderFactory.create('AWS', 'eu-west-1');
|
|
29
|
+
|
|
30
|
+
expect(provider).toBeInstanceOf(AWSProviderAdapter);
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
it('should throw error for GCP (not yet implemented)', () => {
|
|
34
|
+
expect(() => {
|
|
35
|
+
CloudProviderFactory.create('gcp', 'us-central1');
|
|
36
|
+
}).toThrow('GCP provider not yet implemented');
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
it('should throw error for Azure (not yet implemented)', () => {
|
|
40
|
+
expect(() => {
|
|
41
|
+
CloudProviderFactory.create('azure', 'eastus');
|
|
42
|
+
}).toThrow('Azure provider not yet implemented');
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
it('should handle "google" alias for GCP', () => {
|
|
46
|
+
expect(() => {
|
|
47
|
+
CloudProviderFactory.create('google', 'us-central1');
|
|
48
|
+
}).toThrow('GCP provider not yet implemented');
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
it('should handle "microsoft" alias for Azure', () => {
|
|
52
|
+
expect(() => {
|
|
53
|
+
CloudProviderFactory.create('microsoft', 'eastus');
|
|
54
|
+
}).toThrow('Azure provider not yet implemented');
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
it('should throw error for unknown provider', () => {
|
|
58
|
+
expect(() => {
|
|
59
|
+
CloudProviderFactory.create('unknown-cloud', 'region-1');
|
|
60
|
+
}).toThrow('Unknown cloud provider: "unknown-cloud"');
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
it('should pass credentials to provider', () => {
|
|
64
|
+
const credentials = {
|
|
65
|
+
accessKeyId: 'test-key',
|
|
66
|
+
secretAccessKey: 'test-secret',
|
|
67
|
+
};
|
|
68
|
+
|
|
69
|
+
const provider = CloudProviderFactory.create('aws', 'us-east-1', credentials);
|
|
70
|
+
|
|
71
|
+
expect(provider.credentials).toEqual(credentials);
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
describe('getSupportedProviders()', () => {
|
|
76
|
+
it('should return list of supported providers', () => {
|
|
77
|
+
const providers = CloudProviderFactory.getSupportedProviders();
|
|
78
|
+
|
|
79
|
+
expect(Array.isArray(providers)).toBe(true);
|
|
80
|
+
expect(providers.length).toBeGreaterThanOrEqual(3);
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
it('should include AWS as available', () => {
|
|
84
|
+
const providers = CloudProviderFactory.getSupportedProviders();
|
|
85
|
+
const aws = providers.find(p => p.name === 'aws');
|
|
86
|
+
|
|
87
|
+
expect(aws).toBeDefined();
|
|
88
|
+
expect(aws.status).toBe('available');
|
|
89
|
+
expect(aws.displayName).toBe('Amazon Web Services');
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
it('should include GCP as planned', () => {
|
|
93
|
+
const providers = CloudProviderFactory.getSupportedProviders();
|
|
94
|
+
const gcp = providers.find(p => p.name === 'gcp');
|
|
95
|
+
|
|
96
|
+
expect(gcp).toBeDefined();
|
|
97
|
+
expect(gcp.status).toBe('planned');
|
|
98
|
+
expect(gcp.displayName).toBe('Google Cloud Platform');
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
it('should include Azure as planned', () => {
|
|
102
|
+
const providers = CloudProviderFactory.getSupportedProviders();
|
|
103
|
+
const azure = providers.find(p => p.name === 'azure');
|
|
104
|
+
|
|
105
|
+
expect(azure).toBeDefined();
|
|
106
|
+
expect(azure.status).toBe('planned');
|
|
107
|
+
expect(azure.displayName).toBe('Microsoft Azure');
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
|
|
111
|
+
describe('isSupported()', () => {
|
|
112
|
+
it('should return true for aws', () => {
|
|
113
|
+
expect(CloudProviderFactory.isSupported('aws')).toBe(true);
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
it('should return true for gcp', () => {
|
|
117
|
+
expect(CloudProviderFactory.isSupported('gcp')).toBe(true);
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
it('should return true for azure', () => {
|
|
121
|
+
expect(CloudProviderFactory.isSupported('azure')).toBe(true);
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
it('should return true for google (gcp alias)', () => {
|
|
125
|
+
expect(CloudProviderFactory.isSupported('google')).toBe(true);
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
it('should return true for microsoft (azure alias)', () => {
|
|
129
|
+
expect(CloudProviderFactory.isSupported('microsoft')).toBe(true);
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
it('should return false for unknown provider', () => {
|
|
133
|
+
expect(CloudProviderFactory.isSupported('unknown')).toBe(false);
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
it('should be case-insensitive', () => {
|
|
137
|
+
expect(CloudProviderFactory.isSupported('AWS')).toBe(true);
|
|
138
|
+
expect(CloudProviderFactory.isSupported('GCP')).toBe(true);
|
|
139
|
+
});
|
|
140
|
+
|
|
141
|
+
it('should handle null/undefined gracefully', () => {
|
|
142
|
+
expect(CloudProviderFactory.isSupported(null)).toBe(false);
|
|
143
|
+
expect(CloudProviderFactory.isSupported(undefined)).toBe(false);
|
|
144
|
+
});
|
|
145
|
+
});
|
|
146
|
+
|
|
147
|
+
describe('isAvailable()', () => {
|
|
148
|
+
it('should return true only for aws', () => {
|
|
149
|
+
expect(CloudProviderFactory.isAvailable('aws')).toBe(true);
|
|
150
|
+
});
|
|
151
|
+
|
|
152
|
+
it('should return false for gcp (not yet implemented)', () => {
|
|
153
|
+
expect(CloudProviderFactory.isAvailable('gcp')).toBe(false);
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
it('should return false for azure (not yet implemented)', () => {
|
|
157
|
+
expect(CloudProviderFactory.isAvailable('azure')).toBe(false);
|
|
158
|
+
});
|
|
159
|
+
|
|
160
|
+
it('should return false for unknown provider', () => {
|
|
161
|
+
expect(CloudProviderFactory.isAvailable('unknown')).toBe(false);
|
|
162
|
+
});
|
|
163
|
+
|
|
164
|
+
it('should be case-insensitive', () => {
|
|
165
|
+
expect(CloudProviderFactory.isAvailable('AWS')).toBe(true);
|
|
166
|
+
expect(CloudProviderFactory.isAvailable('GCP')).toBe(false);
|
|
167
|
+
});
|
|
168
|
+
});
|
|
169
|
+
});
|
|
170
|
+
|
|
@@ -0,0 +1,306 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AWS Discovery Configuration Tests
|
|
3
|
+
*
|
|
4
|
+
* Tests for AppDefinition-level discovery control options
|
|
5
|
+
* addressing GitHub Issue #481 - Issue 5
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
const { shouldRunDiscovery, gatherDiscoveredResources } = require('./resource-discovery');
|
|
9
|
+
|
|
10
|
+
// Mock dependencies
|
|
11
|
+
jest.mock('./providers/provider-factory');
|
|
12
|
+
jest.mock('./cloudformation-discovery');
|
|
13
|
+
jest.mock('../networking/vpc-discovery');
|
|
14
|
+
jest.mock('../security/kms-discovery');
|
|
15
|
+
jest.mock('../database/aurora-discovery');
|
|
16
|
+
jest.mock('../parameters/ssm-discovery');
|
|
17
|
+
|
|
18
|
+
describe('AWS Discovery Configuration (Issue #481 - Issue 5)', () => {
|
|
19
|
+
beforeEach(() => {
|
|
20
|
+
delete process.env.FRIGG_SKIP_AWS_DISCOVERY;
|
|
21
|
+
jest.clearAllMocks();
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
describe('shouldRunDiscovery - Priority Order', () => {
|
|
25
|
+
it('should use AppDefinition.aws.discovery.enabled when explicitly set to true', () => {
|
|
26
|
+
const appDefinition = {
|
|
27
|
+
aws: { discovery: { enabled: true } },
|
|
28
|
+
vpc: { enable: false }, // Would normally skip
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
const result = shouldRunDiscovery(appDefinition);
|
|
32
|
+
|
|
33
|
+
expect(result).toBe(true);
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
it('should use AppDefinition.aws.discovery.enabled when explicitly set to false', () => {
|
|
37
|
+
const appDefinition = {
|
|
38
|
+
aws: { discovery: { enabled: false } },
|
|
39
|
+
vpc: { enable: true }, // Would normally run
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
const result = shouldRunDiscovery(appDefinition);
|
|
43
|
+
|
|
44
|
+
expect(result).toBe(false);
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
it('should fall back to env var when AppDefinition not set', () => {
|
|
48
|
+
process.env.FRIGG_SKIP_AWS_DISCOVERY = 'true';
|
|
49
|
+
|
|
50
|
+
const appDefinition = {
|
|
51
|
+
vpc: { enable: true }, // Would normally run
|
|
52
|
+
};
|
|
53
|
+
|
|
54
|
+
const result = shouldRunDiscovery(appDefinition);
|
|
55
|
+
|
|
56
|
+
expect(result).toBe(false);
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
it('should auto-detect when neither AppDefinition nor env var is set', () => {
|
|
60
|
+
const appDefinition = {
|
|
61
|
+
vpc: { enable: true },
|
|
62
|
+
};
|
|
63
|
+
|
|
64
|
+
const result = shouldRunDiscovery(appDefinition);
|
|
65
|
+
|
|
66
|
+
expect(result).toBe(true);
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
it('should prioritize AppDefinition over env var', () => {
|
|
70
|
+
process.env.FRIGG_SKIP_AWS_DISCOVERY = 'true';
|
|
71
|
+
|
|
72
|
+
const appDefinition = {
|
|
73
|
+
aws: { discovery: { enabled: true } }, // Explicit override
|
|
74
|
+
};
|
|
75
|
+
|
|
76
|
+
const result = shouldRunDiscovery(appDefinition);
|
|
77
|
+
|
|
78
|
+
expect(result).toBe(true); // AppDefinition wins
|
|
79
|
+
});
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
describe('AppDefinition.aws.discovery.enabled', () => {
|
|
83
|
+
it('should log when using AppDefinition configuration', () => {
|
|
84
|
+
const consoleSpy = jest.spyOn(console, 'log').mockImplementation();
|
|
85
|
+
|
|
86
|
+
const appDefinition = {
|
|
87
|
+
aws: { discovery: { enabled: true } },
|
|
88
|
+
};
|
|
89
|
+
|
|
90
|
+
shouldRunDiscovery(appDefinition);
|
|
91
|
+
|
|
92
|
+
expect(consoleSpy).toHaveBeenCalledWith(
|
|
93
|
+
expect.stringContaining('AppDefinition.aws.discovery.enabled: true')
|
|
94
|
+
);
|
|
95
|
+
|
|
96
|
+
consoleSpy.mockRestore();
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
it('should handle explicit false value', () => {
|
|
100
|
+
const appDefinition = {
|
|
101
|
+
aws: { discovery: { enabled: false } },
|
|
102
|
+
vpc: { enable: true },
|
|
103
|
+
encryption: { fieldLevelEncryptionMethod: 'kms' },
|
|
104
|
+
};
|
|
105
|
+
|
|
106
|
+
const result = shouldRunDiscovery(appDefinition);
|
|
107
|
+
|
|
108
|
+
expect(result).toBe(false);
|
|
109
|
+
});
|
|
110
|
+
|
|
111
|
+
it('should handle undefined correctly (fall through to next priority)', () => {
|
|
112
|
+
const appDefinition = {
|
|
113
|
+
aws: { discovery: { enabled: undefined } },
|
|
114
|
+
vpc: { enable: true },
|
|
115
|
+
};
|
|
116
|
+
|
|
117
|
+
const result = shouldRunDiscovery(appDefinition);
|
|
118
|
+
|
|
119
|
+
// Should skip when undefined (fall through to env var check)
|
|
120
|
+
expect(result).toBe(true); // Auto-detect kicks in
|
|
121
|
+
});
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
describe('Auto-detection based on features', () => {
|
|
125
|
+
it('should run discovery when VPC is enabled', () => {
|
|
126
|
+
const appDefinition = {
|
|
127
|
+
vpc: { enable: true },
|
|
128
|
+
};
|
|
129
|
+
|
|
130
|
+
expect(shouldRunDiscovery(appDefinition)).toBe(true);
|
|
131
|
+
});
|
|
132
|
+
|
|
133
|
+
it('should run discovery when KMS encryption is enabled', () => {
|
|
134
|
+
const appDefinition = {
|
|
135
|
+
encryption: { fieldLevelEncryptionMethod: 'kms' },
|
|
136
|
+
};
|
|
137
|
+
|
|
138
|
+
expect(shouldRunDiscovery(appDefinition)).toBe(true);
|
|
139
|
+
});
|
|
140
|
+
|
|
141
|
+
it('should run discovery when SSM is enabled', () => {
|
|
142
|
+
const appDefinition = {
|
|
143
|
+
ssm: { enable: true },
|
|
144
|
+
};
|
|
145
|
+
|
|
146
|
+
expect(shouldRunDiscovery(appDefinition)).toBe(true);
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
it('should run discovery when PostgreSQL is enabled', () => {
|
|
150
|
+
const appDefinition = {
|
|
151
|
+
database: { postgres: { enable: true } },
|
|
152
|
+
};
|
|
153
|
+
|
|
154
|
+
expect(shouldRunDiscovery(appDefinition)).toBe(true);
|
|
155
|
+
});
|
|
156
|
+
|
|
157
|
+
it('should not run discovery when no features are enabled', () => {
|
|
158
|
+
const appDefinition = {
|
|
159
|
+
vpc: { enable: false },
|
|
160
|
+
encryption: { fieldLevelEncryptionMethod: 'aes' },
|
|
161
|
+
};
|
|
162
|
+
|
|
163
|
+
expect(shouldRunDiscovery(appDefinition)).toBe(false);
|
|
164
|
+
});
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
describe('gatherDiscoveredResources - failOnError behavior', () => {
|
|
168
|
+
const { CloudProviderFactory } = require('./providers/provider-factory');
|
|
169
|
+
const { CloudFormationDiscovery } = require('./cloudformation-discovery');
|
|
170
|
+
|
|
171
|
+
beforeEach(() => {
|
|
172
|
+
const mockProvider = {
|
|
173
|
+
getVpcs: jest.fn(),
|
|
174
|
+
getKmsKeys: jest.fn(),
|
|
175
|
+
};
|
|
176
|
+
|
|
177
|
+
CloudProviderFactory.create = jest.fn().mockReturnValue(mockProvider);
|
|
178
|
+
|
|
179
|
+
// Mock CloudFormation discovery to throw error
|
|
180
|
+
CloudFormationDiscovery.mockImplementation(() => ({
|
|
181
|
+
discoverFromStack: jest.fn().mockRejectedValue(
|
|
182
|
+
new Error('User is not authorized to perform: ec2:DescribeVpcs')
|
|
183
|
+
),
|
|
184
|
+
}));
|
|
185
|
+
});
|
|
186
|
+
|
|
187
|
+
it('should throw error when failOnError is true', async () => {
|
|
188
|
+
const appDefinition = {
|
|
189
|
+
name: 'test-app',
|
|
190
|
+
vpc: { enable: true },
|
|
191
|
+
aws: {
|
|
192
|
+
discovery: {
|
|
193
|
+
enabled: true,
|
|
194
|
+
failOnError: true,
|
|
195
|
+
},
|
|
196
|
+
},
|
|
197
|
+
};
|
|
198
|
+
|
|
199
|
+
await expect(gatherDiscoveredResources(appDefinition)).rejects.toThrow(
|
|
200
|
+
'User is not authorized'
|
|
201
|
+
);
|
|
202
|
+
});
|
|
203
|
+
|
|
204
|
+
it('should return empty object when failOnError is false', async () => {
|
|
205
|
+
const appDefinition = {
|
|
206
|
+
name: 'test-app',
|
|
207
|
+
vpc: { enable: true },
|
|
208
|
+
aws: {
|
|
209
|
+
discovery: {
|
|
210
|
+
enabled: true,
|
|
211
|
+
failOnError: false,
|
|
212
|
+
},
|
|
213
|
+
},
|
|
214
|
+
};
|
|
215
|
+
|
|
216
|
+
const result = await gatherDiscoveredResources(appDefinition);
|
|
217
|
+
|
|
218
|
+
expect(result).toEqual({});
|
|
219
|
+
});
|
|
220
|
+
|
|
221
|
+
it('should default to false when failOnError is not set', async () => {
|
|
222
|
+
const appDefinition = {
|
|
223
|
+
name: 'test-app',
|
|
224
|
+
vpc: { enable: true },
|
|
225
|
+
};
|
|
226
|
+
|
|
227
|
+
const result = await gatherDiscoveredResources(appDefinition);
|
|
228
|
+
|
|
229
|
+
expect(result).toEqual({});
|
|
230
|
+
});
|
|
231
|
+
|
|
232
|
+
it('should log helpful message when failing gracefully', async () => {
|
|
233
|
+
const consoleWarnSpy = jest.spyOn(console, 'warn').mockImplementation();
|
|
234
|
+
|
|
235
|
+
const appDefinition = {
|
|
236
|
+
name: 'test-app',
|
|
237
|
+
vpc: { enable: true },
|
|
238
|
+
aws: { discovery: { failOnError: false } },
|
|
239
|
+
};
|
|
240
|
+
|
|
241
|
+
await gatherDiscoveredResources(appDefinition);
|
|
242
|
+
|
|
243
|
+
expect(consoleWarnSpy).toHaveBeenCalledWith(
|
|
244
|
+
expect.stringContaining('Set aws.discovery.failOnError = true')
|
|
245
|
+
);
|
|
246
|
+
|
|
247
|
+
consoleWarnSpy.mockRestore();
|
|
248
|
+
});
|
|
249
|
+
});
|
|
250
|
+
|
|
251
|
+
describe('Real-world scenarios', () => {
|
|
252
|
+
it('should handle restrictive IAM with explicit disable', () => {
|
|
253
|
+
const appDefinition = {
|
|
254
|
+
vpc: { enable: true },
|
|
255
|
+
aws: {
|
|
256
|
+
discovery: {
|
|
257
|
+
enabled: false, // Explicit disable for restrictive IAM
|
|
258
|
+
},
|
|
259
|
+
},
|
|
260
|
+
};
|
|
261
|
+
|
|
262
|
+
const result = shouldRunDiscovery(appDefinition);
|
|
263
|
+
|
|
264
|
+
expect(result).toBe(false);
|
|
265
|
+
});
|
|
266
|
+
|
|
267
|
+
it('should allow strict mode for production deployments', async () => {
|
|
268
|
+
const { CloudProviderFactory } = require('./providers/provider-factory');
|
|
269
|
+
const { CloudFormationDiscovery } = require('./cloudformation-discovery');
|
|
270
|
+
|
|
271
|
+
CloudFormationDiscovery.mockImplementation(() => ({
|
|
272
|
+
discoverFromStack: jest.fn().mockRejectedValue(new Error('IAM error')),
|
|
273
|
+
}));
|
|
274
|
+
|
|
275
|
+
const appDefinition = {
|
|
276
|
+
name: 'prod-app',
|
|
277
|
+
vpc: { enable: true },
|
|
278
|
+
aws: {
|
|
279
|
+
discovery: {
|
|
280
|
+
enabled: true,
|
|
281
|
+
failOnError: true, // Strict mode for production
|
|
282
|
+
},
|
|
283
|
+
},
|
|
284
|
+
};
|
|
285
|
+
|
|
286
|
+
await expect(gatherDiscoveredResources(appDefinition)).rejects.toThrow();
|
|
287
|
+
});
|
|
288
|
+
|
|
289
|
+
it('should allow graceful degradation for dev environments', async () => {
|
|
290
|
+
const appDefinition = {
|
|
291
|
+
name: 'dev-app',
|
|
292
|
+
vpc: { enable: true },
|
|
293
|
+
aws: {
|
|
294
|
+
discovery: {
|
|
295
|
+
enabled: true,
|
|
296
|
+
failOnError: false, // Graceful for dev
|
|
297
|
+
},
|
|
298
|
+
},
|
|
299
|
+
};
|
|
300
|
+
|
|
301
|
+
const result = await gatherDiscoveredResources(appDefinition);
|
|
302
|
+
|
|
303
|
+
expect(result).toEqual({});
|
|
304
|
+
});
|
|
305
|
+
});
|
|
306
|
+
});
|