@cdklabs/cdk-appmod-catalog-blueprints 1.3.0 → 1.4.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/.jsii +4 -4
- package/README.md +76 -98
- package/lib/document-processing/adapter/queued-s3-adapter.js +1 -1
- package/lib/document-processing/agentic-document-processing.js +1 -1
- package/lib/document-processing/base-document-processing.js +1 -1
- package/lib/document-processing/bedrock-document-processing.js +1 -1
- package/lib/document-processing/default-document-processing-config.js +1 -1
- package/lib/document-processing/tests/agentic-document-processing.test.js +104 -60
- package/lib/document-processing/tests/base-document-processing-nag.test.d.ts +1 -0
- package/lib/document-processing/tests/base-document-processing-nag.test.js +161 -0
- package/lib/document-processing/tests/base-document-processing.test.d.ts +1 -0
- package/lib/document-processing/tests/base-document-processing.test.js +499 -0
- package/lib/document-processing/tests/bedrock-document-processing.test.js +212 -36
- package/lib/document-processing/tests/queued-s3-adapter-nag.test.d.ts +1 -0
- package/lib/document-processing/tests/queued-s3-adapter-nag.test.js +122 -0
- package/lib/document-processing/tests/queued-s3-adapter.test.d.ts +1 -0
- package/lib/document-processing/tests/queued-s3-adapter.test.js +276 -0
- package/lib/framework/agents/base-agent.js +1 -1
- package/lib/framework/agents/batch-agent.js +1 -1
- package/lib/framework/agents/default-agent-config.js +1 -1
- package/lib/framework/bedrock/bedrock.js +1 -1
- package/lib/framework/custom-resource/default-runtimes.js +1 -1
- package/lib/framework/foundation/access-log.js +1 -1
- package/lib/framework/foundation/eventbridge-broker.js +1 -1
- package/lib/framework/foundation/network.js +1 -1
- package/lib/framework/tests/access-log.test.d.ts +1 -0
- package/lib/framework/tests/access-log.test.js +146 -0
- package/lib/framework/tests/batch-agent.test.d.ts +1 -0
- package/lib/framework/tests/batch-agent.test.js +164 -0
- package/lib/framework/tests/bedrock.test.d.ts +1 -0
- package/lib/framework/tests/bedrock.test.js +68 -0
- package/lib/framework/tests/eventbridge-broker.test.d.ts +1 -0
- package/lib/framework/tests/eventbridge-broker.test.js +73 -0
- package/lib/framework/tests/framework-nag.test.d.ts +1 -0
- package/lib/framework/tests/framework-nag.test.js +155 -0
- package/lib/framework/tests/network.test.d.ts +1 -0
- package/lib/framework/tests/network.test.js +120 -0
- package/lib/tsconfig.tsbuildinfo +1 -1
- package/lib/utilities/data-loader.js +1 -1
- package/lib/utilities/lambda-iam-utils.js +1 -1
- package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +1 -1
- package/lib/utilities/observability/default-observability-config.js +1 -1
- package/lib/utilities/observability/lambda-observability-property-injector.js +1 -1
- package/lib/utilities/observability/log-group-data-protection-utils.js +1 -1
- package/lib/utilities/observability/powertools-config.js +1 -1
- package/lib/utilities/observability/state-machine-observability-property-injector.js +1 -1
- package/lib/webapp/frontend-construct.js +1 -1
- package/package.json +8 -8
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
4
|
+
const assertions_1 = require("aws-cdk-lib/assertions");
|
|
5
|
+
const access_log_1 = require("../foundation/access-log");
|
|
6
|
+
describe('AccessLog', () => {
|
|
7
|
+
let stack;
|
|
8
|
+
beforeEach(() => {
|
|
9
|
+
stack = new aws_cdk_lib_1.Stack(undefined, 'TestStack', {
|
|
10
|
+
env: { account: '123456789012', region: 'us-east-1' },
|
|
11
|
+
});
|
|
12
|
+
});
|
|
13
|
+
test('creates bucket with default configuration', () => {
|
|
14
|
+
new access_log_1.AccessLog(stack, 'AccessLog');
|
|
15
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
16
|
+
template.resourceCountIs('AWS::S3::Bucket', 1);
|
|
17
|
+
template.hasResourceProperties('AWS::S3::Bucket', {
|
|
18
|
+
BucketName: 'access-logs-123456789012-us-east-1',
|
|
19
|
+
BucketEncryption: {
|
|
20
|
+
ServerSideEncryptionConfiguration: [{
|
|
21
|
+
ServerSideEncryptionByDefault: { SSEAlgorithm: 'AES256' },
|
|
22
|
+
}],
|
|
23
|
+
},
|
|
24
|
+
PublicAccessBlockConfiguration: assertions_1.Match.objectLike({
|
|
25
|
+
BlockPublicAcls: true,
|
|
26
|
+
BlockPublicPolicy: true,
|
|
27
|
+
IgnorePublicAcls: true,
|
|
28
|
+
RestrictPublicBuckets: true,
|
|
29
|
+
}),
|
|
30
|
+
});
|
|
31
|
+
});
|
|
32
|
+
test('creates bucket with custom name', () => {
|
|
33
|
+
new access_log_1.AccessLog(stack, 'AccessLog', {
|
|
34
|
+
bucketName: 'custom-logs',
|
|
35
|
+
});
|
|
36
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
37
|
+
template.hasResourceProperties('AWS::S3::Bucket', {
|
|
38
|
+
BucketName: 'custom-logs-123456789012-us-east-1',
|
|
39
|
+
});
|
|
40
|
+
});
|
|
41
|
+
test('applies default lifecycle rules', () => {
|
|
42
|
+
new access_log_1.AccessLog(stack, 'AccessLog');
|
|
43
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
44
|
+
template.hasResourceProperties('AWS::S3::Bucket', {
|
|
45
|
+
LifecycleConfiguration: {
|
|
46
|
+
Rules: assertions_1.Match.arrayWith([
|
|
47
|
+
assertions_1.Match.objectLike({
|
|
48
|
+
Status: 'Enabled',
|
|
49
|
+
Transitions: assertions_1.Match.arrayWith([
|
|
50
|
+
{ StorageClass: 'STANDARD_IA', TransitionInDays: 30 },
|
|
51
|
+
{ StorageClass: 'GLACIER', TransitionInDays: 90 },
|
|
52
|
+
]),
|
|
53
|
+
ExpirationInDays: 365,
|
|
54
|
+
}),
|
|
55
|
+
]),
|
|
56
|
+
},
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
test('applies custom lifecycle rules', () => {
|
|
60
|
+
new access_log_1.AccessLog(stack, 'AccessLog', {
|
|
61
|
+
lifecycleRules: [{
|
|
62
|
+
id: 'CustomRule',
|
|
63
|
+
enabled: true,
|
|
64
|
+
expiration: aws_cdk_lib_1.Duration.days(180),
|
|
65
|
+
}],
|
|
66
|
+
});
|
|
67
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
68
|
+
template.hasResourceProperties('AWS::S3::Bucket', {
|
|
69
|
+
LifecycleConfiguration: {
|
|
70
|
+
Rules: [assertions_1.Match.objectLike({ ExpirationInDays: 180 })],
|
|
71
|
+
},
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
test('enables versioning when specified', () => {
|
|
75
|
+
new access_log_1.AccessLog(stack, 'AccessLog', { versioned: true });
|
|
76
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
77
|
+
template.hasResourceProperties('AWS::S3::Bucket', {
|
|
78
|
+
VersioningConfiguration: { Status: 'Enabled' },
|
|
79
|
+
});
|
|
80
|
+
});
|
|
81
|
+
test('adds bucket policy for S3 and CloudWatch Logs', () => {
|
|
82
|
+
new access_log_1.AccessLog(stack, 'AccessLog');
|
|
83
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
84
|
+
template.hasResourceProperties('AWS::S3::BucketPolicy', {
|
|
85
|
+
PolicyDocument: {
|
|
86
|
+
Statement: assertions_1.Match.arrayWith([
|
|
87
|
+
assertions_1.Match.objectLike({
|
|
88
|
+
Sid: 'AllowAccessLogDelivery',
|
|
89
|
+
Effect: 'Allow',
|
|
90
|
+
Principal: {
|
|
91
|
+
Service: assertions_1.Match.arrayWith(['logging.s3.amazonaws.com', 'delivery.logs.amazonaws.com']),
|
|
92
|
+
},
|
|
93
|
+
Action: assertions_1.Match.arrayWith(['s3:PutObject', 's3:GetBucketAcl', 's3:ListBucket']),
|
|
94
|
+
}),
|
|
95
|
+
]),
|
|
96
|
+
},
|
|
97
|
+
});
|
|
98
|
+
});
|
|
99
|
+
test('adds bucket policy for ELB', () => {
|
|
100
|
+
new access_log_1.AccessLog(stack, 'AccessLog');
|
|
101
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
102
|
+
template.hasResourceProperties('AWS::S3::BucketPolicy', {
|
|
103
|
+
PolicyDocument: {
|
|
104
|
+
Statement: assertions_1.Match.arrayWith([
|
|
105
|
+
assertions_1.Match.objectLike({
|
|
106
|
+
Sid: 'AllowELBAccessLogDelivery',
|
|
107
|
+
Effect: 'Allow',
|
|
108
|
+
Principal: { Service: 'elasticloadbalancing.amazonaws.com' },
|
|
109
|
+
Action: 's3:PutObject',
|
|
110
|
+
}),
|
|
111
|
+
]),
|
|
112
|
+
},
|
|
113
|
+
});
|
|
114
|
+
});
|
|
115
|
+
test('enforces SSL', () => {
|
|
116
|
+
new access_log_1.AccessLog(stack, 'AccessLog');
|
|
117
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
118
|
+
template.hasResourceProperties('AWS::S3::BucketPolicy', {
|
|
119
|
+
PolicyDocument: {
|
|
120
|
+
Statement: assertions_1.Match.arrayWith([
|
|
121
|
+
assertions_1.Match.objectLike({
|
|
122
|
+
Effect: 'Deny',
|
|
123
|
+
Condition: { Bool: { 'aws:SecureTransport': 'false' } },
|
|
124
|
+
}),
|
|
125
|
+
]),
|
|
126
|
+
},
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
test('getLogPath returns correct path', () => {
|
|
130
|
+
const accessLog = new access_log_1.AccessLog(stack, 'AccessLog');
|
|
131
|
+
expect(accessLog.getLogPath('alb')).toBe('access-logs-123456789012-us-east-1/access-logs/alb');
|
|
132
|
+
expect(accessLog.getLogPath('alb', 'my-alb')).toBe('access-logs-123456789012-us-east-1/access-logs/alb/my-alb');
|
|
133
|
+
});
|
|
134
|
+
test('getLogUri returns correct S3 URI', () => {
|
|
135
|
+
const accessLog = new access_log_1.AccessLog(stack, 'AccessLog');
|
|
136
|
+
expect(accessLog.getLogUri('cloudfront')).toBe('s3://access-logs-123456789012-us-east-1/access-logs/cloudfront');
|
|
137
|
+
expect(accessLog.getLogUri('cloudfront', 'my-dist')).toBe('s3://access-logs-123456789012-us-east-1/access-logs/cloudfront/my-dist');
|
|
138
|
+
});
|
|
139
|
+
test('uses custom bucket prefix', () => {
|
|
140
|
+
const accessLog = new access_log_1.AccessLog(stack, 'AccessLog', {
|
|
141
|
+
bucketPrefix: 'custom-prefix',
|
|
142
|
+
});
|
|
143
|
+
expect(accessLog.getLogPath('s3')).toBe('access-logs-123456789012-us-east-1/custom-prefix/s3');
|
|
144
|
+
});
|
|
145
|
+
});
|
|
146
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"access-log.test.js","sourceRoot":"","sources":["../../../use-cases/framework/tests/access-log.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack, Duration } from 'aws-cdk-lib';\nimport { Template, Match } from 'aws-cdk-lib/assertions';\nimport { AccessLog } from '../foundation/access-log';\n\ndescribe('AccessLog', () => {\n  let stack: Stack;\n\n  beforeEach(() => {\n    stack = new Stack(undefined, 'TestStack', {\n      env: { account: '123456789012', region: 'us-east-1' },\n    });\n  });\n\n  test('creates bucket with default configuration', () => {\n    new AccessLog(stack, 'AccessLog');\n\n    const template = Template.fromStack(stack);\n    template.resourceCountIs('AWS::S3::Bucket', 1);\n    template.hasResourceProperties('AWS::S3::Bucket', {\n      BucketName: 'access-logs-123456789012-us-east-1',\n      BucketEncryption: {\n        ServerSideEncryptionConfiguration: [{\n          ServerSideEncryptionByDefault: { SSEAlgorithm: 'AES256' },\n        }],\n      },\n      PublicAccessBlockConfiguration: Match.objectLike({\n        BlockPublicAcls: true,\n        BlockPublicPolicy: true,\n        IgnorePublicAcls: true,\n        RestrictPublicBuckets: true,\n      }),\n    });\n  });\n\n  test('creates bucket with custom name', () => {\n    new AccessLog(stack, 'AccessLog', {\n      bucketName: 'custom-logs',\n    });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::S3::Bucket', {\n      BucketName: 'custom-logs-123456789012-us-east-1',\n    });\n  });\n\n  test('applies default lifecycle rules', () => {\n    new AccessLog(stack, 'AccessLog');\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::S3::Bucket', {\n      LifecycleConfiguration: {\n        Rules: Match.arrayWith([\n          Match.objectLike({\n            Status: 'Enabled',\n            Transitions: Match.arrayWith([\n              { StorageClass: 'STANDARD_IA', TransitionInDays: 30 },\n              { StorageClass: 'GLACIER', TransitionInDays: 90 },\n            ]),\n            ExpirationInDays: 365,\n          }),\n        ]),\n      },\n    });\n  });\n\n  test('applies custom lifecycle rules', () => {\n    new AccessLog(stack, 'AccessLog', {\n      lifecycleRules: [{\n        id: 'CustomRule',\n        enabled: true,\n        expiration: Duration.days(180),\n      }],\n    });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::S3::Bucket', {\n      LifecycleConfiguration: {\n        Rules: [Match.objectLike({ ExpirationInDays: 180 })],\n      },\n    });\n  });\n\n  test('enables versioning when specified', () => {\n    new AccessLog(stack, 'AccessLog', { versioned: true });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::S3::Bucket', {\n      VersioningConfiguration: { Status: 'Enabled' },\n    });\n  });\n\n  test('adds bucket policy for S3 and CloudWatch Logs', () => {\n    new AccessLog(stack, 'AccessLog');\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::S3::BucketPolicy', {\n      PolicyDocument: {\n        Statement: Match.arrayWith([\n          Match.objectLike({\n            Sid: 'AllowAccessLogDelivery',\n            Effect: 'Allow',\n            Principal: {\n              Service: Match.arrayWith(['logging.s3.amazonaws.com', 'delivery.logs.amazonaws.com']),\n            },\n            Action: Match.arrayWith(['s3:PutObject', 's3:GetBucketAcl', 's3:ListBucket']),\n          }),\n        ]),\n      },\n    });\n  });\n\n  test('adds bucket policy for ELB', () => {\n    new AccessLog(stack, 'AccessLog');\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::S3::BucketPolicy', {\n      PolicyDocument: {\n        Statement: Match.arrayWith([\n          Match.objectLike({\n            Sid: 'AllowELBAccessLogDelivery',\n            Effect: 'Allow',\n            Principal: { Service: 'elasticloadbalancing.amazonaws.com' },\n            Action: 's3:PutObject',\n          }),\n        ]),\n      },\n    });\n  });\n\n  test('enforces SSL', () => {\n    new AccessLog(stack, 'AccessLog');\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::S3::BucketPolicy', {\n      PolicyDocument: {\n        Statement: Match.arrayWith([\n          Match.objectLike({\n            Effect: 'Deny',\n            Condition: { Bool: { 'aws:SecureTransport': 'false' } },\n          }),\n        ]),\n      },\n    });\n  });\n\n  test('getLogPath returns correct path', () => {\n    const accessLog = new AccessLog(stack, 'AccessLog');\n    expect(accessLog.getLogPath('alb')).toBe('access-logs-123456789012-us-east-1/access-logs/alb');\n    expect(accessLog.getLogPath('alb', 'my-alb')).toBe('access-logs-123456789012-us-east-1/access-logs/alb/my-alb');\n  });\n\n  test('getLogUri returns correct S3 URI', () => {\n    const accessLog = new AccessLog(stack, 'AccessLog');\n    expect(accessLog.getLogUri('cloudfront')).toBe('s3://access-logs-123456789012-us-east-1/access-logs/cloudfront');\n    expect(accessLog.getLogUri('cloudfront', 'my-dist')).toBe('s3://access-logs-123456789012-us-east-1/access-logs/cloudfront/my-dist');\n  });\n\n  test('uses custom bucket prefix', () => {\n    const accessLog = new AccessLog(stack, 'AccessLog', {\n      bucketPrefix: 'custom-prefix',\n    });\n    expect(accessLog.getLogPath('s3')).toBe('access-logs-123456789012-us-east-1/custom-prefix/s3');\n  });\n});\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const path = require("path");
|
|
4
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
5
|
+
const assertions_1 = require("aws-cdk-lib/assertions");
|
|
6
|
+
const aws_bedrock_1 = require("aws-cdk-lib/aws-bedrock");
|
|
7
|
+
const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets");
|
|
8
|
+
const batch_agent_1 = require("../agents/batch-agent");
|
|
9
|
+
const network_1 = require("../foundation/network");
|
|
10
|
+
describe('BatchAgent', () => {
|
|
11
|
+
let stack;
|
|
12
|
+
let systemPrompt;
|
|
13
|
+
const testModel = aws_bedrock_1.FoundationModelIdentifier.ANTHROPIC_CLAUDE_3_SONNET_20240229_V1_0;
|
|
14
|
+
beforeEach(() => {
|
|
15
|
+
stack = new aws_cdk_lib_1.Stack(undefined, 'TestStack', {
|
|
16
|
+
env: { account: '123456789012', region: 'us-east-1' },
|
|
17
|
+
});
|
|
18
|
+
systemPrompt = new aws_s3_assets_1.Asset(stack, 'SystemPrompt', {
|
|
19
|
+
path: path.join(__dirname, '../agents/resources/default-strands-agent/batch.py'),
|
|
20
|
+
});
|
|
21
|
+
});
|
|
22
|
+
test('creates Lambda function with correct configuration', () => {
|
|
23
|
+
new batch_agent_1.BatchAgent(stack, 'Agent', {
|
|
24
|
+
agentName: 'TestAgent',
|
|
25
|
+
prompt: 'Test prompt',
|
|
26
|
+
agentDefinition: {
|
|
27
|
+
bedrockModel: { fmModelId: testModel },
|
|
28
|
+
systemPrompt,
|
|
29
|
+
},
|
|
30
|
+
});
|
|
31
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
32
|
+
template.resourceCountIs('AWS::Lambda::Function', 1);
|
|
33
|
+
template.hasResourceProperties('AWS::Lambda::Function', {
|
|
34
|
+
Runtime: assertions_1.Match.stringLikeRegexp('python3\\.1[23]'),
|
|
35
|
+
Timeout: 600,
|
|
36
|
+
MemorySize: 1024,
|
|
37
|
+
Environment: {
|
|
38
|
+
Variables: assertions_1.Match.objectLike({
|
|
39
|
+
MODEL_ID: testModel.modelId,
|
|
40
|
+
INVOKE_TYPE: 'batch',
|
|
41
|
+
PROMPT: 'Test prompt',
|
|
42
|
+
}),
|
|
43
|
+
},
|
|
44
|
+
});
|
|
45
|
+
});
|
|
46
|
+
test('creates IAM role with Bedrock permissions', () => {
|
|
47
|
+
new batch_agent_1.BatchAgent(stack, 'Agent', {
|
|
48
|
+
agentName: 'TestAgent',
|
|
49
|
+
prompt: 'Test prompt',
|
|
50
|
+
agentDefinition: {
|
|
51
|
+
bedrockModel: { fmModelId: testModel },
|
|
52
|
+
systemPrompt,
|
|
53
|
+
},
|
|
54
|
+
});
|
|
55
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
56
|
+
template.hasResourceProperties('AWS::IAM::Role', {
|
|
57
|
+
AssumeRolePolicyDocument: {
|
|
58
|
+
Statement: assertions_1.Match.arrayWith([
|
|
59
|
+
assertions_1.Match.objectLike({
|
|
60
|
+
Principal: { Service: 'lambda.amazonaws.com' },
|
|
61
|
+
}),
|
|
62
|
+
]),
|
|
63
|
+
},
|
|
64
|
+
});
|
|
65
|
+
template.hasResourceProperties('AWS::IAM::Policy', {
|
|
66
|
+
PolicyDocument: {
|
|
67
|
+
Statement: assertions_1.Match.arrayWith([
|
|
68
|
+
assertions_1.Match.objectLike({
|
|
69
|
+
Action: assertions_1.Match.arrayWith(['bedrock:InvokeModel']),
|
|
70
|
+
}),
|
|
71
|
+
]),
|
|
72
|
+
},
|
|
73
|
+
});
|
|
74
|
+
});
|
|
75
|
+
test('creates KMS key for environment encryption', () => {
|
|
76
|
+
new batch_agent_1.BatchAgent(stack, 'Agent', {
|
|
77
|
+
agentName: 'TestAgent',
|
|
78
|
+
prompt: 'Test prompt',
|
|
79
|
+
agentDefinition: {
|
|
80
|
+
bedrockModel: { fmModelId: testModel },
|
|
81
|
+
systemPrompt,
|
|
82
|
+
},
|
|
83
|
+
});
|
|
84
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
85
|
+
// At least one KMS key exists (agent encryption key, plus potentially asset deployment key)
|
|
86
|
+
const keys = template.findResources('AWS::KMS::Key');
|
|
87
|
+
expect(Object.keys(keys).length).toBeGreaterThanOrEqual(1);
|
|
88
|
+
// Verify at least one key has rotation enabled
|
|
89
|
+
const hasRotationEnabled = Object.values(keys).some((key) => key.Properties.EnableKeyRotation === true);
|
|
90
|
+
expect(hasRotationEnabled).toBe(true);
|
|
91
|
+
});
|
|
92
|
+
test('configures VPC when network provided', () => {
|
|
93
|
+
const network = new network_1.Network(stack, 'Network');
|
|
94
|
+
new batch_agent_1.BatchAgent(stack, 'Agent', {
|
|
95
|
+
agentName: 'TestAgent',
|
|
96
|
+
prompt: 'Test prompt',
|
|
97
|
+
network,
|
|
98
|
+
agentDefinition: {
|
|
99
|
+
bedrockModel: { fmModelId: testModel },
|
|
100
|
+
systemPrompt,
|
|
101
|
+
},
|
|
102
|
+
});
|
|
103
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
104
|
+
template.hasResourceProperties('AWS::Lambda::Function', {
|
|
105
|
+
VpcConfig: assertions_1.Match.objectLike({
|
|
106
|
+
SubnetIds: assertions_1.Match.anyValue(),
|
|
107
|
+
}),
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
test('sets EXPECT_JSON environment variable when expectJson is true', () => {
|
|
111
|
+
new batch_agent_1.BatchAgent(stack, 'Agent', {
|
|
112
|
+
agentName: 'TestAgent',
|
|
113
|
+
prompt: 'Test prompt',
|
|
114
|
+
expectJson: true,
|
|
115
|
+
agentDefinition: {
|
|
116
|
+
bedrockModel: { fmModelId: testModel },
|
|
117
|
+
systemPrompt,
|
|
118
|
+
},
|
|
119
|
+
});
|
|
120
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
121
|
+
template.hasResourceProperties('AWS::Lambda::Function', {
|
|
122
|
+
Environment: {
|
|
123
|
+
Variables: assertions_1.Match.objectLike({
|
|
124
|
+
EXPECT_JSON: 'True',
|
|
125
|
+
}),
|
|
126
|
+
},
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
test('enables observability with Powertools configuration', () => {
|
|
130
|
+
new batch_agent_1.BatchAgent(stack, 'Agent', {
|
|
131
|
+
agentName: 'TestAgent',
|
|
132
|
+
prompt: 'Test prompt',
|
|
133
|
+
enableObservability: true,
|
|
134
|
+
agentDefinition: {
|
|
135
|
+
bedrockModel: { fmModelId: testModel },
|
|
136
|
+
systemPrompt,
|
|
137
|
+
},
|
|
138
|
+
});
|
|
139
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
140
|
+
template.hasResourceProperties('AWS::Lambda::Function', {
|
|
141
|
+
Environment: {
|
|
142
|
+
Variables: assertions_1.Match.objectLike({
|
|
143
|
+
POWERTOOLS_SERVICE_NAME: assertions_1.Match.anyValue(),
|
|
144
|
+
POWERTOOLS_METRICS_NAMESPACE: assertions_1.Match.anyValue(),
|
|
145
|
+
}),
|
|
146
|
+
},
|
|
147
|
+
});
|
|
148
|
+
});
|
|
149
|
+
test('grants read access to system prompt asset', () => {
|
|
150
|
+
new batch_agent_1.BatchAgent(stack, 'Agent', {
|
|
151
|
+
agentName: 'TestAgent',
|
|
152
|
+
prompt: 'Test prompt',
|
|
153
|
+
agentDefinition: {
|
|
154
|
+
bedrockModel: { fmModelId: testModel },
|
|
155
|
+
systemPrompt,
|
|
156
|
+
},
|
|
157
|
+
});
|
|
158
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
159
|
+
// Verify that IAM policies exist (S3 permissions are granted via Asset construct)
|
|
160
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
161
|
+
expect(Object.keys(policies).length).toBeGreaterThan(0);
|
|
162
|
+
});
|
|
163
|
+
});
|
|
164
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"batch-agent.test.js","sourceRoot":"","sources":["../../../use-cases/framework/tests/batch-agent.test.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { Stack } from 'aws-cdk-lib';\nimport { Template, Match } from 'aws-cdk-lib/assertions';\nimport { FoundationModelIdentifier } from 'aws-cdk-lib/aws-bedrock';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { BatchAgent } from '../agents/batch-agent';\nimport { Network } from '../foundation/network';\n\ndescribe('BatchAgent', () => {\n  let stack: Stack;\n  let systemPrompt: Asset;\n  const testModel = FoundationModelIdentifier.ANTHROPIC_CLAUDE_3_SONNET_20240229_V1_0;\n\n  beforeEach(() => {\n    stack = new Stack(undefined, 'TestStack', {\n      env: { account: '123456789012', region: 'us-east-1' },\n    });\n\n    systemPrompt = new Asset(stack, 'SystemPrompt', {\n      path: path.join(__dirname, '../agents/resources/default-strands-agent/batch.py'),\n    });\n  });\n\n  test('creates Lambda function with correct configuration', () => {\n    new BatchAgent(stack, 'Agent', {\n      agentName: 'TestAgent',\n      prompt: 'Test prompt',\n      agentDefinition: {\n        bedrockModel: { fmModelId: testModel },\n        systemPrompt,\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    template.resourceCountIs('AWS::Lambda::Function', 1);\n    template.hasResourceProperties('AWS::Lambda::Function', {\n      Runtime: Match.stringLikeRegexp('python3\\\\.1[23]'),\n      Timeout: 600,\n      MemorySize: 1024,\n      Environment: {\n        Variables: Match.objectLike({\n          MODEL_ID: testModel.modelId,\n          INVOKE_TYPE: 'batch',\n          PROMPT: 'Test prompt',\n        }),\n      },\n    });\n  });\n\n  test('creates IAM role with Bedrock permissions', () => {\n    new BatchAgent(stack, 'Agent', {\n      agentName: 'TestAgent',\n      prompt: 'Test prompt',\n      agentDefinition: {\n        bedrockModel: { fmModelId: testModel },\n        systemPrompt,\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::IAM::Role', {\n      AssumeRolePolicyDocument: {\n        Statement: Match.arrayWith([\n          Match.objectLike({\n            Principal: { Service: 'lambda.amazonaws.com' },\n          }),\n        ]),\n      },\n    });\n\n    template.hasResourceProperties('AWS::IAM::Policy', {\n      PolicyDocument: {\n        Statement: Match.arrayWith([\n          Match.objectLike({\n            Action: Match.arrayWith(['bedrock:InvokeModel']),\n          }),\n        ]),\n      },\n    });\n  });\n\n  test('creates KMS key for environment encryption', () => {\n    new BatchAgent(stack, 'Agent', {\n      agentName: 'TestAgent',\n      prompt: 'Test prompt',\n      agentDefinition: {\n        bedrockModel: { fmModelId: testModel },\n        systemPrompt,\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    // At least one KMS key exists (agent encryption key, plus potentially asset deployment key)\n    const keys = template.findResources('AWS::KMS::Key');\n    expect(Object.keys(keys).length).toBeGreaterThanOrEqual(1);\n\n    // Verify at least one key has rotation enabled\n    const hasRotationEnabled = Object.values(keys).some((key: any) =>\n      key.Properties.EnableKeyRotation === true,\n    );\n    expect(hasRotationEnabled).toBe(true);\n  });\n\n  test('configures VPC when network provided', () => {\n    const network = new Network(stack, 'Network');\n\n    new BatchAgent(stack, 'Agent', {\n      agentName: 'TestAgent',\n      prompt: 'Test prompt',\n      network,\n      agentDefinition: {\n        bedrockModel: { fmModelId: testModel },\n        systemPrompt,\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::Lambda::Function', {\n      VpcConfig: Match.objectLike({\n        SubnetIds: Match.anyValue(),\n      }),\n    });\n  });\n\n  test('sets EXPECT_JSON environment variable when expectJson is true', () => {\n    new BatchAgent(stack, 'Agent', {\n      agentName: 'TestAgent',\n      prompt: 'Test prompt',\n      expectJson: true,\n      agentDefinition: {\n        bedrockModel: { fmModelId: testModel },\n        systemPrompt,\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::Lambda::Function', {\n      Environment: {\n        Variables: Match.objectLike({\n          EXPECT_JSON: 'True',\n        }),\n      },\n    });\n  });\n\n  test('enables observability with Powertools configuration', () => {\n    new BatchAgent(stack, 'Agent', {\n      agentName: 'TestAgent',\n      prompt: 'Test prompt',\n      enableObservability: true,\n      agentDefinition: {\n        bedrockModel: { fmModelId: testModel },\n        systemPrompt,\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::Lambda::Function', {\n      Environment: {\n        Variables: Match.objectLike({\n          POWERTOOLS_SERVICE_NAME: Match.anyValue(),\n          POWERTOOLS_METRICS_NAMESPACE: Match.anyValue(),\n        }),\n      },\n    });\n  });\n\n  test('grants read access to system prompt asset', () => {\n    new BatchAgent(stack, 'Agent', {\n      agentName: 'TestAgent',\n      prompt: 'Test prompt',\n      agentDefinition: {\n        bedrockModel: { fmModelId: testModel },\n        systemPrompt,\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    // Verify that IAM policies exist (S3 permissions are granted via Asset construct)\n    const policies = template.findResources('AWS::IAM::Policy');\n    expect(Object.keys(policies).length).toBeGreaterThan(0);\n  });\n});\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
4
|
+
const aws_bedrock_1 = require("aws-cdk-lib/aws-bedrock");
|
|
5
|
+
const bedrock_1 = require("../bedrock/bedrock");
|
|
6
|
+
describe('BedrockModelUtils', () => {
|
|
7
|
+
let stack;
|
|
8
|
+
const testModel = aws_bedrock_1.FoundationModelIdentifier.ANTHROPIC_CLAUDE_3_SONNET_20240229_V1_0;
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
stack = new aws_cdk_lib_1.Stack(undefined, 'TestStack', {
|
|
11
|
+
env: { account: '123456789012', region: 'us-east-1' },
|
|
12
|
+
});
|
|
13
|
+
});
|
|
14
|
+
test('deriveActualModelId returns model ID without cross-region inference', () => {
|
|
15
|
+
const modelId = bedrock_1.BedrockModelUtils.deriveActualModelId({
|
|
16
|
+
fmModelId: testModel,
|
|
17
|
+
useCrossRegionInference: false,
|
|
18
|
+
});
|
|
19
|
+
expect(modelId).toBe(testModel.modelId);
|
|
20
|
+
});
|
|
21
|
+
test('deriveActualModelId returns prefixed model ID with cross-region inference', () => {
|
|
22
|
+
const modelId = bedrock_1.BedrockModelUtils.deriveActualModelId({
|
|
23
|
+
fmModelId: testModel,
|
|
24
|
+
useCrossRegionInference: true,
|
|
25
|
+
crossRegionInferencePrefix: bedrock_1.BedrockCrossRegionInferencePrefix.US,
|
|
26
|
+
});
|
|
27
|
+
expect(modelId).toBe(`us.${testModel.modelId}`);
|
|
28
|
+
});
|
|
29
|
+
test('deriveActualModelId uses EU prefix when specified', () => {
|
|
30
|
+
const modelId = bedrock_1.BedrockModelUtils.deriveActualModelId({
|
|
31
|
+
fmModelId: testModel,
|
|
32
|
+
useCrossRegionInference: true,
|
|
33
|
+
crossRegionInferencePrefix: bedrock_1.BedrockCrossRegionInferencePrefix.EU,
|
|
34
|
+
});
|
|
35
|
+
expect(modelId).toBe(`eu.${testModel.modelId}`);
|
|
36
|
+
});
|
|
37
|
+
test('deriveActualModelId uses default model when not specified', () => {
|
|
38
|
+
const modelId = bedrock_1.BedrockModelUtils.deriveActualModelId();
|
|
39
|
+
expect(modelId).toBe(aws_bedrock_1.FoundationModelIdentifier.ANTHROPIC_CLAUDE_SONNET_4_20250514_V1_0.modelId);
|
|
40
|
+
});
|
|
41
|
+
test('generateModelIAMPermissions creates policy for foundation model', () => {
|
|
42
|
+
const policy = bedrock_1.BedrockModelUtils.generateModelIAMPermissions(stack, {
|
|
43
|
+
fmModelId: testModel,
|
|
44
|
+
});
|
|
45
|
+
const statement = policy.toJSON();
|
|
46
|
+
expect(statement.Effect).toBe('Allow');
|
|
47
|
+
expect(statement.Action).toEqual([
|
|
48
|
+
'bedrock:InvokeModel',
|
|
49
|
+
'bedrock:InvokeModelWithResponseStream',
|
|
50
|
+
]);
|
|
51
|
+
expect(statement.Resource).toContainEqual(`arn:aws:bedrock:*::foundation-model/${testModel.modelId}`);
|
|
52
|
+
});
|
|
53
|
+
test('generateModelIAMPermissions includes inference profile ARN', () => {
|
|
54
|
+
const policy = bedrock_1.BedrockModelUtils.generateModelIAMPermissions(stack, {
|
|
55
|
+
fmModelId: testModel,
|
|
56
|
+
useCrossRegionInference: true,
|
|
57
|
+
crossRegionInferencePrefix: bedrock_1.BedrockCrossRegionInferencePrefix.US,
|
|
58
|
+
});
|
|
59
|
+
const statement = policy.toJSON();
|
|
60
|
+
expect(statement.Resource).toContainEqual(`arn:aws:bedrock:us-east-1:123456789012:inference-profile/us.${testModel.modelId}`);
|
|
61
|
+
});
|
|
62
|
+
test('generateModelIAMPermissions uses default US prefix', () => {
|
|
63
|
+
const policy = bedrock_1.BedrockModelUtils.generateModelIAMPermissions(stack);
|
|
64
|
+
const statement = policy.toJSON();
|
|
65
|
+
expect(statement.Resource).toContainEqual(expect.stringContaining('inference-profile/us.'));
|
|
66
|
+
});
|
|
67
|
+
});
|
|
68
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmVkcm9jay50ZXN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vdXNlLWNhc2VzL2ZyYW1ld29yay90ZXN0cy9iZWRyb2NrLnRlc3QudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFBQSw2Q0FBb0M7QUFDcEMseURBQW9FO0FBQ3BFLGdEQUEwRjtBQUUxRixRQUFRLENBQUMsbUJBQW1CLEVBQUUsR0FBRyxFQUFFO0lBQ2pDLElBQUksS0FBWSxDQUFDO0lBQ2pCLE1BQU0sU0FBUyxHQUFHLHVDQUF5QixDQUFDLHVDQUF1QyxDQUFDO0lBRXBGLFVBQVUsQ0FBQyxHQUFHLEVBQUU7UUFDZCxLQUFLLEdBQUcsSUFBSSxtQkFBSyxDQUFDLFNBQVMsRUFBRSxXQUFXLEVBQUU7WUFDeEMsR0FBRyxFQUFFLEVBQUUsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLEVBQUUsV0FBVyxFQUFFO1NBQ3RELENBQUMsQ0FBQztJQUNMLENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLHFFQUFxRSxFQUFFLEdBQUcsRUFBRTtRQUMvRSxNQUFNLE9BQU8sR0FBRywyQkFBaUIsQ0FBQyxtQkFBbUIsQ0FBQztZQUNwRCxTQUFTLEVBQUUsU0FBUztZQUNwQix1QkFBdUIsRUFBRSxLQUFLO1NBQy9CLENBQUMsQ0FBQztRQUVILE1BQU0sQ0FBQyxPQUFPLENBQUMsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQzFDLENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLDJFQUEyRSxFQUFFLEdBQUcsRUFBRTtRQUNyRixNQUFNLE9BQU8sR0FBRywyQkFBaUIsQ0FBQyxtQkFBbUIsQ0FBQztZQUNwRCxTQUFTLEVBQUUsU0FBUztZQUNwQix1QkFBdUIsRUFBRSxJQUFJO1lBQzdCLDBCQUEwQixFQUFFLDJDQUFpQyxDQUFDLEVBQUU7U0FDakUsQ0FBQyxDQUFDO1FBRUgsTUFBTSxDQUFDLE9BQU8sQ0FBQyxDQUFDLElBQUksQ0FBQyxNQUFNLFNBQVMsQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDO0lBQ2xELENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLG1EQUFtRCxFQUFFLEdBQUcsRUFBRTtRQUM3RCxNQUFNLE9BQU8sR0FBRywyQkFBaUIsQ0FBQyxtQkFBbUIsQ0FBQztZQUNwRCxTQUFTLEVBQUUsU0FBUztZQUNwQix1QkFBdUIsRUFBRSxJQUFJO1lBQzdCLDBCQUEwQixFQUFFLDJDQUFpQyxDQUFDLEVBQUU7U0FDakUsQ0FBQyxDQUFDO1FBRUgsTUFBTSxDQUFDLE9BQU8sQ0FBQyxDQUFDLElBQUksQ0FBQyxNQUFNLFNBQVMsQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDO0lBQ2xELENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLDJEQUEyRCxFQUFFLEdBQUcsRUFBRTtRQUNyRSxNQUFNLE9BQU8sR0FBRywyQkFBaUIsQ0FBQyxtQkFBbUIsRUFBRSxDQUFDO1FBRXhELE1BQU0sQ0FBQyxPQUFPLENBQUMsQ0FBQyxJQUFJLENBQUMsdUNBQXlCLENBQUMsdUNBQXVDLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDbEcsQ0FBQyxDQUFDLENBQUM7SUFFSCxJQUFJLENBQUMsaUVBQWlFLEVBQUUsR0FBRyxFQUFFO1FBQzNFLE1BQU0sTUFBTSxHQUFHLDJCQUFpQixDQUFDLDJCQUEyQixDQUFDLEtBQUssRUFBRTtZQUNsRSxTQUFTLEVBQUUsU0FBUztTQUNyQixDQUFDLENBQUM7UUFFSCxNQUFNLFNBQVMsR0FBRyxNQUFNLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDbEMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDdkMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxPQUFPLENBQUM7WUFDL0IscUJBQXFCO1lBQ3JCLHVDQUF1QztTQUN4QyxDQUFDLENBQUM7UUFDSCxNQUFNLENBQUMsU0FBUyxDQUFDLFFBQVEsQ0FBQyxDQUFDLGNBQWMsQ0FDdkMsdUNBQXVDLFNBQVMsQ0FBQyxPQUFPLEVBQUUsQ0FDM0QsQ0FBQztJQUNKLENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLDREQUE0RCxFQUFFLEdBQUcsRUFBRTtRQUN0RSxNQUFNLE1BQU0sR0FBRywyQkFBaUIsQ0FBQywyQkFBMkIsQ0FBQyxLQUFLLEVBQUU7WUFDbEUsU0FBUyxFQUFFLFNBQVM7WUFDcEIsdUJBQXVCLEVBQUUsSUFBSTtZQUM3QiwwQkFBMEIsRUFBRSwyQ0FBaUMsQ0FBQyxFQUFFO1NBQ2pFLENBQUMsQ0FBQztRQUVILE1BQU0sU0FBUyxHQUFHLE1BQU0sQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUNsQyxNQUFNLENBQUMsU0FBUyxDQUFDLFFBQVEsQ0FBQyxDQUFDLGNBQWMsQ0FDdkMsK0RBQStELFNBQVMsQ0FBQyxPQUFPLEVBQUUsQ0FDbkYsQ0FBQztJQUNKLENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLG9EQUFvRCxFQUFFLEdBQUcsRUFBRTtRQUM5RCxNQUFNLE1BQU0sR0FBRywyQkFBaUIsQ0FBQywyQkFBMkIsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUVwRSxNQUFNLFNBQVMsR0FBRyxNQUFNLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDbEMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxRQUFRLENBQUMsQ0FBQyxjQUFjLENBQ3ZDLE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQyx1QkFBdUIsQ0FBQyxDQUNqRCxDQUFDO0lBQ0osQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFN0YWNrIH0gZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0IHsgRm91bmRhdGlvbk1vZGVsSWRlbnRpZmllciB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1iZWRyb2NrJztcbmltcG9ydCB7IEJlZHJvY2tDcm9zc1JlZ2lvbkluZmVyZW5jZVByZWZpeCwgQmVkcm9ja01vZGVsVXRpbHMgfSBmcm9tICcuLi9iZWRyb2NrL2JlZHJvY2snO1xuXG5kZXNjcmliZSgnQmVkcm9ja01vZGVsVXRpbHMnLCAoKSA9PiB7XG4gIGxldCBzdGFjazogU3RhY2s7XG4gIGNvbnN0IHRlc3RNb2RlbCA9IEZvdW5kYXRpb25Nb2RlbElkZW50aWZpZXIuQU5USFJPUElDX0NMQVVERV8zX1NPTk5FVF8yMDI0MDIyOV9WMV8wO1xuXG4gIGJlZm9yZUVhY2goKCkgPT4ge1xuICAgIHN0YWNrID0gbmV3IFN0YWNrKHVuZGVmaW5lZCwgJ1Rlc3RTdGFjaycsIHtcbiAgICAgIGVudjogeyBhY2NvdW50OiAnMTIzNDU2Nzg5MDEyJywgcmVnaW9uOiAndXMtZWFzdC0xJyB9LFxuICAgIH0pO1xuICB9KTtcblxuICB0ZXN0KCdkZXJpdmVBY3R1YWxNb2RlbElkIHJldHVybnMgbW9kZWwgSUQgd2l0aG91dCBjcm9zcy1yZWdpb24gaW5mZXJlbmNlJywgKCkgPT4ge1xuICAgIGNvbnN0IG1vZGVsSWQgPSBCZWRyb2NrTW9kZWxVdGlscy5kZXJpdmVBY3R1YWxNb2RlbElkKHtcbiAgICAgIGZtTW9kZWxJZDogdGVzdE1vZGVsLFxuICAgICAgdXNlQ3Jvc3NSZWdpb25JbmZlcmVuY2U6IGZhbHNlLFxuICAgIH0pO1xuXG4gICAgZXhwZWN0KG1vZGVsSWQpLnRvQmUodGVzdE1vZGVsLm1vZGVsSWQpO1xuICB9KTtcblxuICB0ZXN0KCdkZXJpdmVBY3R1YWxNb2RlbElkIHJldHVybnMgcHJlZml4ZWQgbW9kZWwgSUQgd2l0aCBjcm9zcy1yZWdpb24gaW5mZXJlbmNlJywgKCkgPT4ge1xuICAgIGNvbnN0IG1vZGVsSWQgPSBCZWRyb2NrTW9kZWxVdGlscy5kZXJpdmVBY3R1YWxNb2RlbElkKHtcbiAgICAgIGZtTW9kZWxJZDogdGVzdE1vZGVsLFxuICAgICAgdXNlQ3Jvc3NSZWdpb25JbmZlcmVuY2U6IHRydWUsXG4gICAgICBjcm9zc1JlZ2lvbkluZmVyZW5jZVByZWZpeDogQmVkcm9ja0Nyb3NzUmVnaW9uSW5mZXJlbmNlUHJlZml4LlVTLFxuICAgIH0pO1xuXG4gICAgZXhwZWN0KG1vZGVsSWQpLnRvQmUoYHVzLiR7dGVzdE1vZGVsLm1vZGVsSWR9YCk7XG4gIH0pO1xuXG4gIHRlc3QoJ2Rlcml2ZUFjdHVhbE1vZGVsSWQgdXNlcyBFVSBwcmVmaXggd2hlbiBzcGVjaWZpZWQnLCAoKSA9PiB7XG4gICAgY29uc3QgbW9kZWxJZCA9IEJlZHJvY2tNb2RlbFV0aWxzLmRlcml2ZUFjdHVhbE1vZGVsSWQoe1xuICAgICAgZm1Nb2RlbElkOiB0ZXN0TW9kZWwsXG4gICAgICB1c2VDcm9zc1JlZ2lvbkluZmVyZW5jZTogdHJ1ZSxcbiAgICAgIGNyb3NzUmVnaW9uSW5mZXJlbmNlUHJlZml4OiBCZWRyb2NrQ3Jvc3NSZWdpb25JbmZlcmVuY2VQcmVmaXguRVUsXG4gICAgfSk7XG5cbiAgICBleHBlY3QobW9kZWxJZCkudG9CZShgZXUuJHt0ZXN0TW9kZWwubW9kZWxJZH1gKTtcbiAgfSk7XG5cbiAgdGVzdCgnZGVyaXZlQWN0dWFsTW9kZWxJZCB1c2VzIGRlZmF1bHQgbW9kZWwgd2hlbiBub3Qgc3BlY2lmaWVkJywgKCkgPT4ge1xuICAgIGNvbnN0IG1vZGVsSWQgPSBCZWRyb2NrTW9kZWxVdGlscy5kZXJpdmVBY3R1YWxNb2RlbElkKCk7XG5cbiAgICBleHBlY3QobW9kZWxJZCkudG9CZShGb3VuZGF0aW9uTW9kZWxJZGVudGlmaWVyLkFOVEhST1BJQ19DTEFVREVfU09OTkVUXzRfMjAyNTA1MTRfVjFfMC5tb2RlbElkKTtcbiAgfSk7XG5cbiAgdGVzdCgnZ2VuZXJhdGVNb2RlbElBTVBlcm1pc3Npb25zIGNyZWF0ZXMgcG9saWN5IGZvciBmb3VuZGF0aW9uIG1vZGVsJywgKCkgPT4ge1xuICAgIGNvbnN0IHBvbGljeSA9IEJlZHJvY2tNb2RlbFV0aWxzLmdlbmVyYXRlTW9kZWxJQU1QZXJtaXNzaW9ucyhzdGFjaywge1xuICAgICAgZm1Nb2RlbElkOiB0ZXN0TW9kZWwsXG4gICAgfSk7XG5cbiAgICBjb25zdCBzdGF0ZW1lbnQgPSBwb2xpY3kudG9KU09OKCk7XG4gICAgZXhwZWN0KHN0YXRlbWVudC5FZmZlY3QpLnRvQmUoJ0FsbG93Jyk7XG4gICAgZXhwZWN0KHN0YXRlbWVudC5BY3Rpb24pLnRvRXF1YWwoW1xuICAgICAgJ2JlZHJvY2s6SW52b2tlTW9kZWwnLFxuICAgICAgJ2JlZHJvY2s6SW52b2tlTW9kZWxXaXRoUmVzcG9uc2VTdHJlYW0nLFxuICAgIF0pO1xuICAgIGV4cGVjdChzdGF0ZW1lbnQuUmVzb3VyY2UpLnRvQ29udGFpbkVxdWFsKFxuICAgICAgYGFybjphd3M6YmVkcm9jazoqOjpmb3VuZGF0aW9uLW1vZGVsLyR7dGVzdE1vZGVsLm1vZGVsSWR9YCxcbiAgICApO1xuICB9KTtcblxuICB0ZXN0KCdnZW5lcmF0ZU1vZGVsSUFNUGVybWlzc2lvbnMgaW5jbHVkZXMgaW5mZXJlbmNlIHByb2ZpbGUgQVJOJywgKCkgPT4ge1xuICAgIGNvbnN0IHBvbGljeSA9IEJlZHJvY2tNb2RlbFV0aWxzLmdlbmVyYXRlTW9kZWxJQU1QZXJtaXNzaW9ucyhzdGFjaywge1xuICAgICAgZm1Nb2RlbElkOiB0ZXN0TW9kZWwsXG4gICAgICB1c2VDcm9zc1JlZ2lvbkluZmVyZW5jZTogdHJ1ZSxcbiAgICAgIGNyb3NzUmVnaW9uSW5mZXJlbmNlUHJlZml4OiBCZWRyb2NrQ3Jvc3NSZWdpb25JbmZlcmVuY2VQcmVmaXguVVMsXG4gICAgfSk7XG5cbiAgICBjb25zdCBzdGF0ZW1lbnQgPSBwb2xpY3kudG9KU09OKCk7XG4gICAgZXhwZWN0KHN0YXRlbWVudC5SZXNvdXJjZSkudG9Db250YWluRXF1YWwoXG4gICAgICBgYXJuOmF3czpiZWRyb2NrOnVzLWVhc3QtMToxMjM0NTY3ODkwMTI6aW5mZXJlbmNlLXByb2ZpbGUvdXMuJHt0ZXN0TW9kZWwubW9kZWxJZH1gLFxuICAgICk7XG4gIH0pO1xuXG4gIHRlc3QoJ2dlbmVyYXRlTW9kZWxJQU1QZXJtaXNzaW9ucyB1c2VzIGRlZmF1bHQgVVMgcHJlZml4JywgKCkgPT4ge1xuICAgIGNvbnN0IHBvbGljeSA9IEJlZHJvY2tNb2RlbFV0aWxzLmdlbmVyYXRlTW9kZWxJQU1QZXJtaXNzaW9ucyhzdGFjayk7XG5cbiAgICBjb25zdCBzdGF0ZW1lbnQgPSBwb2xpY3kudG9KU09OKCk7XG4gICAgZXhwZWN0KHN0YXRlbWVudC5SZXNvdXJjZSkudG9Db250YWluRXF1YWwoXG4gICAgICBleHBlY3Quc3RyaW5nQ29udGFpbmluZygnaW5mZXJlbmNlLXByb2ZpbGUvdXMuJyksXG4gICAgKTtcbiAgfSk7XG59KTtcbiJdfQ==
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
4
|
+
const assertions_1 = require("aws-cdk-lib/assertions");
|
|
5
|
+
const aws_kms_1 = require("aws-cdk-lib/aws-kms");
|
|
6
|
+
const eventbridge_broker_1 = require("../foundation/eventbridge-broker");
|
|
7
|
+
describe('EventbridgeBroker', () => {
|
|
8
|
+
let stack;
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
stack = new aws_cdk_lib_1.Stack();
|
|
11
|
+
});
|
|
12
|
+
test('creates EventBus with KMS encryption', () => {
|
|
13
|
+
new eventbridge_broker_1.EventbridgeBroker(stack, 'Broker', {
|
|
14
|
+
eventSource: 'test.source',
|
|
15
|
+
});
|
|
16
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
17
|
+
template.resourceCountIs('AWS::Events::EventBus', 1);
|
|
18
|
+
template.resourceCountIs('AWS::KMS::Key', 1);
|
|
19
|
+
template.hasResourceProperties('AWS::KMS::Key', {
|
|
20
|
+
EnableKeyRotation: true,
|
|
21
|
+
});
|
|
22
|
+
});
|
|
23
|
+
test('uses custom KMS key when provided', () => {
|
|
24
|
+
const customKey = new aws_kms_1.Key(stack, 'CustomKey');
|
|
25
|
+
new eventbridge_broker_1.EventbridgeBroker(stack, 'Broker', {
|
|
26
|
+
eventSource: 'test.source',
|
|
27
|
+
kmsKey: customKey,
|
|
28
|
+
});
|
|
29
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
30
|
+
template.resourceCountIs('AWS::KMS::Key', 1);
|
|
31
|
+
});
|
|
32
|
+
test('sets custom event bus name', () => {
|
|
33
|
+
new eventbridge_broker_1.EventbridgeBroker(stack, 'Broker', {
|
|
34
|
+
eventSource: 'test.source',
|
|
35
|
+
name: 'CustomEventBus',
|
|
36
|
+
});
|
|
37
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
38
|
+
template.hasResourceProperties('AWS::Events::EventBus', {
|
|
39
|
+
Name: 'CustomEventBus',
|
|
40
|
+
});
|
|
41
|
+
});
|
|
42
|
+
test('applies RETAIN removal policy', () => {
|
|
43
|
+
new eventbridge_broker_1.EventbridgeBroker(stack, 'Broker', {
|
|
44
|
+
eventSource: 'test.source',
|
|
45
|
+
removalPolicy: aws_cdk_lib_1.RemovalPolicy.RETAIN,
|
|
46
|
+
});
|
|
47
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
48
|
+
template.hasResource('AWS::KMS::Key', {
|
|
49
|
+
DeletionPolicy: 'Retain',
|
|
50
|
+
});
|
|
51
|
+
});
|
|
52
|
+
test('applies DESTROY removal policy by default', () => {
|
|
53
|
+
new eventbridge_broker_1.EventbridgeBroker(stack, 'Broker', {
|
|
54
|
+
eventSource: 'test.source',
|
|
55
|
+
});
|
|
56
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
57
|
+
template.hasResource('AWS::KMS::Key', {
|
|
58
|
+
DeletionPolicy: 'Delete',
|
|
59
|
+
});
|
|
60
|
+
});
|
|
61
|
+
test('sendViaSfnChain creates EventBridge PutEvents task', () => {
|
|
62
|
+
const broker = new eventbridge_broker_1.EventbridgeBroker(stack, 'Broker', {
|
|
63
|
+
eventSource: 'test.source',
|
|
64
|
+
});
|
|
65
|
+
const task = broker.sendViaSfnChain('TestEvent', { key: 'value' });
|
|
66
|
+
expect(task).toBeDefined();
|
|
67
|
+
const stateJson = task.toStateJson();
|
|
68
|
+
expect(stateJson.Type).toBe('Task');
|
|
69
|
+
expect(stateJson.Parameters.Entries[0].Source).toBe('test.source');
|
|
70
|
+
expect(stateJson.Parameters.Entries[0].DetailType).toBe('TestEvent');
|
|
71
|
+
});
|
|
72
|
+
});
|
|
73
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZlbnRicmlkZ2UtYnJva2VyLnRlc3QuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi91c2UtY2FzZXMvZnJhbWV3b3JrL3Rlc3RzL2V2ZW50YnJpZGdlLWJyb2tlci50ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQUEsNkNBQW1EO0FBQ25ELHVEQUFrRDtBQUNsRCxpREFBMEM7QUFDMUMseUVBQXFFO0FBRXJFLFFBQVEsQ0FBQyxtQkFBbUIsRUFBRSxHQUFHLEVBQUU7SUFDakMsSUFBSSxLQUFZLENBQUM7SUFFakIsVUFBVSxDQUFDLEdBQUcsRUFBRTtRQUNkLEtBQUssR0FBRyxJQUFJLG1CQUFLLEVBQUUsQ0FBQztJQUN0QixDQUFDLENBQUMsQ0FBQztJQUVILElBQUksQ0FBQyxzQ0FBc0MsRUFBRSxHQUFHLEVBQUU7UUFDaEQsSUFBSSxzQ0FBaUIsQ0FBQyxLQUFLLEVBQUUsUUFBUSxFQUFFO1lBQ3JDLFdBQVcsRUFBRSxhQUFhO1NBQzNCLENBQUMsQ0FBQztRQUVILE1BQU0sUUFBUSxHQUFHLHFCQUFRLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzNDLFFBQVEsQ0FBQyxlQUFlLENBQUMsdUJBQXVCLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDckQsUUFBUSxDQUFDLGVBQWUsQ0FBQyxlQUFlLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFFN0MsUUFBUSxDQUFDLHFCQUFxQixDQUFDLGVBQWUsRUFBRTtZQUM5QyxpQkFBaUIsRUFBRSxJQUFJO1NBQ3hCLENBQUMsQ0FBQztJQUNMLENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLG1DQUFtQyxFQUFFLEdBQUcsRUFBRTtRQUM3QyxNQUFNLFNBQVMsR0FBRyxJQUFJLGFBQUcsQ0FBQyxLQUFLLEVBQUUsV0FBVyxDQUFDLENBQUM7UUFDOUMsSUFBSSxzQ0FBaUIsQ0FBQyxLQUFLLEVBQUUsUUFBUSxFQUFFO1lBQ3JDLFdBQVcsRUFBRSxhQUFhO1lBQzFCLE1BQU0sRUFBRSxTQUFTO1NBQ2xCLENBQUMsQ0FBQztRQUVILE1BQU0sUUFBUSxHQUFHLHFCQUFRLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzNDLFFBQVEsQ0FBQyxlQUFlLENBQUMsZUFBZSxFQUFFLENBQUMsQ0FBQyxDQUFDO0lBQy9DLENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLDRCQUE0QixFQUFFLEdBQUcsRUFBRTtRQUN0QyxJQUFJLHNDQUFpQixDQUFDLEtBQUssRUFBRSxRQUFRLEVBQUU7WUFDckMsV0FBVyxFQUFFLGFBQWE7WUFDMUIsSUFBSSxFQUFFLGdCQUFnQjtTQUN2QixDQUFDLENBQUM7UUFFSCxNQUFNLFFBQVEsR0FBRyxxQkFBUSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUMzQyxRQUFRLENBQUMscUJBQXFCLENBQUMsdUJBQXVCLEVBQUU7WUFDdEQsSUFBSSxFQUFFLGdCQUFnQjtTQUN2QixDQUFDLENBQUM7SUFDTCxDQUFDLENBQUMsQ0FBQztJQUVILElBQUksQ0FBQywrQkFBK0IsRUFBRSxHQUFHLEVBQUU7UUFDekMsSUFBSSxzQ0FBaUIsQ0FBQyxLQUFLLEVBQUUsUUFBUSxFQUFFO1lBQ3JDLFdBQVcsRUFBRSxhQUFhO1lBQzFCLGFBQWEsRUFBRSwyQkFBYSxDQUFDLE1BQU07U0FDcEMsQ0FBQyxDQUFDO1FBRUgsTUFBTSxRQUFRLEdBQUcscUJBQVEsQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDM0MsUUFBUSxDQUFDLFdBQVcsQ0FBQyxlQUFlLEVBQUU7WUFDcEMsY0FBYyxFQUFFLFFBQVE7U0FDekIsQ0FBQyxDQUFDO0lBQ0wsQ0FBQyxDQUFDLENBQUM7SUFFSCxJQUFJLENBQUMsMkNBQTJDLEVBQUUsR0FBRyxFQUFFO1FBQ3JELElBQUksc0NBQWlCLENBQUMsS0FBSyxFQUFFLFFBQVEsRUFBRTtZQUNyQyxXQUFXLEVBQUUsYUFBYTtTQUMzQixDQUFDLENBQUM7UUFFSCxNQUFNLFFBQVEsR0FBRyxxQkFBUSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUMzQyxRQUFRLENBQUMsV0FBVyxDQUFDLGVBQWUsRUFBRTtZQUNwQyxjQUFjLEVBQUUsUUFBUTtTQUN6QixDQUFDLENBQUM7SUFDTCxDQUFDLENBQUMsQ0FBQztJQUVILElBQUksQ0FBQyxvREFBb0QsRUFBRSxHQUFHLEVBQUU7UUFDOUQsTUFBTSxNQUFNLEdBQUcsSUFBSSxzQ0FBaUIsQ0FBQyxLQUFLLEVBQUUsUUFBUSxFQUFFO1lBQ3BELFdBQVcsRUFBRSxhQUFhO1NBQzNCLENBQUMsQ0FBQztRQUVILE1BQU0sSUFBSSxHQUFHLE1BQU0sQ0FBQyxlQUFlLENBQUMsV0FBVyxFQUFFLEVBQUUsR0FBRyxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUM7UUFFbkUsTUFBTSxDQUFDLElBQUksQ0FBQyxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQzNCLE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxXQUFXLEVBQVMsQ0FBQztRQUM1QyxNQUFNLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUNwQyxNQUFNLENBQUMsU0FBUyxDQUFDLFVBQVUsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxDQUFDO1FBQ25FLE1BQU0sQ0FBQyxTQUFTLENBQUMsVUFBVSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxVQUFVLENBQUMsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLENBQUM7SUFDdkUsQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJlbW92YWxQb2xpY3ksIFN0YWNrIH0gZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0IHsgVGVtcGxhdGUgfSBmcm9tICdhd3MtY2RrLWxpYi9hc3NlcnRpb25zJztcbmltcG9ydCB7IEtleSB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1rbXMnO1xuaW1wb3J0IHsgRXZlbnRicmlkZ2VCcm9rZXIgfSBmcm9tICcuLi9mb3VuZGF0aW9uL2V2ZW50YnJpZGdlLWJyb2tlcic7XG5cbmRlc2NyaWJlKCdFdmVudGJyaWRnZUJyb2tlcicsICgpID0+IHtcbiAgbGV0IHN0YWNrOiBTdGFjaztcblxuICBiZWZvcmVFYWNoKCgpID0+IHtcbiAgICBzdGFjayA9IG5ldyBTdGFjaygpO1xuICB9KTtcblxuICB0ZXN0KCdjcmVhdGVzIEV2ZW50QnVzIHdpdGggS01TIGVuY3J5cHRpb24nLCAoKSA9PiB7XG4gICAgbmV3IEV2ZW50YnJpZGdlQnJva2VyKHN0YWNrLCAnQnJva2VyJywge1xuICAgICAgZXZlbnRTb3VyY2U6ICd0ZXN0LnNvdXJjZScsXG4gICAgfSk7XG5cbiAgICBjb25zdCB0ZW1wbGF0ZSA9IFRlbXBsYXRlLmZyb21TdGFjayhzdGFjayk7XG4gICAgdGVtcGxhdGUucmVzb3VyY2VDb3VudElzKCdBV1M6OkV2ZW50czo6RXZlbnRCdXMnLCAxKTtcbiAgICB0ZW1wbGF0ZS5yZXNvdXJjZUNvdW50SXMoJ0FXUzo6S01TOjpLZXknLCAxKTtcblxuICAgIHRlbXBsYXRlLmhhc1Jlc291cmNlUHJvcGVydGllcygnQVdTOjpLTVM6OktleScsIHtcbiAgICAgIEVuYWJsZUtleVJvdGF0aW9uOiB0cnVlLFxuICAgIH0pO1xuICB9KTtcblxuICB0ZXN0KCd1c2VzIGN1c3RvbSBLTVMga2V5IHdoZW4gcHJvdmlkZWQnLCAoKSA9PiB7XG4gICAgY29uc3QgY3VzdG9tS2V5ID0gbmV3IEtleShzdGFjaywgJ0N1c3RvbUtleScpO1xuICAgIG5ldyBFdmVudGJyaWRnZUJyb2tlcihzdGFjaywgJ0Jyb2tlcicsIHtcbiAgICAgIGV2ZW50U291cmNlOiAndGVzdC5zb3VyY2UnLFxuICAgICAga21zS2V5OiBjdXN0b21LZXksXG4gICAgfSk7XG5cbiAgICBjb25zdCB0ZW1wbGF0ZSA9IFRlbXBsYXRlLmZyb21TdGFjayhzdGFjayk7XG4gICAgdGVtcGxhdGUucmVzb3VyY2VDb3VudElzKCdBV1M6OktNUzo6S2V5JywgMSk7XG4gIH0pO1xuXG4gIHRlc3QoJ3NldHMgY3VzdG9tIGV2ZW50IGJ1cyBuYW1lJywgKCkgPT4ge1xuICAgIG5ldyBFdmVudGJyaWRnZUJyb2tlcihzdGFjaywgJ0Jyb2tlcicsIHtcbiAgICAgIGV2ZW50U291cmNlOiAndGVzdC5zb3VyY2UnLFxuICAgICAgbmFtZTogJ0N1c3RvbUV2ZW50QnVzJyxcbiAgICB9KTtcblxuICAgIGNvbnN0IHRlbXBsYXRlID0gVGVtcGxhdGUuZnJvbVN0YWNrKHN0YWNrKTtcbiAgICB0ZW1wbGF0ZS5oYXNSZXNvdXJjZVByb3BlcnRpZXMoJ0FXUzo6RXZlbnRzOjpFdmVudEJ1cycsIHtcbiAgICAgIE5hbWU6ICdDdXN0b21FdmVudEJ1cycsXG4gICAgfSk7XG4gIH0pO1xuXG4gIHRlc3QoJ2FwcGxpZXMgUkVUQUlOIHJlbW92YWwgcG9saWN5JywgKCkgPT4ge1xuICAgIG5ldyBFdmVudGJyaWRnZUJyb2tlcihzdGFjaywgJ0Jyb2tlcicsIHtcbiAgICAgIGV2ZW50U291cmNlOiAndGVzdC5zb3VyY2UnLFxuICAgICAgcmVtb3ZhbFBvbGljeTogUmVtb3ZhbFBvbGljeS5SRVRBSU4sXG4gICAgfSk7XG5cbiAgICBjb25zdCB0ZW1wbGF0ZSA9IFRlbXBsYXRlLmZyb21TdGFjayhzdGFjayk7XG4gICAgdGVtcGxhdGUuaGFzUmVzb3VyY2UoJ0FXUzo6S01TOjpLZXknLCB7XG4gICAgICBEZWxldGlvblBvbGljeTogJ1JldGFpbicsXG4gICAgfSk7XG4gIH0pO1xuXG4gIHRlc3QoJ2FwcGxpZXMgREVTVFJPWSByZW1vdmFsIHBvbGljeSBieSBkZWZhdWx0JywgKCkgPT4ge1xuICAgIG5ldyBFdmVudGJyaWRnZUJyb2tlcihzdGFjaywgJ0Jyb2tlcicsIHtcbiAgICAgIGV2ZW50U291cmNlOiAndGVzdC5zb3VyY2UnLFxuICAgIH0pO1xuXG4gICAgY29uc3QgdGVtcGxhdGUgPSBUZW1wbGF0ZS5mcm9tU3RhY2soc3RhY2spO1xuICAgIHRlbXBsYXRlLmhhc1Jlc291cmNlKCdBV1M6OktNUzo6S2V5Jywge1xuICAgICAgRGVsZXRpb25Qb2xpY3k6ICdEZWxldGUnLFxuICAgIH0pO1xuICB9KTtcblxuICB0ZXN0KCdzZW5kVmlhU2ZuQ2hhaW4gY3JlYXRlcyBFdmVudEJyaWRnZSBQdXRFdmVudHMgdGFzaycsICgpID0+IHtcbiAgICBjb25zdCBicm9rZXIgPSBuZXcgRXZlbnRicmlkZ2VCcm9rZXIoc3RhY2ssICdCcm9rZXInLCB7XG4gICAgICBldmVudFNvdXJjZTogJ3Rlc3Quc291cmNlJyxcbiAgICB9KTtcblxuICAgIGNvbnN0IHRhc2sgPSBicm9rZXIuc2VuZFZpYVNmbkNoYWluKCdUZXN0RXZlbnQnLCB7IGtleTogJ3ZhbHVlJyB9KTtcblxuICAgIGV4cGVjdCh0YXNrKS50b0JlRGVmaW5lZCgpO1xuICAgIGNvbnN0IHN0YXRlSnNvbiA9IHRhc2sudG9TdGF0ZUpzb24oKSBhcyBhbnk7XG4gICAgZXhwZWN0KHN0YXRlSnNvbi5UeXBlKS50b0JlKCdUYXNrJyk7XG4gICAgZXhwZWN0KHN0YXRlSnNvbi5QYXJhbWV0ZXJzLkVudHJpZXNbMF0uU291cmNlKS50b0JlKCd0ZXN0LnNvdXJjZScpO1xuICAgIGV4cGVjdChzdGF0ZUpzb24uUGFyYW1ldGVycy5FbnRyaWVzWzBdLkRldGFpbFR5cGUpLnRvQmUoJ1Rlc3RFdmVudCcpO1xuICB9KTtcbn0pO1xuIl19
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|