image-scanner-with-trivy 1.3.0 → 1.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/.jsii CHANGED
@@ -3518,7 +3518,7 @@
3518
3518
  },
3519
3519
  "name": "image-scanner-with-trivy",
3520
3520
  "readme": {
3521
- "markdown": "# image-scanner-with-trivy\n\n## Detail Pages\n\nThe detail blog is [here](https://dev.to/aws-builders/container-image-scanning-with-trivy-in-aws-cdk-151h).\n\nTo my surprise, this library was featured on the ecosystem page of [Trivy's official documentation](https://aquasecurity.github.io/trivy/latest/ecosystem/ide/#image-scanner-with-trivy-community)!\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 **prevent the image from being pushed to the ECR for the application**.\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## Usage\n\n- Install\n\n```sh\nnpm install image-scanner-with-trivy\n```\n\n- CDK Code\n\n```ts\nimport { ImageScannerWithTrivy, ScanLogsOutput } 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\n// Add properties you want for trivy options (ignoreUnfixed, severity, scanners, trivyIgnore, etc).\nconst imageScanner = new ImageScannerWithTrivy(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // If you output the scan logs to other than the default log group, you can specify this option.\n scanLogsOutput: ScanLogsOutput.cloudWatchLogs({ logGroup: new LogGroup(this, 'LogGroup') }),\n});\n\n// By adding `addDependency`, if the vulnerabilities are detected by `ImageScannerWithTrivy`, the following `ECRDeployment` will not be executed, deployment will fail.\nconst ecrDeployment = new ECRDeployment(this, 'DeployImage', {\n src: new DockerImageName(image.imageUri),\n dest: new DockerImageName(`${repository.repositoryUri}:latest`),\n});\necrDeployment.node.addDependency(imageScanner);\n```\n\n## API Reference\n\nAPI Reference is [here](./API.md#api-reference-).\n"
3521
+ "markdown": "# image-scanner-with-trivy\n\n## Detail Pages\n\nThe detail blog is [here](https://dev.to/aws-builders/container-image-scanning-with-trivy-in-aws-cdk-151h).\n\nTo my surprise, this library was featured on the ecosystem page of [Trivy's official documentation](https://aquasecurity.github.io/trivy/latest/ecosystem/ide/#image-scanner-with-trivy-community)!\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 **prevent the image from being pushed to the ECR for the application**.\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## Usage\n\n### Install\n\n```sh\nnpm install image-scanner-with-trivy\n```\n\n### CDK Code\n\nThe following code is a minimal example.\n\n```ts\nimport { ImageScannerWithTrivy } 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\n// Add properties you want for trivy options (ignoreUnfixed, severity, scanners, trivyIgnore, etc).\nconst imageScanner = new ImageScannerWithTrivy(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n});\n\n// By adding `addDependency`, if the vulnerabilities are detected by `ImageScannerWithTrivy`, the following `ECRDeployment` will not be executed, deployment will fail.\nconst ecrDeployment = new ECRDeployment(this, 'DeployImage', {\n src: new DockerImageName(image.imageUri),\n dest: new DockerImageName(`${repository.repositoryUri}:latest`),\n});\necrDeployment.node.addDependency(imageScanner);\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\nCurrently, CloudWatch Logs is only supported as an output destination.\n\n```ts\nimport { ImageScannerWithTrivy, ScanLogsOutput } 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 imageScanner = new ImageScannerWithTrivy(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: new LogGroup(this, 'LogGroup') }),\n});\n```\n\n### Default Log Group\n\nIf you customize the default log group for Scanner Lambda, you can specify the `defaultLogGroupRemovalPolicy` and `defaultLogGroupRetentionDays` options.\nCurrently, only changing the removal policy and retention days are supported.\n\nIf the default log group is already created in your AWS Account and you specify the `defaultLogGroupRemovalPolicy` and `defaultLogGroupRetentionDays` options, the deployment will fail because of a conflict with the log group name.\n\n```ts\nimport { ImageScannerWithTrivy } 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\nnew ImageScannerWithTrivy(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // Change the default log group removal policy to `Destroy`.\n defaultLogGroupRemovalPolicy: RemovalPolicy.DESTROY,\n // Change the default log group retention days to `One Year`.\n defaultLogGroupRetentionDays: RetentionDays.ONE_YEAR,\n});\n```\n\nIf you use ImageScannerWithTrivy construct multiple times in the same stack, you have to set the same values for `defaultLogGroupRemovalPolicy` and `defaultLogGroupRetentionDays` for each construct.\nWhen you set the different values for each construct, the first one will be applied to all ImageScannerWithTrivy constructs in the same stack and warning message will be displayed.\n\nThe following code will produce warning message because of the different values of `defaultLogGroupRemovalPolicy` and `defaultLogGroupRetentionDays` for each construct.\n\n```ts\nimport { ImageScannerWithTrivy, ScanLogsOutput } 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\nnew ImageScannerWithTrivy(this, 'ImageScannerWithTrivy', {\n imageUri: image.imageUri,\n repository: image.repository,\n // The following options are applied to all ImageScannerWithTrivy constructs in the same stack.\n defaultLogGroupRemovalPolicy: RemovalPolicy.DESTROY,\n defaultLogGroupRetentionDays: RetentionDays.ONE_YEAR,\n});\n\n// NG example\n// Once you specify the defaultLogGroupRemovalPolicy and defaultLogGroupRetentionDays, you have to set the same values for each construct.\nnew ImageScannerWithTrivy(this, 'ImageScannerWithTrivyWithDifferentDefaultLogGroupOptions', {\n imageUri: image.imageUri,\n repository: image.repository,\n // The following options are different from the above construct, and warning message will be displayed when synthesizing the stack.\n defaultLogGroupRemovalPolicy: RemovalPolicy.RETAIN, // This should be `RemovalPolicy.DESTROY` as the above construct.\n defaultLogGroupRetentionDays: RetentionDays.ONE_MONTH, // This should be `RetentionDays.ONE_YEAR` as the above construct.\n});\nnew ImageScannerWithTrivy(this, 'ImageScannerWithTrivyWithNoDefaultLogGroupOptions', {\n imageUri: image.imageUri,\n repository: image.repository,\n // You should specify the defaultLogGroupRemovalPolicy and defaultLogGroupRetentionDays if you have already set the values.\n});\n```\n\n## API Reference\n\nAPI Reference is [here](./API.md#api-reference-).\n"
3522
3522
  },
3523
3523
  "repository": {
3524
3524
  "type": "git",
@@ -3578,7 +3578,7 @@
3578
3578
  "kind": "interface",
3579
3579
  "locationInModule": {
3580
3580
  "filename": "src/image-scanner-with-trivy.ts",
3581
- "line": 61
3581
+ "line": 70
3582
3582
  },
3583
3583
  "name": "CloudWatchLogsOutputProps",
3584
3584
  "properties": [
@@ -3591,7 +3591,7 @@
3591
3591
  "immutable": true,
3592
3592
  "locationInModule": {
3593
3593
  "filename": "src/image-scanner-with-trivy.ts",
3594
- "line": 65
3594
+ "line": 74
3595
3595
  },
3596
3596
  "name": "logGroup",
3597
3597
  "type": {
@@ -3612,7 +3612,7 @@
3612
3612
  "kind": "enum",
3613
3613
  "locationInModule": {
3614
3614
  "filename": "src/image-scanner-with-trivy.ts",
3615
- "line": 53
3615
+ "line": 62
3616
3616
  },
3617
3617
  "members": [
3618
3618
  {
@@ -3644,7 +3644,7 @@
3644
3644
  },
3645
3645
  "locationInModule": {
3646
3646
  "filename": "src/image-scanner-with-trivy.ts",
3647
- "line": 286
3647
+ "line": 315
3648
3648
  },
3649
3649
  "parameters": [
3650
3650
  {
@@ -3670,7 +3670,7 @@
3670
3670
  "kind": "class",
3671
3671
  "locationInModule": {
3672
3672
  "filename": "src/image-scanner-with-trivy.ts",
3673
- "line": 285
3673
+ "line": 314
3674
3674
  },
3675
3675
  "name": "ImageScannerWithTrivy",
3676
3676
  "symbolId": "src/image-scanner-with-trivy:ImageScannerWithTrivy"
@@ -3685,7 +3685,7 @@
3685
3685
  "kind": "interface",
3686
3686
  "locationInModule": {
3687
3687
  "filename": "src/image-scanner-with-trivy.ts",
3688
- "line": 109
3688
+ "line": 118
3689
3689
  },
3690
3690
  "name": "ImageScannerWithTrivyProps",
3691
3691
  "properties": [
@@ -3698,7 +3698,7 @@
3698
3698
  "immutable": true,
3699
3699
  "locationInModule": {
3700
3700
  "filename": "src/image-scanner-with-trivy.ts",
3701
- "line": 113
3701
+ "line": 122
3702
3702
  },
3703
3703
  "name": "imageUri",
3704
3704
  "type": {
@@ -3715,13 +3715,51 @@
3715
3715
  "immutable": true,
3716
3716
  "locationInModule": {
3717
3717
  "filename": "src/image-scanner-with-trivy.ts",
3718
- "line": 120
3718
+ "line": 129
3719
3719
  },
3720
3720
  "name": "repository",
3721
3721
  "type": {
3722
3722
  "fqn": "aws-cdk-lib.aws_ecr.IRepository"
3723
3723
  }
3724
3724
  },
3725
+ {
3726
+ "abstract": true,
3727
+ "docs": {
3728
+ "default": "- Scanner Lambda creates the default log group(`/aws/lambda/${functionName}`).",
3729
+ "remarks": "If you use ImageScannerWithTrivy construct multiple times in the same stack, you cannot set different removal policies for the default log group.\nSee `Notes` section in the README for more details.",
3730
+ "stability": "stable",
3731
+ "summary": "The removal policy to apply to Scanner Lambda's default log group."
3732
+ },
3733
+ "immutable": true,
3734
+ "locationInModule": {
3735
+ "filename": "src/image-scanner-with-trivy.ts",
3736
+ "line": 285
3737
+ },
3738
+ "name": "defaultLogGroupRemovalPolicy",
3739
+ "optional": true,
3740
+ "type": {
3741
+ "fqn": "aws-cdk-lib.RemovalPolicy"
3742
+ }
3743
+ },
3744
+ {
3745
+ "abstract": true,
3746
+ "docs": {
3747
+ "default": "- Scanner Lambda creates the default log group(`/aws/lambda/${functionName}`) and log events never expire.",
3748
+ "remarks": "If you use ImageScannerWithTrivy construct multiple times in the same stack, you cannot set different retention days for the default log group.\nSee `Notes` section in the README for more details.",
3749
+ "stability": "stable",
3750
+ "summary": "The number of days log events are kept in Scanner Lambda's default log group."
3751
+ },
3752
+ "immutable": true,
3753
+ "locationInModule": {
3754
+ "filename": "src/image-scanner-with-trivy.ts",
3755
+ "line": 295
3756
+ },
3757
+ "name": "defaultLogGroupRetentionDays",
3758
+ "optional": true,
3759
+ "type": {
3760
+ "fqn": "aws-cdk-lib.aws_logs.RetentionDays"
3761
+ }
3762
+ },
3725
3763
  {
3726
3764
  "abstract": true,
3727
3765
  "docs": {
@@ -3734,7 +3772,7 @@
3734
3772
  "immutable": true,
3735
3773
  "locationInModule": {
3736
3774
  "filename": "src/image-scanner-with-trivy.ts",
3737
- "line": 194
3775
+ "line": 203
3738
3776
  },
3739
3777
  "name": "exitCode",
3740
3778
  "optional": true,
@@ -3754,7 +3792,7 @@
3754
3792
  "immutable": true,
3755
3793
  "locationInModule": {
3756
3794
  "filename": "src/image-scanner-with-trivy.ts",
3757
- "line": 212
3795
+ "line": 221
3758
3796
  },
3759
3797
  "name": "exitOnEol",
3760
3798
  "optional": true,
@@ -3774,7 +3812,7 @@
3774
3812
  "immutable": true,
3775
3813
  "locationInModule": {
3776
3814
  "filename": "src/image-scanner-with-trivy.ts",
3777
- "line": 131
3815
+ "line": 140
3778
3816
  },
3779
3817
  "name": "ignoreUnfixed",
3780
3818
  "optional": true,
@@ -3794,7 +3832,7 @@
3794
3832
  "immutable": true,
3795
3833
  "locationInModule": {
3796
3834
  "filename": "src/image-scanner-with-trivy.ts",
3797
- "line": 179
3835
+ "line": 188
3798
3836
  },
3799
3837
  "name": "imageConfigScanners",
3800
3838
  "optional": true,
@@ -3818,7 +3856,7 @@
3818
3856
  "immutable": true,
3819
3857
  "locationInModule": {
3820
3858
  "filename": "src/image-scanner-with-trivy.ts",
3821
- "line": 255
3859
+ "line": 264
3822
3860
  },
3823
3861
  "name": "memorySize",
3824
3862
  "optional": true,
@@ -3837,7 +3875,7 @@
3837
3875
  "immutable": true,
3838
3876
  "locationInModule": {
3839
3877
  "filename": "src/image-scanner-with-trivy.ts",
3840
- "line": 266
3878
+ "line": 275
3841
3879
  },
3842
3880
  "name": "platform",
3843
3881
  "optional": true,
@@ -3856,7 +3894,7 @@
3856
3894
  "immutable": true,
3857
3895
  "locationInModule": {
3858
3896
  "filename": "src/image-scanner-with-trivy.ts",
3859
- "line": 279
3897
+ "line": 308
3860
3898
  },
3861
3899
  "name": "scanLogsOutput",
3862
3900
  "optional": true,
@@ -3876,7 +3914,7 @@
3876
3914
  "immutable": true,
3877
3915
  "locationInModule": {
3878
3916
  "filename": "src/image-scanner-with-trivy.ts",
3879
- "line": 162
3917
+ "line": 171
3880
3918
  },
3881
3919
  "name": "scanners",
3882
3920
  "optional": true,
@@ -3901,7 +3939,7 @@
3901
3939
  "immutable": true,
3902
3940
  "locationInModule": {
3903
3941
  "filename": "src/image-scanner-with-trivy.ts",
3904
- "line": 148
3942
+ "line": 157
3905
3943
  },
3906
3944
  "name": "severity",
3907
3945
  "optional": true,
@@ -3927,7 +3965,7 @@
3927
3965
  "immutable": true,
3928
3966
  "locationInModule": {
3929
3967
  "filename": "src/image-scanner-with-trivy.ts",
3930
- "line": 242
3968
+ "line": 251
3931
3969
  },
3932
3970
  "name": "trivyIgnore",
3933
3971
  "optional": true,
@@ -3959,7 +3997,7 @@
3959
3997
  "kind": "class",
3960
3998
  "locationInModule": {
3961
3999
  "filename": "src/image-scanner-with-trivy.ts",
3962
- "line": 71
4000
+ "line": 80
3963
4001
  },
3964
4002
  "methods": [
3965
4003
  {
@@ -3969,7 +4007,7 @@
3969
4007
  },
3970
4008
  "locationInModule": {
3971
4009
  "filename": "src/image-scanner-with-trivy.ts",
3972
- "line": 75
4010
+ "line": 84
3973
4011
  },
3974
4012
  "name": "cloudWatchLogs",
3975
4013
  "parameters": [
@@ -3995,7 +4033,7 @@
3995
4033
  },
3996
4034
  "locationInModule": {
3997
4035
  "filename": "src/image-scanner-with-trivy.ts",
3998
- "line": 82
4036
+ "line": 91
3999
4037
  },
4000
4038
  "name": "bind",
4001
4039
  "parameters": [
@@ -4083,7 +4121,7 @@
4083
4121
  "kind": "enum",
4084
4122
  "locationInModule": {
4085
4123
  "filename": "src/image-scanner-with-trivy.ts",
4086
- "line": 41
4124
+ "line": 50
4087
4125
  },
4088
4126
  "members": [
4089
4127
  {
@@ -4125,7 +4163,7 @@
4125
4163
  "kind": "enum",
4126
4164
  "locationInModule": {
4127
4165
  "filename": "src/image-scanner-with-trivy.ts",
4128
- "line": 28
4166
+ "line": 37
4129
4167
  },
4130
4168
  "members": [
4131
4169
  {
@@ -4163,6 +4201,6 @@
4163
4201
  "symbolId": "src/image-scanner-with-trivy:Severity"
4164
4202
  }
4165
4203
  },
4166
- "version": "1.3.0",
4167
- "fingerprint": "SsLDuO9DJaCExXTtPKO0crxCBjMa4Z22jTkKsBonYVc="
4204
+ "version": "1.4.0",
4205
+ "fingerprint": "GHI0DiXFw/BG1uuUoyMU3ciRMGMifdtqtkPqVrDGmYY="
4168
4206
  }
package/API.md CHANGED
@@ -192,6 +192,8 @@ const imageScannerWithTrivyProps: ImageScannerWithTrivyProps = { ... }
192
192
  | --- | --- | --- |
193
193
  | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.imageUri">imageUri</a></code> | <code>string</code> | Image URI for scan target. |
194
194
  | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.repository">repository</a></code> | <code>aws-cdk-lib.aws_ecr.IRepository</code> | Repository including the image URI for scan target. |
195
+ | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.defaultLogGroupRemovalPolicy">defaultLogGroupRemovalPolicy</a></code> | <code>aws-cdk-lib.RemovalPolicy</code> | The removal policy to apply to Scanner Lambda's default log group. |
196
+ | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.defaultLogGroupRetentionDays">defaultLogGroupRetentionDays</a></code> | <code>aws-cdk-lib.aws_logs.RetentionDays</code> | The number of days log events are kept in Scanner Lambda's default log group. |
195
197
  | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.exitCode">exitCode</a></code> | <code>number</code> | Exit Code. |
196
198
  | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.exitOnEol">exitOnEol</a></code> | <code>number</code> | Exit on EOL. |
197
199
  | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.ignoreUnfixed">ignoreUnfixed</a></code> | <code>boolean</code> | The unfixed/unfixable vulnerabilities mean that the patch has not yet been provided on their distribution. |
@@ -231,6 +233,38 @@ Because of grantPull to CustomResourceLambda.
231
233
 
232
234
  ---
233
235
 
236
+ ##### `defaultLogGroupRemovalPolicy`<sup>Optional</sup> <a name="defaultLogGroupRemovalPolicy" id="image-scanner-with-trivy.ImageScannerWithTrivyProps.property.defaultLogGroupRemovalPolicy"></a>
237
+
238
+ ```typescript
239
+ public readonly defaultLogGroupRemovalPolicy: RemovalPolicy;
240
+ ```
241
+
242
+ - *Type:* aws-cdk-lib.RemovalPolicy
243
+ - *Default:* Scanner Lambda creates the default log group(`/aws/lambda/${functionName}`).
244
+
245
+ The removal policy to apply to Scanner Lambda's default log group.
246
+
247
+ If you use ImageScannerWithTrivy construct multiple times in the same stack, you cannot set different removal policies for the default log group.
248
+ See `Notes` section in the README for more details.
249
+
250
+ ---
251
+
252
+ ##### `defaultLogGroupRetentionDays`<sup>Optional</sup> <a name="defaultLogGroupRetentionDays" id="image-scanner-with-trivy.ImageScannerWithTrivyProps.property.defaultLogGroupRetentionDays"></a>
253
+
254
+ ```typescript
255
+ public readonly defaultLogGroupRetentionDays: RetentionDays;
256
+ ```
257
+
258
+ - *Type:* aws-cdk-lib.aws_logs.RetentionDays
259
+ - *Default:* Scanner Lambda creates the default log group(`/aws/lambda/${functionName}`) and log events never expire.
260
+
261
+ The number of days log events are kept in Scanner Lambda's default log group.
262
+
263
+ If you use ImageScannerWithTrivy construct multiple times in the same stack, you cannot set different retention days for the default log group.
264
+ See `Notes` section in the README for more details.
265
+
266
+ ---
267
+
234
268
  ##### `exitCode`<sup>Optional</sup> <a name="exitCode" id="image-scanner-with-trivy.ImageScannerWithTrivyProps.property.exitCode"></a>
235
269
 
236
270
  ```typescript
package/README.md CHANGED
@@ -20,16 +20,18 @@ Since it takes an `imageUri` for ECR as an argument, it can also be used to **si
20
20
 
21
21
  ## Usage
22
22
 
23
- - Install
23
+ ### Install
24
24
 
25
25
  ```sh
26
26
  npm install image-scanner-with-trivy
27
27
  ```
28
28
 
29
- - CDK Code
29
+ ### CDK Code
30
+
31
+ The following code is a minimal example.
30
32
 
31
33
  ```ts
32
- import { ImageScannerWithTrivy, ScanLogsOutput } from 'image-scanner-with-trivy';
34
+ import { ImageScannerWithTrivy } from 'image-scanner-with-trivy';
33
35
 
34
36
  const repository = new Repository(this, 'ImageRepository', {
35
37
  removalPolicy: RemovalPolicy.DESTROY,
@@ -44,8 +46,6 @@ const image = new DockerImageAsset(this, 'DockerImage', {
44
46
  const imageScanner = new ImageScannerWithTrivy(this, 'ImageScannerWithTrivy', {
45
47
  imageUri: image.imageUri,
46
48
  repository: image.repository,
47
- // If you output the scan logs to other than the default log group, you can specify this option.
48
- scanLogsOutput: ScanLogsOutput.cloudWatchLogs({ logGroup: new LogGroup(this, 'LogGroup') }),
49
49
  });
50
50
 
51
51
  // By adding `addDependency`, if the vulnerabilities are detected by `ImageScannerWithTrivy`, the following `ECRDeployment` will not be executed, deployment will fail.
@@ -56,6 +56,104 @@ const ecrDeployment = new ECRDeployment(this, 'DeployImage', {
56
56
  ecrDeployment.node.addDependency(imageScanner);
57
57
  ```
58
58
 
59
+ ### Scan Logs Output
60
+
61
+ If you output the scan logs to other than the default log group, you can specify the `scanLogsOutput` option.
62
+
63
+ This option is useful when you want to choose where to output the scan logs.
64
+
65
+ Currently, CloudWatch Logs is only supported as an output destination.
66
+
67
+ ```ts
68
+ import { ImageScannerWithTrivy, ScanLogsOutput } from 'image-scanner-with-trivy';
69
+
70
+ const repository = new Repository(this, 'ImageRepository', {
71
+ removalPolicy: RemovalPolicy.DESTROY,
72
+ autoDeleteImages: true,
73
+ });
74
+
75
+ const image = new DockerImageAsset(this, 'DockerImage', {
76
+ directory: resolve(__dirname, './'),
77
+ });
78
+
79
+ const imageScanner = new ImageScannerWithTrivy(this, 'ImageScannerWithTrivy', {
80
+ imageUri: image.imageUri,
81
+ repository: image.repository,
82
+ // Use `ScanLogsOutput.cloudWatchLogs` method to specify the log group.
83
+ scanLogsOutput: ScanLogsOutput.cloudWatchLogs({ logGroup: new LogGroup(this, 'LogGroup') }),
84
+ });
85
+ ```
86
+
87
+ ### Default Log Group
88
+
89
+ If you customize the default log group for Scanner Lambda, you can specify the `defaultLogGroupRemovalPolicy` and `defaultLogGroupRetentionDays` options.
90
+ Currently, only changing the removal policy and retention days are supported.
91
+
92
+ If the default log group is already created in your AWS Account and you specify the `defaultLogGroupRemovalPolicy` and `defaultLogGroupRetentionDays` options, the deployment will fail because of a conflict with the log group name.
93
+
94
+ ```ts
95
+ import { ImageScannerWithTrivy } from 'image-scanner-with-trivy';
96
+
97
+ const repository = new Repository(this, 'ImageRepository', {
98
+ removalPolicy: RemovalPolicy.DESTROY,
99
+ autoDeleteImages: true,
100
+ });
101
+
102
+ const image = new DockerImageAsset(this, 'DockerImage', {
103
+ directory: resolve(__dirname, './'),
104
+ });
105
+
106
+ new ImageScannerWithTrivy(this, 'ImageScannerWithTrivy', {
107
+ imageUri: image.imageUri,
108
+ repository: image.repository,
109
+ // Change the default log group removal policy to `Destroy`.
110
+ defaultLogGroupRemovalPolicy: RemovalPolicy.DESTROY,
111
+ // Change the default log group retention days to `One Year`.
112
+ defaultLogGroupRetentionDays: RetentionDays.ONE_YEAR,
113
+ });
114
+ ```
115
+
116
+ If you use ImageScannerWithTrivy construct multiple times in the same stack, you have to set the same values for `defaultLogGroupRemovalPolicy` and `defaultLogGroupRetentionDays` for each construct.
117
+ When you set the different values for each construct, the first one will be applied to all ImageScannerWithTrivy constructs in the same stack and warning message will be displayed.
118
+
119
+ The following code will produce warning message because of the different values of `defaultLogGroupRemovalPolicy` and `defaultLogGroupRetentionDays` for each construct.
120
+
121
+ ```ts
122
+ import { ImageScannerWithTrivy, ScanLogsOutput } from 'image-scanner-with-trivy';
123
+
124
+ const repository = new Repository(this, 'ImageRepository', {
125
+ removalPolicy: RemovalPolicy.DESTROY,
126
+ autoDeleteImages: true,
127
+ });
128
+
129
+ const image = new DockerImageAsset(this, 'DockerImage', {
130
+ directory: resolve(__dirname, './'),
131
+ });
132
+
133
+ new ImageScannerWithTrivy(this, 'ImageScannerWithTrivy', {
134
+ imageUri: image.imageUri,
135
+ repository: image.repository,
136
+ // The following options are applied to all ImageScannerWithTrivy constructs in the same stack.
137
+ defaultLogGroupRemovalPolicy: RemovalPolicy.DESTROY,
138
+ defaultLogGroupRetentionDays: RetentionDays.ONE_YEAR,
139
+ });
140
+
141
+ // NG example
142
+ // Once you specify the defaultLogGroupRemovalPolicy and defaultLogGroupRetentionDays, you have to set the same values for each construct.
143
+ new ImageScannerWithTrivy(this, 'ImageScannerWithTrivyWithDifferentDefaultLogGroupOptions', {
144
+ imageUri: image.imageUri,
145
+ repository: image.repository,
146
+ // The following options are different from the above construct, and warning message will be displayed when synthesizing the stack.
147
+ defaultLogGroupRemovalPolicy: RemovalPolicy.RETAIN, // This should be `RemovalPolicy.DESTROY` as the above construct.
148
+ defaultLogGroupRetentionDays: RetentionDays.ONE_MONTH, // This should be `RetentionDays.ONE_YEAR` as the above construct.
149
+ });
150
+ new ImageScannerWithTrivy(this, 'ImageScannerWithTrivyWithNoDefaultLogGroupOptions', {
151
+ imageUri: image.imageUri,
152
+ repository: image.repository,
153
+ // You should specify the defaultLogGroupRemovalPolicy and defaultLogGroupRetentionDays if you have already set the values.
154
+ });
155
+ ```
156
+
59
157
  ## API Reference
60
158
 
61
159
  API Reference is [here](./API.md#api-reference-).
@@ -1,6 +1,7 @@
1
+ import { RemovalPolicy } from 'aws-cdk-lib';
1
2
  import { IRepository } from 'aws-cdk-lib/aws-ecr';
2
3
  import { IGrantable } from 'aws-cdk-lib/aws-iam';
3
- import { ILogGroup } from 'aws-cdk-lib/aws-logs';
4
+ import { ILogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';
4
5
  import { Construct } from 'constructs';
5
6
  import { ScanLogsOutputOptions } from './types';
6
7
  /**
@@ -205,6 +206,24 @@ export interface ImageScannerWithTrivyProps {
205
206
  * @default -
206
207
  */
207
208
  readonly platform?: string;
209
+ /**
210
+ * The removal policy to apply to Scanner Lambda's default log group
211
+ *
212
+ * If you use ImageScannerWithTrivy construct multiple times in the same stack, you cannot set different removal policies for the default log group.
213
+ * See `Notes` section in the README for more details.
214
+ *
215
+ * @default - Scanner Lambda creates the default log group(`/aws/lambda/${functionName}`).
216
+ */
217
+ readonly defaultLogGroupRemovalPolicy?: RemovalPolicy;
218
+ /**
219
+ * The number of days log events are kept in Scanner Lambda's default log group
220
+ *
221
+ * If you use ImageScannerWithTrivy construct multiple times in the same stack, you cannot set different retention days for the default log group.
222
+ * See `Notes` section in the README for more details.
223
+ *
224
+ * @default - Scanner Lambda creates the default log group(`/aws/lambda/${functionName}`) and log events never expire.
225
+ */
226
+ readonly defaultLogGroupRetentionDays?: RetentionDays;
208
227
  /**
209
228
  * Configuration for scan logs output
210
229
  *
@@ -220,4 +239,16 @@ export interface ImageScannerWithTrivyProps {
220
239
  }
221
240
  export declare class ImageScannerWithTrivy extends Construct {
222
241
  constructor(scope: Construct, id: string, props: ImageScannerWithTrivyProps);
242
+ /**
243
+ * Validates that specified default log group options are the same for existing default log group.
244
+ */
245
+ private validateLambdaDefaultLogGroupOptions;
246
+ /**
247
+ * Creates the default log group for Scanner Lambda if it does not exist.
248
+ *
249
+ * This method checks if the default log group for Scanner Lambda exists in children of the stack construct.
250
+ * If it does not exist, it creates the default log group for Scanner Lambda as a child of the stack construct.
251
+ */
252
+ private ensureLambdaDefaultLogGroup;
253
+ private isSameResourceDeletionBehavior;
223
254
  }
@@ -7,6 +7,7 @@ const path_1 = require("path");
7
7
  const aws_cdk_lib_1 = require("aws-cdk-lib");
8
8
  const aws_ecr_assets_1 = require("aws-cdk-lib/aws-ecr-assets");
9
9
  const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
10
+ const aws_logs_1 = require("aws-cdk-lib/aws-logs");
10
11
  const custom_resources_1 = require("aws-cdk-lib/custom-resources");
11
12
  const constructs_1 = require("constructs");
12
13
  const types_1 = require("./types");
@@ -57,7 +58,7 @@ class ScanLogsOutput {
57
58
  }
58
59
  }
59
60
  _a = JSII_RTTI_SYMBOL_1;
60
- ScanLogsOutput[_a] = { fqn: "image-scanner-with-trivy.ScanLogsOutput", version: "1.3.0" };
61
+ ScanLogsOutput[_a] = { fqn: "image-scanner-with-trivy.ScanLogsOutput", version: "1.4.0" };
61
62
  exports.ScanLogsOutput = ScanLogsOutput;
62
63
  class CloudWatchLogsOutput extends ScanLogsOutput {
63
64
  constructor(options) {
@@ -84,9 +85,10 @@ class ImageScannerWithTrivy extends constructs_1.Construct {
84
85
  (props.memorySize < 3008 || props.memorySize > 10240)) {
85
86
  throw new Error(`You can specify between \`3008\` and \`10240\` for \`memorySize\`, got ${props.memorySize}.`);
86
87
  }
88
+ const lambdaPurpose = 'Custom::ImageScannerWithTrivyCustomResourceLambda';
87
89
  const customResourceLambda = new aws_lambda_1.SingletonFunction(this, 'CustomResourceLambda', {
88
90
  uuid: '470b6343-d267-f753-226c-1e99f09f319a',
89
- lambdaPurpose: 'Custom::ImageScannerWithTrivyCustomResourceLambda',
91
+ lambdaPurpose,
90
92
  runtime: aws_lambda_1.Runtime.FROM_IMAGE,
91
93
  handler: aws_lambda_1.Handler.FROM_IMAGE,
92
94
  code: aws_lambda_1.AssetCode.fromAssetImage((0, path_1.join)(__dirname, '../assets/lambda'), {
@@ -103,6 +105,11 @@ class ImageScannerWithTrivy extends constructs_1.Construct {
103
105
  ephemeralStorageSize: aws_cdk_lib_1.Size.gibibytes(10), // for cases that need to update trivy DB: /tmp/trivy/db/trivy.db
104
106
  });
105
107
  props.repository.grantPull(customResourceLambda);
108
+ const customResourceLambdaLogGroupConstructName = `DefaultLogGroupFor${lambdaPurpose}`;
109
+ this.validateLambdaDefaultLogGroupOptions(customResourceLambdaLogGroupConstructName, props);
110
+ if (props.defaultLogGroupRemovalPolicy || props.defaultLogGroupRetentionDays) {
111
+ this.ensureLambdaDefaultLogGroup(customResourceLambda, customResourceLambdaLogGroupConstructName, props);
112
+ }
106
113
  const imageScannerProvider = new custom_resources_1.Provider(this, 'Provider', {
107
114
  onEventHandler: customResourceLambda,
108
115
  });
@@ -125,8 +132,54 @@ class ImageScannerWithTrivy extends constructs_1.Construct {
125
132
  serviceToken: imageScannerProvider.serviceToken,
126
133
  });
127
134
  }
135
+ /**
136
+ * Validates that specified default log group options are the same for existing default log group.
137
+ */
138
+ validateLambdaDefaultLogGroupOptions(logGroupConstructName, props) {
139
+ const existing = aws_cdk_lib_1.Stack.of(this).node.tryFindChild(logGroupConstructName);
140
+ if (!existing)
141
+ return;
142
+ const cfnLogGroup = existing.node.defaultChild;
143
+ if (!this.isSameResourceDeletionBehavior(props.defaultLogGroupRemovalPolicy, cfnLogGroup.cfnOptions.deletionPolicy) ||
144
+ cfnLogGroup.retentionInDays !== props.defaultLogGroupRetentionDays) {
145
+ aws_cdk_lib_1.Annotations.of(this).addWarningV2('@image-scanner-with-trivy:duplicateLambdaDefaultLogGroupOptions', "You have to set the same values for 'defaultLogGroupRemovalPolicy' and 'defaultLogGroupRetentionDays' for each ImageScannerWithTrivy construct in the same stack.");
146
+ }
147
+ }
148
+ /**
149
+ * Creates the default log group for Scanner Lambda if it does not exist.
150
+ *
151
+ * This method checks if the default log group for Scanner Lambda exists in children of the stack construct.
152
+ * If it does not exist, it creates the default log group for Scanner Lambda as a child of the stack construct.
153
+ */
154
+ ensureLambdaDefaultLogGroup(singletonFunction, logGroupConstructName, props) {
155
+ const existing = aws_cdk_lib_1.Stack.of(this).node.tryFindChild(logGroupConstructName);
156
+ if (existing) {
157
+ return existing;
158
+ }
159
+ return new aws_logs_1.LogGroup(aws_cdk_lib_1.Stack.of(this), logGroupConstructName, {
160
+ logGroupName: `/aws/lambda/${singletonFunction.functionName}`,
161
+ retention: props.defaultLogGroupRetentionDays,
162
+ removalPolicy: props.defaultLogGroupRemovalPolicy,
163
+ });
164
+ }
165
+ isSameResourceDeletionBehavior(removalPolicy, deletionPolicy) {
166
+ switch (removalPolicy) {
167
+ case aws_cdk_lib_1.RemovalPolicy.DESTROY:
168
+ return deletionPolicy === aws_cdk_lib_1.CfnDeletionPolicy.DELETE;
169
+ case aws_cdk_lib_1.RemovalPolicy.RETAIN:
170
+ return deletionPolicy === aws_cdk_lib_1.CfnDeletionPolicy.RETAIN;
171
+ case aws_cdk_lib_1.RemovalPolicy.SNAPSHOT:
172
+ return deletionPolicy === aws_cdk_lib_1.CfnDeletionPolicy.SNAPSHOT;
173
+ case aws_cdk_lib_1.RemovalPolicy.RETAIN_ON_UPDATE_OR_DELETE:
174
+ return deletionPolicy === aws_cdk_lib_1.CfnDeletionPolicy.RETAIN_EXCEPT_ON_CREATE;
175
+ case undefined:
176
+ return deletionPolicy === undefined;
177
+ default:
178
+ return removalPolicy;
179
+ }
180
+ }
128
181
  }
129
182
  _b = JSII_RTTI_SYMBOL_1;
130
- ImageScannerWithTrivy[_b] = { fqn: "image-scanner-with-trivy.ImageScannerWithTrivy", version: "1.3.0" };
183
+ ImageScannerWithTrivy[_b] = { fqn: "image-scanner-with-trivy.ImageScannerWithTrivy", version: "1.4.0" };
131
184
  exports.ImageScannerWithTrivy = ImageScannerWithTrivy;
132
- //# 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 { CustomResource, Duration, Size, Token } from 'aws-cdk-lib';\nimport { IRepository } from 'aws-cdk-lib/aws-ecr';\nimport { Platform } from 'aws-cdk-lib/aws-ecr-assets';\nimport { IGrantable } 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 { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\nimport {\n  CloudWatchLogsOutputOptions,\n  ScanLogsOutputOptions,\n  ScanLogsOutputType,\n  ScannerCustomResourceProps,\n} from './types';\n\n/**\n * Enum for Severity Selection\n *\n * @see https://aquasecurity.github.io/trivy/latest/docs/scanner/vulnerability/#severity-selection\n */\nexport enum Severity {\n  UNKNOWN = 'UNKNOWN',\n  LOW = 'LOW',\n  MEDIUM = 'MEDIUM',\n  HIGH = 'HIGH',\n  CRITICAL = 'CRITICAL',\n}\n\n/**\n * Enum for Scanners\n *\n * @see https://aquasecurity.github.io/trivy/latest/docs/configuration/others/#enabledisable-scanners\n */\nexport enum Scanners {\n  VULN = 'vuln',\n  CONFIG = 'config',\n  SECRET = 'secret',\n  LICENSE = 'license',\n}\n\n/**\n * Enum for ImageConfigScanners\n *\n * @see https://aquasecurity.github.io/trivy/latest/docs/target/container_image/#container-image-metadata\n */\nexport enum ImageConfigScanners {\n  CONFIG = 'config',\n  SECRET = 'secret',\n}\n\n/**\n * Configuration for scan logs output to CloudWatch Logs log group.\n */\nexport interface CloudWatchLogsOutputProps {\n  /**\n   * The log group to output scan logs.\n   */\n  readonly logGroup: ILogGroup;\n}\n\n/**\n * Represents the output of the scan logs.\n */\nexport abstract class ScanLogsOutput {\n  /**\n   * Scan logs output to CloudWatch Logs log group.\n   */\n  public static cloudWatchLogs(options: CloudWatchLogsOutputProps): ScanLogsOutput {\n    return new CloudWatchLogsOutput(options);\n  }\n\n  /**\n   * Returns the output configuration for scan logs.\n   */\n  public abstract bind(grantee: IGrantable): ScanLogsOutputOptions;\n}\n\nclass CloudWatchLogsOutput extends ScanLogsOutput {\n  /**\n   * The log group to output scan logs.\n   */\n  private readonly logGroup: ILogGroup;\n\n  constructor(options: CloudWatchLogsOutputProps) {\n    super();\n\n    this.logGroup = options.logGroup;\n  }\n\n  public bind(grantee: IGrantable): CloudWatchLogsOutputOptions {\n    // Most Lambdas are granted AWSLambdaBasicExecutionRole and can write to any CloudWatch Logs.\n    // However, just in case AWSLambdaBasicExecutionRole is not granted, allow writing to CloudWatch Logs.\n    this.logGroup.grantWrite(grantee);\n\n    return {\n      type: ScanLogsOutputType.CLOUDWATCH_LOGS,\n      logGroupName: this.logGroup.logGroupName,\n    };\n  }\n}\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   * 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// Maximum Lambda memory size for default AWS account without quota limit increase\nconst DEFAULT_MEMORY_SIZE = 3008;\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 customResourceLambda = new SingletonFunction(this, 'CustomResourceLambda', {\n      uuid: '470b6343-d267-f753-226c-1e99f09f319a',\n      lambdaPurpose: 'Custom::ImageScannerWithTrivyCustomResourceLambda',\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    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,\n      exitOnEol: props.exitOnEol ?? 1,\n      trivyIgnore: props.trivyIgnore ?? [],\n      platform: props.platform ?? '',\n      output: props.scanLogsOutput?.bind(customResourceLambda),\n    };\n\n    new CustomResource(this, 'Resource', {\n      resourceType: 'Custom::ImageScannerWithTrivy',\n      properties: imageScannerProperties,\n      serviceToken: imageScannerProvider.serviceToken,\n    });\n  }\n}\n"]}
185
+ //# 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 { IGrantable } 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, ILogGroup, LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\nimport {\n  CloudWatchLogsOutputOptions,\n  ScanLogsOutputOptions,\n  ScanLogsOutputType,\n  ScannerCustomResourceProps,\n} from './types';\n\n/**\n * Enum for Severity Selection\n *\n * @see https://aquasecurity.github.io/trivy/latest/docs/scanner/vulnerability/#severity-selection\n */\nexport enum Severity {\n  UNKNOWN = 'UNKNOWN',\n  LOW = 'LOW',\n  MEDIUM = 'MEDIUM',\n  HIGH = 'HIGH',\n  CRITICAL = 'CRITICAL',\n}\n\n/**\n * Enum for Scanners\n *\n * @see https://aquasecurity.github.io/trivy/latest/docs/configuration/others/#enabledisable-scanners\n */\nexport enum Scanners {\n  VULN = 'vuln',\n  CONFIG = 'config',\n  SECRET = 'secret',\n  LICENSE = 'license',\n}\n\n/**\n * Enum for ImageConfigScanners\n *\n * @see https://aquasecurity.github.io/trivy/latest/docs/target/container_image/#container-image-metadata\n */\nexport enum ImageConfigScanners {\n  CONFIG = 'config',\n  SECRET = 'secret',\n}\n\n/**\n * Configuration for scan logs output to CloudWatch Logs log group.\n */\nexport interface CloudWatchLogsOutputProps {\n  /**\n   * The log group to output scan logs.\n   */\n  readonly logGroup: ILogGroup;\n}\n\n/**\n * Represents the output of the scan logs.\n */\nexport abstract class ScanLogsOutput {\n  /**\n   * Scan logs output to CloudWatch Logs log group.\n   */\n  public static cloudWatchLogs(options: CloudWatchLogsOutputProps): ScanLogsOutput {\n    return new CloudWatchLogsOutput(options);\n  }\n\n  /**\n   * Returns the output configuration for scan logs.\n   */\n  public abstract bind(grantee: IGrantable): ScanLogsOutputOptions;\n}\n\nclass CloudWatchLogsOutput extends ScanLogsOutput {\n  /**\n   * The log group to output scan logs.\n   */\n  private readonly logGroup: ILogGroup;\n\n  constructor(options: CloudWatchLogsOutputProps) {\n    super();\n\n    this.logGroup = options.logGroup;\n  }\n\n  public bind(grantee: IGrantable): CloudWatchLogsOutputOptions {\n    // Most Lambdas are granted AWSLambdaBasicExecutionRole and can write to any CloudWatch Logs.\n    // However, just in case AWSLambdaBasicExecutionRole is not granted, allow writing to CloudWatch Logs.\n    this.logGroup.grantWrite(grantee);\n\n    return {\n      type: ScanLogsOutputType.CLOUDWATCH_LOGS,\n      logGroupName: this.logGroup.logGroupName,\n    };\n  }\n}\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// Maximum Lambda memory size for default AWS account without quota limit increase\nconst DEFAULT_MEMORY_SIZE = 3008;\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    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,\n      exitOnEol: props.exitOnEol ?? 1,\n      trivyIgnore: props.trivyIgnore ?? [],\n      platform: props.platform ?? '',\n      output: props.scanLogsOutput?.bind(customResourceLambda),\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"]}
package/package.json CHANGED
@@ -91,7 +91,7 @@
91
91
  "publishConfig": {
92
92
  "access": "public"
93
93
  },
94
- "version": "1.3.0",
94
+ "version": "1.4.0",
95
95
  "types": "lib/index.d.ts",
96
96
  "stability": "stable",
97
97
  "jsii": {