@cdklabs/cdk-appmod-catalog-blueprints 1.2.2 → 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 +804 -173
- package/README.md +76 -98
- package/lib/document-processing/adapter/queued-s3-adapter.js +1 -1
- package/lib/document-processing/agentic-document-processing.d.ts +5 -28
- package/lib/document-processing/agentic-document-processing.js +8 -63
- package/lib/document-processing/base-document-processing.js +4 -20
- package/lib/document-processing/bedrock-document-processing.d.ts +4 -32
- package/lib/document-processing/bedrock-document-processing.js +10 -37
- package/lib/document-processing/default-document-processing-config.js +1 -1
- package/lib/document-processing/tests/agentic-document-processing-nag.test.js +12 -11
- package/lib/document-processing/tests/agentic-document-processing.test.js +136 -67
- 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-nag.test.js +3 -2
- package/lib/document-processing/tests/bedrock-document-processing.test.js +221 -40
- 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.d.ts +90 -0
- package/lib/framework/agents/base-agent.js +55 -0
- package/lib/framework/agents/batch-agent.d.ts +11 -0
- package/lib/framework/agents/batch-agent.js +64 -0
- package/lib/framework/agents/default-agent-config.d.ts +3 -0
- package/lib/framework/agents/default-agent-config.js +12 -0
- package/lib/framework/agents/index.d.ts +3 -0
- package/lib/framework/agents/index.js +20 -0
- package/lib/framework/agents/resources/default-strands-agent/batch.py +99 -0
- package/lib/framework/agents/resources/default-strands-agent/models.py +7 -0
- package/lib/framework/agents/resources/default-strands-agent/requirements.txt +7 -0
- package/lib/framework/agents/resources/default-strands-agent/utils.py +36 -0
- package/lib/framework/bedrock/bedrock.d.ts +38 -0
- package/lib/framework/bedrock/bedrock.js +54 -0
- package/lib/framework/bedrock/index.d.ts +1 -0
- package/lib/framework/bedrock/index.js +18 -0
- 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/index.d.ts +2 -0
- package/lib/framework/index.js +3 -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 +4 -3
- 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/index.d.ts +1 -0
- package/lib/utilities/observability/index.js +2 -1
- package/lib/utilities/observability/lambda-observability-property-injector.js +1 -1
- package/lib/utilities/observability/log-group-data-protection-utils.d.ts +6 -0
- package/lib/utilities/observability/log-group-data-protection-utils.js +37 -0
- 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
|
@@ -3,6 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
3
3
|
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
4
4
|
const assertions_1 = require("aws-cdk-lib/assertions");
|
|
5
5
|
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
|
|
6
|
+
const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets");
|
|
6
7
|
const cdk_nag_1 = require("cdk-nag");
|
|
7
8
|
const framework_1 = require("../../framework");
|
|
8
9
|
const adapter_1 = require("../adapter");
|
|
@@ -25,23 +26,23 @@ const bucket = new aws_s3_1.Bucket(stack, 'AgenticDocumentProcessingBucket', {
|
|
|
25
26
|
const adapter = new adapter_1.QueuedS3Adapter({
|
|
26
27
|
bucket,
|
|
27
28
|
});
|
|
29
|
+
const systemPrompt = new aws_s3_assets_1.Asset(stack, 'SystemPrompt', {
|
|
30
|
+
path: __dirname + '/../resources/default-strands-agent',
|
|
31
|
+
});
|
|
28
32
|
// Create the main AgenticDocumentProcessing construct
|
|
29
33
|
new agentic_document_processing_1.AgenticDocumentProcessing(stack, 'AgenticDocumentProcessing', {
|
|
30
34
|
ingressAdapter: adapter,
|
|
31
|
-
useCrossRegionInference: true,
|
|
32
35
|
processingAgentParameters: {
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
],
|
|
40
|
-
},
|
|
41
|
-
processingPrompt: `
|
|
36
|
+
agentName: 'ClaimsSpecialist',
|
|
37
|
+
agentDefinition: {
|
|
38
|
+
bedrockModel: { useCrossRegionInference: true },
|
|
39
|
+
systemPrompt,
|
|
40
|
+
},
|
|
41
|
+
prompt: `
|
|
42
42
|
Analyze the attached insurance claim document and check if this is a valid claim or not.
|
|
43
43
|
Final output should in JSON format with claim_approved and justification fields.
|
|
44
44
|
`,
|
|
45
|
+
},
|
|
45
46
|
enableObservability: true,
|
|
46
47
|
});
|
|
47
48
|
// Suppress CDK-managed BucketNotificationsHandler AWS managed policy
|
|
@@ -108,4 +109,4 @@ test('No unsuppressed errors', () => {
|
|
|
108
109
|
}
|
|
109
110
|
expect(errors).toHaveLength(0);
|
|
110
111
|
});
|
|
111
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"agentic-document-processing-nag.test.js","sourceRoot":"","sources":["../../../use-cases/document-processing/tests/agentic-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 { Bucket } from 'aws-cdk-lib/aws-s3';\nimport { AwsSolutionsChecks, NagSuppressions } from 'cdk-nag';\nimport { AccessLog } from '../../framework';\nimport { QueuedS3Adapter } from '../adapter';\nimport { AgenticDocumentProcessing } from '../agentic-document-processing';\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\nconst accessLog = new AccessLog(stack, 'AccessLog');\n\n// Create S3 bucket for agentic tools\nconst bucket = new Bucket(stack, 'AgenticDocumentProcessingBucket', {\n  serverAccessLogsBucket: accessLog.bucket,\n  serverAccessLogsPrefix: accessLog.bucketPrefix,\n  enforceSSL: true,\n});\n\nconst adapter = new QueuedS3Adapter({\n  bucket,\n});\n\n// Create the main AgenticDocumentProcessing construct\nnew AgenticDocumentProcessing(stack, 'AgenticDocumentProcessing', {\n  ingressAdapter: adapter,\n  useCrossRegionInference: true,\n  processingAgentParameters: {\n    agentSystemPrompt: `\n      You're an insurance claims specialist. Use the provided tools to ensure that the submitted claims and supporting documents are valid and there are no discrepancies.\n    `,\n    toolsLocation: [\n      `s3://${bucket.bucketName}/agentic-tools/download_policy.py`,\n      `s3://${bucket.bucketName}/agentic-tools/download_supporting_documents.py`,\n    ],\n  },\n  processingPrompt: `\n    Analyze the attached insurance claim document and check if this is a valid claim or not.\n    Final output should in JSON format with claim_approved and justification fields.\n  `,\n  enableObservability: true,\n});\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::<AgenticDocumentProcessingBucketC4E254EC.Arn>/*'],\n    },\n  ],\n  true,\n);\n\n// Suppress Bedrock foundation model wildcard permissions\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Cross-region inference requires wildcard region access to Bedrock foundation models',\n      appliesTo: ['Resource::arn:aws:bedrock:*::foundation-model/anthropic.claude-3-7-sonnet-20250219-v1:0'],\n    },\n  ],\n  true,\n);\n\n// Suppress SQSConsumerRole wildcard permissions for Lambda 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/AgenticDocumentProcessing/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\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"]}
|
|
112
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"agentic-document-processing-nag.test.js","sourceRoot":"","sources":["../../../use-cases/document-processing/tests/agentic-document-processing-nag.test.ts"],"names":[],"mappings":";;AAAA,6CAAkD;AAClD,uDAA4D;AAC5D,+CAA4C;AAC5C,6DAAkD;AAClD,qCAA8D;AAC9D,+CAA4C;AAC5C,wCAA6C;AAC7C,gFAA2E;AAE3E,uBAAuB;AACvB,MAAM,GAAG,GAAG,IAAI,iBAAG,EAAE,CAAC;AACtB,MAAM,KAAK,GAAG,IAAI,mBAAK,CAAC,GAAG,EAAE,WAAW,EAAE;IACxC,GAAG,EAAE;QACH,OAAO,EAAE,cAAc;QACvB,MAAM,EAAE,WAAW;KACpB;CACF,CAAC,CAAC;AAEH,MAAM,SAAS,GAAG,IAAI,qBAAS,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC;AAEpD,qCAAqC;AACrC,MAAM,MAAM,GAAG,IAAI,eAAM,CAAC,KAAK,EAAE,iCAAiC,EAAE;IAClE,sBAAsB,EAAE,SAAS,CAAC,MAAM;IACxC,sBAAsB,EAAE,SAAS,CAAC,YAAY;IAC9C,UAAU,EAAE,IAAI;CACjB,CAAC,CAAC;AAEH,MAAM,OAAO,GAAG,IAAI,yBAAe,CAAC;IAClC,MAAM;CACP,CAAC,CAAC;AAEH,MAAM,YAAY,GAAG,IAAI,qBAAK,CAAC,KAAK,EAAE,cAAc,EAAE;IACpD,IAAI,EAAE,SAAS,GAAG,qCAAqC;CACxD,CAAC,CAAC;AAEH,sDAAsD;AACtD,IAAI,uDAAyB,CAAC,KAAK,EAAE,2BAA2B,EAAE;IAChE,cAAc,EAAE,OAAO;IACvB,yBAAyB,EAAE;QACzB,SAAS,EAAE,kBAAkB;QAC7B,eAAe,EAAE;YACf,YAAY,EAAE,EAAE,uBAAuB,EAAE,IAAI,EAAE;YAC/C,YAAY;SACb;QACD,MAAM,EAAE;;;GAGT;KACA;IACD,mBAAmB,EAAE,IAAI;CAC1B,CAAC,CAAC;AAEH,qEAAqE;AACrE,yBAAe,CAAC,6BAA6B,CAC3C,KAAK,EACL,4EAA4E,EAC5E;IACE;QACE,EAAE,EAAE,mBAAmB;QACvB,MAAM,EAAE,qGAAqG;QAC7G,SAAS,EAAE,CAAC,uFAAuF,CAAC;KACrG;CACF,CACF,CAAC;AAEF,2DAA2D;AAC3D,yBAAe,CAAC,uBAAuB,CACrC,KAAK,EACL;IACE;QACE,EAAE,EAAE,mBAAmB;QACvB,MAAM,EAAE,uFAAuF;QAC/F,SAAS,EAAE,CAAC,2DAA2D,CAAC;KACzE;CACF,EACD,IAAI,CACL,CAAC;AAEF,yDAAyD;AACzD,yBAAe,CAAC,uBAAuB,CACrC,KAAK,EACL;IACE;QACE,EAAE,EAAE,mBAAmB;QACvB,MAAM,EAAE,qFAAqF;QAC7F,SAAS,EAAE,CAAC,yFAAyF,CAAC;KACvG;CACF,EACD,IAAI,CACL,CAAC;AAEF,uEAAuE;AACvE,yBAAe,CAAC,uBAAuB,CACrC,KAAK,EACL;IACE;QACE,EAAE,EAAE,mBAAmB;QACvB,MAAM,EAAE,8FAA8F;KACvG;CACF,EACD,IAAI,CACL,CAAC;AAEF,uEAAuE;AACvE,yBAAe,CAAC,6BAA6B,CAC3C,KAAK,EACL,qEAAqE,EACrE;IACE;QACE,EAAE,EAAE,mBAAmB;QACvB,MAAM,EAAE,oGAAoG;KAC7G;CACF,CACF,CAAC;AAEF,iDAAiD;AACjD,yBAAe,CAAC,uBAAuB,CACrC,KAAK,EACL;IACE;QACE,EAAE,EAAE,mBAAmB;QACvB,MAAM,EAAE,gGAAgG;KACzG;CACF,EACD,IAAI,CACL,CAAC;AAGF,uBAAuB;AACvB,qBAAO,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,IAAI,4BAAkB,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;AAE/D,sEAAsE;AACtE,MAAM,QAAQ,GAAG,wBAAW,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,WAAW,CAAC,GAAG,EAAE,kBAAK,CAAC,gBAAgB,CAAC,iBAAiB,CAAC,CAAC,CAAC;AAC1G,MAAM,MAAM,GAAG,wBAAW,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,kBAAK,CAAC,gBAAgB,CAAC,iBAAiB,CAAC,CAAC,CAAC;AAEtG,iCAAiC;AACjC,IAAI,CAAC,0BAA0B,EAAE,GAAG,EAAE;IACpC,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACxB,OAAO,CAAC,GAAG,CAAC,mBAAmB,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;IACtE,CAAC;IACD,MAAM,CAAC,QAAQ,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;AACnC,CAAC,CAAC,CAAC;AAEH,+BAA+B;AAC/B,IAAI,CAAC,wBAAwB,EAAE,GAAG,EAAE;IAClC,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACtB,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;IAClE,CAAC;IACD,MAAM,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;AACjC,CAAC,CAAC,CAAC","sourcesContent":["import { App, Stack, Aspects } from 'aws-cdk-lib';\nimport { Annotations, Match } from 'aws-cdk-lib/assertions';\nimport { Bucket } from 'aws-cdk-lib/aws-s3';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { AwsSolutionsChecks, NagSuppressions } from 'cdk-nag';\nimport { AccessLog } from '../../framework';\nimport { QueuedS3Adapter } from '../adapter';\nimport { AgenticDocumentProcessing } from '../agentic-document-processing';\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\nconst accessLog = new AccessLog(stack, 'AccessLog');\n\n// Create S3 bucket for agentic tools\nconst bucket = new Bucket(stack, 'AgenticDocumentProcessingBucket', {\n  serverAccessLogsBucket: accessLog.bucket,\n  serverAccessLogsPrefix: accessLog.bucketPrefix,\n  enforceSSL: true,\n});\n\nconst adapter = new QueuedS3Adapter({\n  bucket,\n});\n\nconst systemPrompt = new Asset(stack, 'SystemPrompt', {\n  path: __dirname + '/../resources/default-strands-agent',\n});\n\n// Create the main AgenticDocumentProcessing construct\nnew AgenticDocumentProcessing(stack, 'AgenticDocumentProcessing', {\n  ingressAdapter: adapter,\n  processingAgentParameters: {\n    agentName: 'ClaimsSpecialist',\n    agentDefinition: {\n      bedrockModel: { useCrossRegionInference: true },\n      systemPrompt,\n    },\n    prompt: `\n    Analyze the attached insurance claim document and check if this is a valid claim or not.\n    Final output should in JSON format with claim_approved and justification fields.\n  `,\n  },\n  enableObservability: true,\n});\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::<AgenticDocumentProcessingBucketC4E254EC.Arn>/*'],\n    },\n  ],\n  true,\n);\n\n// Suppress Bedrock foundation model wildcard permissions\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Cross-region inference requires wildcard region access to Bedrock foundation models',\n      appliesTo: ['Resource::arn:aws:bedrock:*::foundation-model/anthropic.claude-3-7-sonnet-20250219-v1:0'],\n    },\n  ],\n  true,\n);\n\n// Suppress SQSConsumerRole wildcard permissions for Lambda 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/AgenticDocumentProcessing/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\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"]}
|
|
@@ -2,6 +2,7 @@
|
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
4
4
|
const assertions_1 = require("aws-cdk-lib/assertions");
|
|
5
|
+
const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets");
|
|
5
6
|
const agentic_document_processing_1 = require("../agentic-document-processing");
|
|
6
7
|
describe('AgenticDocumentProcessing', () => {
|
|
7
8
|
let basicStack;
|
|
@@ -11,101 +12,169 @@ describe('AgenticDocumentProcessing', () => {
|
|
|
11
12
|
let crossRegionTemplate;
|
|
12
13
|
let defaultModelTemplate;
|
|
13
14
|
beforeAll(() => {
|
|
14
|
-
// Create all stacks and constructs first
|
|
15
15
|
basicStack = new aws_cdk_lib_1.Stack();
|
|
16
|
+
const systemPrompt = new aws_s3_assets_1.Asset(basicStack, 'SystemPrompt', {
|
|
17
|
+
path: __dirname + '/../resources/default-strands-agent',
|
|
18
|
+
});
|
|
16
19
|
new agentic_document_processing_1.AgenticDocumentProcessing(basicStack, 'BasicTest', {
|
|
17
|
-
processingPrompt: 'Custom processing prompt',
|
|
18
20
|
processingAgentParameters: {
|
|
19
|
-
|
|
20
|
-
|
|
21
|
+
agentName: 'TestAgent',
|
|
22
|
+
agentDefinition: {
|
|
23
|
+
bedrockModel: {},
|
|
24
|
+
systemPrompt,
|
|
25
|
+
},
|
|
26
|
+
prompt: 'Custom processing prompt',
|
|
21
27
|
},
|
|
22
28
|
});
|
|
23
29
|
crossRegionStack = new aws_cdk_lib_1.Stack();
|
|
30
|
+
const crossRegionPrompt = new aws_s3_assets_1.Asset(crossRegionStack, 'SystemPrompt', {
|
|
31
|
+
path: __dirname + '/../resources/default-strands-agent',
|
|
32
|
+
});
|
|
24
33
|
new agentic_document_processing_1.AgenticDocumentProcessing(crossRegionStack, 'CrossRegionTest', {
|
|
25
|
-
|
|
34
|
+
processingAgentParameters: {
|
|
35
|
+
agentName: 'CrossRegionAgent',
|
|
36
|
+
agentDefinition: {
|
|
37
|
+
bedrockModel: { useCrossRegionInference: true },
|
|
38
|
+
systemPrompt: crossRegionPrompt,
|
|
39
|
+
},
|
|
40
|
+
prompt: 'Test prompt',
|
|
41
|
+
},
|
|
26
42
|
});
|
|
27
43
|
defaultModelStack = new aws_cdk_lib_1.Stack();
|
|
44
|
+
const defaultPrompt = new aws_s3_assets_1.Asset(defaultModelStack, 'SystemPrompt', {
|
|
45
|
+
path: __dirname + '/../resources/default-strands-agent',
|
|
46
|
+
});
|
|
28
47
|
new agentic_document_processing_1.AgenticDocumentProcessing(defaultModelStack, 'DefaultModelTest', {
|
|
29
|
-
|
|
48
|
+
processingAgentParameters: {
|
|
49
|
+
agentName: 'DefaultAgent',
|
|
50
|
+
agentDefinition: {
|
|
51
|
+
bedrockModel: { useCrossRegionInference: false },
|
|
52
|
+
systemPrompt: defaultPrompt,
|
|
53
|
+
},
|
|
54
|
+
prompt: 'Test prompt',
|
|
55
|
+
},
|
|
30
56
|
});
|
|
31
|
-
// Generate templates once after all constructs are created
|
|
32
57
|
basicTemplate = assertions_1.Template.fromStack(basicStack);
|
|
33
58
|
crossRegionTemplate = assertions_1.Template.fromStack(crossRegionStack);
|
|
34
59
|
defaultModelTemplate = assertions_1.Template.fromStack(defaultModelStack);
|
|
35
60
|
});
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
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', {});
|
|
40
74
|
});
|
|
41
|
-
basicTemplate.resourceCountIs('AWS::Lambda::Function', 5);
|
|
42
75
|
});
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
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
|
+
},
|
|
50
83
|
},
|
|
51
|
-
}
|
|
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
|
+
});
|
|
52
109
|
});
|
|
53
110
|
});
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
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
|
+
]),
|
|
68
133
|
},
|
|
134
|
+
PolicyName: 'BedrockInvokePolicy',
|
|
69
135
|
}],
|
|
70
|
-
}
|
|
136
|
+
});
|
|
71
137
|
});
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
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
|
+
});
|
|
84
153
|
});
|
|
85
154
|
});
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
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
|
+
});
|
|
90
162
|
});
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
},
|
|
98
|
-
},
|
|
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
|
+
});
|
|
99
169
|
});
|
|
100
170
|
});
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
},
|
|
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);
|
|
108
177
|
});
|
|
109
178
|
});
|
|
110
179
|
});
|
|
111
|
-
//# 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 { 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    new AgenticDocumentProcessing(basicStack, 'BasicTest', {\n      processingPrompt: 'Custom processing prompt',\n      processingAgentParameters: {\n        agentSystemPrompt: 'Test system prompt',\n        toolsLocation: ['tool1', 'tool2'],\n      },\n    });\n\n    crossRegionStack = new Stack();\n    new AgenticDocumentProcessing(crossRegionStack, 'CrossRegionTest', {\n      useCrossRegionInference: true,\n    });\n\n    defaultModelStack = new Stack();\n    new AgenticDocumentProcessing(defaultModelStack, 'DefaultModelTest', {\n      useCrossRegionInference: false,\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          SYSTEM_PROMPT: 'Test system prompt',\n          TOOLS_CONFIG: '[\"tool1\",\"tool2\"]',\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: 300,\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-3-7-sonnet-20250219-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-3-7-sonnet-20250219-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 @@
|
|
|
1
|
+
export {};
|
|
@@ -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"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|