@cdklabs/cdk-appmod-catalog-blueprints 1.3.0 → 1.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. package/.jsii +4 -4
  2. package/README.md +76 -98
  3. package/lib/document-processing/adapter/queued-s3-adapter.js +1 -1
  4. package/lib/document-processing/agentic-document-processing.js +1 -1
  5. package/lib/document-processing/base-document-processing.js +1 -1
  6. package/lib/document-processing/bedrock-document-processing.js +1 -1
  7. package/lib/document-processing/default-document-processing-config.js +1 -1
  8. package/lib/document-processing/tests/agentic-document-processing.test.js +104 -60
  9. package/lib/document-processing/tests/base-document-processing-nag.test.d.ts +1 -0
  10. package/lib/document-processing/tests/base-document-processing-nag.test.js +161 -0
  11. package/lib/document-processing/tests/base-document-processing.test.d.ts +1 -0
  12. package/lib/document-processing/tests/base-document-processing.test.js +499 -0
  13. package/lib/document-processing/tests/bedrock-document-processing.test.js +212 -36
  14. package/lib/document-processing/tests/queued-s3-adapter-nag.test.d.ts +1 -0
  15. package/lib/document-processing/tests/queued-s3-adapter-nag.test.js +122 -0
  16. package/lib/document-processing/tests/queued-s3-adapter.test.d.ts +1 -0
  17. package/lib/document-processing/tests/queued-s3-adapter.test.js +276 -0
  18. package/lib/framework/agents/base-agent.js +1 -1
  19. package/lib/framework/agents/batch-agent.js +1 -1
  20. package/lib/framework/agents/default-agent-config.js +1 -1
  21. package/lib/framework/bedrock/bedrock.js +1 -1
  22. package/lib/framework/custom-resource/default-runtimes.js +1 -1
  23. package/lib/framework/foundation/access-log.js +1 -1
  24. package/lib/framework/foundation/eventbridge-broker.js +1 -1
  25. package/lib/framework/foundation/network.js +1 -1
  26. package/lib/framework/tests/access-log.test.d.ts +1 -0
  27. package/lib/framework/tests/access-log.test.js +146 -0
  28. package/lib/framework/tests/batch-agent.test.d.ts +1 -0
  29. package/lib/framework/tests/batch-agent.test.js +164 -0
  30. package/lib/framework/tests/bedrock.test.d.ts +1 -0
  31. package/lib/framework/tests/bedrock.test.js +68 -0
  32. package/lib/framework/tests/eventbridge-broker.test.d.ts +1 -0
  33. package/lib/framework/tests/eventbridge-broker.test.js +73 -0
  34. package/lib/framework/tests/framework-nag.test.d.ts +1 -0
  35. package/lib/framework/tests/framework-nag.test.js +155 -0
  36. package/lib/framework/tests/network.test.d.ts +1 -0
  37. package/lib/framework/tests/network.test.js +120 -0
  38. package/lib/tsconfig.tsbuildinfo +1 -1
  39. package/lib/utilities/data-loader.js +1 -1
  40. package/lib/utilities/lambda-iam-utils.js +1 -1
  41. package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +1 -1
  42. package/lib/utilities/observability/default-observability-config.js +1 -1
  43. package/lib/utilities/observability/lambda-observability-property-injector.js +1 -1
  44. package/lib/utilities/observability/log-group-data-protection-utils.js +1 -1
  45. package/lib/utilities/observability/powertools-config.js +1 -1
  46. package/lib/utilities/observability/state-machine-observability-property-injector.js +1 -1
  47. package/lib/webapp/frontend-construct.js +1 -1
  48. 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 {};