image-scanner-with-trivy 2.0.0 → 2.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
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## 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"
3965
+ "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### 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\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 // 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## API Reference\n\nAPI Reference is [here](./API.md#api-reference-).\n"
3966
3966
  },
3967
3967
  "repository": {
3968
3968
  "type": "git",
@@ -4088,7 +4088,7 @@
4088
4088
  },
4089
4089
  "locationInModule": {
4090
4090
  "filename": "src/image-scanner-with-trivy.ts",
4091
- "line": 315
4091
+ "line": 330
4092
4092
  },
4093
4093
  "parameters": [
4094
4094
  {
@@ -4114,7 +4114,7 @@
4114
4114
  "kind": "class",
4115
4115
  "locationInModule": {
4116
4116
  "filename": "src/image-scanner-with-trivy.ts",
4117
- "line": 314
4117
+ "line": 329
4118
4118
  },
4119
4119
  "name": "ImageScannerWithTrivy",
4120
4120
  "symbolId": "src/image-scanner-with-trivy:ImageScannerWithTrivy"
@@ -4396,6 +4396,25 @@
4396
4396
  }
4397
4397
  }
4398
4398
  },
4399
+ {
4400
+ "abstract": true,
4401
+ "docs": {
4402
+ "default": "true",
4403
+ "remarks": "When image scanning fails, CloudFormation triggers a rollback and executes the previous\nversion of the scanner Lambda. If this property is set to `true`, the previous version of\nthe scanner Lambda will not throw an error, even if the image scanning for the previous version\nfails.\n\nThis allows the rollback to complete successfully, avoiding ROLLBACK_FAILED state\nwhen image scanning failures occur.",
4404
+ "stability": "stable",
4405
+ "summary": "Suppress errors during rollback scanner Lambda execution."
4406
+ },
4407
+ "immutable": true,
4408
+ "locationInModule": {
4409
+ "filename": "src/image-scanner-with-trivy.ts",
4410
+ "line": 323
4411
+ },
4412
+ "name": "suppressErrorOnRollback",
4413
+ "optional": true,
4414
+ "type": {
4415
+ "primitive": "boolean"
4416
+ }
4417
+ },
4399
4418
  {
4400
4419
  "abstract": true,
4401
4420
  "docs": {
@@ -4645,6 +4664,6 @@
4645
4664
  "symbolId": "src/image-scanner-with-trivy:Severity"
4646
4665
  }
4647
4666
  },
4648
- "version": "2.0.0",
4649
- "fingerprint": "AgLZ3VCxzpZE+D9DPljC53OKCs6bcGXU0vjOIwofx5w="
4667
+ "version": "2.1.1",
4668
+ "fingerprint": "UxJGqLDdTlaUSu2uNRizQj8PQSz6boH8jdb01Unk+3s="
4650
4669
  }
package/API.md CHANGED
@@ -203,6 +203,7 @@ const imageScannerWithTrivyProps: ImageScannerWithTrivyProps = { ... }
203
203
  | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.scanLogsOutput">scanLogsOutput</a></code> | <code><a href="#image-scanner-with-trivy.ScanLogsOutput">ScanLogsOutput</a></code> | Configuration for scan logs output. |
204
204
  | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.scanners">scanners</a></code> | <code><a href="#image-scanner-with-trivy.Scanners">Scanners</a>[]</code> | Enable/Disable Scanners. |
205
205
  | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.severity">severity</a></code> | <code><a href="#image-scanner-with-trivy.Severity">Severity</a>[]</code> | Severity Selection. |
206
+ | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.suppressErrorOnRollback">suppressErrorOnRollback</a></code> | <code>boolean</code> | Suppress errors during rollback scanner Lambda execution. |
206
207
  | <code><a href="#image-scanner-with-trivy.ImageScannerWithTrivyProps.property.trivyIgnore">trivyIgnore</a></code> | <code>string[]</code> | By Finding IDs. |
207
208
 
208
209
  ---
@@ -448,6 +449,27 @@ It defaults to `CRITICAL` IN THIS CONSTRUCT for safety in CI/CD, but the default
448
449
 
449
450
  ---
450
451
 
452
+ ##### `suppressErrorOnRollback`<sup>Optional</sup> <a name="suppressErrorOnRollback" id="image-scanner-with-trivy.ImageScannerWithTrivyProps.property.suppressErrorOnRollback"></a>
453
+
454
+ ```typescript
455
+ public readonly suppressErrorOnRollback: boolean;
456
+ ```
457
+
458
+ - *Type:* boolean
459
+ - *Default:* true
460
+
461
+ Suppress errors during rollback scanner Lambda execution.
462
+
463
+ When image scanning fails, CloudFormation triggers a rollback and executes the previous
464
+ version of the scanner Lambda. If this property is set to `true`, the previous version of
465
+ the scanner Lambda will not throw an error, even if the image scanning for the previous version
466
+ fails.
467
+
468
+ This allows the rollback to complete successfully, avoiding ROLLBACK_FAILED state
469
+ when image scanning failures occur.
470
+
471
+ ---
472
+
451
473
  ##### `trivyIgnore`<sup>Optional</sup> <a name="trivyIgnore" id="image-scanner-with-trivy.ImageScannerWithTrivyProps.property.trivyIgnore"></a>
452
474
 
453
475
  ```typescript
package/CONTRIBUTION.md CHANGED
@@ -1,5 +1,11 @@
1
1
  ## Contribution
2
2
 
3
+ ### Build for Code Bundle and API.md
4
+
5
+ ```sh
6
+ yarn build
7
+ ```
8
+
3
9
  ### Update Snapshots in Integration Tests
4
10
 
5
11
  ```sh
package/README.md CHANGED
@@ -154,6 +154,39 @@ new ImageScannerWithTrivy(this, 'ImageScannerWithTrivyWithNoDefaultLogGroupOptio
154
154
  });
155
155
  ```
156
156
 
157
+ ### Rollback Error Suppression
158
+
159
+ By default, the `suppressErrorOnRollback` property is set to `true`.
160
+
161
+ When image scanning fails, CloudFormation triggers a rollback and executes the previous version
162
+ of the scanner Lambda. If this property is set to `true`, the previous version of the scanner
163
+ Lambda will not throw an error, even if the image scanning for the previous version fails.
164
+
165
+ This allows the rollback to complete successfully, avoiding ROLLBACK_FAILED state
166
+ when image scanning failures occur.
167
+
168
+ ```ts
169
+ import { ImageScannerWithTrivy } from 'image-scanner-with-trivy';
170
+
171
+ const repository = new Repository(this, 'ImageRepository', {
172
+ removalPolicy: RemovalPolicy.DESTROY,
173
+ autoDeleteImages: true,
174
+ });
175
+
176
+ const image = new DockerImageAsset(this, 'DockerImage', {
177
+ directory: resolve(__dirname, './'),
178
+ });
179
+
180
+ new ImageScannerWithTrivy(this, 'ImageScannerWithTrivy', {
181
+ imageUri: image.imageUri,
182
+ repository: image.repository,
183
+ // Default is true - suppress errors during rollback to prevent ROLLBACK_FAILED
184
+ suppressErrorOnRollback: true,
185
+ // Set to false if you want rollback errors to be thrown
186
+ suppressErrorOnRollback: false,
187
+ });
188
+ ```
189
+
157
190
  ## API Reference
158
191
 
159
192
  API Reference is [here](./API.md#api-reference-).
@@ -1,9 +1,10 @@
1
- "use strict";var g=Object.defineProperty;var f=Object.getOwnPropertyDescriptor;var S=Object.getOwnPropertyNames;var C=Object.prototype.hasOwnProperty;var h=(e,t)=>{for(var n in t)g(e,n,{get:t[n],enumerable:!0})},O=(e,t,n,i)=>{if(t&&typeof t=="object"||typeof t=="function")for(let o of S(t))!C.call(e,o)&&o!==n&&g(e,o,{get:()=>t[o],enumerable:!(i=f(t,o))||i.enumerable});return e};var L=e=>O(g({},"__esModule",{value:!0}),e);var E={};h(E,{handler:()=>w});module.exports=L(E);var l=require("child_process"),m=require("fs"),s=require("@aws-sdk/client-cloudwatch-logs");var d="/tmp/.trivyignore",c=new s.CloudWatchLogsClient,w=async function(e){let t=e.RequestType,n=e.ResourceProperties;if(!n.addr||!n.imageUri)throw new Error("addr and imageUri are required.");let i={PhysicalResourceId:n.addr,Data:{}};if(t==="Create"||t==="Update"){let o=x(n);n.trivyIgnore.length&&(console.log("trivyignore: "+JSON.stringify(n.trivyIgnore)),R(n.trivyIgnore));let a=`/opt/trivy image --no-progress ${o.join(" ")} ${n.imageUri}`;console.log("command: "+a),console.log("imageUri: "+n.imageUri);let r=(0,l.spawnSync)(a,{shell:!0});if(await $(r,n.imageUri,n.output),r.status!==0)throw new Error(`Error: ${r.error}
2
- Signal: ${r.signal}
3
- Status: ${r.status}
4
- Image Scanner returned fatal errors. You may have vulnerabilities. See logs.`)}return i},x=e=>{let t=[];return e.ignoreUnfixed==="true"&&t.push("--ignore-unfixed"),e.severity.length&&t.push(`--severity ${e.severity.join(",")}`),e.scanners.length&&t.push(`--scanners ${e.scanners.join(",")}`),e.imageConfigScanners.length&&t.push(`--image-config-scanners ${e.imageConfigScanners.join(",")}`),e.exitCode&&t.push(`--exit-code ${e.exitCode}`),e.exitOnEol&&t.push(`--exit-on-eol ${e.exitOnEol}`),e.trivyIgnore.length&&t.push(`--ignorefile ${d}`),e.platform&&t.push(`--platform ${e.platform}`),t},R=e=>{let t=e.join(`
5
- `);(0,m.writeFileSync)(d,t,"utf-8")},$=async(e,t,n)=>{switch(n?.type){case"cloudWatchLogs":await v(e,n,t);break;default:console.log(`stderr:
1
+ "use strict";var g=Object.defineProperty;var y=Object.getOwnPropertyDescriptor;var C=Object.getOwnPropertyNames;var O=Object.prototype.hasOwnProperty;var h=(e,t)=>{for(var n in t)g(e,n,{get:t[n],enumerable:!0})},L=(e,t,n,o)=>{if(t&&typeof t=="object"||typeof t=="function")for(let r of C(t))!O.call(e,r)&&r!==n&&g(e,r,{get:()=>t[r],enumerable:!(o=y(t,r))||o.enumerable});return e};var R=e=>L(g({},"__esModule",{value:!0}),e);var b={};h(b,{handler:()=>k});module.exports=R(b);var d=require("child_process"),p=require("fs"),a=require("@aws-sdk/client-cloudwatch-logs"),c=require("@aws-sdk/client-cloudformation");var S="/tmp/.trivyignore",m=new a.CloudWatchLogsClient,w=new c.CloudFormationClient,k=async function(e){let t=e.RequestType,n=e.ResourceProperties;if(!n.addr||!n.imageUri)throw new Error("addr and imageUri are required.");let o={PhysicalResourceId:n.addr,Data:{}};if(t==="Create"||t==="Update"){let r=$(n);n.trivyIgnore.length&&(console.log("trivyignore: "+JSON.stringify(n.trivyIgnore)),x(n.trivyIgnore));let i=`/opt/trivy image --no-progress ${r.join(" ")} ${n.imageUri}`;console.log("command: "+i),console.log("imageUri: "+n.imageUri);let s=(0,d.spawnSync)(i,{shell:!0});if(await E(s,n.imageUri,n.output),s.status===0)return o;let u=`Error: ${s.error}
2
+ Signal: ${s.signal}
3
+ Status: ${s.status}
4
+ Image Scanner returned fatal errors. You may have vulnerabilities. See logs.`;if(n.suppressErrorOnRollback==="true"&&await I(e.StackId))return console.log(`Vulnerabilities may be detected, but suppressing errors during rollback (suppressErrorOnRollback=true).
5
+ ${u}`),o;throw new Error(u)}return o},$=e=>{let t=[];return e.ignoreUnfixed==="true"&&t.push("--ignore-unfixed"),e.severity.length&&t.push(`--severity ${e.severity.join(",")}`),e.scanners.length&&t.push(`--scanners ${e.scanners.join(",")}`),e.imageConfigScanners.length&&t.push(`--image-config-scanners ${e.imageConfigScanners.join(",")}`),e.exitCode&&t.push(`--exit-code ${e.exitCode}`),e.exitOnEol&&t.push(`--exit-on-eol ${e.exitOnEol}`),e.trivyIgnore.length&&t.push(`--ignorefile ${S}`),e.platform&&t.push(`--platform ${e.platform}`),t},x=e=>{let t=e.join(`
6
+ `);(0,p.writeFileSync)(S,t,"utf-8")},E=async(e,t,n)=>{switch(n?.type){case"cloudWatchLogs":await v(e,n,t);break;default:console.log(`stderr:
6
7
  `+e.stderr.toString()),console.log(`stdout:
7
- `+e.stdout.toString())}},v=async(e,t,n)=>{let[i,o]=n.split(":"),a=o?`uri=${i},tag=${o}`:`uri=${i}`;try{await c.send(new s.CreateLogStreamCommand({logGroupName:t.logGroupName,logStreamName:a}))}catch(u){if(u instanceof s.ResourceAlreadyExistsException)console.log(`Log stream ${a} already exists in log group ${t.logGroupName}.`);else throw u}let r=new Date().getTime(),p={logGroupName:t.logGroupName,logStreamName:a,logEvents:[{timestamp:r,message:`stderr:
8
- `+e.stderr.toString()},{timestamp:r,message:`stdout:
9
- `+e.stdout.toString()}]},y=new s.PutLogEventsCommand(p);await c.send(y),console.log(`Scan logs output to the log group: ${t.logGroupName}, log stream: ${a}`)};0&&(module.exports={handler});
8
+ `+e.stdout.toString())}},v=async(e,t,n)=>{let[o,r]=n.split(":"),i=r?`uri=${o},tag=${r}`:`uri=${o}`;try{await m.send(new a.CreateLogStreamCommand({logGroupName:t.logGroupName,logStreamName:i}))}catch(l){if(l instanceof a.ResourceAlreadyExistsException)console.log(`Log stream ${i} already exists in log group ${t.logGroupName}.`);else throw l}let s=new Date().getTime(),u={logGroupName:t.logGroupName,logStreamName:i,logEvents:[{timestamp:s,message:`stderr:
9
+ `+e.stderr.toString()},{timestamp:s,message:`stdout:
10
+ `+e.stdout.toString()}]},f=new a.PutLogEventsCommand(u);await m.send(f),console.log(`Scan logs output to the log group: ${t.logGroupName}, log stream: ${i}`)},I=async e=>{let t=new c.DescribeStacksCommand({StackName:e}),n=await w.send(t);if(n.Stacks&&n.Stacks.length>0)return n.Stacks[0].StackStatus===c.ResourceStatus.ROLLBACK_IN_PROGRESS;throw new Error(`Stack not found or no stacks returned from DescribeStacks command, stackId: ${e}`)};0&&(module.exports={handler});
@@ -7,6 +7,11 @@ import {
7
7
  PutLogEventsCommandInput,
8
8
  ResourceAlreadyExistsException,
9
9
  } from '@aws-sdk/client-cloudwatch-logs';
10
+ import {
11
+ CloudFormationClient,
12
+ DescribeStacksCommand,
13
+ ResourceStatus,
14
+ } from '@aws-sdk/client-cloudformation';
10
15
  import { CdkCustomResourceHandler, CdkCustomResourceResponse } from 'aws-lambda';
11
16
  import {
12
17
  ScanLogsOutputOptions,
@@ -18,6 +23,7 @@ import {
18
23
  const TRIVY_IGNORE_FILE_PATH = '/tmp/.trivyignore';
19
24
 
20
25
  const cwClient = new CloudWatchLogsClient();
26
+ const cfnClient = new CloudFormationClient();
21
27
 
22
28
  export const handler: CdkCustomResourceHandler = async function (event) {
23
29
  const requestType = event.RequestType;
@@ -48,10 +54,18 @@ export const handler: CdkCustomResourceHandler = async function (event) {
48
54
 
49
55
  await outputScanLogs(response, props.imageUri, props.output);
50
56
 
51
- if (response.status !== 0)
52
- throw new Error(
53
- `Error: ${response.error}\nSignal: ${response.signal}\nStatus: ${response.status}\nImage Scanner returned fatal errors. You may have vulnerabilities. See logs.`,
57
+ if (response.status === 0) return funcResponse;
58
+
59
+ const errorMessage = `Error: ${response.error}\nSignal: ${response.signal}\nStatus: ${response.status}\nImage Scanner returned fatal errors. You may have vulnerabilities. See logs.`;
60
+
61
+ if (props.suppressErrorOnRollback === 'true' && (await isRollbackInProgress(event.StackId))) {
62
+ console.log(
63
+ `Vulnerabilities may be detected, but suppressing errors during rollback (suppressErrorOnRollback=true).\n${errorMessage}`,
54
64
  );
65
+ return funcResponse;
66
+ }
67
+
68
+ throw new Error(errorMessage);
55
69
  }
56
70
 
57
71
  return funcResponse;
@@ -145,3 +159,19 @@ const outputScanLogsToCWLogs = async (
145
159
  `Scan logs output to the log group: ${output.logGroupName}, log stream: ${logStreamName}`,
146
160
  );
147
161
  };
162
+
163
+ const isRollbackInProgress = async (stackId: string): Promise<boolean> => {
164
+ const command = new DescribeStacksCommand({
165
+ StackName: stackId,
166
+ });
167
+ const response = await cfnClient.send(command);
168
+
169
+ if (response.Stacks && response.Stacks.length > 0) {
170
+ const stackStatus = response.Stacks[0].StackStatus;
171
+ return stackStatus === ResourceStatus.ROLLBACK_IN_PROGRESS;
172
+ }
173
+
174
+ throw new Error(
175
+ `Stack not found or no stacks returned from DescribeStacks command, stackId: ${stackId}`,
176
+ );
177
+ };
@@ -236,6 +236,20 @@ export interface ImageScannerWithTrivyProps {
236
236
  * @default - scan logs output to default log group created by Scanner Lambda(`/aws/lambda/${functionName}`)
237
237
  */
238
238
  readonly scanLogsOutput?: ScanLogsOutput;
239
+ /**
240
+ * Suppress errors during rollback scanner Lambda execution
241
+ *
242
+ * When image scanning fails, CloudFormation triggers a rollback and executes the previous
243
+ * version of the scanner Lambda. If this property is set to `true`, the previous version of
244
+ * the scanner Lambda will not throw an error, even if the image scanning for the previous version
245
+ * fails.
246
+ *
247
+ * This allows the rollback to complete successfully, avoiding ROLLBACK_FAILED state
248
+ * when image scanning failures occur.
249
+ *
250
+ * @default true
251
+ */
252
+ readonly suppressErrorOnRollback?: boolean;
239
253
  }
240
254
  export declare class ImageScannerWithTrivy extends Construct {
241
255
  constructor(scope: Construct, id: string, props: ImageScannerWithTrivyProps);
@@ -59,7 +59,7 @@ class ScanLogsOutput {
59
59
  }
60
60
  exports.ScanLogsOutput = ScanLogsOutput;
61
61
  _a = JSII_RTTI_SYMBOL_1;
62
- ScanLogsOutput[_a] = { fqn: "image-scanner-with-trivy.ScanLogsOutput", version: "2.0.0" };
62
+ ScanLogsOutput[_a] = { fqn: "image-scanner-with-trivy.ScanLogsOutput", version: "2.1.1" };
63
63
  class CloudWatchLogsOutput extends ScanLogsOutput {
64
64
  constructor(options) {
65
65
  super();
@@ -125,6 +125,7 @@ class ImageScannerWithTrivy extends constructs_1.Construct {
125
125
  trivyIgnore: props.trivyIgnore ?? [],
126
126
  platform: props.platform ?? '',
127
127
  output: props.scanLogsOutput?.bind(customResourceLambda),
128
+ suppressErrorOnRollback: String(props.suppressErrorOnRollback ?? true),
128
129
  };
129
130
  new aws_cdk_lib_1.CustomResource(this, 'Resource', {
130
131
  resourceType: 'Custom::ImageScannerWithTrivy',
@@ -181,5 +182,5 @@ class ImageScannerWithTrivy extends constructs_1.Construct {
181
182
  }
182
183
  exports.ImageScannerWithTrivy = ImageScannerWithTrivy;
183
184
  _b = JSII_RTTI_SYMBOL_1;
184
- ImageScannerWithTrivy[_b] = { fqn: "image-scanner-with-trivy.ImageScannerWithTrivy", version: "2.0.0" };
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"]}
185
+ ImageScannerWithTrivy[_b] = { fqn: "image-scanner-with-trivy.ImageScannerWithTrivy", version: "2.1.1" };
186
+ //# 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   * 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\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      suppressErrorOnRollback: String(props.suppressErrorOnRollback ?? true),\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/lib/types.d.ts CHANGED
@@ -31,4 +31,5 @@ export interface ScannerCustomResourceProps {
31
31
  readonly trivyIgnore: string[];
32
32
  readonly platform: string;
33
33
  readonly output?: ScanLogsOutputOptions;
34
+ readonly suppressErrorOnRollback: string;
34
35
  }
package/lib/types.js CHANGED
@@ -8,4 +8,4 @@ var ScanLogsOutputType;
8
8
  (function (ScanLogsOutputType) {
9
9
  ScanLogsOutputType["CLOUDWATCH_LOGS"] = "cloudWatchLogs";
10
10
  })(ScanLogsOutputType || (exports.ScanLogsOutputType = ScanLogsOutputType = {}));
11
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUE7O0dBRUc7QUFDSCxJQUFZLGtCQUVYO0FBRkQsV0FBWSxrQkFBa0I7SUFDNUIsd0RBQWtDLENBQUE7QUFDcEMsQ0FBQyxFQUZXLGtCQUFrQixrQ0FBbEIsa0JBQWtCLFFBRTdCIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBFbnVtIGZvciBTY2FuTG9nc091dHB1dFR5cGVcbiAqL1xuZXhwb3J0IGVudW0gU2NhbkxvZ3NPdXRwdXRUeXBlIHtcbiAgQ0xPVURXQVRDSF9MT0dTID0gJ2Nsb3VkV2F0Y2hMb2dzJyxcbn1cblxuLyoqXG4gKiBPdXRwdXQgY29uZmlndXJhdGlvbnMgZm9yIHNjYW4gbG9ncy5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBTY2FuTG9nc091dHB1dE9wdGlvbnMge1xuICByZWFkb25seSB0eXBlOiBTY2FuTG9nc091dHB1dFR5cGU7XG59XG5cbi8qKlxuICogT3V0cHV0IGNvbmZpZ3VyYXRpb24gZm9yIHNjYW4gbG9ncyB0byBDbG91ZFdhdGNoIExvZ3MuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ2xvdWRXYXRjaExvZ3NPdXRwdXRPcHRpb25zIGV4dGVuZHMgU2NhbkxvZ3NPdXRwdXRPcHRpb25zIHtcbiAgcmVhZG9ubHkgbG9nR3JvdXBOYW1lOiBzdHJpbmc7XG59XG5cbi8qKlxuICogTGFtYmRhIGZ1bmN0aW9uIGV2ZW50IG9iamVjdCBmb3IgU2Nhbm5lciBDdXN0b20gUmVzb3VyY2UuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU2Nhbm5lckN1c3RvbVJlc291cmNlUHJvcHMge1xuICByZWFkb25seSBhZGRyOiBzdHJpbmc7XG4gIHJlYWRvbmx5IGltYWdlVXJpOiBzdHJpbmc7XG4gIHJlYWRvbmx5IGlnbm9yZVVuZml4ZWQ6IHN0cmluZztcbiAgcmVhZG9ubHkgc2V2ZXJpdHk6IHN0cmluZ1tdO1xuICByZWFkb25seSBzY2FubmVyczogc3RyaW5nW107XG4gIHJlYWRvbmx5IGltYWdlQ29uZmlnU2Nhbm5lcnM6IHN0cmluZ1tdO1xuICByZWFkb25seSBleGl0Q29kZTogbnVtYmVyO1xuICByZWFkb25seSBleGl0T25Fb2w6IG51bWJlcjtcbiAgcmVhZG9ubHkgdHJpdnlJZ25vcmU6IHN0cmluZ1tdO1xuICByZWFkb25seSBwbGF0Zm9ybTogc3RyaW5nO1xuICByZWFkb25seSBvdXRwdXQ/OiBTY2FuTG9nc091dHB1dE9wdGlvbnM7XG59XG4iXX0=
11
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUE7O0dBRUc7QUFDSCxJQUFZLGtCQUVYO0FBRkQsV0FBWSxrQkFBa0I7SUFDNUIsd0RBQWtDLENBQUE7QUFDcEMsQ0FBQyxFQUZXLGtCQUFrQixrQ0FBbEIsa0JBQWtCLFFBRTdCIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBFbnVtIGZvciBTY2FuTG9nc091dHB1dFR5cGVcbiAqL1xuZXhwb3J0IGVudW0gU2NhbkxvZ3NPdXRwdXRUeXBlIHtcbiAgQ0xPVURXQVRDSF9MT0dTID0gJ2Nsb3VkV2F0Y2hMb2dzJyxcbn1cblxuLyoqXG4gKiBPdXRwdXQgY29uZmlndXJhdGlvbnMgZm9yIHNjYW4gbG9ncy5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBTY2FuTG9nc091dHB1dE9wdGlvbnMge1xuICByZWFkb25seSB0eXBlOiBTY2FuTG9nc091dHB1dFR5cGU7XG59XG5cbi8qKlxuICogT3V0cHV0IGNvbmZpZ3VyYXRpb24gZm9yIHNjYW4gbG9ncyB0byBDbG91ZFdhdGNoIExvZ3MuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ2xvdWRXYXRjaExvZ3NPdXRwdXRPcHRpb25zIGV4dGVuZHMgU2NhbkxvZ3NPdXRwdXRPcHRpb25zIHtcbiAgcmVhZG9ubHkgbG9nR3JvdXBOYW1lOiBzdHJpbmc7XG59XG5cbi8qKlxuICogTGFtYmRhIGZ1bmN0aW9uIGV2ZW50IG9iamVjdCBmb3IgU2Nhbm5lciBDdXN0b20gUmVzb3VyY2UuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU2Nhbm5lckN1c3RvbVJlc291cmNlUHJvcHMge1xuICByZWFkb25seSBhZGRyOiBzdHJpbmc7XG4gIHJlYWRvbmx5IGltYWdlVXJpOiBzdHJpbmc7XG4gIHJlYWRvbmx5IGlnbm9yZVVuZml4ZWQ6IHN0cmluZztcbiAgcmVhZG9ubHkgc2V2ZXJpdHk6IHN0cmluZ1tdO1xuICByZWFkb25seSBzY2FubmVyczogc3RyaW5nW107XG4gIHJlYWRvbmx5IGltYWdlQ29uZmlnU2Nhbm5lcnM6IHN0cmluZ1tdO1xuICByZWFkb25seSBleGl0Q29kZTogbnVtYmVyO1xuICByZWFkb25seSBleGl0T25Fb2w6IG51bWJlcjtcbiAgcmVhZG9ubHkgdHJpdnlJZ25vcmU6IHN0cmluZ1tdO1xuICByZWFkb25seSBwbGF0Zm9ybTogc3RyaW5nO1xuICByZWFkb25seSBvdXRwdXQ/OiBTY2FuTG9nc091dHB1dE9wdGlvbnM7XG4gIHJlYWRvbmx5IHN1cHByZXNzRXJyb3JPblJvbGxiYWNrOiBzdHJpbmc7XG59XG4iXX0=
package/package.json CHANGED
@@ -88,7 +88,7 @@
88
88
  "publishConfig": {
89
89
  "access": "public"
90
90
  },
91
- "version": "2.0.0",
91
+ "version": "2.1.1",
92
92
  "types": "lib/index.d.ts",
93
93
  "stability": "stable",
94
94
  "jsii": {