@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,155 @@
|
|
|
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 cdk_nag_1 = require("cdk-nag");
|
|
9
|
+
const batch_agent_1 = require("../agents/batch-agent");
|
|
10
|
+
const access_log_1 = require("../foundation/access-log");
|
|
11
|
+
const eventbridge_broker_1 = require("../foundation/eventbridge-broker");
|
|
12
|
+
const network_1 = require("../foundation/network");
|
|
13
|
+
const testModel = aws_bedrock_1.FoundationModelIdentifier.ANTHROPIC_CLAUDE_3_SONNET_20240229_V1_0;
|
|
14
|
+
describe('Framework CDK Nag Tests', () => {
|
|
15
|
+
describe('AccessLog', () => {
|
|
16
|
+
test('passes CDK Nag checks', () => {
|
|
17
|
+
const stack = new aws_cdk_lib_1.Stack(undefined, 'TestStack', {
|
|
18
|
+
env: { account: '123456789012', region: 'us-east-1' },
|
|
19
|
+
});
|
|
20
|
+
new access_log_1.AccessLog(stack, 'AccessLog');
|
|
21
|
+
aws_cdk_lib_1.Aspects.of(stack).add(new cdk_nag_1.AwsSolutionsChecks({ verbose: true }));
|
|
22
|
+
cdk_nag_1.NagSuppressions.addStackSuppressions(stack, [
|
|
23
|
+
{
|
|
24
|
+
id: 'AwsSolutions-S1',
|
|
25
|
+
reason: 'Access log bucket does not need its own access logging to avoid circular dependency',
|
|
26
|
+
},
|
|
27
|
+
]);
|
|
28
|
+
const errors = assertions_1.Annotations.fromStack(stack).findError('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
|
|
29
|
+
expect(errors).toHaveLength(0);
|
|
30
|
+
});
|
|
31
|
+
});
|
|
32
|
+
describe('Network', () => {
|
|
33
|
+
test('passes CDK Nag checks for public VPC', () => {
|
|
34
|
+
const stack = new aws_cdk_lib_1.Stack();
|
|
35
|
+
new network_1.Network(stack, 'Network');
|
|
36
|
+
aws_cdk_lib_1.Aspects.of(stack).add(new cdk_nag_1.AwsSolutionsChecks({ verbose: true }));
|
|
37
|
+
cdk_nag_1.NagSuppressions.addStackSuppressions(stack, [
|
|
38
|
+
{
|
|
39
|
+
id: 'AwsSolutions-VPC7',
|
|
40
|
+
reason: 'VPC Flow Logs are optional and should be enabled by users based on requirements',
|
|
41
|
+
},
|
|
42
|
+
]);
|
|
43
|
+
const errors = assertions_1.Annotations.fromStack(stack).findError('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
|
|
44
|
+
expect(errors).toHaveLength(0);
|
|
45
|
+
});
|
|
46
|
+
test('passes CDK Nag checks for private VPC', () => {
|
|
47
|
+
const stack = new aws_cdk_lib_1.Stack();
|
|
48
|
+
new network_1.Network(stack, 'Network', { private: true });
|
|
49
|
+
aws_cdk_lib_1.Aspects.of(stack).add(new cdk_nag_1.AwsSolutionsChecks({ verbose: true }));
|
|
50
|
+
cdk_nag_1.NagSuppressions.addStackSuppressions(stack, [
|
|
51
|
+
{
|
|
52
|
+
id: 'AwsSolutions-VPC7',
|
|
53
|
+
reason: 'VPC Flow Logs are optional and should be enabled by users based on requirements',
|
|
54
|
+
},
|
|
55
|
+
]);
|
|
56
|
+
const errors = assertions_1.Annotations.fromStack(stack).findError('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
|
|
57
|
+
expect(errors).toHaveLength(0);
|
|
58
|
+
});
|
|
59
|
+
});
|
|
60
|
+
describe('EventbridgeBroker', () => {
|
|
61
|
+
test('passes CDK Nag checks', () => {
|
|
62
|
+
const stack = new aws_cdk_lib_1.Stack();
|
|
63
|
+
new eventbridge_broker_1.EventbridgeBroker(stack, 'Broker', {
|
|
64
|
+
eventSource: 'test.source',
|
|
65
|
+
});
|
|
66
|
+
aws_cdk_lib_1.Aspects.of(stack).add(new cdk_nag_1.AwsSolutionsChecks({ verbose: true }));
|
|
67
|
+
const errors = assertions_1.Annotations.fromStack(stack).findError('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
|
|
68
|
+
expect(errors).toHaveLength(0);
|
|
69
|
+
});
|
|
70
|
+
});
|
|
71
|
+
describe('BatchAgent', () => {
|
|
72
|
+
test('passes CDK Nag checks', () => {
|
|
73
|
+
const stack = new aws_cdk_lib_1.Stack(undefined, 'TestStack', {
|
|
74
|
+
env: { account: '123456789012', region: 'us-east-1' },
|
|
75
|
+
});
|
|
76
|
+
const systemPrompt = new aws_s3_assets_1.Asset(stack, 'SystemPrompt', {
|
|
77
|
+
path: path.join(__dirname, '../agents/resources/default-strands-agent/batch.py'),
|
|
78
|
+
});
|
|
79
|
+
new batch_agent_1.BatchAgent(stack, 'Agent', {
|
|
80
|
+
agentName: 'TestAgent',
|
|
81
|
+
prompt: 'Test prompt',
|
|
82
|
+
agentDefinition: {
|
|
83
|
+
bedrockModel: { fmModelId: testModel },
|
|
84
|
+
systemPrompt,
|
|
85
|
+
},
|
|
86
|
+
});
|
|
87
|
+
aws_cdk_lib_1.Aspects.of(stack).add(new cdk_nag_1.AwsSolutionsChecks({ verbose: true }));
|
|
88
|
+
cdk_nag_1.NagSuppressions.addStackSuppressions(stack, [
|
|
89
|
+
{
|
|
90
|
+
id: 'AwsSolutions-IAM4',
|
|
91
|
+
reason: 'Managed policies are acceptable for Lambda execution role',
|
|
92
|
+
appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],
|
|
93
|
+
},
|
|
94
|
+
{
|
|
95
|
+
id: 'AwsSolutions-IAM5',
|
|
96
|
+
reason: 'Wildcard permissions required for S3 asset access, Bedrock model invocation, and CloudWatch Logs',
|
|
97
|
+
appliesTo: [
|
|
98
|
+
'Action::s3:GetObject*',
|
|
99
|
+
'Action::s3:GetBucket*',
|
|
100
|
+
'Action::s3:List*',
|
|
101
|
+
'Resource::*',
|
|
102
|
+
'Resource::arn:aws:bedrock:*::foundation-model/anthropic.claude-3-sonnet-20240229-v1:0',
|
|
103
|
+
'Resource::arn:aws:logs:us-east-1:123456789012:log-group:/aws/lambda/TestAgent-teststackagentaa2af0c4:*',
|
|
104
|
+
],
|
|
105
|
+
},
|
|
106
|
+
{
|
|
107
|
+
id: 'AwsSolutions-L1',
|
|
108
|
+
reason: 'Lambda runtime is managed by construct defaults',
|
|
109
|
+
},
|
|
110
|
+
]);
|
|
111
|
+
const errors = assertions_1.Annotations.fromStack(stack).findError('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
|
|
112
|
+
expect(errors).toHaveLength(0);
|
|
113
|
+
});
|
|
114
|
+
test('passes CDK Nag checks with VPC configuration', () => {
|
|
115
|
+
const stack = new aws_cdk_lib_1.Stack(undefined, 'TestStack', {
|
|
116
|
+
env: { account: '123456789012', region: 'us-east-1' },
|
|
117
|
+
});
|
|
118
|
+
const network = new network_1.Network(stack, 'Network');
|
|
119
|
+
const systemPrompt = new aws_s3_assets_1.Asset(stack, 'SystemPrompt', {
|
|
120
|
+
path: path.join(__dirname, '../agents/resources/default-strands-agent/batch.py'),
|
|
121
|
+
});
|
|
122
|
+
new batch_agent_1.BatchAgent(stack, 'Agent', {
|
|
123
|
+
agentName: 'TestAgent',
|
|
124
|
+
prompt: 'Test prompt',
|
|
125
|
+
network,
|
|
126
|
+
agentDefinition: {
|
|
127
|
+
bedrockModel: { fmModelId: testModel },
|
|
128
|
+
systemPrompt,
|
|
129
|
+
},
|
|
130
|
+
});
|
|
131
|
+
aws_cdk_lib_1.Aspects.of(stack).add(new cdk_nag_1.AwsSolutionsChecks({ verbose: true }));
|
|
132
|
+
cdk_nag_1.NagSuppressions.addStackSuppressions(stack, [
|
|
133
|
+
{
|
|
134
|
+
id: 'AwsSolutions-IAM4',
|
|
135
|
+
reason: 'Managed policies are acceptable for Lambda execution role',
|
|
136
|
+
},
|
|
137
|
+
{
|
|
138
|
+
id: 'AwsSolutions-IAM5',
|
|
139
|
+
reason: 'Wildcard permissions required for VPC, S3, and Bedrock access',
|
|
140
|
+
},
|
|
141
|
+
{
|
|
142
|
+
id: 'AwsSolutions-L1',
|
|
143
|
+
reason: 'Lambda runtime is managed by construct defaults',
|
|
144
|
+
},
|
|
145
|
+
{
|
|
146
|
+
id: 'AwsSolutions-VPC7',
|
|
147
|
+
reason: 'VPC Flow Logs are optional',
|
|
148
|
+
},
|
|
149
|
+
]);
|
|
150
|
+
const errors = assertions_1.Annotations.fromStack(stack).findError('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
|
|
151
|
+
expect(errors).toHaveLength(0);
|
|
152
|
+
});
|
|
153
|
+
});
|
|
154
|
+
});
|
|
155
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"framework-nag.test.js","sourceRoot":"","sources":["../../../use-cases/framework/tests/framework-nag.test.ts"],"names":[],"mappings":";;AAAA,6BAA6B;AAC7B,6CAA6C;AAC7C,uDAA4D;AAC5D,yDAAoE;AACpE,6DAAkD;AAClD,qCAA8D;AAC9D,uDAAmD;AACnD,yDAAqD;AACrD,yEAAqE;AACrE,mDAAgD;AAEhD,MAAM,SAAS,GAAG,uCAAyB,CAAC,uCAAuC,CAAC;AAEpF,QAAQ,CAAC,yBAAyB,EAAE,GAAG,EAAE;IACvC,QAAQ,CAAC,WAAW,EAAE,GAAG,EAAE;QACzB,IAAI,CAAC,uBAAuB,EAAE,GAAG,EAAE;YACjC,MAAM,KAAK,GAAG,IAAI,mBAAK,CAAC,SAAS,EAAE,WAAW,EAAE;gBAC9C,GAAG,EAAE,EAAE,OAAO,EAAE,cAAc,EAAE,MAAM,EAAE,WAAW,EAAE;aACtD,CAAC,CAAC;YAEH,IAAI,sBAAS,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC;YAElC,qBAAO,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,IAAI,4BAAkB,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;YAEjE,yBAAe,CAAC,oBAAoB,CAAC,KAAK,EAAE;gBAC1C;oBACE,EAAE,EAAE,iBAAiB;oBACrB,MAAM,EAAE,qFAAqF;iBAC9F;aACF,CAAC,CAAC;YAEH,MAAM,MAAM,GAAG,wBAAW,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,kBAAK,CAAC,gBAAgB,CAAC,iBAAiB,CAAC,CAAC,CAAC;YACtG,MAAM,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;QACjC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,SAAS,EAAE,GAAG,EAAE;QACvB,IAAI,CAAC,sCAAsC,EAAE,GAAG,EAAE;YAChD,MAAM,KAAK,GAAG,IAAI,mBAAK,EAAE,CAAC;YAC1B,IAAI,iBAAO,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;YAE9B,qBAAO,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,IAAI,4BAAkB,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;YAEjE,yBAAe,CAAC,oBAAoB,CAAC,KAAK,EAAE;gBAC1C;oBACE,EAAE,EAAE,mBAAmB;oBACvB,MAAM,EAAE,iFAAiF;iBAC1F;aACF,CAAC,CAAC;YAEH,MAAM,MAAM,GAAG,wBAAW,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,kBAAK,CAAC,gBAAgB,CAAC,iBAAiB,CAAC,CAAC,CAAC;YACtG,MAAM,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;QACjC,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,uCAAuC,EAAE,GAAG,EAAE;YACjD,MAAM,KAAK,GAAG,IAAI,mBAAK,EAAE,CAAC;YAC1B,IAAI,iBAAO,CAAC,KAAK,EAAE,SAAS,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC;YAEjD,qBAAO,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,IAAI,4BAAkB,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;YAEjE,yBAAe,CAAC,oBAAoB,CAAC,KAAK,EAAE;gBAC1C;oBACE,EAAE,EAAE,mBAAmB;oBACvB,MAAM,EAAE,iFAAiF;iBAC1F;aACF,CAAC,CAAC;YAEH,MAAM,MAAM,GAAG,wBAAW,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,kBAAK,CAAC,gBAAgB,CAAC,iBAAiB,CAAC,CAAC,CAAC;YACtG,MAAM,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;QACjC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,mBAAmB,EAAE,GAAG,EAAE;QACjC,IAAI,CAAC,uBAAuB,EAAE,GAAG,EAAE;YACjC,MAAM,KAAK,GAAG,IAAI,mBAAK,EAAE,CAAC;YAC1B,IAAI,sCAAiB,CAAC,KAAK,EAAE,QAAQ,EAAE;gBACrC,WAAW,EAAE,aAAa;aAC3B,CAAC,CAAC;YAEH,qBAAO,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,IAAI,4BAAkB,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;YACjE,MAAM,MAAM,GAAG,wBAAW,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,kBAAK,CAAC,gBAAgB,CAAC,iBAAiB,CAAC,CAAC,CAAC;YACtG,MAAM,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;QACjC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,YAAY,EAAE,GAAG,EAAE;QAC1B,IAAI,CAAC,uBAAuB,EAAE,GAAG,EAAE;YACjC,MAAM,KAAK,GAAG,IAAI,mBAAK,CAAC,SAAS,EAAE,WAAW,EAAE;gBAC9C,GAAG,EAAE,EAAE,OAAO,EAAE,cAAc,EAAE,MAAM,EAAE,WAAW,EAAE;aACtD,CAAC,CAAC;YAEH,MAAM,YAAY,GAAG,IAAI,qBAAK,CAAC,KAAK,EAAE,cAAc,EAAE;gBACpD,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,oDAAoD,CAAC;aACjF,CAAC,CAAC;YAEH,IAAI,wBAAU,CAAC,KAAK,EAAE,OAAO,EAAE;gBAC7B,SAAS,EAAE,WAAW;gBACtB,MAAM,EAAE,aAAa;gBACrB,eAAe,EAAE;oBACf,YAAY,EAAE,EAAE,SAAS,EAAE,SAAS,EAAE;oBACtC,YAAY;iBACb;aACF,CAAC,CAAC;YAEH,qBAAO,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,IAAI,4BAAkB,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;YAEjE,yBAAe,CAAC,oBAAoB,CAAC,KAAK,EAAE;gBAC1C;oBACE,EAAE,EAAE,mBAAmB;oBACvB,MAAM,EAAE,2DAA2D;oBACnE,SAAS,EAAE,CAAC,uFAAuF,CAAC;iBACrG;gBACD;oBACE,EAAE,EAAE,mBAAmB;oBACvB,MAAM,EAAE,kGAAkG;oBAC1G,SAAS,EAAE;wBACT,uBAAuB;wBACvB,uBAAuB;wBACvB,kBAAkB;wBAClB,aAAa;wBACb,uFAAuF;wBACvF,wGAAwG;qBACzG;iBACF;gBACD;oBACE,EAAE,EAAE,iBAAiB;oBACrB,MAAM,EAAE,iDAAiD;iBAC1D;aACF,CAAC,CAAC;YAEH,MAAM,MAAM,GAAG,wBAAW,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,kBAAK,CAAC,gBAAgB,CAAC,iBAAiB,CAAC,CAAC,CAAC;YACtG,MAAM,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;QACjC,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,8CAA8C,EAAE,GAAG,EAAE;YACxD,MAAM,KAAK,GAAG,IAAI,mBAAK,CAAC,SAAS,EAAE,WAAW,EAAE;gBAC9C,GAAG,EAAE,EAAE,OAAO,EAAE,cAAc,EAAE,MAAM,EAAE,WAAW,EAAE;aACtD,CAAC,CAAC;YAEH,MAAM,OAAO,GAAG,IAAI,iBAAO,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;YAC9C,MAAM,YAAY,GAAG,IAAI,qBAAK,CAAC,KAAK,EAAE,cAAc,EAAE;gBACpD,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,oDAAoD,CAAC;aACjF,CAAC,CAAC;YAEH,IAAI,wBAAU,CAAC,KAAK,EAAE,OAAO,EAAE;gBAC7B,SAAS,EAAE,WAAW;gBACtB,MAAM,EAAE,aAAa;gBACrB,OAAO;gBACP,eAAe,EAAE;oBACf,YAAY,EAAE,EAAE,SAAS,EAAE,SAAS,EAAE;oBACtC,YAAY;iBACb;aACF,CAAC,CAAC;YAEH,qBAAO,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,IAAI,4BAAkB,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;YAEjE,yBAAe,CAAC,oBAAoB,CAAC,KAAK,EAAE;gBAC1C;oBACE,EAAE,EAAE,mBAAmB;oBACvB,MAAM,EAAE,2DAA2D;iBACpE;gBACD;oBACE,EAAE,EAAE,mBAAmB;oBACvB,MAAM,EAAE,+DAA+D;iBACxE;gBACD;oBACE,EAAE,EAAE,iBAAiB;oBACrB,MAAM,EAAE,iDAAiD;iBAC1D;gBACD;oBACE,EAAE,EAAE,mBAAmB;oBACvB,MAAM,EAAE,4BAA4B;iBACrC;aACF,CAAC,CAAC;YAEH,MAAM,MAAM,GAAG,wBAAW,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,kBAAK,CAAC,gBAAgB,CAAC,iBAAiB,CAAC,CAAC,CAAC;YACtG,MAAM,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;QACjC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["import * as path from 'path';\nimport { Aspects, Stack } from 'aws-cdk-lib';\nimport { Annotations, Match } from 'aws-cdk-lib/assertions';\nimport { FoundationModelIdentifier } from 'aws-cdk-lib/aws-bedrock';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { AwsSolutionsChecks, NagSuppressions } from 'cdk-nag';\nimport { BatchAgent } from '../agents/batch-agent';\nimport { AccessLog } from '../foundation/access-log';\nimport { EventbridgeBroker } from '../foundation/eventbridge-broker';\nimport { Network } from '../foundation/network';\n\nconst testModel = FoundationModelIdentifier.ANTHROPIC_CLAUDE_3_SONNET_20240229_V1_0;\n\ndescribe('Framework CDK Nag Tests', () => {\n  describe('AccessLog', () => {\n    test('passes CDK Nag checks', () => {\n      const stack = new Stack(undefined, 'TestStack', {\n        env: { account: '123456789012', region: 'us-east-1' },\n      });\n\n      new AccessLog(stack, 'AccessLog');\n\n      Aspects.of(stack).add(new AwsSolutionsChecks({ verbose: true }));\n\n      NagSuppressions.addStackSuppressions(stack, [\n        {\n          id: 'AwsSolutions-S1',\n          reason: 'Access log bucket does not need its own access logging to avoid circular dependency',\n        },\n      ]);\n\n      const errors = Annotations.fromStack(stack).findError('*', Match.stringLikeRegexp('AwsSolutions-.*'));\n      expect(errors).toHaveLength(0);\n    });\n  });\n\n  describe('Network', () => {\n    test('passes CDK Nag checks for public VPC', () => {\n      const stack = new Stack();\n      new Network(stack, 'Network');\n\n      Aspects.of(stack).add(new AwsSolutionsChecks({ verbose: true }));\n\n      NagSuppressions.addStackSuppressions(stack, [\n        {\n          id: 'AwsSolutions-VPC7',\n          reason: 'VPC Flow Logs are optional and should be enabled by users based on requirements',\n        },\n      ]);\n\n      const errors = Annotations.fromStack(stack).findError('*', Match.stringLikeRegexp('AwsSolutions-.*'));\n      expect(errors).toHaveLength(0);\n    });\n\n    test('passes CDK Nag checks for private VPC', () => {\n      const stack = new Stack();\n      new Network(stack, 'Network', { private: true });\n\n      Aspects.of(stack).add(new AwsSolutionsChecks({ verbose: true }));\n\n      NagSuppressions.addStackSuppressions(stack, [\n        {\n          id: 'AwsSolutions-VPC7',\n          reason: 'VPC Flow Logs are optional and should be enabled by users based on requirements',\n        },\n      ]);\n\n      const errors = Annotations.fromStack(stack).findError('*', Match.stringLikeRegexp('AwsSolutions-.*'));\n      expect(errors).toHaveLength(0);\n    });\n  });\n\n  describe('EventbridgeBroker', () => {\n    test('passes CDK Nag checks', () => {\n      const stack = new Stack();\n      new EventbridgeBroker(stack, 'Broker', {\n        eventSource: 'test.source',\n      });\n\n      Aspects.of(stack).add(new AwsSolutionsChecks({ verbose: true }));\n      const errors = Annotations.fromStack(stack).findError('*', Match.stringLikeRegexp('AwsSolutions-.*'));\n      expect(errors).toHaveLength(0);\n    });\n  });\n\n  describe('BatchAgent', () => {\n    test('passes CDK Nag checks', () => {\n      const stack = new Stack(undefined, 'TestStack', {\n        env: { account: '123456789012', region: 'us-east-1' },\n      });\n\n      const systemPrompt = new Asset(stack, 'SystemPrompt', {\n        path: path.join(__dirname, '../agents/resources/default-strands-agent/batch.py'),\n      });\n\n      new BatchAgent(stack, 'Agent', {\n        agentName: 'TestAgent',\n        prompt: 'Test prompt',\n        agentDefinition: {\n          bedrockModel: { fmModelId: testModel },\n          systemPrompt,\n        },\n      });\n\n      Aspects.of(stack).add(new AwsSolutionsChecks({ verbose: true }));\n\n      NagSuppressions.addStackSuppressions(stack, [\n        {\n          id: 'AwsSolutions-IAM4',\n          reason: 'Managed policies are acceptable for Lambda execution role',\n          appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],\n        },\n        {\n          id: 'AwsSolutions-IAM5',\n          reason: 'Wildcard permissions required for S3 asset access, Bedrock model invocation, and CloudWatch Logs',\n          appliesTo: [\n            'Action::s3:GetObject*',\n            'Action::s3:GetBucket*',\n            'Action::s3:List*',\n            'Resource::*',\n            'Resource::arn:aws:bedrock:*::foundation-model/anthropic.claude-3-sonnet-20240229-v1:0',\n            'Resource::arn:aws:logs:us-east-1:123456789012:log-group:/aws/lambda/TestAgent-teststackagentaa2af0c4:*',\n          ],\n        },\n        {\n          id: 'AwsSolutions-L1',\n          reason: 'Lambda runtime is managed by construct defaults',\n        },\n      ]);\n\n      const errors = Annotations.fromStack(stack).findError('*', Match.stringLikeRegexp('AwsSolutions-.*'));\n      expect(errors).toHaveLength(0);\n    });\n\n    test('passes CDK Nag checks with VPC configuration', () => {\n      const stack = new Stack(undefined, 'TestStack', {\n        env: { account: '123456789012', region: 'us-east-1' },\n      });\n\n      const network = new Network(stack, 'Network');\n      const systemPrompt = new Asset(stack, 'SystemPrompt', {\n        path: path.join(__dirname, '../agents/resources/default-strands-agent/batch.py'),\n      });\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      Aspects.of(stack).add(new AwsSolutionsChecks({ verbose: true }));\n\n      NagSuppressions.addStackSuppressions(stack, [\n        {\n          id: 'AwsSolutions-IAM4',\n          reason: 'Managed policies are acceptable for Lambda execution role',\n        },\n        {\n          id: 'AwsSolutions-IAM5',\n          reason: 'Wildcard permissions required for VPC, S3, and Bedrock access',\n        },\n        {\n          id: 'AwsSolutions-L1',\n          reason: 'Lambda runtime is managed by construct defaults',\n        },\n        {\n          id: 'AwsSolutions-VPC7',\n          reason: 'VPC Flow Logs are optional',\n        },\n      ]);\n\n      const errors = Annotations.fromStack(stack).findError('*', Match.stringLikeRegexp('AwsSolutions-.*'));\n      expect(errors).toHaveLength(0);\n    });\n  });\n});\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,120 @@
|
|
|
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_ec2_1 = require("aws-cdk-lib/aws-ec2");
|
|
6
|
+
const network_1 = require("../foundation/network");
|
|
7
|
+
describe('Network', () => {
|
|
8
|
+
let stack;
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
stack = new aws_cdk_lib_1.Stack();
|
|
11
|
+
});
|
|
12
|
+
test('creates VPC with default configuration', () => {
|
|
13
|
+
new network_1.Network(stack, 'Network');
|
|
14
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
15
|
+
template.resourceCountIs('AWS::EC2::VPC', 1);
|
|
16
|
+
template.hasResourceProperties('AWS::EC2::VPC', {
|
|
17
|
+
CidrBlock: '10.0.0.0/16',
|
|
18
|
+
});
|
|
19
|
+
});
|
|
20
|
+
test('creates public, private, and isolated subnets by default', () => {
|
|
21
|
+
new network_1.Network(stack, 'Network');
|
|
22
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
23
|
+
const subnets = template.findResources('AWS::EC2::Subnet');
|
|
24
|
+
expect(Object.keys(subnets).length).toBeGreaterThan(0);
|
|
25
|
+
const subnetTags = Object.values(subnets).map((s) => s.Properties.Tags?.find((t) => t.Key === 'aws-cdk:subnet-name')?.Value);
|
|
26
|
+
expect(subnetTags).toContain('Public');
|
|
27
|
+
expect(subnetTags).toContain('Private');
|
|
28
|
+
expect(subnetTags).toContain('Isolated');
|
|
29
|
+
});
|
|
30
|
+
test('creates NAT gateway by default', () => {
|
|
31
|
+
new network_1.Network(stack, 'Network');
|
|
32
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
33
|
+
template.resourceCountIs('AWS::EC2::NatGateway', 1);
|
|
34
|
+
});
|
|
35
|
+
test('creates private VPC without NAT gateway', () => {
|
|
36
|
+
new network_1.Network(stack, 'Network', { private: true });
|
|
37
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
38
|
+
template.resourceCountIs('AWS::EC2::NatGateway', 0);
|
|
39
|
+
template.resourceCountIs('AWS::EC2::InternetGateway', 0);
|
|
40
|
+
});
|
|
41
|
+
test('creates only isolated subnets for private VPC', () => {
|
|
42
|
+
new network_1.Network(stack, 'Network', { private: true });
|
|
43
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
44
|
+
const subnets = template.findResources('AWS::EC2::Subnet');
|
|
45
|
+
const subnetTags = Object.values(subnets).map((s) => s.Properties.Tags?.find((t) => t.Key === 'aws-cdk:subnet-name')?.Value);
|
|
46
|
+
expect(subnetTags.every((tag) => tag === 'Isolated')).toBe(true);
|
|
47
|
+
});
|
|
48
|
+
test('uses custom CIDR block', () => {
|
|
49
|
+
new network_1.Network(stack, 'Network', {
|
|
50
|
+
ipAddresses: aws_ec2_1.IpAddresses.cidr('172.16.0.0/16'),
|
|
51
|
+
});
|
|
52
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
53
|
+
template.hasResourceProperties('AWS::EC2::VPC', {
|
|
54
|
+
CidrBlock: '172.16.0.0/16',
|
|
55
|
+
});
|
|
56
|
+
});
|
|
57
|
+
test('configures custom NAT gateways count', () => {
|
|
58
|
+
new network_1.Network(stack, 'Network', { natGateways: 2 });
|
|
59
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
60
|
+
template.resourceCountIs('AWS::EC2::NatGateway', 2);
|
|
61
|
+
});
|
|
62
|
+
test('sets custom VPC name', () => {
|
|
63
|
+
new network_1.Network(stack, 'Network', { vpcName: 'MyCustomVPC' });
|
|
64
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
65
|
+
template.hasResourceProperties('AWS::EC2::VPC', {
|
|
66
|
+
Tags: assertions_1.Match.arrayWith([
|
|
67
|
+
{ Key: 'Name', Value: 'MyCustomVPC' },
|
|
68
|
+
]),
|
|
69
|
+
});
|
|
70
|
+
});
|
|
71
|
+
test('creates interface endpoint with security group', () => {
|
|
72
|
+
const network = new network_1.Network(stack, 'Network');
|
|
73
|
+
network.createServiceEndpoint('Lambda', aws_ec2_1.InterfaceVpcEndpointAwsService.LAMBDA);
|
|
74
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
75
|
+
template.resourceCountIs('AWS::EC2::VPCEndpoint', 1);
|
|
76
|
+
template.hasResourceProperties('AWS::EC2::VPCEndpoint', {
|
|
77
|
+
VpcEndpointType: 'Interface',
|
|
78
|
+
});
|
|
79
|
+
template.resourceCountIs('AWS::EC2::SecurityGroup', 1);
|
|
80
|
+
});
|
|
81
|
+
test('creates S3 gateway endpoint when S3 service requested', () => {
|
|
82
|
+
const network = new network_1.Network(stack, 'Network');
|
|
83
|
+
network.createServiceEndpoint('S3', aws_ec2_1.InterfaceVpcEndpointAwsService.S3);
|
|
84
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
85
|
+
template.hasResourceProperties('AWS::EC2::VPCEndpoint', {
|
|
86
|
+
ServiceName: assertions_1.Match.objectLike({
|
|
87
|
+
'Fn::Join': assertions_1.Match.arrayWith([
|
|
88
|
+
assertions_1.Match.arrayWith([assertions_1.Match.stringLikeRegexp('s3')]),
|
|
89
|
+
]),
|
|
90
|
+
}),
|
|
91
|
+
VpcEndpointType: 'Gateway',
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
test('security group allows HTTPS from custom peer', () => {
|
|
95
|
+
const network = new network_1.Network(stack, 'Network');
|
|
96
|
+
network.createServiceEndpoint('Lambda', aws_ec2_1.InterfaceVpcEndpointAwsService.LAMBDA, aws_ec2_1.Peer.ipv4('10.0.0.0/8'));
|
|
97
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
98
|
+
template.hasResourceProperties('AWS::EC2::SecurityGroup', {
|
|
99
|
+
SecurityGroupIngress: assertions_1.Match.arrayWith([
|
|
100
|
+
assertions_1.Match.objectLike({
|
|
101
|
+
IpProtocol: 'tcp',
|
|
102
|
+
FromPort: 443,
|
|
103
|
+
ToPort: 443,
|
|
104
|
+
CidrIp: '10.0.0.0/8',
|
|
105
|
+
}),
|
|
106
|
+
]),
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
test('applicationSubnetSelection returns correct subnet type for public VPC', () => {
|
|
110
|
+
const network = new network_1.Network(stack, 'Network');
|
|
111
|
+
const selection = network.applicationSubnetSelection();
|
|
112
|
+
expect(selection.subnetType).toBe(aws_ec2_1.SubnetType.PRIVATE_WITH_EGRESS);
|
|
113
|
+
});
|
|
114
|
+
test('applicationSubnetSelection returns correct subnet type for private VPC', () => {
|
|
115
|
+
const network = new network_1.Network(stack, 'Network', { private: true });
|
|
116
|
+
const selection = network.applicationSubnetSelection();
|
|
117
|
+
expect(selection.subnetType).toBe(aws_ec2_1.SubnetType.PRIVATE_ISOLATED);
|
|
118
|
+
});
|
|
119
|
+
});
|
|
120
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"network.test.js","sourceRoot":"","sources":["../../../use-cases/framework/tests/network.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack } from 'aws-cdk-lib';\nimport { Template, Match } from 'aws-cdk-lib/assertions';\nimport { InterfaceVpcEndpointAwsService, IpAddresses, Peer, SubnetType } from 'aws-cdk-lib/aws-ec2';\nimport { Network } from '../foundation/network';\n\ndescribe('Network', () => {\n  let stack: Stack;\n\n  beforeEach(() => {\n    stack = new Stack();\n  });\n\n  test('creates VPC with default configuration', () => {\n    new Network(stack, 'Network');\n\n    const template = Template.fromStack(stack);\n    template.resourceCountIs('AWS::EC2::VPC', 1);\n    template.hasResourceProperties('AWS::EC2::VPC', {\n      CidrBlock: '10.0.0.0/16',\n    });\n  });\n\n  test('creates public, private, and isolated subnets by default', () => {\n    new Network(stack, 'Network');\n\n    const template = Template.fromStack(stack);\n    const subnets = template.findResources('AWS::EC2::Subnet');\n    expect(Object.keys(subnets).length).toBeGreaterThan(0);\n\n    const subnetTags = Object.values(subnets).map((s: any) =>\n      s.Properties.Tags?.find((t: any) => t.Key === 'aws-cdk:subnet-name')?.Value,\n    );\n\n    expect(subnetTags).toContain('Public');\n    expect(subnetTags).toContain('Private');\n    expect(subnetTags).toContain('Isolated');\n  });\n\n  test('creates NAT gateway by default', () => {\n    new Network(stack, 'Network');\n\n    const template = Template.fromStack(stack);\n    template.resourceCountIs('AWS::EC2::NatGateway', 1);\n  });\n\n  test('creates private VPC without NAT gateway', () => {\n    new Network(stack, 'Network', { private: true });\n\n    const template = Template.fromStack(stack);\n    template.resourceCountIs('AWS::EC2::NatGateway', 0);\n    template.resourceCountIs('AWS::EC2::InternetGateway', 0);\n  });\n\n  test('creates only isolated subnets for private VPC', () => {\n    new Network(stack, 'Network', { private: true });\n\n    const template = Template.fromStack(stack);\n    const subnets = template.findResources('AWS::EC2::Subnet');\n    const subnetTags = Object.values(subnets).map((s: any) =>\n      s.Properties.Tags?.find((t: any) => t.Key === 'aws-cdk:subnet-name')?.Value,\n    );\n\n    expect(subnetTags.every((tag: string) => tag === 'Isolated')).toBe(true);\n  });\n\n  test('uses custom CIDR block', () => {\n    new Network(stack, 'Network', {\n      ipAddresses: IpAddresses.cidr('172.16.0.0/16'),\n    });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::EC2::VPC', {\n      CidrBlock: '172.16.0.0/16',\n    });\n  });\n\n  test('configures custom NAT gateways count', () => {\n    new Network(stack, 'Network', { natGateways: 2 });\n\n    const template = Template.fromStack(stack);\n    template.resourceCountIs('AWS::EC2::NatGateway', 2);\n  });\n\n  test('sets custom VPC name', () => {\n    new Network(stack, 'Network', { vpcName: 'MyCustomVPC' });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::EC2::VPC', {\n      Tags: Match.arrayWith([\n        { Key: 'Name', Value: 'MyCustomVPC' },\n      ]),\n    });\n  });\n\n  test('creates interface endpoint with security group', () => {\n    const network = new Network(stack, 'Network');\n    network.createServiceEndpoint('Lambda', InterfaceVpcEndpointAwsService.LAMBDA);\n\n    const template = Template.fromStack(stack);\n    template.resourceCountIs('AWS::EC2::VPCEndpoint', 1);\n    template.hasResourceProperties('AWS::EC2::VPCEndpoint', {\n      VpcEndpointType: 'Interface',\n    });\n    template.resourceCountIs('AWS::EC2::SecurityGroup', 1);\n  });\n\n  test('creates S3 gateway endpoint when S3 service requested', () => {\n    const network = new Network(stack, 'Network');\n    network.createServiceEndpoint('S3', InterfaceVpcEndpointAwsService.S3);\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::EC2::VPCEndpoint', {\n      ServiceName: Match.objectLike({\n        'Fn::Join': Match.arrayWith([\n          Match.arrayWith([Match.stringLikeRegexp('s3')]),\n        ]),\n      }),\n      VpcEndpointType: 'Gateway',\n    });\n  });\n\n  test('security group allows HTTPS from custom peer', () => {\n    const network = new Network(stack, 'Network');\n    network.createServiceEndpoint('Lambda', InterfaceVpcEndpointAwsService.LAMBDA, Peer.ipv4('10.0.0.0/8'));\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties('AWS::EC2::SecurityGroup', {\n      SecurityGroupIngress: Match.arrayWith([\n        Match.objectLike({\n          IpProtocol: 'tcp',\n          FromPort: 443,\n          ToPort: 443,\n          CidrIp: '10.0.0.0/8',\n        }),\n      ]),\n    });\n  });\n\n  test('applicationSubnetSelection returns correct subnet type for public VPC', () => {\n    const network = new Network(stack, 'Network');\n    const selection = network.applicationSubnetSelection();\n    expect(selection.subnetType).toBe(SubnetType.PRIVATE_WITH_EGRESS);\n  });\n\n  test('applicationSubnetSelection returns correct subnet type for private VPC', () => {\n    const network = new Network(stack, 'Network', { private: true });\n    const selection = network.applicationSubnetSelection();\n    expect(selection.subnetType).toBe(SubnetType.PRIVATE_ISOLATED);\n  });\n});\n"]}
|