@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
@@ -12,7 +12,6 @@ describe('AgenticDocumentProcessing', () => {
12
12
  let crossRegionTemplate;
13
13
  let defaultModelTemplate;
14
14
  beforeAll(() => {
15
- // Create all stacks and constructs first
16
15
  basicStack = new aws_cdk_lib_1.Stack();
17
16
  const systemPrompt = new aws_s3_assets_1.Asset(basicStack, 'SystemPrompt', {
18
17
  path: __dirname + '/../resources/default-strands-agent',
@@ -55,82 +54,127 @@ describe('AgenticDocumentProcessing', () => {
55
54
  prompt: 'Test prompt',
56
55
  },
57
56
  });
58
- // Generate templates once after all constructs are created
59
57
  basicTemplate = assertions_1.Template.fromStack(basicStack);
60
58
  crossRegionTemplate = assertions_1.Template.fromStack(crossRegionStack);
61
59
  defaultModelTemplate = assertions_1.Template.fromStack(defaultModelStack);
62
60
  });
63
- test('creates basic infrastructure', () => {
64
- basicTemplate.hasResourceProperties('AWS::S3::Bucket', {});
65
- basicTemplate.hasResourceProperties('AWS::Lambda::Function', {
66
- Runtime: 'python3.13',
61
+ describe('Basic infrastructure', () => {
62
+ test('creates basic infrastructure', () => {
63
+ basicTemplate.hasResourceProperties('AWS::S3::Bucket', {});
64
+ basicTemplate.hasResourceProperties('AWS::Lambda::Function', {
65
+ Runtime: 'python3.13',
66
+ });
67
+ basicTemplate.resourceCountIs('AWS::Lambda::Function', 5);
68
+ });
69
+ test('inherits bedrock document processing functionality', () => {
70
+ basicTemplate.hasResourceProperties('AWS::S3::Bucket', {});
71
+ basicTemplate.hasResourceProperties('AWS::DynamoDB::Table', {});
72
+ basicTemplate.hasResourceProperties('AWS::SQS::Queue', {});
73
+ basicTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {});
67
74
  });
68
- basicTemplate.resourceCountIs('AWS::Lambda::Function', 5);
69
75
  });
70
- test('configures all agent parameters', () => {
71
- basicTemplate.hasResourceProperties('AWS::Lambda::Function', {
72
- Environment: {
73
- Variables: {
74
- PROMPT: 'Custom processing prompt',
76
+ describe('Agent configuration', () => {
77
+ test('configures all agent parameters', () => {
78
+ basicTemplate.hasResourceProperties('AWS::Lambda::Function', {
79
+ Environment: {
80
+ Variables: {
81
+ PROMPT: 'Custom processing prompt',
82
+ },
75
83
  },
76
- },
84
+ });
85
+ });
86
+ test('configures timeout and memory for processing function', () => {
87
+ basicTemplate.hasResourceProperties('AWS::Lambda::Function', {
88
+ Timeout: 600,
89
+ MemorySize: 1024,
90
+ });
91
+ });
92
+ test('uses cross-region inference when enabled', () => {
93
+ crossRegionTemplate.hasResourceProperties('AWS::Lambda::Function', {
94
+ Environment: {
95
+ Variables: {
96
+ MODEL_ID: 'us.anthropic.claude-sonnet-4-20250514-v1:0',
97
+ },
98
+ },
99
+ });
100
+ });
101
+ test('uses default model when cross-region inference is disabled', () => {
102
+ defaultModelTemplate.hasResourceProperties('AWS::Lambda::Function', {
103
+ Environment: {
104
+ Variables: {
105
+ MODEL_ID: 'anthropic.claude-sonnet-4-20250514-v1:0',
106
+ },
107
+ },
108
+ });
77
109
  });
78
110
  });
79
- test('inherits bedrock document processing functionality', () => {
80
- basicTemplate.hasResourceProperties('AWS::S3::Bucket', {});
81
- basicTemplate.hasResourceProperties('AWS::DynamoDB::Table', {});
82
- basicTemplate.hasResourceProperties('AWS::SQS::Queue', {});
83
- basicTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {});
84
- });
85
- test('creates IAM role with correct permissions', () => {
86
- basicTemplate.hasResourceProperties('AWS::IAM::Role', {
87
- AssumeRolePolicyDocument: {
88
- Statement: [{
89
- Action: 'sts:AssumeRole',
90
- Effect: 'Allow',
91
- Principal: {
92
- Service: 'lambda.amazonaws.com',
111
+ describe('IAM permissions', () => {
112
+ test('creates IAM role with correct permissions', () => {
113
+ basicTemplate.hasResourceProperties('AWS::IAM::Role', {
114
+ AssumeRolePolicyDocument: {
115
+ Statement: [{
116
+ Action: 'sts:AssumeRole',
117
+ Effect: 'Allow',
118
+ Principal: {
119
+ Service: 'lambda.amazonaws.com',
120
+ },
121
+ }],
122
+ },
123
+ });
124
+ basicTemplate.hasResourceProperties('AWS::IAM::Role', {
125
+ Policies: [{
126
+ PolicyDocument: {
127
+ Statement: assertions_1.Match.arrayWith([
128
+ assertions_1.Match.objectLike({
129
+ Effect: 'Allow',
130
+ Action: ['bedrock:InvokeModel', 'bedrock:InvokeModelWithResponseStream'],
131
+ }),
132
+ ]),
93
133
  },
134
+ PolicyName: 'BedrockInvokePolicy',
94
135
  }],
95
- },
136
+ });
96
137
  });
97
- basicTemplate.hasResourceProperties('AWS::IAM::Role', {
98
- Policies: [{
99
- PolicyDocument: {
100
- Statement: assertions_1.Match.arrayWith([
101
- assertions_1.Match.objectLike({
102
- Effect: 'Allow',
103
- Action: ['bedrock:InvokeModel', 'bedrock:InvokeModelWithResponseStream'],
104
- }),
105
- ]),
106
- },
107
- PolicyName: 'BedrockInvokePolicy',
108
- }],
138
+ test('grants S3 access to agent function', () => {
139
+ basicTemplate.hasResourceProperties('AWS::IAM::Role', {
140
+ Policies: assertions_1.Match.arrayWith([
141
+ assertions_1.Match.objectLike({
142
+ PolicyDocument: {
143
+ Statement: assertions_1.Match.arrayWith([
144
+ assertions_1.Match.objectLike({
145
+ Action: assertions_1.Match.arrayWith(['s3:GetObject']),
146
+ Effect: 'Allow',
147
+ }),
148
+ ]),
149
+ },
150
+ }),
151
+ ]),
152
+ });
109
153
  });
110
154
  });
111
- test('configures timeout and memory for processing function', () => {
112
- basicTemplate.hasResourceProperties('AWS::Lambda::Function', {
113
- Timeout: 600,
114
- MemorySize: 1024,
155
+ describe('State machine integration', () => {
156
+ test('integrates agent as processing step in workflow', () => {
157
+ basicTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
158
+ DefinitionString: assertions_1.Match.objectLike({
159
+ 'Fn::Join': assertions_1.Match.arrayWith(['']),
160
+ }),
161
+ });
115
162
  });
116
- });
117
- test('uses cross-region inference when enabled', () => {
118
- crossRegionTemplate.hasResourceProperties('AWS::Lambda::Function', {
119
- Environment: {
120
- Variables: {
121
- MODEL_ID: 'us.anthropic.claude-sonnet-4-20250514-v1:0',
122
- },
123
- },
163
+ test('creates state machine with encryption', () => {
164
+ basicTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
165
+ EncryptionConfiguration: assertions_1.Match.objectLike({
166
+ Type: 'CUSTOMER_MANAGED_KMS_KEY',
167
+ }),
168
+ });
124
169
  });
125
170
  });
126
- test('uses default model when cross-region inference is disabled', () => {
127
- defaultModelTemplate.hasResourceProperties('AWS::Lambda::Function', {
128
- Environment: {
129
- Variables: {
130
- MODEL_ID: 'anthropic.claude-sonnet-4-20250514-v1:0',
131
- },
132
- },
171
+ describe('Resource counts', () => {
172
+ test('creates expected number of Lambda functions', () => {
173
+ basicTemplate.resourceCountIs('AWS::Lambda::Function', 5);
174
+ });
175
+ test('creates single state machine', () => {
176
+ basicTemplate.resourceCountIs('AWS::StepFunctions::StateMachine', 1);
133
177
  });
134
178
  });
135
179
  });
136
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"agentic-document-processing.test.js","sourceRoot":"","sources":["../../../use-cases/document-processing/tests/agentic-document-processing.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack } from 'aws-cdk-lib';\nimport { Template, Match } from 'aws-cdk-lib/assertions';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { AgenticDocumentProcessing } from '../agentic-document-processing';\n\ndescribe('AgenticDocumentProcessing', () => {\n  let basicStack: Stack;\n  let crossRegionStack: Stack;\n  let defaultModelStack: Stack;\n  let basicTemplate: Template;\n  let crossRegionTemplate: Template;\n  let defaultModelTemplate: Template;\n\n  beforeAll(() => {\n    // Create all stacks and constructs first\n    basicStack = new Stack();\n    const systemPrompt = new Asset(basicStack, 'SystemPrompt', {\n      path: __dirname + '/../resources/default-strands-agent',\n    });\n    new AgenticDocumentProcessing(basicStack, 'BasicTest', {\n      processingAgentParameters: {\n        agentName: 'TestAgent',\n        agentDefinition: {\n          bedrockModel: {},\n          systemPrompt,\n        },\n        prompt: 'Custom processing prompt',\n      },\n    });\n\n    crossRegionStack = new Stack();\n    const crossRegionPrompt = new Asset(crossRegionStack, 'SystemPrompt', {\n      path: __dirname + '/../resources/default-strands-agent',\n    });\n    new AgenticDocumentProcessing(crossRegionStack, 'CrossRegionTest', {\n      processingAgentParameters: {\n        agentName: 'CrossRegionAgent',\n        agentDefinition: {\n          bedrockModel: { useCrossRegionInference: true },\n          systemPrompt: crossRegionPrompt,\n        },\n        prompt: 'Test prompt',\n      },\n    });\n\n    defaultModelStack = new Stack();\n    const defaultPrompt = new Asset(defaultModelStack, 'SystemPrompt', {\n      path: __dirname + '/../resources/default-strands-agent',\n    });\n    new AgenticDocumentProcessing(defaultModelStack, 'DefaultModelTest', {\n      processingAgentParameters: {\n        agentName: 'DefaultAgent',\n        agentDefinition: {\n          bedrockModel: { useCrossRegionInference: false },\n          systemPrompt: defaultPrompt,\n        },\n        prompt: 'Test prompt',\n      },\n    });\n\n    // Generate templates once after all constructs are created\n    basicTemplate = Template.fromStack(basicStack);\n    crossRegionTemplate = Template.fromStack(crossRegionStack);\n    defaultModelTemplate = Template.fromStack(defaultModelStack);\n  });\n\n  test('creates basic infrastructure', () => {\n    basicTemplate.hasResourceProperties('AWS::S3::Bucket', {});\n    basicTemplate.hasResourceProperties('AWS::Lambda::Function', {\n      Runtime: 'python3.13',\n    });\n    basicTemplate.resourceCountIs('AWS::Lambda::Function', 5);\n  });\n\n  test('configures all agent parameters', () => {\n    basicTemplate.hasResourceProperties('AWS::Lambda::Function', {\n      Environment: {\n        Variables: {\n          PROMPT: 'Custom processing prompt',\n        },\n      },\n    });\n  });\n\n  test('inherits bedrock document processing functionality', () => {\n    basicTemplate.hasResourceProperties('AWS::S3::Bucket', {});\n    basicTemplate.hasResourceProperties('AWS::DynamoDB::Table', {});\n    basicTemplate.hasResourceProperties('AWS::SQS::Queue', {});\n    basicTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {});\n  });\n\n  test('creates IAM role with correct permissions', () => {\n    basicTemplate.hasResourceProperties('AWS::IAM::Role', {\n      AssumeRolePolicyDocument: {\n        Statement: [{\n          Action: 'sts:AssumeRole',\n          Effect: 'Allow',\n          Principal: {\n            Service: 'lambda.amazonaws.com',\n          },\n        }],\n      },\n    });\n\n    basicTemplate.hasResourceProperties('AWS::IAM::Role', {\n      Policies: [{\n        PolicyDocument: {\n          Statement: Match.arrayWith([\n            Match.objectLike({\n              Effect: 'Allow',\n              Action: ['bedrock:InvokeModel', 'bedrock:InvokeModelWithResponseStream'],\n            }),\n          ]),\n        },\n        PolicyName: 'BedrockInvokePolicy',\n      }],\n    });\n  });\n\n  test('configures timeout and memory for processing function', () => {\n    basicTemplate.hasResourceProperties('AWS::Lambda::Function', {\n      Timeout: 600,\n      MemorySize: 1024,\n    });\n  });\n\n  test('uses cross-region inference when enabled', () => {\n    crossRegionTemplate.hasResourceProperties('AWS::Lambda::Function', {\n      Environment: {\n        Variables: {\n          MODEL_ID: 'us.anthropic.claude-sonnet-4-20250514-v1:0',\n        },\n      },\n    });\n  });\n\n  test('uses default model when cross-region inference is disabled', () => {\n    defaultModelTemplate.hasResourceProperties('AWS::Lambda::Function', {\n      Environment: {\n        Variables: {\n          MODEL_ID: 'anthropic.claude-sonnet-4-20250514-v1:0',\n        },\n      },\n    });\n  });\n});\n"]}
180
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"agentic-document-processing.test.js","sourceRoot":"","sources":["../../../use-cases/document-processing/tests/agentic-document-processing.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack } from 'aws-cdk-lib';\nimport { Match, Template } from 'aws-cdk-lib/assertions';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { AgenticDocumentProcessing } from '../agentic-document-processing';\n\ndescribe('AgenticDocumentProcessing', () => {\n  let basicStack: Stack;\n  let crossRegionStack: Stack;\n  let defaultModelStack: Stack;\n  let basicTemplate: Template;\n  let crossRegionTemplate: Template;\n  let defaultModelTemplate: Template;\n\n  beforeAll(() => {\n    basicStack = new Stack();\n    const systemPrompt = new Asset(basicStack, 'SystemPrompt', {\n      path: __dirname + '/../resources/default-strands-agent',\n    });\n    new AgenticDocumentProcessing(basicStack, 'BasicTest', {\n      processingAgentParameters: {\n        agentName: 'TestAgent',\n        agentDefinition: {\n          bedrockModel: {},\n          systemPrompt,\n        },\n        prompt: 'Custom processing prompt',\n      },\n    });\n\n    crossRegionStack = new Stack();\n    const crossRegionPrompt = new Asset(crossRegionStack, 'SystemPrompt', {\n      path: __dirname + '/../resources/default-strands-agent',\n    });\n    new AgenticDocumentProcessing(crossRegionStack, 'CrossRegionTest', {\n      processingAgentParameters: {\n        agentName: 'CrossRegionAgent',\n        agentDefinition: {\n          bedrockModel: { useCrossRegionInference: true },\n          systemPrompt: crossRegionPrompt,\n        },\n        prompt: 'Test prompt',\n      },\n    });\n\n    defaultModelStack = new Stack();\n    const defaultPrompt = new Asset(defaultModelStack, 'SystemPrompt', {\n      path: __dirname + '/../resources/default-strands-agent',\n    });\n    new AgenticDocumentProcessing(defaultModelStack, 'DefaultModelTest', {\n      processingAgentParameters: {\n        agentName: 'DefaultAgent',\n        agentDefinition: {\n          bedrockModel: { useCrossRegionInference: false },\n          systemPrompt: defaultPrompt,\n        },\n        prompt: 'Test prompt',\n      },\n    });\n\n    basicTemplate = Template.fromStack(basicStack);\n    crossRegionTemplate = Template.fromStack(crossRegionStack);\n    defaultModelTemplate = Template.fromStack(defaultModelStack);\n  });\n\n  describe('Basic infrastructure', () => {\n    test('creates basic infrastructure', () => {\n      basicTemplate.hasResourceProperties('AWS::S3::Bucket', {});\n      basicTemplate.hasResourceProperties('AWS::Lambda::Function', {\n        Runtime: 'python3.13',\n      });\n      basicTemplate.resourceCountIs('AWS::Lambda::Function', 5);\n    });\n\n    test('inherits bedrock document processing functionality', () => {\n      basicTemplate.hasResourceProperties('AWS::S3::Bucket', {});\n      basicTemplate.hasResourceProperties('AWS::DynamoDB::Table', {});\n      basicTemplate.hasResourceProperties('AWS::SQS::Queue', {});\n      basicTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {});\n    });\n  });\n\n  describe('Agent configuration', () => {\n    test('configures all agent parameters', () => {\n      basicTemplate.hasResourceProperties('AWS::Lambda::Function', {\n        Environment: {\n          Variables: {\n            PROMPT: 'Custom processing prompt',\n          },\n        },\n      });\n    });\n\n    test('configures timeout and memory for processing function', () => {\n      basicTemplate.hasResourceProperties('AWS::Lambda::Function', {\n        Timeout: 600,\n        MemorySize: 1024,\n      });\n    });\n\n    test('uses cross-region inference when enabled', () => {\n      crossRegionTemplate.hasResourceProperties('AWS::Lambda::Function', {\n        Environment: {\n          Variables: {\n            MODEL_ID: 'us.anthropic.claude-sonnet-4-20250514-v1:0',\n          },\n        },\n      });\n    });\n\n    test('uses default model when cross-region inference is disabled', () => {\n      defaultModelTemplate.hasResourceProperties('AWS::Lambda::Function', {\n        Environment: {\n          Variables: {\n            MODEL_ID: 'anthropic.claude-sonnet-4-20250514-v1:0',\n          },\n        },\n      });\n    });\n  });\n\n  describe('IAM permissions', () => {\n    test('creates IAM role with correct permissions', () => {\n      basicTemplate.hasResourceProperties('AWS::IAM::Role', {\n        AssumeRolePolicyDocument: {\n          Statement: [{\n            Action: 'sts:AssumeRole',\n            Effect: 'Allow',\n            Principal: {\n              Service: 'lambda.amazonaws.com',\n            },\n          }],\n        },\n      });\n\n      basicTemplate.hasResourceProperties('AWS::IAM::Role', {\n        Policies: [{\n          PolicyDocument: {\n            Statement: Match.arrayWith([\n              Match.objectLike({\n                Effect: 'Allow',\n                Action: ['bedrock:InvokeModel', 'bedrock:InvokeModelWithResponseStream'],\n              }),\n            ]),\n          },\n          PolicyName: 'BedrockInvokePolicy',\n        }],\n      });\n    });\n\n    test('grants S3 access to agent function', () => {\n      basicTemplate.hasResourceProperties('AWS::IAM::Role', {\n        Policies: Match.arrayWith([\n          Match.objectLike({\n            PolicyDocument: {\n              Statement: Match.arrayWith([\n                Match.objectLike({\n                  Action: Match.arrayWith(['s3:GetObject']),\n                  Effect: 'Allow',\n                }),\n              ]),\n            },\n          }),\n        ]),\n      });\n    });\n  });\n\n  describe('State machine integration', () => {\n    test('integrates agent as processing step in workflow', () => {\n      basicTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        DefinitionString: Match.objectLike({\n          'Fn::Join': Match.arrayWith(['']),\n        }),\n      });\n    });\n\n    test('creates state machine with encryption', () => {\n      basicTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        EncryptionConfiguration: Match.objectLike({\n          Type: 'CUSTOMER_MANAGED_KMS_KEY',\n        }),\n      });\n    });\n  });\n\n  describe('Resource counts', () => {\n    test('creates expected number of Lambda functions', () => {\n      basicTemplate.resourceCountIs('AWS::Lambda::Function', 5);\n    });\n\n    test('creates single state machine', () => {\n      basicTemplate.resourceCountIs('AWS::StepFunctions::StateMachine', 1);\n    });\n  });\n});\n"]}
@@ -0,0 +1,161 @@
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_lambda_1 = require("aws-cdk-lib/aws-lambda");
6
+ const aws_s3_1 = require("aws-cdk-lib/aws-s3");
7
+ const aws_stepfunctions_tasks_1 = require("aws-cdk-lib/aws-stepfunctions-tasks");
8
+ const cdk_nag_1 = require("cdk-nag");
9
+ const framework_1 = require("../../framework");
10
+ const eventbridge_broker_1 = require("../../framework/foundation/eventbridge-broker");
11
+ const adapter_1 = require("../adapter");
12
+ const base_document_processing_1 = require("../base-document-processing");
13
+ // Concrete test implementation of BaseDocumentProcessing for CDK Nag testing
14
+ class TestDocumentProcessing extends base_document_processing_1.BaseDocumentProcessing {
15
+ constructor(scope, id, props) {
16
+ super(scope, id, props);
17
+ this.classificationFn = new aws_lambda_1.Function(this, 'ClassificationFn', {
18
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
19
+ handler: 'index.handler',
20
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ documentClassification: "TEST" });'),
21
+ });
22
+ this.processingFn = new aws_lambda_1.Function(this, 'ProcessingFn', {
23
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
24
+ handler: 'index.handler',
25
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ result: {} });'),
26
+ });
27
+ }
28
+ classificationStep() {
29
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockClassification', {
30
+ lambdaFunction: this.classificationFn,
31
+ resultPath: '$.classificationResult',
32
+ });
33
+ }
34
+ processingStep() {
35
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockProcessing', {
36
+ lambdaFunction: this.processingFn,
37
+ resultPath: '$.processingResult',
38
+ });
39
+ }
40
+ enrichmentStep() {
41
+ return undefined;
42
+ }
43
+ postProcessingStep() {
44
+ return undefined;
45
+ }
46
+ createStateMachine() {
47
+ return this.handleStateMachineCreation('test-state-machine');
48
+ }
49
+ }
50
+ // Create app and stack
51
+ const app = new aws_cdk_lib_1.App();
52
+ const stack = new aws_cdk_lib_1.Stack(app, 'TestStack', {
53
+ env: {
54
+ account: '123456789012',
55
+ region: 'us-east-1',
56
+ },
57
+ });
58
+ // Create access log bucket
59
+ const accessLog = new framework_1.AccessLog(stack, 'AccessLog');
60
+ // Create S3 bucket with proper configuration
61
+ const bucket = new aws_s3_1.Bucket(stack, 'BaseDocumentProcessingBucket', {
62
+ serverAccessLogsBucket: accessLog.bucket,
63
+ serverAccessLogsPrefix: accessLog.bucketPrefix,
64
+ enforceSSL: true,
65
+ });
66
+ // Create EventBridge broker
67
+ const broker = new eventbridge_broker_1.EventbridgeBroker(stack, 'TestBroker', {
68
+ name: 'test-broker',
69
+ eventSource: 'test-source',
70
+ });
71
+ // Create adapter with custom bucket
72
+ const adapter = new adapter_1.QueuedS3Adapter({
73
+ bucket,
74
+ });
75
+ // Create the BaseDocumentProcessing construct
76
+ const construct = new TestDocumentProcessing(stack, 'BaseDocumentProcessing', {
77
+ ingressAdapter: adapter,
78
+ eventbridgeBroker: broker,
79
+ enableObservability: true,
80
+ });
81
+ // Create the state machine
82
+ construct.createStateMachine();
83
+ // Suppress CDK-managed BucketNotificationsHandler AWS managed policy
84
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/BucketNotificationsHandler050a0587b7544547bf325f094a3db834/Role', [
85
+ {
86
+ id: 'AwsSolutions-IAM4',
87
+ reason: 'CDK-managed BucketNotificationsHandler requires AWSLambdaBasicExecutionRole for S3 event processing',
88
+ appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],
89
+ },
90
+ ]);
91
+ // Suppress S3 bucket wildcard permissions for Lambda roles
92
+ cdk_nag_1.NagSuppressions.addResourceSuppressions(stack, [
93
+ {
94
+ id: 'AwsSolutions-IAM5',
95
+ reason: 'Lambda functions require wildcard access to S3 bucket objects for document processing',
96
+ appliesTo: ['Resource::<BaseDocumentProcessingBucketE8E0F6F5.Arn>/*'],
97
+ },
98
+ ], true);
99
+ // Suppress SQS consumer Lambda wildcard permissions for log streams
100
+ cdk_nag_1.NagSuppressions.addResourceSuppressions(stack, [
101
+ {
102
+ id: 'AwsSolutions-IAM5',
103
+ reason: 'Lambda log stream ARN is only known at runtime, wildcard required for CloudWatch Logs access',
104
+ },
105
+ ], true);
106
+ // Suppress StateMachineRole wildcard permissions for Lambda invocation
107
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/BaseDocumentProcessing/StateMachineRole/DefaultPolicy', [
108
+ {
109
+ id: 'AwsSolutions-IAM5',
110
+ reason: 'Step Functions requires wildcard permissions to invoke Lambda functions with version-specific ARNs',
111
+ },
112
+ ]);
113
+ // Suppress Lambda log group wildcard permissions
114
+ cdk_nag_1.NagSuppressions.addResourceSuppressions(stack, [
115
+ {
116
+ id: 'AwsSolutions-IAM5',
117
+ reason: 'Lambda log stream names are generated at runtime, wildcard required for CloudWatch Logs access',
118
+ },
119
+ ], true);
120
+ // Suppress Lambda basic execution role
121
+ cdk_nag_1.NagSuppressions.addResourceSuppressions(stack, [
122
+ {
123
+ id: 'AwsSolutions-IAM4',
124
+ reason: 'Test Lambda functions use AWS managed policies for basic execution',
125
+ appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],
126
+ },
127
+ ], true);
128
+ // Suppress Lambda runtime version for test functions
129
+ cdk_nag_1.NagSuppressions.addResourceSuppressions(stack, [
130
+ {
131
+ id: 'AwsSolutions-L1',
132
+ reason: 'Test Lambda functions use Node.js 20 which is a supported runtime',
133
+ },
134
+ ], true);
135
+ // Suppress KMS key rotation for test encryption key
136
+ cdk_nag_1.NagSuppressions.addResourceSuppressions(stack, [
137
+ {
138
+ id: 'AwsSolutions-KMS5',
139
+ reason: 'KMS key rotation is enabled by default in BaseDocumentProcessing construct',
140
+ },
141
+ ], true);
142
+ // Apply CDK Nag checks
143
+ aws_cdk_lib_1.Aspects.of(app).add(new cdk_nag_1.AwsSolutionsChecks({ verbose: true }));
144
+ // Synthesize the stack and check for unsuppressed warnings and errors
145
+ const warnings = assertions_1.Annotations.fromStack(stack).findWarning('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
146
+ const errors = assertions_1.Annotations.fromStack(stack).findError('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
147
+ // Test: No unsuppressed warnings
148
+ test('No unsuppressed warnings', () => {
149
+ if (warnings.length > 0) {
150
+ console.log('CDK Nag Warnings:', JSON.stringify(warnings, null, 2));
151
+ }
152
+ expect(warnings).toHaveLength(0);
153
+ });
154
+ // Test: No unsuppressed errors
155
+ test('No unsuppressed errors', () => {
156
+ if (errors.length > 0) {
157
+ console.log('CDK Nag Errors:', JSON.stringify(errors, null, 2));
158
+ }
159
+ expect(errors).toHaveLength(0);
160
+ });
161
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-document-processing-nag.test.js","sourceRoot":"","sources":["../../../use-cases/document-processing/tests/base-document-processing-nag.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { App, Stack, Aspects } from 'aws-cdk-lib';\nimport { Annotations, Match } from 'aws-cdk-lib/assertions';\nimport { Code, Function, Runtime } from 'aws-cdk-lib/aws-lambda';\nimport { Bucket } from 'aws-cdk-lib/aws-s3';\nimport { LambdaInvoke } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { AwsSolutionsChecks, NagSuppressions } from 'cdk-nag';\nimport { AccessLog } from '../../framework';\nimport { EventbridgeBroker } from '../../framework/foundation/eventbridge-broker';\nimport { QueuedS3Adapter } from '../adapter';\nimport { BaseDocumentProcessing, DocumentProcessingStepType } from '../base-document-processing';\n\n// Concrete test implementation of BaseDocumentProcessing for CDK Nag testing\nclass TestDocumentProcessing extends BaseDocumentProcessing {\n  private classificationFn: Function;\n  private processingFn: Function;\n\n  constructor(scope: any, id: string, props: any) {\n    super(scope, id, props);\n\n    this.classificationFn = new Function(this, 'ClassificationFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ documentClassification: \"TEST\" });'),\n    });\n\n    this.processingFn = new Function(this, 'ProcessingFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ result: {} });'),\n    });\n  }\n\n  protected classificationStep(): DocumentProcessingStepType {\n    return new LambdaInvoke(this, 'MockClassification', {\n      lambdaFunction: this.classificationFn,\n      resultPath: '$.classificationResult',\n    });\n  }\n\n  protected processingStep(): DocumentProcessingStepType {\n    return new LambdaInvoke(this, 'MockProcessing', {\n      lambdaFunction: this.processingFn,\n      resultPath: '$.processingResult',\n    });\n  }\n\n  protected enrichmentStep(): DocumentProcessingStepType | undefined {\n    return undefined;\n  }\n\n  protected postProcessingStep(): DocumentProcessingStepType | undefined {\n    return undefined;\n  }\n\n  public createStateMachine() {\n    return this.handleStateMachineCreation('test-state-machine');\n  }\n}\n\n// Create app and stack\nconst app = new App();\nconst stack = new Stack(app, 'TestStack', {\n  env: {\n    account: '123456789012',\n    region: 'us-east-1',\n  },\n});\n\n// Create access log bucket\nconst accessLog = new AccessLog(stack, 'AccessLog');\n\n// Create S3 bucket with proper configuration\nconst bucket = new Bucket(stack, 'BaseDocumentProcessingBucket', {\n  serverAccessLogsBucket: accessLog.bucket,\n  serverAccessLogsPrefix: accessLog.bucketPrefix,\n  enforceSSL: true,\n});\n\n// Create EventBridge broker\nconst broker = new EventbridgeBroker(stack, 'TestBroker', {\n  name: 'test-broker',\n  eventSource: 'test-source',\n});\n\n// Create adapter with custom bucket\nconst adapter = new QueuedS3Adapter({\n  bucket,\n});\n\n// Create the BaseDocumentProcessing construct\nconst construct = new TestDocumentProcessing(stack, 'BaseDocumentProcessing', {\n  ingressAdapter: adapter,\n  eventbridgeBroker: broker,\n  enableObservability: true,\n});\n\n// Create the state machine\nconstruct.createStateMachine();\n\n// Suppress CDK-managed BucketNotificationsHandler AWS managed policy\nNagSuppressions.addResourceSuppressionsByPath(\n  stack,\n  '/TestStack/BucketNotificationsHandler050a0587b7544547bf325f094a3db834/Role',\n  [\n    {\n      id: 'AwsSolutions-IAM4',\n      reason: 'CDK-managed BucketNotificationsHandler requires AWSLambdaBasicExecutionRole for S3 event processing',\n      appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],\n    },\n  ],\n);\n\n// Suppress S3 bucket wildcard permissions for Lambda roles\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Lambda functions require wildcard access to S3 bucket objects for document processing',\n      appliesTo: ['Resource::<BaseDocumentProcessingBucketE8E0F6F5.Arn>/*'],\n    },\n  ],\n  true,\n);\n\n// Suppress SQS consumer Lambda wildcard permissions for log streams\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Lambda log stream ARN is only known at runtime, wildcard required for CloudWatch Logs access',\n    },\n  ],\n  true,\n);\n\n// Suppress StateMachineRole wildcard permissions for Lambda invocation\nNagSuppressions.addResourceSuppressionsByPath(\n  stack,\n  '/TestStack/BaseDocumentProcessing/StateMachineRole/DefaultPolicy',\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Step Functions requires wildcard permissions to invoke Lambda functions with version-specific ARNs',\n    },\n  ],\n);\n\n// Suppress Lambda log group wildcard permissions\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Lambda log stream names are generated at runtime, wildcard required for CloudWatch Logs access',\n    },\n  ],\n  true,\n);\n\n// Suppress Lambda basic execution role\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM4',\n      reason: 'Test Lambda functions use AWS managed policies for basic execution',\n      appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],\n    },\n  ],\n  true,\n);\n\n// Suppress Lambda runtime version for test functions\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-L1',\n      reason: 'Test Lambda functions use Node.js 20 which is a supported runtime',\n    },\n  ],\n  true,\n);\n\n// Suppress KMS key rotation for test encryption key\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-KMS5',\n      reason: 'KMS key rotation is enabled by default in BaseDocumentProcessing construct',\n    },\n  ],\n  true,\n);\n\n// Apply CDK Nag checks\nAspects.of(app).add(new AwsSolutionsChecks({ verbose: true }));\n\n// Synthesize the stack and check for unsuppressed warnings and errors\nconst warnings = Annotations.fromStack(stack).findWarning('*', Match.stringLikeRegexp('AwsSolutions-.*'));\nconst errors = Annotations.fromStack(stack).findError('*', Match.stringLikeRegexp('AwsSolutions-.*'));\n\n// Test: No unsuppressed warnings\ntest('No unsuppressed warnings', () => {\n  if (warnings.length > 0) {\n    console.log('CDK Nag Warnings:', JSON.stringify(warnings, null, 2));\n  }\n  expect(warnings).toHaveLength(0);\n});\n\n// Test: No unsuppressed errors\ntest('No unsuppressed errors', () => {\n  if (errors.length > 0) {\n    console.log('CDK Nag Errors:', JSON.stringify(errors, null, 2));\n  }\n  expect(errors).toHaveLength(0);\n});\n"]}