image-scanner-with-trivy 2.4.0 → 2.4.2

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 CHANGED
@@ -3962,7 +3962,7 @@
3962
3962
  },
3963
3963
  "name": "image-scanner-with-trivy",
3964
3964
  "readme": {
3965
- "markdown": "# image-scanner-with-trivy\n\n## What is\n\nThis is an AWS CDK Construct that allows you to **scan container images with Trivy in CDK deployment layer**.\n\nIf it detects vulnerabilities, it can **block deployments to ECS, Lambda, and other services, or prevent the image from being pushed to ECR**. You can also choose to **receive notifications without failing the deployment**.\n\nScan results and **SBOM (Software Bill of Materials) can be output to S3** for further analysis and compliance reporting.\n\nSince it takes an `imageUri` for ECR as an argument, it can also be used to **simply scan an existing image in the repository**.\n\n## Trivy\n\n[Trivy](https://github.com/aquasecurity/trivy) is a comprehensive and versatile security scanner.\n\n**This library is featured on the ecosystem page of [Trivy's official documentation](https://trivy.dev/docs/latest/ecosystem/ide/#image-scanner-with-trivy-community)!**\n\n## Usage\n\n### Install\n\n```sh\nnpm install image-scanner-with-trivy\n```\n\n### CDK Code\n\n**Note: We recommend using `ImageScannerWithTrivyV2`.** See the [V2 Construct](#v2-construct) section for details and migration guide.\n\nThe following code is a minimal example that scans the image and blocks the ECS deployment if vulnerabilities are detected.\n\n```ts\nimport { ImageScannerWithTrivyV2 } from 'image-scanner-with-trivy';\n\n// Target image to scan\nconst image = new DockerImageAsset(this, 'DockerImage', {\n directory: resolve(__dirname, './'),\n});\n\n// Example of an ECS construct that uses the image\nconst ecs = new YourECSConstruct(this, 'YourECSConstruct', {\n dockerImage: image,\n // ...\n});\n\n// Scan the image before deploying to ECS\nconst imageScanner = new ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // If vulnerabilities are detected, the ECS deployment will be blocked\n // Note: This option only works when `failOnVulnerability` is `true` (default).\n blockConstructs: [ecs],\n});\n```\n\n#### Other Use Cases\n\nFor scanning the image and blocking `ECRDeployment` (copying images from `DockerImageAsset` ECR to another ECR repository) if vulnerabilities are detected:\n\n```ts\nimport { ImageScannerWithTrivyV2 } from 'image-scanner-with-trivy';\n\nconst repository = new Repository(this, 'ImageRepository', {\n removalPolicy: RemovalPolicy.DESTROY,\n autoDeleteImages: true,\n});\n\nconst image = new DockerImageAsset(this, 'DockerImage', {\n directory: resolve(__dirname, './'),\n});\n\nconst ecrDeployment = new ECRDeployment(this, 'DeployImage', {\n src: new DockerImageName(image.imageUri),\n dest: new DockerImageName(`${repository.repositoryUri}:latest`),\n});\n\nconst imageScanner = new ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // If vulnerabilities are detected, the ECRDeployment will be blocked\n blockConstructs: [ecrDeployment],\n});\n```\n\n### Default Log Group\n\nIf you want to use a custom log group for the Scanner Lambda function's default log group, you can specify the `defaultLogGroup` option.\n\nIf you use ImageScannerWithTrivyV2 construct multiple times in the same stack, you have to set the same log group for `defaultLogGroup` for each construct.\nWhen you set different log groups for each construct, a warning message will be displayed.\n\n```ts\nimport { ImageScannerWithTrivyV2 } from 'image-scanner-with-trivy';\n\nconst logGroup = new LogGroup(this, 'LogGroup');\n\nnew ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Specify the log group to use as the default log group for Scanner Lambda.\n defaultLogGroup: logGroup,\n});\n\n// NG example\n// When multiple ImageScannerWithTrivyV2 constructs have different default log groups, a warning will be displayed.\nnew ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivyWithAnotherDefaultLogGroup', {\n imageUri: anotherImage.imageUri,\n repository: anotherImage.repository,\n defaultLogGroup: new LogGroup(this, 'AnotherDefaultLogGroup'), // NG example - different log group from the previous construct\n // defaultLogGroup: logGroup, // OK example - use the same log group for all constructs\n});\n```\n\n### Scan Logs Output\n\nIf you output the scan logs to other than the default log group, you can specify the `scanLogsOutput` option.\n\nThis option is useful when you want to choose where to output the scan logs.\n\nYou can output scan logs to CloudWatch Logs or S3.\n\n#### CloudWatch Logs\n\nYou can output scan logs to a specific CloudWatch Logs log group using `ScanLogsOutput.cloudWatchLogs`.\n\n```ts\nconst scanLogsLogGroup = new LogGroup(this, 'ScanLogsLogGroup');\n\nconst imageScanner = new ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Use `ScanLogsOutput.cloudWatchLogs` method to specify the log group.\n scanLogsOutput: ScanLogsOutput.cloudWatchLogs({ logGroup: scanLogsLogGroup }),\n});\n```\n\n**Note on Large Scan Results**: CloudWatch Logs has a limit of 1 MB per log event. If Trivy scan results exceed this limit, they will be automatically split into multiple log events. Each chunk will be prefixed with `[part X/Y]` to indicate the sequence, ensuring no data loss while staying within CloudWatch Logs quotas. **For large scan results, we recommend using S3 output instead** to avoid fragmentation and make it easier to view complete results.\n\n#### S3\n\nYou can output scan logs to an S3 bucket using `ScanLogsOutput.s3`.\n\n```ts\nconst scanLogsBucket = new Bucket(this, 'ScanLogsBucket');\n\nconst imageScanner = new ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Use `ScanLogsOutput.s3` method to specify the S3 bucket.\n scanLogsOutput: ScanLogsOutput.s3({\n bucket: scanLogsBucket,\n prefix: 'scan-logs/', // Optional: specify a prefix for S3 objects\n }),\n});\n```\n\nAdditionally, you can output SBOM (Software Bill of Materials) in various formats by specifying the `sbomFormat` option.\n\nAvailable SBOM formats:\n\n- `SbomFormat.CYCLONEDX` - CycloneDX JSON format\n- `SbomFormat.SPDX_JSON` - SPDX JSON format\n- `SbomFormat.SPDX` - SPDX Tag-Value format (human-readable)\n\n**Important**: SBOM generation is not a vulnerability scan. When `sbomFormat` is specified:\n\n- Trivy generates a Software Bill of Materials (SBOM) instead of performing a vulnerability scan\n- The scan will not fail regardless of the `failOnVulnerability` setting\n- SNS notifications (`vulnsNotificationTopic`) will not be sent since no vulnerabilities are detected\n- The SBOM file and stderr logs will be uploaded to S3\n\n```ts\nconst scanLogsBucket = new Bucket(this, 'ScanLogsBucket');\n\nconst imageScanner = new ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Use `ScanLogsOutput.s3` method to specify the S3 bucket.\n scanLogsOutput: ScanLogsOutput.s3({\n bucket: scanLogsBucket,\n sbomFormat: SbomFormat.CYCLONEDX, // Optional: output SBOM in CycloneDX format\n }),\n});\n```\n\n### SNS Notification for Vulnerabilities\n\nYou can configure an SNS topic to receive notifications when vulnerabilities or EOL (End of Life) OS are detected.\n\nThe notification is sent **regardless of the `failOnVulnerability` setting**. This means you can receive notifications even when you don't want the deployment to fail.\n\n```ts\nconst notificationTopic = new Topic(this, 'VulnerabilityNotificationTopic');\n\nnew ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Receive notifications for vulnerabilities and EOL detection\n vulnsNotificationTopic: notificationTopic,\n // You can choose not to fail the deployment while still receiving notifications\n failOnVulnerability: false,\n});\n```\n\nYou can specify an SNS topic associated with AWS Chatbot, as notifications are sent in AWS Chatbot message format.\n\n### Rollback Error Suppression\n\nBy default, the `suppressErrorOnRollback` property is set to `true`.\n\nWhen image scanning fails, CloudFormation triggers a rollback and executes the previous version\nof the scanner Lambda. If this property is set to `true`, the previous version of the scanner\nLambda will not throw an error, even if the image scanning for the previous version fails.\n\nThis allows the rollback to complete successfully, avoiding ROLLBACK_FAILED state\nwhen image scanning failures occur.\n\n```ts\nnew ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Default is true - suppress errors during rollback to prevent ROLLBACK_FAILED\n suppressErrorOnRollback: true,\n // Set to false if you want rollback errors to be thrown\n suppressErrorOnRollback: false,\n});\n```\n\n## V2 Construct\n\n### What's changed in V2?\n\nThe `ImageScannerWithTrivyV2` construct introduces several API improvements and new features while maintaining the same core functionality:\n\n1. **Improved API Design**\n - `exitCode` and `exitOnEol` → `failOnVulnerability` (boolean): More intuitive boolean property to control whether to fail on vulnerabilities or EOL\n - `platform` → `targetImagePlatform`: Uses the new `TargetImagePlatform` class for better type safety\n\n2. **Improved Trivy Ignore Configuration**\n - New `TrivyIgnore` class for more flexible ignore configuration\n - Support for both inline rules (`TrivyIgnore.fromRules()`) and file paths (`TrivyIgnore.fromFilePath()`)\n - Support for both `.trivyignore` and `.trivyignore.yaml` formats\n\n3. **Enhanced Log Management**\n - `defaultLogGroup` property for Scanner Lambda's default log group\n - Simplified log configuration by consolidating previous separate properties\n - CloudWatch Logs output now supports separate log streams for stdout and stderr\n\n4. **New Features**\n - S3 support for scan logs output with SBOM format support (`ScanLogsOutput.s3()`)\n - SNS notification support for vulnerabilities (`vulnsNotificationTopic`): Receive notifications even when you don't want the deployment to fail by setting `failOnVulnerability: false`\n - `blockConstructs` property to automatically block dependent constructs on vulnerability detection\n\n### Migration from V1 to V2\n\nTo migrate from V1 to V2, follow these steps:\n\n**Update the import statement**:\n\n```ts\n// Before (V1)\nimport { ImageScannerWithTrivy } from 'image-scanner-with-trivy';\n\n// After (V2)\nimport { ImageScannerWithTrivyV2 } from 'image-scanner-with-trivy';\n```\n\n**Update to new properties**:\n\n```ts\n// Before (V1)\nnew ImageScannerWithTrivy(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n exitCode: 1, // or 0\n exitOnEol: 1, // or 0\n platform: 'linux/amd64',\n trivyIgnore: [\n 'CVE-2021-12345',\n 'CVE-2021-67890',\n ],\n});\n\n// After (V2)\nnew ImageScannerWithTrivyV2(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n failOnVulnerability: true, // or false\n // failOnEol behavior is now included in failOnVulnerability\n targetImagePlatform: TargetImagePlatform.LINUX_AMD64,\n trivyIgnore: TrivyIgnore.fromRules([\n 'CVE-2021-12345',\n 'CVE-2021-67890',\n ]),\n});\n```\n\n**Update log group configuration** (if you were using custom log settings):\n\n```ts\n// Before (V1)\nnew ImageScannerWithTrivy(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n defaultLogGroupRetentionDays: RetentionDays.ONE_WEEK,\n defaultLogGroupRemovalPolicy: RemovalPolicy.DESTROY,\n});\n\n// After (V2)\n// Create a custom log group with retention and removal policy\nconst logGroup = new LogGroup(this, 'DefaultLogGroup', {\n retention: RetentionDays.ONE_WEEK,\n removalPolicy: RemovalPolicy.DESTROY,\n});\n\n// Use `defaultLogGroup` for Scanner Lambda's default log group\nnew ImageScannerWithTrivyV2(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n defaultLogGroup: logGroup,\n});\n```\n\n**Update dependencies using `blockConstructs`** (if you were using `construct.node.addDependency()` to block deployments):\n\n```ts\n// Before (V1)\nconst constructToBlock = new YourConstruct(this, 'YourConstruct');\nconst scanner = new ImageScannerWithTrivy(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n});\n\n// Manually add dependencies to block constructs on vulnerability detection\nconstructToBlock.node.addDependency(scanner);\n\n// After (V2)\nnew ImageScannerWithTrivyV2(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n blockConstructs: [constructToBlock], // Automatically block the construct on vulnerability detection\n});\n\n// constructToBlock.node.addDependency(scanner); // No longer needed - blockConstructs handles this automatically\n```\n\n### Important Notes on Migration\n\n⚠️ **CloudWatch Logs Considerations**\n\nWhen you migrate from V1 to V2, the following log behavior changes occur:\n\n1. **Different Log Group**: V2 uses a new internal Lambda function (with different UUID). As a result, **scan logs will be output to a different log group** than V1.\n\n2. **Previous Log Group Retention**: The V1 log group behavior after migration depends on your `defaultLogGroupRemovalPolicy` setting:\n\n```ts\n// V1 construct configuration\nnew ImageScannerWithTrivy(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n defaultLogGroupRemovalPolicy: RemovalPolicy.DESTROY, // or RETAIN (default)\n});\n```\n\n- If you set `RemovalPolicy.DESTROY`, the V1 log group will be deleted when you remove the V1 construct.\n- If you used the **default** (`RemovalPolicy.RETAIN`), **the V1 log group will be retained** in your AWS account. You'll need to manually delete it if desired, or keep it for log history.\n\nIf you want to preserve your V1 scan logs, make sure to use `RemovalPolicy.RETAIN` or back them up before migration.\n\n## API Reference\n\nAPI Reference is [here](./API.md#api-reference-).\n"
3965
+ "markdown": "# image-scanner-with-trivy\n\n## What is\n\nThis is an AWS CDK Construct that allows you to **scan container images with [Trivy](https://github.com/aquasecurity/trivy) in CDK deployment layer**.\n\nIf it detects vulnerabilities, it can **block deployments** to ECS, Lambda, and other services, or prevent the image from being pushed to the application ECR. You can also choose to receive **notifications without failing the deployment**.\n\n- **Block deployments on vulnerability detection** — works with ECS, Lambda, application ECR push, or any construct\n- **Notify without failing** — get alerts via SNS without blocking deployment. Great for gradual adoption\n- **Scan logs output** — results go to CloudWatch Logs or S3\n- **SBOM generation** — output Software Bill of Materials in CycloneDX or SPDX format to S3 for compliance reporting\n- **Works with any ECR image** — not just images built in the same stack. Scan existing repository images too\n\n**This library is featured on the ecosystem page of [Trivy's official documentation](https://trivy.dev/docs/latest/ecosystem/ide/#image-scanner-with-trivy-community)!**\n\n## Why not ECR scanning?\n\nECR's basic scanning is asynchronous and does not block deployments. ECR Enhanced Scanning (Amazon Inspector) costs money per scan and still does not natively integrate with CloudFormation deployment flow.\n\nThis construct runs Trivy as a CloudFormation Custom Resource during `cdk deploy`. If vulnerabilities are found, **CloudFormation rolls back automatically** — no one can bypass the scan. It gives you **synchronous, deployment-blocking scans at no additional cost**.\n\n## Usage\n\n### Install\n\n```sh\nnpm install image-scanner-with-trivy\n```\n\n### CDK Code\n\n**Note: We recommend using `ImageScannerWithTrivyV2`.** See the [V2 Construct](#v2-construct) section for details and migration guide.\n\nThe following code is a minimal example that scans the image and blocks the ECS deployment if vulnerabilities are detected.\n\n```ts\nimport { ImageScannerWithTrivyV2 } from 'image-scanner-with-trivy';\n\n// Target image to scan\nconst image = new DockerImageAsset(this, 'DockerImage', {\n directory: resolve(__dirname, './'),\n});\n\n// Example of an ECS construct that uses the image\nconst ecs = new YourECSConstruct(this, 'YourECSConstruct', {\n dockerImage: image,\n // ...\n});\n\n// Scan the image before deploying to ECS\nconst imageScanner = new ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // If vulnerabilities are detected, the ECS deployment will be blocked\n // Note: This option only works when `failOnVulnerability` is `true` (default).\n blockConstructs: [ecs],\n});\n```\n\n#### Other Use Cases\n\nFor scanning the image and blocking `ECRDeployment` (copying images from `DockerImageAsset` ECR to another ECR repository) if vulnerabilities are detected:\n\n```ts\nimport { ImageScannerWithTrivyV2 } from 'image-scanner-with-trivy';\n\nconst repository = new Repository(this, 'ImageRepository', {\n removalPolicy: RemovalPolicy.DESTROY,\n autoDeleteImages: true,\n});\n\nconst image = new DockerImageAsset(this, 'DockerImage', {\n directory: resolve(__dirname, './'),\n});\n\nconst ecrDeployment = new ECRDeployment(this, 'DeployImage', {\n src: new DockerImageName(image.imageUri),\n dest: new DockerImageName(`${repository.repositoryUri}:latest`),\n});\n\nconst imageScanner = new ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // If vulnerabilities are detected, the ECRDeployment will be blocked\n blockConstructs: [ecrDeployment],\n});\n```\n\n### Default Log Group\n\nIf you want to use a custom log group for the Scanner Lambda function's default log group, you can specify the `defaultLogGroup` option.\n\nIf you use ImageScannerWithTrivyV2 construct multiple times in the same stack, you have to set the same log group for `defaultLogGroup` for each construct.\nWhen you set different log groups for each construct, a warning message will be displayed.\n\n```ts\nimport { ImageScannerWithTrivyV2 } from 'image-scanner-with-trivy';\n\nconst logGroup = new LogGroup(this, 'LogGroup');\n\nnew ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Specify the log group to use as the default log group for Scanner Lambda.\n defaultLogGroup: logGroup,\n});\n\n// NG example\n// When multiple ImageScannerWithTrivyV2 constructs have different default log groups, a warning will be displayed.\nnew ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivyWithAnotherDefaultLogGroup', {\n imageUri: anotherImage.imageUri,\n repository: anotherImage.repository,\n defaultLogGroup: new LogGroup(this, 'AnotherDefaultLogGroup'), // NG example - different log group from the previous construct\n // defaultLogGroup: logGroup, // OK example - use the same log group for all constructs\n});\n```\n\n### Scan Logs Output\n\nIf you output the scan logs to other than the default log group, you can specify the `scanLogsOutput` option.\n\nThis option is useful when you want to choose where to output the scan logs.\n\nYou can output scan logs to CloudWatch Logs or S3.\n\n#### CloudWatch Logs\n\nYou can output scan logs to a specific CloudWatch Logs log group using `ScanLogsOutput.cloudWatchLogs`.\n\n```ts\nconst scanLogsLogGroup = new LogGroup(this, 'ScanLogsLogGroup');\n\nconst imageScanner = new ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Use `ScanLogsOutput.cloudWatchLogs` method to specify the log group.\n scanLogsOutput: ScanLogsOutput.cloudWatchLogs({ logGroup: scanLogsLogGroup }),\n});\n```\n\n**Note on Large Scan Results**: CloudWatch Logs has a limit of 1 MB per log event. If Trivy scan results exceed this limit, they will be automatically split into multiple log events. Each chunk will be prefixed with `[part X/Y]` to indicate the sequence, ensuring no data loss while staying within CloudWatch Logs quotas. **For large scan results, we recommend using S3 output instead** to avoid fragmentation and make it easier to view complete results.\n\n#### S3\n\nYou can output scan logs to an S3 bucket using `ScanLogsOutput.s3`.\n\n```ts\nconst scanLogsBucket = new Bucket(this, 'ScanLogsBucket');\n\nconst imageScanner = new ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Use `ScanLogsOutput.s3` method to specify the S3 bucket.\n scanLogsOutput: ScanLogsOutput.s3({\n bucket: scanLogsBucket,\n prefix: 'scan-logs/', // Optional: specify a prefix for S3 objects\n }),\n});\n```\n\nAdditionally, you can output SBOM (Software Bill of Materials) in various formats by specifying the `sbomFormat` option.\n\nAvailable SBOM formats:\n\n- `SbomFormat.CYCLONEDX` - CycloneDX JSON format\n- `SbomFormat.SPDX_JSON` - SPDX JSON format\n- `SbomFormat.SPDX` - SPDX Tag-Value format (human-readable)\n\n**Important**: SBOM generation is not a vulnerability scan. When `sbomFormat` is specified:\n\n- Trivy generates a Software Bill of Materials (SBOM) instead of performing a vulnerability scan\n- The scan will not fail regardless of the `failOnVulnerability` setting\n- SNS notifications (`vulnsNotificationTopic`) will not be sent since no vulnerabilities are detected\n- The SBOM file and stderr logs will be uploaded to S3\n\n```ts\nconst scanLogsBucket = new Bucket(this, 'ScanLogsBucket');\n\nconst imageScanner = new ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Use `ScanLogsOutput.s3` method to specify the S3 bucket.\n scanLogsOutput: ScanLogsOutput.s3({\n bucket: scanLogsBucket,\n sbomFormat: SbomFormat.CYCLONEDX, // Optional: output SBOM in CycloneDX format\n }),\n});\n```\n\n### SNS Notification for Vulnerabilities\n\nYou can configure an SNS topic to receive notifications when vulnerabilities or EOL (End of Life) OS are detected.\n\nThe notification is sent **regardless of the `failOnVulnerability` setting**. This means you can receive notifications even when you don't want the deployment to fail.\n\n```ts\nconst notificationTopic = new Topic(this, 'VulnerabilityNotificationTopic');\n\nnew ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Receive notifications for vulnerabilities and EOL detection\n vulnsNotificationTopic: notificationTopic,\n // You can choose not to fail the deployment while still receiving notifications\n failOnVulnerability: false,\n});\n```\n\nYou can specify an SNS topic associated with AWS Chatbot, as notifications are sent in AWS Chatbot message format.\n\n### Rollback Error Suppression\n\nBy default, the `suppressErrorOnRollback` property is set to `true`.\n\nWhen image scanning fails, CloudFormation triggers a rollback and executes the previous version\nof the scanner Lambda. If this property is set to `true`, the previous version of the scanner\nLambda will not throw an error, even if the image scanning for the previous version fails.\n\nThis allows the rollback to complete successfully, avoiding ROLLBACK_FAILED state\nwhen image scanning failures occur.\n\n```ts\nnew ImageScannerWithTrivyV2(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Default is true - suppress errors during rollback to prevent ROLLBACK_FAILED\n suppressErrorOnRollback: true,\n // Set to false if you want rollback errors to be thrown\n suppressErrorOnRollback: false,\n});\n```\n\n## V2 Construct\n\n### What's changed in V2?\n\nThe `ImageScannerWithTrivyV2` construct introduces several API improvements and new features while maintaining the same core functionality:\n\n1. **Improved API Design**\n - `exitCode` and `exitOnEol` → `failOnVulnerability` (boolean): More intuitive boolean property to control whether to fail on vulnerabilities or EOL\n - `platform` → `targetImagePlatform`: Uses the new `TargetImagePlatform` class for better type safety\n\n2. **Improved Trivy Ignore Configuration**\n - New `TrivyIgnore` class for more flexible ignore configuration\n - Support for both inline rules (`TrivyIgnore.fromRules()`) and file paths (`TrivyIgnore.fromFilePath()`)\n - Support for both `.trivyignore` and `.trivyignore.yaml` formats\n\n3. **Enhanced Log Management**\n - `defaultLogGroup` property for Scanner Lambda's default log group\n - Simplified log configuration by consolidating previous separate properties\n - CloudWatch Logs output now supports separate log streams for stdout and stderr\n\n4. **New Features**\n - S3 support for scan logs output with SBOM format support (`ScanLogsOutput.s3()`)\n - SNS notification support for vulnerabilities (`vulnsNotificationTopic`): Receive notifications even when you don't want the deployment to fail by setting `failOnVulnerability: false`\n - `blockConstructs` property to automatically block dependent constructs on vulnerability detection\n\n### Migration from V1 to V2\n\nTo migrate from V1 to V2, follow these steps:\n\n**Update the import statement**:\n\n```ts\n// Before (V1)\nimport { ImageScannerWithTrivy } from 'image-scanner-with-trivy';\n\n// After (V2)\nimport { ImageScannerWithTrivyV2 } from 'image-scanner-with-trivy';\n```\n\n**Update to new properties**:\n\n```ts\n// Before (V1)\nnew ImageScannerWithTrivy(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n exitCode: 1, // or 0\n exitOnEol: 1, // or 0\n platform: 'linux/amd64',\n trivyIgnore: [\n 'CVE-2021-12345',\n 'CVE-2021-67890',\n ],\n});\n\n// After (V2)\nnew ImageScannerWithTrivyV2(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n failOnVulnerability: true, // or false\n // failOnEol behavior is now included in failOnVulnerability\n targetImagePlatform: TargetImagePlatform.LINUX_AMD64,\n trivyIgnore: TrivyIgnore.fromRules([\n 'CVE-2021-12345',\n 'CVE-2021-67890',\n ]),\n});\n```\n\n**Update log group configuration** (if you were using custom log settings):\n\n```ts\n// Before (V1)\nnew ImageScannerWithTrivy(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n defaultLogGroupRetentionDays: RetentionDays.ONE_WEEK,\n defaultLogGroupRemovalPolicy: RemovalPolicy.DESTROY,\n});\n\n// After (V2)\n// Create a custom log group with retention and removal policy\nconst logGroup = new LogGroup(this, 'DefaultLogGroup', {\n retention: RetentionDays.ONE_WEEK,\n removalPolicy: RemovalPolicy.DESTROY,\n});\n\n// Use `defaultLogGroup` for Scanner Lambda's default log group\nnew ImageScannerWithTrivyV2(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n defaultLogGroup: logGroup,\n});\n```\n\n**Update dependencies using `blockConstructs`** (if you were using `construct.node.addDependency()` to block deployments):\n\n```ts\n// Before (V1)\nconst constructToBlock = new YourConstruct(this, 'YourConstruct');\nconst scanner = new ImageScannerWithTrivy(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n});\n\n// Manually add dependencies to block constructs on vulnerability detection\nconstructToBlock.node.addDependency(scanner);\n\n// After (V2)\nnew ImageScannerWithTrivyV2(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n blockConstructs: [constructToBlock], // Automatically block the construct on vulnerability detection\n});\n\n// constructToBlock.node.addDependency(scanner); // No longer needed - blockConstructs handles this automatically\n```\n\n### Important Notes on Migration\n\n⚠️ **CloudWatch Logs Considerations**\n\nWhen you migrate from V1 to V2, the following log behavior changes occur:\n\n1. **Different Log Group**: V2 uses a new internal Lambda function (with different UUID). As a result, **scan logs will be output to a different log group** than V1.\n\n2. **Previous Log Group Retention**: The V1 log group behavior after migration depends on your `defaultLogGroupRemovalPolicy` setting:\n\n```ts\n// V1 construct configuration\nnew ImageScannerWithTrivy(this, 'Scanner', {\n imageUri: image.imageUri,\n repository: image.repository,\n defaultLogGroupRemovalPolicy: RemovalPolicy.DESTROY, // or RETAIN (default)\n});\n```\n\n- If you set `RemovalPolicy.DESTROY`, the V1 log group will be deleted when you remove the V1 construct.\n- If you used the **default** (`RemovalPolicy.RETAIN`), **the V1 log group will be retained** in your AWS account. You'll need to manually delete it if desired, or keep it for log history.\n\nIf you want to preserve your V1 scan logs, make sure to use `RemovalPolicy.RETAIN` or back them up before migration.\n\n## API Reference\n\nAPI Reference is [here](./API.md#api-reference-).\n"
3966
3966
  },
3967
3967
  "repository": {
3968
3968
  "type": "git",
@@ -5508,6 +5508,6 @@
5508
5508
  "symbolId": "src/image-scanner-with-trivy-v2:TrivyIgnoreFileType"
5509
5509
  }
5510
5510
  },
5511
- "version": "2.4.0",
5512
- "fingerprint": "EF/SVlRMgfMGP+2Tq1Qrps4sJDgJvyLVMpC5EI3ycys="
5511
+ "version": "2.4.2",
5512
+ "fingerprint": "ZXp5q+AJ4QwC6QLS1+QUrVkR/HYHvOe3nzjybMuyQTg="
5513
5513
  }
package/README.md CHANGED
@@ -2,19 +2,23 @@
2
2
 
3
3
  ## What is
4
4
 
5
- This is an AWS CDK Construct that allows you to **scan container images with Trivy in CDK deployment layer**.
5
+ This is an AWS CDK Construct that allows you to **scan container images with [Trivy](https://github.com/aquasecurity/trivy) in CDK deployment layer**.
6
6
 
7
- If it detects vulnerabilities, it can **block deployments to ECS, Lambda, and other services, or prevent the image from being pushed to ECR**. You can also choose to **receive notifications without failing the deployment**.
7
+ If it detects vulnerabilities, it can **block deployments** to ECS, Lambda, and other services, or prevent the image from being pushed to the application ECR. You can also choose to receive **notifications without failing the deployment**.
8
8
 
9
- Scan results and **SBOM (Software Bill of Materials) can be output to S3** for further analysis and compliance reporting.
9
+ - **Block deployments on vulnerability detection** works with ECS, Lambda, application ECR push, or any construct
10
+ - **Notify without failing** — get alerts via SNS without blocking deployment. Great for gradual adoption
11
+ - **Scan logs output** — results go to CloudWatch Logs or S3
12
+ - **SBOM generation** — output Software Bill of Materials in CycloneDX or SPDX format to S3 for compliance reporting
13
+ - **Works with any ECR image** — not just images built in the same stack. Scan existing repository images too
10
14
 
11
- Since it takes an `imageUri` for ECR as an argument, it can also be used to **simply scan an existing image in the repository**.
15
+ **This library is featured on the ecosystem page of [Trivy's official documentation](https://trivy.dev/docs/latest/ecosystem/ide/#image-scanner-with-trivy-community)!**
12
16
 
13
- ## Trivy
17
+ ## Why not ECR scanning?
14
18
 
15
- [Trivy](https://github.com/aquasecurity/trivy) is a comprehensive and versatile security scanner.
19
+ ECR's basic scanning is asynchronous and does not block deployments. ECR Enhanced Scanning (Amazon Inspector) costs money per scan and still does not natively integrate with CloudFormation deployment flow.
16
20
 
17
- **This library is featured on the ecosystem page of [Trivy's official documentation](https://trivy.dev/docs/latest/ecosystem/ide/#image-scanner-with-trivy-community)!**
21
+ This construct runs Trivy as a CloudFormation Custom Resource during `cdk deploy`. If vulnerabilities are found, **CloudFormation rolls back automatically** — no one can bypass the scan. It gives you **synchronous, deployment-blocking scans at no additional cost**.
18
22
 
19
23
  ## Usage
20
24
 
@@ -4,6 +4,6 @@ WORKDIR /
4
4
 
5
5
  COPY dist/index.js /var/task/index.js
6
6
 
7
- COPY --from=aquasec/trivy:latest /usr/local/bin/trivy /opt
7
+ COPY --from=ghcr.io/aquasecurity/trivy:0.69.3@sha256:bcc376de8d77cfe086a917230e818dc9f8528e3c852f7b1aff648949b6258d1c /usr/local/bin/trivy /opt
8
8
 
9
9
  CMD ["index.handler"]
@@ -68,7 +68,7 @@ class TrivyIgnore {
68
68
  }
69
69
  exports.TrivyIgnore = TrivyIgnore;
70
70
  _a = JSII_RTTI_SYMBOL_1;
71
- TrivyIgnore[_a] = { fqn: "image-scanner-with-trivy.TrivyIgnore", version: "2.4.0" };
71
+ TrivyIgnore[_a] = { fqn: "image-scanner-with-trivy.TrivyIgnore", version: "2.4.2" };
72
72
  /**
73
73
  * Enum for Target Image Platform
74
74
  */
@@ -87,7 +87,7 @@ class TargetImagePlatform {
87
87
  }
88
88
  exports.TargetImagePlatform = TargetImagePlatform;
89
89
  _b = JSII_RTTI_SYMBOL_1;
90
- TargetImagePlatform[_b] = { fqn: "image-scanner-with-trivy.TargetImagePlatform", version: "2.4.0" };
90
+ TargetImagePlatform[_b] = { fqn: "image-scanner-with-trivy.TargetImagePlatform", version: "2.4.2" };
91
91
  /**
92
92
  * Linux AMD64 platform
93
93
  */
@@ -185,5 +185,5 @@ class ImageScannerWithTrivyV2 extends constructs_1.Construct {
185
185
  }
186
186
  exports.ImageScannerWithTrivyV2 = ImageScannerWithTrivyV2;
187
187
  _c = JSII_RTTI_SYMBOL_1;
188
- ImageScannerWithTrivyV2[_c] = { fqn: "image-scanner-with-trivy.ImageScannerWithTrivyV2", version: "2.4.0" };
188
+ ImageScannerWithTrivyV2[_c] = { fqn: "image-scanner-with-trivy.ImageScannerWithTrivyV2", version: "2.4.2" };
189
189
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-scanner-with-trivy-v2.js","sourceRoot":"","sources":["../src/image-scanner-with-trivy-v2.ts"],"names":[],"mappings":"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","sourcesContent":["import { readFileSync } from 'fs';\nimport { join } from 'path';\nimport {\n  Annotations,\n  Aspects,\n  CustomResource,\n  Duration,\n  IgnoreMode,\n  Size,\n  Stack,\n  Token,\n} from 'aws-cdk-lib';\nimport { IRepository } from 'aws-cdk-lib/aws-ecr';\nimport { Platform } from 'aws-cdk-lib/aws-ecr-assets';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport {\n  Architecture,\n  AssetCode,\n  Handler,\n  Runtime,\n  SingletonFunction,\n} from 'aws-cdk-lib/aws-lambda';\nimport { ILogGroup } from 'aws-cdk-lib/aws-logs';\nimport { ITopic } from 'aws-cdk-lib/aws-sns';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct, IConstruct } from 'constructs';\nimport { ScannerCustomResourceProps } from './custom-resource-props';\nimport { ScanLogsOutput } from './scan-logs-output';\nimport { Severity, Scanners, ImageConfigScanners } from './types';\n\n/**\n * File type for TrivyIgnore file path\n */\nexport enum TrivyIgnoreFileType {\n  /**\n   * .trivyignore file\n   *\n   * @see https://trivy.dev/docs/latest/configuration/filtering/#trivyignore\n   */\n  TRIVYIGNORE = 'TRIVYIGNORE',\n\n  /**\n   * .trivyignore.yaml file\n   *\n   * @see https://trivy.dev/docs/latest/configuration/filtering/#trivyignoreyaml\n   */\n  TRIVYIGNORE_YAML = 'TRIVYIGNORE_YAML',\n}\n\n/**\n * Union-like class for specifying Trivy ignore configuration.\n *\n * You can either specify ignore rules inline, or point to an existing ignore file.\n */\nexport class TrivyIgnore {\n  /**\n   * Specify ignore rules inline (equivalent to writing lines in a .trivyignore file).\n   *\n   * @param rules Each element corresponds to one line in the .trivyignore file.\n   *\n   * @see https://trivy.dev/docs/latest/configuration/filtering/#trivyignore\n   */\n  public static fromRules(rules: string[]): TrivyIgnore {\n    return new TrivyIgnore(rules);\n  }\n\n  /**\n   * Specify the path to an existing trivyignore file.\n   *\n   * @param path Path to the ignore file.\n   * @param fileType File format. Defaults to `TrivyIgnoreFileType.TRIVYIGNORE`.\n   *\n   * @see https://trivy.dev/docs/latest/configuration/filtering/#trivyignore\n   * @see https://trivy.dev/docs/latest/configuration/filtering/#trivyignoreyaml\n   */\n  public static fromFilePath(\n    path: string,\n    fileType: TrivyIgnoreFileType = TrivyIgnoreFileType.TRIVYIGNORE,\n  ): TrivyIgnore {\n    const content = readFileSync(path, 'utf-8');\n    // Pass lines as-is without stripping comments or empty lines\n    // because Trivy itself handles comment lines (starting with `#`) and empty lines when reading the ignore file.\n    return new TrivyIgnore(content.split('\\n'), fileType);\n  }\n\n  private constructor(\n    public readonly rules: string[],\n    public readonly fileType?: TrivyIgnoreFileType,\n  ) {}\n}\n\n/**\n * Enum for Target Image Platform\n */\nexport class TargetImagePlatform {\n  /**\n   * Linux AMD64 platform\n   */\n  public static readonly LINUX_AMD64 = new TargetImagePlatform('linux/amd64');\n\n  /**\n   * Linux ARM64 platform\n   */\n  public static readonly LINUX_ARM64 = new TargetImagePlatform('linux/arm64');\n\n  /**\n   * Custom value for target image platform\n   *\n   * The value should be in the format OS/Architecture for the image, such as `linux/arm64`.\n   */\n  public static custom(value: string): TargetImagePlatform {\n    return new TargetImagePlatform(value);\n  }\n\n  private constructor(public readonly value: string) {}\n}\n\n/**\n * Properties for ImageScannerWithTrivyV2 Construct.\n */\nexport interface ImageScannerWithTrivyV2Props {\n  /**\n   * Image URI for scan target.\n   */\n  readonly imageUri: string;\n\n  /**\n   * Repository including the image URI for scan target.\n   *\n   * Because of grantPull to CustomResourceLambda.\n   */\n  readonly repository: IRepository;\n\n  /**\n   * The unfixed/unfixable vulnerabilities mean that the patch has not yet been provided on their distribution.\n   *\n   * To hide unfixed/unfixable vulnerabilities, you can use the `--ignore-unfixed` flag.\n   *\n   * @default false\n   *\n   * @see https://trivy.dev/docs/latest/scanner/vulnerability/#unfixed-vulnerabilities\n   */\n  readonly ignoreUnfixed?: boolean;\n\n  /**\n   * Severity Selection\n   *\n   * The severity is taken from the selected data source since the severity from vendors is more accurate.\n   * Using CVE-2023-0464 as an example, while it is rated as \"HIGH\" in NVD, Red Hat has marked its 'Impact' as \"Low\". As a result, Trivy will display it as \"Low\".\n   *\n   * The severity depends on the compile option, the default configuration, etc. NVD doesn't know how the vendor distributes the software.\n   * Red Hat evaluates the severity more accurately. That's why Trivy prefers vendor scores over NVD.\n   *\n   * It defaults to `CRITICAL` IN THIS CONSTRUCT for safety in CI/CD, but the default configuration of Trivy is \"CRITICAL,HIGH,MEDIUM,LOW,UNKNOWN\".\n   *\n   * @default [Severity.CRITICAL]\n   *\n   * @see https://trivy.dev/docs/latest/scanner/vulnerability/#severity-selection\n   */\n  readonly severity?: Severity[];\n\n  /**\n   * Enable/Disable Scanners\n   *\n   * You can enable/disable scanners with the `scanners`.\n   *\n   * For example, container image scanning enables vulnerability (VULN) and secret scanners (SECRET) by default.\n   * If you don't need secret scanning, it can be disabled by specifying Scanners.VULN only.\n   *\n   * @default [Security.VULN,Scanners.SECRET]\n   *\n   * @see https://trivy.dev/docs/latest/configuration/others/#enabledisable-scanners\n   */\n  readonly scanners?: Scanners[];\n\n  /**\n   * Enum for ImageConfigScanners\n   *\n   * Container images have configuration. docker inspect and `docker history` show the information according to the configuration.\n   * Trivy scans the configuration of container images for\n   *\n   * - Misconfigurations\n   * - Secrets\n   *\n   * They are disabled by default. You can enable them with `imageConfigScanners`.\n   *\n   * @default []\n   *\n   * @see https://trivy.dev/docs/latest/target/container_image/#container-image-metadata\n   */\n  readonly imageConfigScanners?: ImageConfigScanners[];\n\n  /**\n   * Whether to fail on vulnerabilities or EOL (End of Life) images\n   *\n   * If set to `true`, Trivy exits with a non-zero exit code when vulnerabilities or EOL images are detected.\n   *\n   * If set to `false`, Trivy exits with a zero exit code even when vulnerabilities or EOL images are detected.\n   *\n   * It defaults to `true` IN THIS CONSTRUCT for safety in CI/CD. In the original trivy, it is `false` (exit code 0).\n   *\n   * **Note**: When `sbomFormat` is specified in `scanLogsOutput.s3()`, SBOM generation mode is used instead of\n   * vulnerability scanning. In SBOM mode, Trivy always exits with code 0 regardless of this setting, and\n   * no SNS notifications will be sent even if `vulnsNotificationTopic` is configured.\n   *\n   * @default true\n   *\n   * @see https://trivy.dev/docs/latest/configuration/others/#exit-code\n   */\n  readonly failOnVulnerability?: boolean;\n\n  /**\n   * Ignore rules or ignore file for Trivy.\n   *\n   * Use `TrivyIgnore.fromRules()` to specify inline ignore rules (equivalent to writing lines\n   * in a `.trivyignore` file), or `TrivyIgnore.fromFilePath()` to point to an existing ignore file.\n   *\n   * @default - no ignore rules\n   *\n   * @see https://trivy.dev/docs/latest/configuration/filtering/#trivyignore\n   * @see https://trivy.dev/docs/latest/configuration/filtering/#trivyignoreyaml\n   */\n  readonly trivyIgnore?: TrivyIgnore;\n\n  /**\n   * Memory Size (MB) for Scanner Lambda\n   *\n   * You can specify between `3008` and `10240`.\n   *\n   * If this Construct execution terminates abnormally due to SIGKILL, try a larger size.\n   *\n   * Default value (`3008` MB) is Maximum Lambda memory size for default AWS account without quota limit increase.\n   *\n   * @default 3008\n   */\n  readonly memorySize?: number;\n\n  /**\n   * Scan Image on a specific Architecture and OS\n   *\n   * @default - Trivy loads an image on a `linux/amd64` machine.\n   */\n  readonly targetImagePlatform?: TargetImagePlatform;\n\n  /**\n   * The Scanner Lambda function's default log group\n   *\n   * If you use ImageScannerWithTrivyV2 construct multiple times in the same stack,\n   * you must specify the same log group for each construct.\n   *\n   * See `Default Log Group` section in the README for more details.\n   *\n   * @default - Scanner Lambda creates the default log group(`/aws/lambda/${functionName}`).\n   */\n  readonly defaultLogGroup?: ILogGroup;\n\n  /**\n   * Configuration for scan logs output\n   *\n   * By default, scan logs are output to default log group created by Scanner Lambda.\n   *\n   * Specify this if you want to send scan logs to other than the default log group.\n   *\n   * **Note**: CloudWatch Logs has a 1 MB per log event limit. Large scan results will be\n   * automatically split into multiple events with `[part X/Y]` prefixes. **For large scan\n   * results, we recommend using S3 output** to avoid fragmentation and make it easier to\n   * view complete results.\n   *\n   * @default - scan logs output to `defaultLogGroup` if specified, otherwise to the default\n   * log group created by Scanner Lambda.\n   */\n  readonly scanLogsOutput?: ScanLogsOutput;\n\n  /**\n   * Suppress errors during rollback scanner Lambda execution\n   *\n   * When image scanning fails, CloudFormation triggers a rollback and executes the previous\n   * version of the scanner Lambda. If this property is set to `true`, the previous version of\n   * the scanner Lambda will not throw an error, even if the image scanning for the previous version\n   * fails.\n   *\n   * This allows the rollback to complete successfully, avoiding ROLLBACK_FAILED state\n   * when image scanning failures occur.\n   *\n   * @default true\n   */\n  readonly suppressErrorOnRollback?: boolean;\n\n  /**\n   * SNS topic for vulnerabilities notification\n   *\n   * If specified, an SNS topic notification will be sent when vulnerabilities or EOL (End of Life) OS are detected.\n   *\n   * The notification is sent regardless of the `failOnVulnerability` setting.\n   * This means you can choose to receive notifications even when you don't want the deployment to fail.\n   *\n   * You can specify an SNS topic associated with AWS Chatbot, as notifications are sent in AWS Chatbot message format.\n   *\n   * @default - no notification\n   */\n  readonly vulnsNotificationTopic?: ITopic;\n\n  /**\n   * Constructs to block if vulnerabilities are detected.\n   *\n   * This is equivalent to calling `construct.node.addDependency(imageScanner)` for each construct.\n   *\n   * Note: This option only works when `failOnVulnerability` is `true` (default).\n   * If `failOnVulnerability` is set to `false`, the scanner will not fail on vulnerabilities,\n   * and the specified constructs will not be blocked.\n   *\n   * @default - no constructs to block\n   */\n  readonly blockConstructs?: IConstruct[];\n}\n\n// Maximum Lambda memory size for default AWS account without quota limit increase\nconst DEFAULT_MEMORY_SIZE = 3008;\n\n/**\n * A Construct that scans container images with Trivy.\n * It uses a Lambda function as a Custom Resource provider to run Trivy and scan container images.\n */\nexport class ImageScannerWithTrivyV2 extends Construct {\n  private readonly defaultLogGroup?: ILogGroup;\n\n  constructor(scope: Construct, id: string, props: ImageScannerWithTrivyV2Props) {\n    super(scope, id);\n\n    if (\n      props.memorySize &&\n      !Token.isUnresolved(props.memorySize) &&\n      (props.memorySize < 3008 || props.memorySize > 10240)\n    ) {\n      throw new Error(\n        `You can specify between \\`3008\\` and \\`10240\\` for \\`memorySize\\`, got ${props.memorySize}.`,\n      );\n    }\n\n    this.defaultLogGroup = props.defaultLogGroup;\n    const lambdaPurpose = 'Custom::ImageScannerWithTrivyV2CustomResourceLambda';\n\n    const customResourceLambda = new SingletonFunction(this, 'CustomResourceLambda', {\n      uuid: 'cc3b41b5-4701-d86f-fe24-3a04f4a573f1',\n      lambdaPurpose,\n      runtime: Runtime.FROM_IMAGE,\n      handler: Handler.FROM_IMAGE,\n      code: AssetCode.fromAssetImage(join(__dirname, '../assets/lambda'), {\n        platform: Platform.LINUX_ARM64,\n        ignoreMode: IgnoreMode.DOCKER,\n      }),\n      architecture: Architecture.ARM_64,\n      timeout: Duration.seconds(900),\n      retryAttempts: 0,\n      memorySize: props.memorySize ?? DEFAULT_MEMORY_SIZE,\n      ephemeralStorageSize: Size.gibibytes(10), // for cases that need to update trivy DB: /tmp/trivy/db/trivy.db\n      logGroup: this.defaultLogGroup,\n    });\n\n    props.repository.grantPull(customResourceLambda);\n\n    if (props.vulnsNotificationTopic) {\n      props.vulnsNotificationTopic.grantPublish(customResourceLambda);\n    }\n\n    // Grant CloudFormation DescribeStacks permission for rollback detection when suppressErrorOnRollback is enabled\n    const suppressErrorOnRollback = props.suppressErrorOnRollback ?? true;\n    if (suppressErrorOnRollback) {\n      customResourceLambda.addToRolePolicy(\n        new PolicyStatement({\n          actions: ['cloudformation:DescribeStacks'],\n          resources: [Stack.of(this).stackId],\n        }),\n      );\n    }\n\n    // If multiple ImageScannerWithTrivyV2 constructs in the same stack have different default log groups, add a warning annotation.\n    Aspects.of(Stack.of(this)).add({\n      visit: (node) => {\n        if (\n          node instanceof ImageScannerWithTrivyV2 &&\n          node._defaultLogGroup?.node.path !== this.defaultLogGroup?.node.path\n        ) {\n          Annotations.of(this).addWarningV2(\n            '@image-scanner-with-trivy:duplicateLambdaDefaultLogGroup',\n            \"You have to set the same log group for 'defaultLogGroup' for each ImageScannerWithTrivyV2 construct in the same stack.\",\n          );\n        }\n      },\n    });\n\n    const imageScannerProvider = new Provider(this, 'Provider', {\n      onEventHandler: customResourceLambda,\n    });\n\n    const imageScannerProperties: ScannerCustomResourceProps = {\n      addr: this.node.addr,\n      imageUri: props.imageUri,\n      ignoreUnfixed: String(props.ignoreUnfixed ?? false),\n      severity: props.severity ?? [Severity.CRITICAL],\n      scanners: props.scanners ?? [],\n      imageConfigScanners: props.imageConfigScanners ?? [],\n      failOnVulnerability: String(props.failOnVulnerability ?? true),\n      trivyIgnore: props.trivyIgnore?.rules ?? [],\n      trivyIgnoreFileType: props.trivyIgnore?.fileType,\n      platform: props.targetImagePlatform?.value ?? '',\n      output: props.scanLogsOutput?.bind(customResourceLambda),\n      suppressErrorOnRollback: String(suppressErrorOnRollback),\n      vulnsTopicArn: props.vulnsNotificationTopic?.topicArn,\n      defaultLogGroupName:\n        this.defaultLogGroup?.logGroupName ?? `/aws/lambda/${customResourceLambda.functionName}`,\n    };\n\n    new CustomResource(this, 'Resource', {\n      resourceType: 'Custom::ImageScannerWithTrivyV2',\n      properties: imageScannerProperties,\n      serviceToken: imageScannerProvider.serviceToken,\n    });\n\n    // Block constructs if vulnerabilities are detected\n    props.blockConstructs?.forEach((construct) => {\n      construct.node.addDependency(this);\n    });\n  }\n\n  /** @internal */\n  get _defaultLogGroup(): ILogGroup | undefined {\n    return this.defaultLogGroup;\n  }\n}\n"]}
@@ -135,5 +135,5 @@ class ImageScannerWithTrivy extends constructs_1.Construct {
135
135
  }
136
136
  exports.ImageScannerWithTrivy = ImageScannerWithTrivy;
137
137
  _a = JSII_RTTI_SYMBOL_1;
138
- ImageScannerWithTrivy[_a] = { fqn: "image-scanner-with-trivy.ImageScannerWithTrivy", version: "2.4.0" };
138
+ ImageScannerWithTrivy[_a] = { fqn: "image-scanner-with-trivy.ImageScannerWithTrivy", version: "2.4.2" };
139
139
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-scanner-with-trivy.js","sourceRoot":"","sources":["../src/image-scanner-with-trivy.ts"],"names":[],"mappings":"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","sourcesContent":["import { join } from 'path';\nimport {\n  Annotations,\n  CfnDeletionPolicy,\n  CustomResource,\n  Duration,\n  RemovalPolicy,\n  Size,\n  Stack,\n  Token,\n} from 'aws-cdk-lib';\nimport { IRepository } from 'aws-cdk-lib/aws-ecr';\nimport { Platform } from 'aws-cdk-lib/aws-ecr-assets';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport {\n  Architecture,\n  AssetCode,\n  Handler,\n  Runtime,\n  SingletonFunction,\n} from 'aws-cdk-lib/aws-lambda';\nimport { CfnLogGroup, LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\nimport { ScannerCustomResourceProps } from './custom-resource-props';\nimport { ScanLogsOutput } from './scan-logs-output';\nimport { Severity, Scanners, ImageConfigScanners } from './types';\n\n/**\n * Properties for ImageScannerWithTrivy Construct.\n *\n * @deprecated Use ImageScannerWithTrivyV2Props instead. This will be removed in the next major version.\n */\nexport interface ImageScannerWithTrivyProps {\n  /**\n   * Image URI for scan target.\n   */\n  readonly imageUri: string;\n\n  /**\n   * Repository including the image URI for scan target.\n   *\n   * Because of grantPull to CustomResourceLambda.\n   */\n  readonly repository: IRepository;\n\n  /**\n   * The unfixed/unfixable vulnerabilities mean that the patch has not yet been provided on their distribution.\n   *\n   * To hide unfixed/unfixable vulnerabilities, you can use the `--ignore-unfixed` flag.\n   *\n   * @default false\n   *\n   * @see https://aquasecurity.github.io/trivy/latest/docs/scanner/vulnerability/#unfixed-vulnerabilities\n   */\n  readonly ignoreUnfixed?: boolean;\n\n  /**\n   * Severity Selection\n   *\n   * The severity is taken from the selected data source since the severity from vendors is more accurate.\n   * Using CVE-2023-0464 as an example, while it is rated as \"HIGH\" in NVD, Red Hat has marked its 'Impact' as \"Low\". As a result, Trivy will display it as \"Low\".\n   *\n   * The severity depends on the compile option, the default configuration, etc. NVD doesn't know how the vendor distributes the software.\n   * Red Hat evaluates the severity more accurately. That's why Trivy prefers vendor scores over NVD.\n   *\n   * It defaults to `CRITICAL` IN THIS CONSTRUCT for safety in CI/CD, but the default configuration of Trivy is \"CRITICAL,HIGH,MEDIUM,LOW,UNKNOWN\".\n   *\n   * @default [Severity.CRITICAL]\n   *\n   * @see https://aquasecurity.github.io/trivy/latest/docs/scanner/vulnerability/#severity-selection\n   */\n  readonly severity?: Severity[];\n\n  /**\n   * Enable/Disable Scanners\n   *\n   * You can enable/disable scanners with the `scanners`.\n   *\n   * For example, container image scanning enables vulnerability (VULN) and secret scanners (SECRET) by default.\n   * If you don't need secret scanning, it can be disabled by specifying Scanners.VULN only.\n   *\n   * @default [Security.VULN,Scanners.SECRET]\n   *\n   * @see https://aquasecurity.github.io/trivy/latest/docs/configuration/others/#enabledisable-scanners\n   */\n  readonly scanners?: Scanners[];\n\n  /**\n   * Enum for ImageConfigScanners\n   *\n   * Container images have configuration. docker inspect and `docker history` show the information according to the configuration.\n   * Trivy scans the configuration of container images for\n   *\n   * - Misconfigurations\n   * - Secrets\n   *\n   * They are disabled by default. You can enable them with `imageConfigScanners`.\n   *\n   * @default []\n   *\n   * @see https://aquasecurity.github.io/trivy/latest/docs/target/container_image/#container-image-metadata\n   */\n  readonly imageConfigScanners?: ImageConfigScanners[];\n\n  /**\n   * Exit Code\n   *\n   * Use the `exitCode` option if you want to exit with a non-zero exit code.\n   *\n   * You can specify 0 if you do not want to exit even when vulnerabilities are detected.\n   *\n   * It defaults to 1 IN THIS CONSTRUCT for safety in CI/CD. In the original trivy, it is 0.\n   *\n   * @default 1\n   *\n   * @see https://aquasecurity.github.io/trivy/latest/docs/configuration/others/#exit-code\n   */\n  readonly exitCode?: number;\n\n  /**\n   * Exit on EOL\n   *\n   * Sometimes you may surprisingly get 0 vulnerabilities in an old image:\n   *  - Enabling --ignore-unfixed option while all packages have no fixed versions.\n   *  - Scanning a rather outdated OS (e.g. Ubuntu 10.04).\n   *\n   * An OS at the end of service/life (EOL) usually gets into this situation, which is definitely full of vulnerabilities.\n   * `exitOnEol` can fail scanning on EOL OS with a non-zero code.\n   *\n   * It defaults to 1 IN THIS CONSTRUCT for safety in CI/CD. In the original trivy, it is 0.\n   *\n   * @default 1\n   *\n   * @see https://aquasecurity.github.io/trivy/latest/docs/configuration/others/#exit-on-eol\n   */\n  readonly exitOnEol?: number;\n\n  /**\n   * By Finding IDs\n   *\n   * The ignore rules written to the .trivyignore in trivy.\n   * Put each line you write in the file into one element of the array.\n   *\n   * @example\n   *     $ cat .trivyignore\n   *     # Accept the risk\n   *     CVE-2018-14618\n   *\n   *     # Accept the risk until 2023-01-01\n   *     CVE-2019-14697 exp:2023-01-01\n   *\n   *     # No impact in our settings\n   *     CVE-2019-1543\n   *\n   *     # Ignore misconfigurations\n   *     AVD-DS-0002\n   *\n   *     # Ignore secrets\n   *     generic-unwanted-rule\n   *     aws-account-id\n   *\n   * @default []\n   *\n   * @see https://aquasecurity.github.io/trivy/latest/docs/configuration/filtering/#trivyignore\n   */\n  readonly trivyIgnore?: string[];\n\n  /**\n   * Memory Size (MB) for Scanner Lambda\n   *\n   * You can specify between `3008` and `10240`.\n   *\n   * If this Construct execution terminates abnormally due to SIGKILL, try a larger size.\n   *\n   * Default value (`3008` MB) is Maximum Lambda memory size for default AWS account without quota limit increase.\n   *\n   * @default 3008\n   */\n  readonly memorySize?: number;\n\n  /**\n   * Scan Image on a specific Architecture and OS\n   *\n   * By default, Trivy loads an image on a `linux/amd64` machine.\n   *\n   * To customize this, pass a `platform` argument in the format OS/Architecture for the image, such as `linux/arm64`\n   *\n   * @default -\n   */\n  readonly platform?: string;\n\n  /**\n   * The removal policy to apply to Scanner Lambda's default log group\n   *\n   * If you use ImageScannerWithTrivy construct multiple times in the same stack, you cannot set different removal policies for the default log group.\n   * See `Notes` section in the README for more details.\n   *\n   * @default - Scanner Lambda creates the default log group(`/aws/lambda/${functionName}`).\n   */\n  readonly defaultLogGroupRemovalPolicy?: RemovalPolicy;\n\n  /**\n   * The number of days log events are kept in Scanner Lambda's default log group\n   *\n   * If you use ImageScannerWithTrivy construct multiple times in the same stack, you cannot set different retention days for the default log group.\n   * See `Notes` section in the README for more details.\n   *\n   * @default - Scanner Lambda creates the default log group(`/aws/lambda/${functionName}`) and log events never expire.\n   */\n  readonly defaultLogGroupRetentionDays?: RetentionDays;\n\n  /**\n   * Configuration for scan logs output\n   *\n   * By default, scan logs are output to default log group created by Scanner Lambda.\n   *\n   * Specify this if you want to send scan logs to other than the default log group.\n   *\n   * Currently, only `cloudWatchLogs` is supported.\n   *\n   * @default - scan logs output to default log group created by Scanner Lambda(`/aws/lambda/${functionName}`)\n   */\n  readonly scanLogsOutput?: ScanLogsOutput;\n\n  /**\n   * Suppress errors during rollback scanner Lambda execution\n   *\n   * When image scanning fails, CloudFormation triggers a rollback and executes the previous\n   * version of the scanner Lambda. If this property is set to `true`, the previous version of\n   * the scanner Lambda will not throw an error, even if the image scanning for the previous version\n   * fails.\n   *\n   * This allows the rollback to complete successfully, avoiding ROLLBACK_FAILED state\n   * when image scanning failures occur.\n   *\n   * @default true\n   */\n  readonly suppressErrorOnRollback?: boolean;\n}\n\n// Maximum Lambda memory size for default AWS account without quota limit increase\nconst DEFAULT_MEMORY_SIZE = 3008;\n\n/**\n * A Construct that scans container images with Trivy.\n * It uses a Lambda function as a Custom Resource provider to run Trivy and scan container images.\n *\n * @deprecated Use ImageScannerWithTrivyV2 instead. This will be removed in the next major version.\n */\nexport class ImageScannerWithTrivy extends Construct {\n  constructor(scope: Construct, id: string, props: ImageScannerWithTrivyProps) {\n    super(scope, id);\n\n    if (\n      props.memorySize &&\n      !Token.isUnresolved(props.memorySize) &&\n      (props.memorySize < 3008 || props.memorySize > 10240)\n    ) {\n      throw new Error(\n        `You can specify between \\`3008\\` and \\`10240\\` for \\`memorySize\\`, got ${props.memorySize}.`,\n      );\n    }\n\n    const lambdaPurpose = 'Custom::ImageScannerWithTrivyCustomResourceLambda';\n    const customResourceLambda = new SingletonFunction(this, 'CustomResourceLambda', {\n      uuid: '470b6343-d267-f753-226c-1e99f09f319a',\n      lambdaPurpose,\n      runtime: Runtime.FROM_IMAGE,\n      handler: Handler.FROM_IMAGE,\n      code: AssetCode.fromAssetImage(join(__dirname, '../assets/lambda'), {\n        platform: Platform.LINUX_ARM64,\n        // exclude node_modules\n        // because the native binary of the installed esbuild changes depending on the cpu architecture\n        // and the hash value of the image asset changes depending on the execution environment.\n        exclude: ['node_modules'],\n      }),\n      architecture: Architecture.ARM_64,\n      timeout: Duration.seconds(900),\n      retryAttempts: 0,\n      memorySize: props.memorySize ?? DEFAULT_MEMORY_SIZE,\n      ephemeralStorageSize: Size.gibibytes(10), // for cases that need to update trivy DB: /tmp/trivy/db/trivy.db\n    });\n    props.repository.grantPull(customResourceLambda);\n\n    // Grant CloudFormation DescribeStacks permission for rollback detection when suppressErrorOnRollback is enabled\n    const suppressErrorOnRollback = props.suppressErrorOnRollback ?? true;\n    if (suppressErrorOnRollback) {\n      customResourceLambda.addToRolePolicy(\n        new PolicyStatement({\n          actions: ['cloudformation:DescribeStacks'],\n          resources: [Stack.of(this).stackId],\n        }),\n      );\n    }\n\n    const customResourceLambdaLogGroupConstructName = `DefaultLogGroupFor${lambdaPurpose}`;\n\n    this.validateLambdaDefaultLogGroupOptions(customResourceLambdaLogGroupConstructName, props);\n\n    if (props.defaultLogGroupRemovalPolicy || props.defaultLogGroupRetentionDays) {\n      this.ensureLambdaDefaultLogGroup(\n        customResourceLambda,\n        customResourceLambdaLogGroupConstructName,\n        props,\n      );\n    }\n\n    const imageScannerProvider = new Provider(this, 'Provider', {\n      onEventHandler: customResourceLambda,\n    });\n\n    const imageScannerProperties: ScannerCustomResourceProps = {\n      addr: this.node.addr,\n      imageUri: props.imageUri,\n      ignoreUnfixed: String(props.ignoreUnfixed ?? false),\n      severity: props.severity ?? [Severity.CRITICAL],\n      scanners: props.scanners ?? [],\n      imageConfigScanners: props.imageConfigScanners ?? [],\n      exitCode: props.exitCode ?? 1, // TODO: Remove exitCode and exitOnEol properties in the next major version, as they are now controlled by failOnVulnerability.\n      exitOnEol: props.exitOnEol ?? 1, // TODO: Remove exitCode and exitOnEol properties in the next major version, as they are now controlled by failOnVulnerability.\n      failOnVulnerability: 'true', // dummy value for v2 property\n      trivyIgnore: props.trivyIgnore ?? [],\n      platform: props.platform ?? '',\n      output: props.scanLogsOutput?.bind(customResourceLambda),\n      suppressErrorOnRollback: String(suppressErrorOnRollback),\n      defaultLogGroupName: `/aws/lambda/${customResourceLambda.functionName}`,\n    };\n\n    new CustomResource(this, 'Resource', {\n      resourceType: 'Custom::ImageScannerWithTrivy',\n      properties: imageScannerProperties,\n      serviceToken: imageScannerProvider.serviceToken,\n    });\n  }\n\n  /**\n   * Validates that specified default log group options are the same for existing default log group.\n   */\n  private validateLambdaDefaultLogGroupOptions(\n    logGroupConstructName: string,\n    props: ImageScannerWithTrivyProps,\n  ): void {\n    const existing = Stack.of(this).node.tryFindChild(logGroupConstructName) as\n      | LogGroup\n      | undefined;\n    if (!existing) return;\n\n    const cfnLogGroup = existing.node.defaultChild as CfnLogGroup;\n\n    if (\n      !this.isSameResourceDeletionBehavior(\n        props.defaultLogGroupRemovalPolicy,\n        cfnLogGroup.cfnOptions.deletionPolicy,\n      ) ||\n      cfnLogGroup.retentionInDays !== props.defaultLogGroupRetentionDays\n    ) {\n      Annotations.of(this).addWarningV2(\n        '@image-scanner-with-trivy:duplicateLambdaDefaultLogGroupOptions',\n        \"You have to set the same values for 'defaultLogGroupRemovalPolicy' and 'defaultLogGroupRetentionDays' for each ImageScannerWithTrivy construct in the same stack.\",\n      );\n    }\n  }\n\n  /**\n   * Creates the default log group for Scanner Lambda if it does not exist.\n   *\n   * This method checks if the default log group for Scanner Lambda exists in children of the stack construct.\n   * If it does not exist, it creates the default log group for Scanner Lambda as a child of the stack construct.\n   */\n  private ensureLambdaDefaultLogGroup(\n    singletonFunction: SingletonFunction,\n    logGroupConstructName: string,\n    props: ImageScannerWithTrivyProps,\n  ): LogGroup {\n    const existing = Stack.of(this).node.tryFindChild(logGroupConstructName) as\n      | LogGroup\n      | undefined;\n    if (existing) {\n      return existing;\n    }\n\n    return new LogGroup(Stack.of(this), logGroupConstructName, {\n      logGroupName: `/aws/lambda/${singletonFunction.functionName}`,\n      retention: props.defaultLogGroupRetentionDays,\n      removalPolicy: props.defaultLogGroupRemovalPolicy,\n    });\n  }\n\n  private isSameResourceDeletionBehavior(\n    removalPolicy?: RemovalPolicy,\n    deletionPolicy?: CfnDeletionPolicy,\n  ): boolean {\n    switch (removalPolicy) {\n      case RemovalPolicy.DESTROY:\n        return deletionPolicy === CfnDeletionPolicy.DELETE;\n      case RemovalPolicy.RETAIN:\n        return deletionPolicy === CfnDeletionPolicy.RETAIN;\n      case RemovalPolicy.SNAPSHOT:\n        return deletionPolicy === CfnDeletionPolicy.SNAPSHOT;\n      case RemovalPolicy.RETAIN_ON_UPDATE_OR_DELETE:\n        return deletionPolicy === CfnDeletionPolicy.RETAIN_EXCEPT_ON_CREATE;\n      case undefined:\n        return deletionPolicy === undefined;\n      default:\n        return removalPolicy satisfies never;\n    }\n  }\n}\n"]}
@@ -61,7 +61,7 @@ class ScanLogsOutput {
61
61
  }
62
62
  exports.ScanLogsOutput = ScanLogsOutput;
63
63
  _a = JSII_RTTI_SYMBOL_1;
64
- ScanLogsOutput[_a] = { fqn: "image-scanner-with-trivy.ScanLogsOutput", version: "2.4.0" };
64
+ ScanLogsOutput[_a] = { fqn: "image-scanner-with-trivy.ScanLogsOutput", version: "2.4.2" };
65
65
  class CloudWatchLogsOutput extends ScanLogsOutput {
66
66
  constructor(options) {
67
67
  super();
package/package.json CHANGED
@@ -60,7 +60,7 @@
60
60
  "publishConfig": {
61
61
  "access": "public"
62
62
  },
63
- "version": "2.4.0",
63
+ "version": "2.4.2",
64
64
  "types": "lib/index.d.ts",
65
65
  "stability": "stable",
66
66
  "jsii": {
package/renovate.json ADDED
@@ -0,0 +1,12 @@
1
+ {
2
+ "$schema": "https://docs.renovatebot.com/renovate-schema.json",
3
+ "extends": ["config:recommended"],
4
+ "packageRules": [
5
+ {
6
+ "matchDatasources": ["docker"],
7
+ "matchPackageNames": ["ghcr.io/aquasecurity/trivy"],
8
+ "stabilityDays": 3,
9
+ "internalChecksFilter": "strict"
10
+ }
11
+ ]
12
+ }