@cdklabs/cdk-appmod-catalog-blueprints 1.4.1 → 1.6.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.
Files changed (93) hide show
  1. package/.jsii +2579 -194
  2. package/lib/document-processing/adapter/adapter.d.ts +4 -2
  3. package/lib/document-processing/adapter/adapter.js +1 -1
  4. package/lib/document-processing/adapter/queued-s3-adapter.d.ts +9 -2
  5. package/lib/document-processing/adapter/queued-s3-adapter.js +29 -15
  6. package/lib/document-processing/agentic-document-processing.d.ts +4 -0
  7. package/lib/document-processing/agentic-document-processing.js +20 -10
  8. package/lib/document-processing/base-document-processing.d.ts +54 -2
  9. package/lib/document-processing/base-document-processing.js +136 -82
  10. package/lib/document-processing/bedrock-document-processing.d.ts +202 -2
  11. package/lib/document-processing/bedrock-document-processing.js +717 -77
  12. package/lib/document-processing/chunking-config.d.ts +614 -0
  13. package/lib/document-processing/chunking-config.js +5 -0
  14. package/lib/document-processing/default-document-processing-config.js +1 -1
  15. package/lib/document-processing/index.d.ts +1 -0
  16. package/lib/document-processing/index.js +2 -1
  17. package/lib/document-processing/resources/aggregation/handler.py +567 -0
  18. package/lib/document-processing/resources/aggregation/requirements.txt +7 -0
  19. package/lib/document-processing/resources/aggregation/test_handler.py +362 -0
  20. package/lib/document-processing/resources/cleanup/handler.py +276 -0
  21. package/lib/document-processing/resources/cleanup/requirements.txt +5 -0
  22. package/lib/document-processing/resources/cleanup/test_handler.py +436 -0
  23. package/lib/document-processing/resources/default-bedrock-invoke/index.py +85 -3
  24. package/lib/document-processing/resources/default-bedrock-invoke/test_index.py +622 -0
  25. package/lib/document-processing/resources/pdf-chunking/README.md +313 -0
  26. package/lib/document-processing/resources/pdf-chunking/chunking_strategies.py +460 -0
  27. package/lib/document-processing/resources/pdf-chunking/error_handling.py +491 -0
  28. package/lib/document-processing/resources/pdf-chunking/handler.py +958 -0
  29. package/lib/document-processing/resources/pdf-chunking/metrics.py +435 -0
  30. package/lib/document-processing/resources/pdf-chunking/requirements.txt +3 -0
  31. package/lib/document-processing/resources/pdf-chunking/strategy_selection.py +420 -0
  32. package/lib/document-processing/resources/pdf-chunking/structured_logging.py +457 -0
  33. package/lib/document-processing/resources/pdf-chunking/test_chunking_strategies.py +353 -0
  34. package/lib/document-processing/resources/pdf-chunking/test_error_handling.py +487 -0
  35. package/lib/document-processing/resources/pdf-chunking/test_handler.py +609 -0
  36. package/lib/document-processing/resources/pdf-chunking/test_integration.py +694 -0
  37. package/lib/document-processing/resources/pdf-chunking/test_metrics.py +532 -0
  38. package/lib/document-processing/resources/pdf-chunking/test_strategy_selection.py +471 -0
  39. package/lib/document-processing/resources/pdf-chunking/test_structured_logging.py +449 -0
  40. package/lib/document-processing/resources/pdf-chunking/test_token_estimation.py +374 -0
  41. package/lib/document-processing/resources/pdf-chunking/token_estimation.py +189 -0
  42. package/lib/document-processing/tests/agentic-document-processing-nag.test.js +4 -3
  43. package/lib/document-processing/tests/agentic-document-processing.test.js +488 -4
  44. package/lib/document-processing/tests/base-document-processing-nag.test.js +9 -2
  45. package/lib/document-processing/tests/base-document-processing-schema.test.d.ts +1 -0
  46. package/lib/document-processing/tests/base-document-processing-schema.test.js +337 -0
  47. package/lib/document-processing/tests/base-document-processing.test.js +114 -8
  48. package/lib/document-processing/tests/bedrock-document-processing-chunking-nag.test.d.ts +1 -0
  49. package/lib/document-processing/tests/bedrock-document-processing-chunking-nag.test.js +382 -0
  50. package/lib/document-processing/tests/bedrock-document-processing-nag.test.js +4 -3
  51. package/lib/document-processing/tests/bedrock-document-processing-security.test.d.ts +1 -0
  52. package/lib/document-processing/tests/bedrock-document-processing-security.test.js +389 -0
  53. package/lib/document-processing/tests/bedrock-document-processing.test.js +808 -8
  54. package/lib/document-processing/tests/chunking-config.test.d.ts +1 -0
  55. package/lib/document-processing/tests/chunking-config.test.js +238 -0
  56. package/lib/document-processing/tests/queued-s3-adapter-nag.test.js +9 -2
  57. package/lib/document-processing/tests/queued-s3-adapter.test.js +17 -6
  58. package/lib/framework/agents/base-agent.js +1 -1
  59. package/lib/framework/agents/batch-agent.js +1 -1
  60. package/lib/framework/agents/default-agent-config.js +1 -1
  61. package/lib/framework/bedrock/bedrock.js +1 -1
  62. package/lib/framework/custom-resource/default-runtimes.js +1 -1
  63. package/lib/framework/foundation/access-log.js +1 -1
  64. package/lib/framework/foundation/eventbridge-broker.js +1 -1
  65. package/lib/framework/foundation/network.d.ts +4 -2
  66. package/lib/framework/foundation/network.js +52 -41
  67. package/lib/framework/tests/access-log.test.js +5 -2
  68. package/lib/framework/tests/batch-agent.test.js +5 -2
  69. package/lib/framework/tests/bedrock.test.js +5 -2
  70. package/lib/framework/tests/eventbridge-broker.test.js +5 -2
  71. package/lib/framework/tests/framework-nag.test.js +26 -7
  72. package/lib/framework/tests/network.test.js +30 -2
  73. package/lib/tsconfig.tsbuildinfo +1 -1
  74. package/lib/utilities/data-loader.js +1 -1
  75. package/lib/utilities/lambda-iam-utils.js +1 -1
  76. package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +1 -1
  77. package/lib/utilities/observability/default-observability-config.js +1 -1
  78. package/lib/utilities/observability/lambda-observability-property-injector.js +1 -1
  79. package/lib/utilities/observability/log-group-data-protection-utils.js +1 -1
  80. package/lib/utilities/observability/powertools-config.d.ts +10 -1
  81. package/lib/utilities/observability/powertools-config.js +19 -3
  82. package/lib/utilities/observability/state-machine-observability-property-injector.js +1 -1
  83. package/lib/utilities/test-utils.d.ts +43 -0
  84. package/lib/utilities/test-utils.js +56 -0
  85. package/lib/utilities/tests/data-loader-nag.test.js +3 -2
  86. package/lib/utilities/tests/data-loader.test.js +3 -2
  87. package/lib/webapp/frontend-construct.js +1 -1
  88. package/lib/webapp/tests/frontend-construct-nag.test.js +3 -2
  89. package/lib/webapp/tests/frontend-construct.test.js +3 -2
  90. package/package.json +6 -5
  91. package/lib/document-processing/resources/default-error-handler/index.js +0 -46
  92. package/lib/document-processing/resources/default-pdf-processor/index.js +0 -46
  93. package/lib/document-processing/resources/default-pdf-validator/index.js +0 -36
@@ -8,6 +8,7 @@ const aws_stepfunctions_tasks_1 = require("aws-cdk-lib/aws-stepfunctions-tasks")
8
8
  const cdk_nag_1 = require("cdk-nag");
9
9
  const framework_1 = require("../../framework");
10
10
  const eventbridge_broker_1 = require("../../framework/foundation/eventbridge-broker");
11
+ const test_utils_1 = require("../../utilities/test-utils");
11
12
  const adapter_1 = require("../adapter");
12
13
  const base_document_processing_1 = require("../base-document-processing");
13
14
  // Concrete test implementation of BaseDocumentProcessing for CDK Nag testing
@@ -43,12 +44,18 @@ class TestDocumentProcessing extends base_document_processing_1.BaseDocumentProc
43
44
  postProcessingStep() {
44
45
  return undefined;
45
46
  }
47
+ preprocessingStep() {
48
+ return undefined;
49
+ }
50
+ createProcessingWorkflow() {
51
+ return this.createStandardProcessingWorkflow();
52
+ }
46
53
  createStateMachine() {
47
54
  return this.handleStateMachineCreation('test-state-machine');
48
55
  }
49
56
  }
50
57
  // Create app and stack
51
- const app = new aws_cdk_lib_1.App();
58
+ const app = (0, test_utils_1.createTestApp)();
52
59
  const stack = new aws_cdk_lib_1.Stack(app, 'TestStack', {
53
60
  env: {
54
61
  account: '123456789012',
@@ -158,4 +165,4 @@ test('No unsuppressed errors', () => {
158
165
  }
159
166
  expect(errors).toHaveLength(0);
160
167
  });
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"]}
168
+ //# 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 { 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 { createTestApp } from '../../utilities/test-utils';\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  protected preprocessingStep(): DocumentProcessingStepType | undefined {\n    return undefined;\n  }\n\n  protected createProcessingWorkflow() {\n    return this.createStandardProcessingWorkflow();\n  }\n\n  public createStateMachine() {\n    return this.handleStateMachineCreation('test-state-machine');\n  }\n}\n\n// Create app and stack\nconst app = createTestApp();\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,337 @@
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_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
7
+ const aws_stepfunctions_tasks_1 = require("aws-cdk-lib/aws-stepfunctions-tasks");
8
+ const test_utils_1 = require("../../utilities/test-utils");
9
+ const base_document_processing_1 = require("../base-document-processing");
10
+ /**
11
+ * Test implementation WITHOUT preprocessing (backward compatible)
12
+ * This simulates existing implementations that don't use any preprocessing
13
+ */
14
+ class TestDocumentProcessingWithoutPreprocessing 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
+ preprocessingStep() {
29
+ // No preprocessing - backward compatible
30
+ return undefined;
31
+ }
32
+ // No override of preprocessingMetadata() - uses default (empty)
33
+ classificationStep() {
34
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockClassification', {
35
+ lambdaFunction: this.classificationFn,
36
+ resultPath: '$.classificationResult',
37
+ });
38
+ }
39
+ processingStep() {
40
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockProcessing', {
41
+ lambdaFunction: this.processingFn,
42
+ resultPath: '$.processingResult',
43
+ });
44
+ }
45
+ enrichmentStep() {
46
+ return undefined;
47
+ }
48
+ postProcessingStep() {
49
+ return undefined;
50
+ }
51
+ createProcessingWorkflow() {
52
+ return this.createStandardProcessingWorkflow();
53
+ }
54
+ createStateMachine() {
55
+ return this.handleStateMachineCreation('test-state-machine-no-preprocessing');
56
+ }
57
+ }
58
+ /**
59
+ * Test implementation WITH preprocessing and custom metadata
60
+ * This simulates implementations that add their own preprocessing-specific fields
61
+ * (e.g., chunking metadata) without the base class knowing about them
62
+ */
63
+ class TestDocumentProcessingWithCustomMetadata extends base_document_processing_1.BaseDocumentProcessing {
64
+ constructor(scope, id, props) {
65
+ super(scope, id, props);
66
+ this.preprocessingFn = new aws_lambda_1.Function(this, 'PreprocessingFn', {
67
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
68
+ handler: 'index.handler',
69
+ code: aws_lambda_1.Code.fromInline(`
70
+ exports.handler = async () => ({
71
+ customField1: 'value1',
72
+ customField2: { nested: 'data' },
73
+ customField3: true
74
+ });
75
+ `),
76
+ });
77
+ this.classificationFn = new aws_lambda_1.Function(this, 'ClassificationFn', {
78
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
79
+ handler: 'index.handler',
80
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ documentClassification: "TEST" });'),
81
+ });
82
+ this.processingFn = new aws_lambda_1.Function(this, 'ProcessingFn', {
83
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
84
+ handler: 'index.handler',
85
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ result: {} });'),
86
+ });
87
+ }
88
+ preprocessingStep() {
89
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockPreprocessing', {
90
+ lambdaFunction: this.preprocessingFn,
91
+ resultPath: '$.preprocessingResult',
92
+ });
93
+ }
94
+ // Override to add custom metadata fields
95
+ preprocessingMetadata() {
96
+ return {
97
+ CustomField1: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.preprocessingResult.customField1')),
98
+ CustomField2: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.jsonToString(aws_stepfunctions_1.JsonPath.objectAt('$.preprocessingResult.customField2'))),
99
+ CustomField3: aws_stepfunctions_tasks_1.DynamoAttributeValue.booleanFromJsonPath(aws_stepfunctions_1.JsonPath.stringAt('$.preprocessingResult.customField3')),
100
+ };
101
+ }
102
+ classificationStep() {
103
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockClassification', {
104
+ lambdaFunction: this.classificationFn,
105
+ resultPath: '$.classificationResult',
106
+ });
107
+ }
108
+ processingStep() {
109
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockProcessing', {
110
+ lambdaFunction: this.processingFn,
111
+ resultPath: '$.processingResult',
112
+ });
113
+ }
114
+ enrichmentStep() {
115
+ return undefined;
116
+ }
117
+ postProcessingStep() {
118
+ return undefined;
119
+ }
120
+ createProcessingWorkflow() {
121
+ return this.createStandardProcessingWorkflow();
122
+ }
123
+ createStateMachine() {
124
+ return this.handleStateMachineCreation('test-state-machine-with-custom-metadata');
125
+ }
126
+ }
127
+ describe('BaseDocumentProcessing - DynamoDB Schema Extension Mechanism', () => {
128
+ let app;
129
+ let stackWithoutPreprocessing;
130
+ let stackWithCustomMetadata;
131
+ let templateWithoutPreprocessing;
132
+ let templateWithCustomMetadata;
133
+ beforeAll(() => {
134
+ // Use createTestApp() to skip bundling and speed up tests
135
+ app = (0, test_utils_1.createTestApp)();
136
+ // Stack without preprocessing (backward compatible)
137
+ stackWithoutPreprocessing = new aws_cdk_lib_1.Stack(app, 'WithoutPreprocessingStack');
138
+ const constructWithoutPreprocessing = new TestDocumentProcessingWithoutPreprocessing(stackWithoutPreprocessing, 'TestWithoutPreprocessing', {});
139
+ constructWithoutPreprocessing.createStateMachine();
140
+ // Stack with custom metadata
141
+ stackWithCustomMetadata = new aws_cdk_lib_1.Stack(app, 'WithCustomMetadataStack');
142
+ const constructWithCustomMetadata = new TestDocumentProcessingWithCustomMetadata(stackWithCustomMetadata, 'TestWithCustomMetadata', {});
143
+ constructWithCustomMetadata.createStateMachine();
144
+ // Generate templates
145
+ templateWithoutPreprocessing = assertions_1.Template.fromStack(stackWithoutPreprocessing);
146
+ templateWithCustomMetadata = assertions_1.Template.fromStack(stackWithCustomMetadata);
147
+ });
148
+ describe('Backward Compatibility - Documents without preprocessing', () => {
149
+ test('InitMetadata only includes base fields when no preprocessing', () => {
150
+ const stateMachines = templateWithoutPreprocessing.findResources('AWS::StepFunctions::StateMachine');
151
+ const stateMachineKey = Object.keys(stateMachines)[0];
152
+ const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;
153
+ // Parse the definition
154
+ const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));
155
+ // Find the InitMetadataEntry state by name
156
+ const initMetadataState = definition.States.InitMetadataEntry;
157
+ expect(initMetadataState).toBeDefined();
158
+ expect(initMetadataState.Type).toBe('Task');
159
+ // Verify base fields are present
160
+ expect(initMetadataState.Parameters.Item.DocumentId).toBeDefined();
161
+ expect(initMetadataState.Parameters.Item.ContentType).toBeDefined();
162
+ expect(initMetadataState.Parameters.Item.Content).toBeDefined();
163
+ expect(initMetadataState.Parameters.Item.WorkflowStatus).toBeDefined();
164
+ expect(initMetadataState.Parameters.Item.StateMachineExecId).toBeDefined();
165
+ // Verify NO custom fields are present (backward compatibility)
166
+ expect(initMetadataState.Parameters.Item.CustomField1).toBeUndefined();
167
+ expect(initMetadataState.Parameters.Item.CustomField2).toBeUndefined();
168
+ expect(initMetadataState.Parameters.Item.CustomField3).toBeUndefined();
169
+ });
170
+ test('workflow starts with InitMetadata when no preprocessing', () => {
171
+ const stateMachines = templateWithoutPreprocessing.findResources('AWS::StepFunctions::StateMachine');
172
+ const stateMachineKey = Object.keys(stateMachines)[0];
173
+ const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;
174
+ const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));
175
+ // Should start with InitMetadata when no preprocessing
176
+ expect(definition.StartAt).toMatch(/InitMetadata/);
177
+ });
178
+ test('DynamoDB table schema remains unchanged', () => {
179
+ // Verify table is created with same schema as before
180
+ templateWithoutPreprocessing.hasResourceProperties('AWS::DynamoDB::Table', {
181
+ KeySchema: [{
182
+ AttributeName: 'DocumentId',
183
+ KeyType: 'HASH',
184
+ }],
185
+ BillingMode: 'PAY_PER_REQUEST',
186
+ });
187
+ });
188
+ });
189
+ describe('Schema Extension Mechanism - Generic preprocessing metadata', () => {
190
+ test('preprocessingMetadata() hook allows subclasses to extend schema', () => {
191
+ // The key design principle: subclasses can override preprocessingMetadata()
192
+ // to add their own fields to InitMetadata without base class knowing the details
193
+ // Verify the construct with custom metadata was created successfully
194
+ expect(stackWithCustomMetadata).toBeDefined();
195
+ // Verify it has a state machine
196
+ templateWithCustomMetadata.resourceCountIs('AWS::StepFunctions::StateMachine', 1);
197
+ // Verify it has the same DynamoDB table structure as the base implementation
198
+ templateWithCustomMetadata.hasResourceProperties('AWS::DynamoDB::Table', {
199
+ KeySchema: [{
200
+ AttributeName: 'DocumentId',
201
+ KeyType: 'HASH',
202
+ }],
203
+ });
204
+ });
205
+ test('workflow includes preprocessing step when provided', () => {
206
+ const stateMachines = templateWithCustomMetadata.findResources('AWS::StepFunctions::StateMachine');
207
+ const stateMachineKey = Object.keys(stateMachines)[0];
208
+ const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;
209
+ const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));
210
+ // Verify InitMetadata step exists (core functionality)
211
+ expect(definition.States.InitMetadataEntry).toBeDefined();
212
+ // When preprocessing is provided, it should be in the workflow
213
+ // The exact structure depends on the subclass implementation
214
+ expect(definition.States).toBeDefined();
215
+ expect(Object.keys(definition.States).length).toBeGreaterThan(1);
216
+ });
217
+ test('InitMetadata can accept additional fields via preprocessingMetadata()', () => {
218
+ const stateMachines = templateWithCustomMetadata.findResources('AWS::StepFunctions::StateMachine');
219
+ const stateMachineKey = Object.keys(stateMachines)[0];
220
+ const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;
221
+ const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));
222
+ const initMetadataState = definition.States.InitMetadataEntry;
223
+ // Verify base fields are always present
224
+ expect(initMetadataState.Parameters.Item.DocumentId).toBeDefined();
225
+ expect(initMetadataState.Parameters.Item.ContentType).toBeDefined();
226
+ expect(initMetadataState.Parameters.Item.Content).toBeDefined();
227
+ expect(initMetadataState.Parameters.Item.WorkflowStatus).toBeDefined();
228
+ expect(initMetadataState.Parameters.Item.StateMachineExecId).toBeDefined();
229
+ // The mechanism allows adding custom fields (implementation detail of subclass)
230
+ // DynamoDB's schemaless nature means any fields can be added at runtime
231
+ });
232
+ });
233
+ describe('Schema Compatibility', () => {
234
+ test('both implementations use the same DynamoDB table structure', () => {
235
+ // Both should create tables with same partition key
236
+ templateWithoutPreprocessing.hasResourceProperties('AWS::DynamoDB::Table', {
237
+ KeySchema: [{
238
+ AttributeName: 'DocumentId',
239
+ KeyType: 'HASH',
240
+ }],
241
+ });
242
+ templateWithCustomMetadata.hasResourceProperties('AWS::DynamoDB::Table', {
243
+ KeySchema: [{
244
+ AttributeName: 'DocumentId',
245
+ KeyType: 'HASH',
246
+ }],
247
+ });
248
+ });
249
+ test('both implementations create encrypted tables', () => {
250
+ templateWithoutPreprocessing.hasResourceProperties('AWS::DynamoDB::Table', {
251
+ SSESpecification: {
252
+ SSEEnabled: true,
253
+ SSEType: 'KMS',
254
+ },
255
+ });
256
+ templateWithCustomMetadata.hasResourceProperties('AWS::DynamoDB::Table', {
257
+ SSESpecification: {
258
+ SSEEnabled: true,
259
+ SSEType: 'KMS',
260
+ },
261
+ });
262
+ });
263
+ test('both implementations use PAY_PER_REQUEST billing', () => {
264
+ templateWithoutPreprocessing.hasResourceProperties('AWS::DynamoDB::Table', {
265
+ BillingMode: 'PAY_PER_REQUEST',
266
+ });
267
+ templateWithCustomMetadata.hasResourceProperties('AWS::DynamoDB::Table', {
268
+ BillingMode: 'PAY_PER_REQUEST',
269
+ });
270
+ });
271
+ });
272
+ describe('Design Principle: Base class has no knowledge of preprocessing specifics', () => {
273
+ test('base class provides generic extension mechanism via preprocessingMetadata()', () => {
274
+ // This test verifies the design principle:
275
+ // - Base class doesn't know about chunking, validation, or any specific preprocessing
276
+ // - Base class only provides a hook (preprocessingMetadata) for subclasses to extend
277
+ // - Subclasses are responsible for their own metadata fields
278
+ // Verify that without overriding preprocessingMetadata(), workflow works correctly
279
+ const stateMachines = templateWithoutPreprocessing.findResources('AWS::StepFunctions::StateMachine');
280
+ const stateMachineKey = Object.keys(stateMachines)[0];
281
+ const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;
282
+ const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));
283
+ const initMetadataState = definition.States.InitMetadataEntry;
284
+ // Base fields are always present
285
+ expect(initMetadataState.Parameters.Item.DocumentId).toBeDefined();
286
+ expect(initMetadataState.Parameters.Item.ContentType).toBeDefined();
287
+ expect(initMetadataState.Parameters.Item.Content).toBeDefined();
288
+ expect(initMetadataState.Parameters.Item.WorkflowStatus).toBeDefined();
289
+ expect(initMetadataState.Parameters.Item.StateMachineExecId).toBeDefined();
290
+ });
291
+ test('subclasses can extend via preprocessingMetadata() without breaking base functionality', () => {
292
+ // Verify that the construct with custom metadata still has all base functionality
293
+ templateWithCustomMetadata.resourceCountIs('AWS::StepFunctions::StateMachine', 1);
294
+ templateWithCustomMetadata.resourceCountIs('AWS::DynamoDB::Table', 1);
295
+ templateWithCustomMetadata.resourceCountIs('AWS::S3::Bucket', 1);
296
+ // Verify state machine has the same base structure
297
+ const stateMachines = templateWithCustomMetadata.findResources('AWS::StepFunctions::StateMachine');
298
+ const stateMachineKey = Object.keys(stateMachines)[0];
299
+ const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;
300
+ const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));
301
+ // Core states should exist
302
+ expect(definition.States.InitMetadataEntry).toBeDefined();
303
+ expect(definition.States.MockClassification).toBeDefined();
304
+ expect(definition.States.MockProcessing).toBeDefined();
305
+ });
306
+ });
307
+ describe('DynamoDB schemaless nature supports runtime field addition', () => {
308
+ test('DynamoDB table supports any additional fields at runtime', () => {
309
+ // DynamoDB is schemaless, so any fields can be added at runtime
310
+ // This test verifies the table is created correctly and can accept any attributes
311
+ templateWithCustomMetadata.hasResourceProperties('AWS::DynamoDB::Table', {
312
+ KeySchema: [{
313
+ AttributeName: 'DocumentId',
314
+ KeyType: 'HASH',
315
+ }],
316
+ BillingMode: 'PAY_PER_REQUEST',
317
+ });
318
+ // No attribute definitions needed for non-key attributes in DynamoDB
319
+ // Fields like AggregatedResult can be added via DynamoUpdateItem during processing
320
+ });
321
+ test('state machine has permissions to update DynamoDB items', () => {
322
+ // Verify the state machine role has UpdateItem permissions
323
+ // This allows adding fields like AggregatedResult during processing
324
+ const policies = templateWithCustomMetadata.findResources('AWS::IAM::Policy');
325
+ // Find the state machine role policy
326
+ const stateMachinePolicy = Object.values(policies).find((policy) => policy.Properties.PolicyName.includes('StateMachineRole'));
327
+ expect(stateMachinePolicy).toBeDefined();
328
+ // Check that it has DynamoDB UpdateItem permissions
329
+ const statements = stateMachinePolicy.Properties.PolicyDocument.Statement;
330
+ const dynamoStatement = statements.find((stmt) => stmt.Action && ((Array.isArray(stmt.Action) && stmt.Action.includes('dynamodb:UpdateItem')) ||
331
+ stmt.Action === 'dynamodb:UpdateItem'));
332
+ expect(dynamoStatement).toBeDefined();
333
+ expect(dynamoStatement.Effect).toBe('Allow');
334
+ });
335
+ });
336
+ });
337
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-document-processing-schema.test.js","sourceRoot":"","sources":["../../../use-cases/document-processing/tests/base-document-processing-schema.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { App, Stack } from 'aws-cdk-lib';\nimport { Template } from 'aws-cdk-lib/assertions';\nimport { Code, Function, Runtime } from 'aws-cdk-lib/aws-lambda';\nimport { JsonPath } from 'aws-cdk-lib/aws-stepfunctions';\nimport { DynamoAttributeValue, LambdaInvoke } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { createTestApp } from '../../utilities/test-utils';\nimport { BaseDocumentProcessing, DocumentProcessingStepType } from '../base-document-processing';\n\n/**\n * Test implementation WITHOUT preprocessing (backward compatible)\n * This simulates existing implementations that don't use any preprocessing\n */\nclass TestDocumentProcessingWithoutPreprocessing 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 preprocessingStep(): DocumentProcessingStepType | undefined {\n    // No preprocessing - backward compatible\n    return undefined;\n  }\n\n  // No override of preprocessingMetadata() - uses default (empty)\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  protected createProcessingWorkflow() {\n    return this.createStandardProcessingWorkflow();\n  }\n\n  public createStateMachine() {\n    return this.handleStateMachineCreation('test-state-machine-no-preprocessing');\n  }\n}\n\n/**\n * Test implementation WITH preprocessing and custom metadata\n * This simulates implementations that add their own preprocessing-specific fields\n * (e.g., chunking metadata) without the base class knowing about them\n */\nclass TestDocumentProcessingWithCustomMetadata extends BaseDocumentProcessing {\n  private preprocessingFn: Function;\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.preprocessingFn = new Function(this, 'PreprocessingFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline(`\n        exports.handler = async () => ({\n          customField1: 'value1',\n          customField2: { nested: 'data' },\n          customField3: true\n        });\n      `),\n    });\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 preprocessingStep(): DocumentProcessingStepType | undefined {\n    return new LambdaInvoke(this, 'MockPreprocessing', {\n      lambdaFunction: this.preprocessingFn,\n      resultPath: '$.preprocessingResult',\n    });\n  }\n\n  // Override to add custom metadata fields\n  protected preprocessingMetadata(): Record<string, DynamoAttributeValue> {\n    return {\n      CustomField1: DynamoAttributeValue.fromString(JsonPath.stringAt('$.preprocessingResult.customField1')),\n      CustomField2: DynamoAttributeValue.fromString(JsonPath.jsonToString(JsonPath.objectAt('$.preprocessingResult.customField2'))),\n      CustomField3: DynamoAttributeValue.booleanFromJsonPath(JsonPath.stringAt('$.preprocessingResult.customField3')),\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  protected createProcessingWorkflow() {\n    return this.createStandardProcessingWorkflow();\n  }\n\n  public createStateMachine() {\n    return this.handleStateMachineCreation('test-state-machine-with-custom-metadata');\n  }\n}\n\ndescribe('BaseDocumentProcessing - DynamoDB Schema Extension Mechanism', () => {\n  let app: App;\n  let stackWithoutPreprocessing: Stack;\n  let stackWithCustomMetadata: Stack;\n  let templateWithoutPreprocessing: Template;\n  let templateWithCustomMetadata: Template;\n\n  beforeAll(() => {\n    // Use createTestApp() to skip bundling and speed up tests\n    app = createTestApp();\n\n    // Stack without preprocessing (backward compatible)\n    stackWithoutPreprocessing = new Stack(app, 'WithoutPreprocessingStack');\n    const constructWithoutPreprocessing = new TestDocumentProcessingWithoutPreprocessing(\n      stackWithoutPreprocessing,\n      'TestWithoutPreprocessing',\n      {},\n    );\n    constructWithoutPreprocessing.createStateMachine();\n\n    // Stack with custom metadata\n    stackWithCustomMetadata = new Stack(app, 'WithCustomMetadataStack');\n    const constructWithCustomMetadata = new TestDocumentProcessingWithCustomMetadata(\n      stackWithCustomMetadata,\n      'TestWithCustomMetadata',\n      {},\n    );\n    constructWithCustomMetadata.createStateMachine();\n\n    // Generate templates\n    templateWithoutPreprocessing = Template.fromStack(stackWithoutPreprocessing);\n    templateWithCustomMetadata = Template.fromStack(stackWithCustomMetadata);\n  });\n\n  describe('Backward Compatibility - Documents without preprocessing', () => {\n    test('InitMetadata only includes base fields when no preprocessing', () => {\n      const stateMachines = templateWithoutPreprocessing.findResources('AWS::StepFunctions::StateMachine');\n      const stateMachineKey = Object.keys(stateMachines)[0];\n      const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;\n\n      // Parse the definition\n      const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));\n\n      // Find the InitMetadataEntry state by name\n      const initMetadataState = definition.States.InitMetadataEntry;\n\n      expect(initMetadataState).toBeDefined();\n      expect(initMetadataState.Type).toBe('Task');\n\n      // Verify base fields are present\n      expect(initMetadataState.Parameters.Item.DocumentId).toBeDefined();\n      expect(initMetadataState.Parameters.Item.ContentType).toBeDefined();\n      expect(initMetadataState.Parameters.Item.Content).toBeDefined();\n      expect(initMetadataState.Parameters.Item.WorkflowStatus).toBeDefined();\n      expect(initMetadataState.Parameters.Item.StateMachineExecId).toBeDefined();\n\n      // Verify NO custom fields are present (backward compatibility)\n      expect(initMetadataState.Parameters.Item.CustomField1).toBeUndefined();\n      expect(initMetadataState.Parameters.Item.CustomField2).toBeUndefined();\n      expect(initMetadataState.Parameters.Item.CustomField3).toBeUndefined();\n    });\n\n    test('workflow starts with InitMetadata when no preprocessing', () => {\n      const stateMachines = templateWithoutPreprocessing.findResources('AWS::StepFunctions::StateMachine');\n      const stateMachineKey = Object.keys(stateMachines)[0];\n      const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;\n\n      const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));\n\n      // Should start with InitMetadata when no preprocessing\n      expect(definition.StartAt).toMatch(/InitMetadata/);\n    });\n\n    test('DynamoDB table schema remains unchanged', () => {\n      // Verify table is created with same schema as before\n      templateWithoutPreprocessing.hasResourceProperties('AWS::DynamoDB::Table', {\n        KeySchema: [{\n          AttributeName: 'DocumentId',\n          KeyType: 'HASH',\n        }],\n        BillingMode: 'PAY_PER_REQUEST',\n      });\n    });\n  });\n\n  describe('Schema Extension Mechanism - Generic preprocessing metadata', () => {\n    test('preprocessingMetadata() hook allows subclasses to extend schema', () => {\n      // The key design principle: subclasses can override preprocessingMetadata()\n      // to add their own fields to InitMetadata without base class knowing the details\n\n      // Verify the construct with custom metadata was created successfully\n      expect(stackWithCustomMetadata).toBeDefined();\n\n      // Verify it has a state machine\n      templateWithCustomMetadata.resourceCountIs('AWS::StepFunctions::StateMachine', 1);\n\n      // Verify it has the same DynamoDB table structure as the base implementation\n      templateWithCustomMetadata.hasResourceProperties('AWS::DynamoDB::Table', {\n        KeySchema: [{\n          AttributeName: 'DocumentId',\n          KeyType: 'HASH',\n        }],\n      });\n    });\n\n    test('workflow includes preprocessing step when provided', () => {\n      const stateMachines = templateWithCustomMetadata.findResources('AWS::StepFunctions::StateMachine');\n      const stateMachineKey = Object.keys(stateMachines)[0];\n      const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;\n\n      const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));\n\n      // Verify InitMetadata step exists (core functionality)\n      expect(definition.States.InitMetadataEntry).toBeDefined();\n\n      // When preprocessing is provided, it should be in the workflow\n      // The exact structure depends on the subclass implementation\n      expect(definition.States).toBeDefined();\n      expect(Object.keys(definition.States).length).toBeGreaterThan(1);\n    });\n\n    test('InitMetadata can accept additional fields via preprocessingMetadata()', () => {\n      const stateMachines = templateWithCustomMetadata.findResources('AWS::StepFunctions::StateMachine');\n      const stateMachineKey = Object.keys(stateMachines)[0];\n      const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;\n\n      const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));\n\n      const initMetadataState = definition.States.InitMetadataEntry;\n\n      // Verify base fields are always present\n      expect(initMetadataState.Parameters.Item.DocumentId).toBeDefined();\n      expect(initMetadataState.Parameters.Item.ContentType).toBeDefined();\n      expect(initMetadataState.Parameters.Item.Content).toBeDefined();\n      expect(initMetadataState.Parameters.Item.WorkflowStatus).toBeDefined();\n      expect(initMetadataState.Parameters.Item.StateMachineExecId).toBeDefined();\n\n      // The mechanism allows adding custom fields (implementation detail of subclass)\n      // DynamoDB's schemaless nature means any fields can be added at runtime\n    });\n  });\n\n  describe('Schema Compatibility', () => {\n    test('both implementations use the same DynamoDB table structure', () => {\n      // Both should create tables with same partition key\n      templateWithoutPreprocessing.hasResourceProperties('AWS::DynamoDB::Table', {\n        KeySchema: [{\n          AttributeName: 'DocumentId',\n          KeyType: 'HASH',\n        }],\n      });\n\n      templateWithCustomMetadata.hasResourceProperties('AWS::DynamoDB::Table', {\n        KeySchema: [{\n          AttributeName: 'DocumentId',\n          KeyType: 'HASH',\n        }],\n      });\n    });\n\n    test('both implementations create encrypted tables', () => {\n      templateWithoutPreprocessing.hasResourceProperties('AWS::DynamoDB::Table', {\n        SSESpecification: {\n          SSEEnabled: true,\n          SSEType: 'KMS',\n        },\n      });\n\n      templateWithCustomMetadata.hasResourceProperties('AWS::DynamoDB::Table', {\n        SSESpecification: {\n          SSEEnabled: true,\n          SSEType: 'KMS',\n        },\n      });\n    });\n\n    test('both implementations use PAY_PER_REQUEST billing', () => {\n      templateWithoutPreprocessing.hasResourceProperties('AWS::DynamoDB::Table', {\n        BillingMode: 'PAY_PER_REQUEST',\n      });\n\n      templateWithCustomMetadata.hasResourceProperties('AWS::DynamoDB::Table', {\n        BillingMode: 'PAY_PER_REQUEST',\n      });\n    });\n  });\n\n  describe('Design Principle: Base class has no knowledge of preprocessing specifics', () => {\n    test('base class provides generic extension mechanism via preprocessingMetadata()', () => {\n      // This test verifies the design principle:\n      // - Base class doesn't know about chunking, validation, or any specific preprocessing\n      // - Base class only provides a hook (preprocessingMetadata) for subclasses to extend\n      // - Subclasses are responsible for their own metadata fields\n\n      // Verify that without overriding preprocessingMetadata(), workflow works correctly\n      const stateMachines = templateWithoutPreprocessing.findResources('AWS::StepFunctions::StateMachine');\n      const stateMachineKey = Object.keys(stateMachines)[0];\n      const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;\n      const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));\n\n      const initMetadataState = definition.States.InitMetadataEntry;\n\n      // Base fields are always present\n      expect(initMetadataState.Parameters.Item.DocumentId).toBeDefined();\n      expect(initMetadataState.Parameters.Item.ContentType).toBeDefined();\n      expect(initMetadataState.Parameters.Item.Content).toBeDefined();\n      expect(initMetadataState.Parameters.Item.WorkflowStatus).toBeDefined();\n      expect(initMetadataState.Parameters.Item.StateMachineExecId).toBeDefined();\n    });\n\n    test('subclasses can extend via preprocessingMetadata() without breaking base functionality', () => {\n      // Verify that the construct with custom metadata still has all base functionality\n      templateWithCustomMetadata.resourceCountIs('AWS::StepFunctions::StateMachine', 1);\n      templateWithCustomMetadata.resourceCountIs('AWS::DynamoDB::Table', 1);\n      templateWithCustomMetadata.resourceCountIs('AWS::S3::Bucket', 1);\n\n      // Verify state machine has the same base structure\n      const stateMachines = templateWithCustomMetadata.findResources('AWS::StepFunctions::StateMachine');\n      const stateMachineKey = Object.keys(stateMachines)[0];\n      const definitionString = stateMachines[stateMachineKey].Properties.DefinitionString;\n      const definition = JSON.parse(definitionString['Fn::Join'][1].join(''));\n\n      // Core states should exist\n      expect(definition.States.InitMetadataEntry).toBeDefined();\n      expect(definition.States.MockClassification).toBeDefined();\n      expect(definition.States.MockProcessing).toBeDefined();\n    });\n  });\n\n  describe('DynamoDB schemaless nature supports runtime field addition', () => {\n    test('DynamoDB table supports any additional fields at runtime', () => {\n      // DynamoDB is schemaless, so any fields can be added at runtime\n      // This test verifies the table is created correctly and can accept any attributes\n\n      templateWithCustomMetadata.hasResourceProperties('AWS::DynamoDB::Table', {\n        KeySchema: [{\n          AttributeName: 'DocumentId',\n          KeyType: 'HASH',\n        }],\n        BillingMode: 'PAY_PER_REQUEST',\n      });\n\n      // No attribute definitions needed for non-key attributes in DynamoDB\n      // Fields like AggregatedResult can be added via DynamoUpdateItem during processing\n    });\n\n    test('state machine has permissions to update DynamoDB items', () => {\n      // Verify the state machine role has UpdateItem permissions\n      // This allows adding fields like AggregatedResult during processing\n      const policies = templateWithCustomMetadata.findResources('AWS::IAM::Policy');\n\n      // Find the state machine role policy\n      const stateMachinePolicy = Object.values(policies).find((policy: any) =>\n        policy.Properties.PolicyName.includes('StateMachineRole'),\n      );\n\n      expect(stateMachinePolicy).toBeDefined();\n\n      // Check that it has DynamoDB UpdateItem permissions\n      const statements = (stateMachinePolicy as any).Properties.PolicyDocument.Statement;\n      const dynamoStatement = statements.find((stmt: any) =>\n        stmt.Action && (\n          (Array.isArray(stmt.Action) && stmt.Action.includes('dynamodb:UpdateItem')) ||\n          stmt.Action === 'dynamodb:UpdateItem'\n        ),\n      );\n\n      expect(dynamoStatement).toBeDefined();\n      expect(dynamoStatement.Effect).toBe('Allow');\n    });\n  });\n});\n"]}