image-scanner-with-trivy 1.0.0 → 1.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +4 -4
- package/.projenrc.ts +8 -0
- package/API.md +5 -1
- package/README.md +5 -1
- package/assets/lambda/dist/index.js +7 -7
- package/assets/lambda/index.ts +39 -29
- package/lib/index.js +1 -1
- package/lib/integ.js +7 -3
- package/package.json +2 -1
package/.jsii
CHANGED
|
@@ -3489,7 +3489,7 @@
|
|
|
3489
3489
|
}
|
|
3490
3490
|
}
|
|
3491
3491
|
},
|
|
3492
|
-
"description": "
|
|
3492
|
+
"description": "Scan container images with Trivy in CDK deployment",
|
|
3493
3493
|
"docs": {
|
|
3494
3494
|
"stability": "stable"
|
|
3495
3495
|
},
|
|
@@ -3518,7 +3518,7 @@
|
|
|
3518
3518
|
},
|
|
3519
3519
|
"name": "image-scanner-with-trivy",
|
|
3520
3520
|
"readme": {
|
|
3521
|
-
"markdown": "# image-scanner-with-trivy\n\n## What is\n\nThis is an AWS CDK Construct that allows you to **scan
|
|
3521
|
+
"markdown": "# image-scanner-with-trivy\n\n## What is\n\nThis is an AWS CDK Construct that allows you to **scan container images with Trivy in CDK deployment layer**.\n\nIf it detects vulnerabilities, it can **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 } 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// Please grant the necessary options on a case-by-case basis.\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## API Reference\n\nAPI Reference is [here](./API.md#api-reference-).\n"
|
|
3522
3522
|
},
|
|
3523
3523
|
"repository": {
|
|
3524
3524
|
"type": "git",
|
|
@@ -3945,6 +3945,6 @@
|
|
|
3945
3945
|
"symbolId": "src/index:Severity"
|
|
3946
3946
|
}
|
|
3947
3947
|
},
|
|
3948
|
-
"version": "1.0.
|
|
3949
|
-
"fingerprint": "
|
|
3948
|
+
"version": "1.0.2",
|
|
3949
|
+
"fingerprint": "mbTIFBgcYh4wOS2dkAXmKqFjjCRN3dbMDghOQgjw1Rw="
|
|
3950
3950
|
}
|
package/.projenrc.ts
CHANGED
|
@@ -11,6 +11,7 @@ const project = new awscdk.AwsCdkConstructLibrary({
|
|
|
11
11
|
name: 'image-scanner-with-trivy',
|
|
12
12
|
projenrcTs: true,
|
|
13
13
|
repositoryUrl: 'https://github.com/go-to-k/image-scanner-with-trivy',
|
|
14
|
+
description: 'Scan container images with Trivy in CDK deployment',
|
|
14
15
|
prettier: true,
|
|
15
16
|
prettierOptions: {
|
|
16
17
|
settings: {
|
|
@@ -61,6 +62,13 @@ const project = new awscdk.AwsCdkConstructLibrary({
|
|
|
61
62
|
bin: {
|
|
62
63
|
0: './assets',
|
|
63
64
|
},
|
|
65
|
+
githubOptions: {
|
|
66
|
+
pullRequestLintOptions: {
|
|
67
|
+
semanticTitleOptions: {
|
|
68
|
+
types: ['feat', 'fix', 'chore', 'docs', 'test', 'refactor', 'ci'],
|
|
69
|
+
},
|
|
70
|
+
},
|
|
71
|
+
},
|
|
64
72
|
// deps: [], /* Runtime dependencies of this module. */
|
|
65
73
|
// description: undefined, /* The description is just a string that helps people understand the purpose of the package. */
|
|
66
74
|
// devDeps: [], /* Build dependencies for this module. */
|
package/API.md
CHANGED
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
|
|
3
3
|
## What is
|
|
4
4
|
|
|
5
|
-
This is an AWS CDK Construct that allows you to **scan
|
|
5
|
+
This is an AWS CDK Construct that allows you to **scan container images with Trivy in CDK deployment layer**.
|
|
6
6
|
|
|
7
7
|
If it detects vulnerabilities, it can **prevent the image from being pushed to the ECR for the application**.
|
|
8
8
|
|
|
@@ -48,6 +48,10 @@ const ecrDeployment = new ECRDeployment(this, 'DeployImage', {
|
|
|
48
48
|
ecrDeployment.node.addDependency(imageScanner);
|
|
49
49
|
```
|
|
50
50
|
|
|
51
|
+
## API Reference
|
|
52
|
+
|
|
53
|
+
API Reference is [here](./API.md#api-reference-).
|
|
54
|
+
|
|
51
55
|
# API Reference <a name="API Reference" id="api-reference"></a>
|
|
52
56
|
|
|
53
57
|
## Constructs <a name="Constructs" id="Constructs"></a>
|
package/README.md
CHANGED
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
|
|
3
3
|
## What is
|
|
4
4
|
|
|
5
|
-
This is an AWS CDK Construct that allows you to **scan
|
|
5
|
+
This is an AWS CDK Construct that allows you to **scan container images with Trivy in CDK deployment layer**.
|
|
6
6
|
|
|
7
7
|
If it detects vulnerabilities, it can **prevent the image from being pushed to the ECR for the application**.
|
|
8
8
|
|
|
@@ -47,3 +47,7 @@ const ecrDeployment = new ECRDeployment(this, 'DeployImage', {
|
|
|
47
47
|
});
|
|
48
48
|
ecrDeployment.node.addDependency(imageScanner);
|
|
49
49
|
```
|
|
50
|
+
|
|
51
|
+
## API Reference
|
|
52
|
+
|
|
53
|
+
API Reference is [here](./API.md#api-reference-).
|
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
"use strict";var
|
|
2
|
-
`+
|
|
3
|
-
`+
|
|
4
|
-
Signal: ${
|
|
5
|
-
Status: ${
|
|
6
|
-
Image Scanner returned fatal errors. You may have vulnerabilities. See logs.`)}return
|
|
7
|
-
`)}),
|
|
1
|
+
"use strict";var u=Object.create;var s=Object.defineProperty;var d=Object.getOwnPropertyDescriptor;var f=Object.getOwnPropertyNames;var y=Object.getPrototypeOf,h=Object.prototype.hasOwnProperty;var v=(e,n)=>{for(var r in n)s(e,r,{get:n[r],enumerable:!0})},g=(e,n,r,t)=>{if(n&&typeof n=="object"||typeof n=="function")for(let i of f(n))!h.call(e,i)&&i!==r&&s(e,i,{get:()=>n[i],enumerable:!(t=d(n,i))||t.enumerable});return e};var C=(e,n,r)=>(r=e!=null?u(y(e)):{},g(n||!e||!e.__esModule?s(r,"default",{value:e,enumerable:!0}):r,e)),S=e=>g(s({},"__esModule",{value:!0}),e);var I={};v(I,{handler:()=>p});module.exports=S(I);var c=require("child_process"),l=require("fs"),m=C(require("path"),1),p=async function(e){let n=e.RequestType,r=e.ResourceProperties;if(!r.addr||!r.imageUri)throw new Error("addr and imageUri are required.");let t={PhysicalResourceId:r.addr,Data:{}};if(n==="Create"||n==="Update"){let i=x(r);r.trivyIgnore.length&&(console.log("trivyignore: "+JSON.stringify(r.trivyIgnore)),$(r.trivyIgnore));let a=`/opt/trivy image --no-progress ${i.join(" ")} ${r.imageUri}`;console.log("command: "+a),console.log("imageUri: "+r.imageUri);let o=(0,c.spawnSync)(a,{shell:!0});if(console.log(`stderr:
|
|
2
|
+
`+o.stderr?.toString()),console.log(`stdout:
|
|
3
|
+
`+o.stdout?.toString()),o.status!==0)throw new Error(`Error: ${o.error}
|
|
4
|
+
Signal: ${o.signal}
|
|
5
|
+
Status: ${o.status}
|
|
6
|
+
Image Scanner returned fatal errors. You may have vulnerabilities. See logs.`)}return t},x=e=>{let n=[];return e.ignoreUnfixed==="true"&&n.push("--ignore-unfixed"),e.severity.length&&n.push(`--severity ${e.severity.join(",")}`),e.scanners.length&&n.push(`--scanners ${e.scanners.join(",")}`),e.imageConfigScanners.length&&n.push(`--image-config-scanners ${e.imageConfigScanners.join(",")}`),e.exitCode&&n.push(`--exit-code ${e.exitCode}`),e.exitOnEol&&n.push(`--exit-on-eol ${e.exitOnEol}`),e.trivyIgnore.length&&n.push("--ignorefile /tmp/.trivyignore"),e.platform&&n.push(`--platform ${e.platform}`),n},$=e=>{let n=m.default.join("/tmp",".trivyignore"),r=(0,l.createWriteStream)(n);e.forEach(t=>{r.write(t+`
|
|
7
|
+
`)}),r.end()};0&&(module.exports={handler});
|
package/assets/lambda/index.ts
CHANGED
|
@@ -3,47 +3,41 @@ import { createWriteStream } from 'fs';
|
|
|
3
3
|
import path from 'path';
|
|
4
4
|
import { CdkCustomResourceHandler, CdkCustomResourceResponse } from 'aws-lambda';
|
|
5
5
|
|
|
6
|
+
interface ScannerProps {
|
|
7
|
+
addr: string;
|
|
8
|
+
imageUri: string;
|
|
9
|
+
ignoreUnfixed: string;
|
|
10
|
+
severity: string[];
|
|
11
|
+
scanners: string[];
|
|
12
|
+
imageConfigScanners: string[];
|
|
13
|
+
exitCode: number;
|
|
14
|
+
exitOnEol: number;
|
|
15
|
+
trivyIgnore: string[];
|
|
16
|
+
platform: string;
|
|
17
|
+
}
|
|
18
|
+
|
|
6
19
|
export const handler: CdkCustomResourceHandler = async function (event) {
|
|
7
20
|
const requestType = event.RequestType;
|
|
21
|
+
const props = event.ResourceProperties as unknown as ScannerProps;
|
|
8
22
|
|
|
9
|
-
|
|
10
|
-
const imageUri = event.ResourceProperties.imageUri as string;
|
|
11
|
-
const ignoreUnfixed = event.ResourceProperties.ignoreUnfixed as string;
|
|
12
|
-
const severity = event.ResourceProperties.severity as string[];
|
|
13
|
-
const scanners = event.ResourceProperties.scanners as string[];
|
|
14
|
-
const imageConfigScanners = event.ResourceProperties.imageConfigScanners as string[];
|
|
15
|
-
const exitCode = event.ResourceProperties.exitCode as number;
|
|
16
|
-
const exitOnEol = event.ResourceProperties.exitOnEol as number;
|
|
17
|
-
const trivyIgnore = event.ResourceProperties.trivyIgnore as string[];
|
|
18
|
-
const platform = event.ResourceProperties.platform as string;
|
|
19
|
-
|
|
20
|
-
if (!addr || !imageUri) throw new Error('addr and imageUri are required.');
|
|
23
|
+
if (!props.addr || !props.imageUri) throw new Error('addr and imageUri are required.');
|
|
21
24
|
|
|
22
25
|
const funcResponse: CdkCustomResourceResponse = {
|
|
23
|
-
PhysicalResourceId: addr,
|
|
26
|
+
PhysicalResourceId: props.addr,
|
|
24
27
|
Data: {} as { [key: string]: string },
|
|
25
28
|
};
|
|
26
29
|
|
|
27
30
|
if (requestType === 'Create' || requestType === 'Update') {
|
|
28
|
-
const options
|
|
29
|
-
|
|
30
|
-
if (
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
options.push(`--image-config-scanners ${imageConfigScanners.join(',')}`);
|
|
34
|
-
if (exitCode) options.push(`--exit-code ${exitCode}`);
|
|
35
|
-
if (exitOnEol) options.push(`--exit-on-eol ${exitOnEol}`);
|
|
36
|
-
if (trivyIgnore.length) options.push('--ignorefile /tmp/.trivyignore');
|
|
37
|
-
if (platform) options.push(`--platform ${platform}`);
|
|
38
|
-
|
|
39
|
-
if (trivyIgnore.length) {
|
|
40
|
-
console.log('trivyignore: ' + JSON.stringify(trivyIgnore));
|
|
41
|
-
makeTrivyIgnoreFile(trivyIgnore);
|
|
31
|
+
const options = makeOptions(props);
|
|
32
|
+
|
|
33
|
+
if (props.trivyIgnore.length) {
|
|
34
|
+
console.log('trivyignore: ' + JSON.stringify(props.trivyIgnore));
|
|
35
|
+
makeTrivyIgnoreFile(props.trivyIgnore);
|
|
42
36
|
}
|
|
43
37
|
|
|
44
|
-
const cmd = `/opt/trivy image --no-progress ${options.join(' ')} ${imageUri}`;
|
|
38
|
+
const cmd = `/opt/trivy image --no-progress ${options.join(' ')} ${props.imageUri}`;
|
|
45
39
|
console.log('command: ' + cmd);
|
|
46
|
-
console.log('imageUri: ' + imageUri);
|
|
40
|
+
console.log('imageUri: ' + props.imageUri);
|
|
47
41
|
|
|
48
42
|
const response = spawnSync(cmd, {
|
|
49
43
|
shell: true,
|
|
@@ -61,6 +55,22 @@ export const handler: CdkCustomResourceHandler = async function (event) {
|
|
|
61
55
|
return funcResponse;
|
|
62
56
|
};
|
|
63
57
|
|
|
58
|
+
const makeOptions = (props: ScannerProps): string[] => {
|
|
59
|
+
const options: string[] = [];
|
|
60
|
+
|
|
61
|
+
if (props.ignoreUnfixed === 'true') options.push('--ignore-unfixed');
|
|
62
|
+
if (props.severity.length) options.push(`--severity ${props.severity.join(',')}`);
|
|
63
|
+
if (props.scanners.length) options.push(`--scanners ${props.scanners.join(',')}`);
|
|
64
|
+
if (props.imageConfigScanners.length)
|
|
65
|
+
options.push(`--image-config-scanners ${props.imageConfigScanners.join(',')}`);
|
|
66
|
+
if (props.exitCode) options.push(`--exit-code ${props.exitCode}`);
|
|
67
|
+
if (props.exitOnEol) options.push(`--exit-on-eol ${props.exitOnEol}`);
|
|
68
|
+
if (props.trivyIgnore.length) options.push('--ignorefile /tmp/.trivyignore');
|
|
69
|
+
if (props.platform) options.push(`--platform ${props.platform}`);
|
|
70
|
+
|
|
71
|
+
return options;
|
|
72
|
+
};
|
|
73
|
+
|
|
64
74
|
const makeTrivyIgnoreFile = (trivyIgnore: string[]) => {
|
|
65
75
|
const trivyIgnoreFilePath = path.join('/tmp', '.trivyignore');
|
|
66
76
|
const trivyIgnoreFile = createWriteStream(trivyIgnoreFilePath);
|
package/lib/index.js
CHANGED
|
@@ -87,6 +87,6 @@ class ImageScannerWithTrivy extends constructs_1.Construct {
|
|
|
87
87
|
}
|
|
88
88
|
}
|
|
89
89
|
_a = JSII_RTTI_SYMBOL_1;
|
|
90
|
-
ImageScannerWithTrivy[_a] = { fqn: "image-scanner-with-trivy.ImageScannerWithTrivy", version: "1.0.
|
|
90
|
+
ImageScannerWithTrivy[_a] = { fqn: "image-scanner-with-trivy.ImageScannerWithTrivy", version: "1.0.2" };
|
|
91
91
|
exports.ImageScannerWithTrivy = ImageScannerWithTrivy;
|
|
92
92
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { join } from 'path';\nimport { CustomResource, Duration } from 'aws-cdk-lib';\nimport { IRepository } from 'aws-cdk-lib/aws-ecr';\nimport {\n  Architecture,\n  AssetCode,\n  Handler,\n  Runtime,\n  SingletonFunction,\n} from 'aws-cdk-lib/aws-lambda';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\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\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// 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 (props.memorySize) {\n      if (props.memorySize < 3008 || props.memorySize > 10240) {\n        throw new Error('You can specify between `3008` and `10240` for `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      architecture: Architecture.ARM_64,\n      timeout: Duration.seconds(900),\n      retryAttempts: 0,\n      memorySize: props.memorySize ?? DEFAULT_MEMORY_SIZE,\n    });\n    props.repository.grantPull(customResourceLambda);\n\n    const imageScannerProvider = new Provider(this, 'Provider', {\n      onEventHandler: customResourceLambda,\n    });\n\n    const imageScannerProperties: { [key: string]: string | string[] | boolean | number } = {};\n    imageScannerProperties.addr = this.node.addr;\n    imageScannerProperties.imageUri = props.imageUri;\n    imageScannerProperties.ignoreUnfixed = props.ignoreUnfixed ?? false;\n    imageScannerProperties.severity = props.severity ?? [Severity.CRITICAL];\n    imageScannerProperties.scanners = props.scanners ?? [];\n    imageScannerProperties.imageConfigScanners = props.imageConfigScanners ?? [];\n    imageScannerProperties.exitCode = props.exitCode ?? 1;\n    imageScannerProperties.exitOnEol = props.exitOnEol ?? 1;\n    imageScannerProperties.trivyIgnore = props.trivyIgnore ?? [];\n    imageScannerProperties.platform = props.platform ?? '';\n\n    new CustomResource(this, 'Default', {\n      resourceType: 'Custom::ImageScannerWithTrivy',\n      properties: imageScannerProperties,\n      serviceToken: imageScannerProvider.serviceToken,\n    });\n  }\n}\n"]}
|
package/lib/integ.js
CHANGED
|
@@ -10,7 +10,11 @@ const image = new aws_ecr_assets_1.DockerImageAsset(stack, 'DockerImage', {
|
|
|
10
10
|
directory: (0, path_1.resolve)(__dirname, '../assets/lambda'),
|
|
11
11
|
platform: aws_ecr_assets_1.Platform.LINUX_ARM64,
|
|
12
12
|
});
|
|
13
|
-
new _1.ImageScannerWithTrivy(stack, '
|
|
13
|
+
new _1.ImageScannerWithTrivy(stack, 'ImageScannerWithTrivy1', {
|
|
14
|
+
imageUri: image.imageUri,
|
|
15
|
+
repository: image.repository,
|
|
16
|
+
});
|
|
17
|
+
new _1.ImageScannerWithTrivy(stack, 'ImageScannerWithTrivy2', {
|
|
14
18
|
imageUri: image.imageUri,
|
|
15
19
|
repository: image.repository,
|
|
16
20
|
ignoreUnfixed: false,
|
|
@@ -18,8 +22,8 @@ new _1.ImageScannerWithTrivy(stack, 'ImageScannerWithTrivy4', {
|
|
|
18
22
|
scanners: [_1.Scanners.VULN, _1.Scanners.SECRET],
|
|
19
23
|
exitCode: 1,
|
|
20
24
|
exitOnEol: 1,
|
|
21
|
-
trivyIgnore: ['CVE-2023-37920', 'CVE-2019-14697 exp:2023-01-01'],
|
|
25
|
+
trivyIgnore: ['CVE-2023-37920', 'CVE-2019-14697 exp:2023-01-01', 'generic-unwanted-rule'],
|
|
22
26
|
memorySize: 3008,
|
|
23
27
|
platform: 'linux/arm64',
|
|
24
28
|
});
|
|
25
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
29
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50ZWcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW50ZWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFBQSwrQkFBK0I7QUFDL0IsNkNBQXlDO0FBQ3pDLCtEQUF3RTtBQUN4RSx3QkFBOEQ7QUFFOUQsTUFBTSxHQUFHLEdBQUcsSUFBSSxpQkFBRyxFQUFFLENBQUM7QUFDdEIsTUFBTSxLQUFLLEdBQUcsSUFBSSxtQkFBSyxDQUFDLEdBQUcsRUFBRSw0QkFBNEIsQ0FBQyxDQUFDO0FBRTNELE1BQU0sS0FBSyxHQUFHLElBQUksaUNBQWdCLENBQUMsS0FBSyxFQUFFLGFBQWEsRUFBRTtJQUN2RCxTQUFTLEVBQUUsSUFBQSxjQUFPLEVBQUMsU0FBUyxFQUFFLGtCQUFrQixDQUFDO0lBQ2pELFFBQVEsRUFBRSx5QkFBUSxDQUFDLFdBQVc7Q0FDL0IsQ0FBQyxDQUFDO0FBRUgsSUFBSSx3QkFBcUIsQ0FBQyxLQUFLLEVBQUUsd0JBQXdCLEVBQUU7SUFDekQsUUFBUSxFQUFFLEtBQUssQ0FBQyxRQUFRO0lBQ3hCLFVBQVUsRUFBRSxLQUFLLENBQUMsVUFBVTtDQUM3QixDQUFDLENBQUM7QUFFSCxJQUFJLHdCQUFxQixDQUFDLEtBQUssRUFBRSx3QkFBd0IsRUFBRTtJQUN6RCxRQUFRLEVBQUUsS0FBSyxDQUFDLFFBQVE7SUFDeEIsVUFBVSxFQUFFLEtBQUssQ0FBQyxVQUFVO0lBQzVCLGFBQWEsRUFBRSxLQUFLO0lBQ3BCLFFBQVEsRUFBRSxDQUFDLFdBQVEsQ0FBQyxRQUFRLENBQUM7SUFDN0IsUUFBUSxFQUFFLENBQUMsV0FBUSxDQUFDLElBQUksRUFBRSxXQUFRLENBQUMsTUFBTSxDQUFDO0lBQzFDLFFBQVEsRUFBRSxDQUFDO0lBQ1gsU0FBUyxFQUFFLENBQUM7SUFDWixXQUFXLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSwrQkFBK0IsRUFBRSx1QkFBdUIsQ0FBQztJQUN6RixVQUFVLEVBQUUsSUFBSTtJQUNoQixRQUFRLEVBQUUsYUFBYTtDQUN4QixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyByZXNvbHZlIH0gZnJvbSAncGF0aCc7XG5pbXBvcnQgeyBBcHAsIFN0YWNrIH0gZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0IHsgRG9ja2VySW1hZ2VBc3NldCwgUGxhdGZvcm0gfSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtZWNyLWFzc2V0cyc7XG5pbXBvcnQgeyBJbWFnZVNjYW5uZXJXaXRoVHJpdnksIFNjYW5uZXJzLCBTZXZlcml0eSB9IGZyb20gJy4nO1xuXG5jb25zdCBhcHAgPSBuZXcgQXBwKCk7XG5jb25zdCBzdGFjayA9IG5ldyBTdGFjayhhcHAsICdJbWFnZVNjYW5uZXJXaXRoVHJpdnlTdGFjaycpO1xuXG5jb25zdCBpbWFnZSA9IG5ldyBEb2NrZXJJbWFnZUFzc2V0KHN0YWNrLCAnRG9ja2VySW1hZ2UnLCB7XG4gIGRpcmVjdG9yeTogcmVzb2x2ZShfX2Rpcm5hbWUsICcuLi9hc3NldHMvbGFtYmRhJyksXG4gIHBsYXRmb3JtOiBQbGF0Zm9ybS5MSU5VWF9BUk02NCxcbn0pO1xuXG5uZXcgSW1hZ2VTY2FubmVyV2l0aFRyaXZ5KHN0YWNrLCAnSW1hZ2VTY2FubmVyV2l0aFRyaXZ5MScsIHtcbiAgaW1hZ2VVcmk6IGltYWdlLmltYWdlVXJpLFxuICByZXBvc2l0b3J5OiBpbWFnZS5yZXBvc2l0b3J5LFxufSk7XG5cbm5ldyBJbWFnZVNjYW5uZXJXaXRoVHJpdnkoc3RhY2ssICdJbWFnZVNjYW5uZXJXaXRoVHJpdnkyJywge1xuICBpbWFnZVVyaTogaW1hZ2UuaW1hZ2VVcmksXG4gIHJlcG9zaXRvcnk6IGltYWdlLnJlcG9zaXRvcnksXG4gIGlnbm9yZVVuZml4ZWQ6IGZhbHNlLFxuICBzZXZlcml0eTogW1NldmVyaXR5LkNSSVRJQ0FMXSxcbiAgc2Nhbm5lcnM6IFtTY2FubmVycy5WVUxOLCBTY2FubmVycy5TRUNSRVRdLFxuICBleGl0Q29kZTogMSxcbiAgZXhpdE9uRW9sOiAxLFxuICB0cml2eUlnbm9yZTogWydDVkUtMjAyMy0zNzkyMCcsICdDVkUtMjAxOS0xNDY5NyBleHA6MjAyMy0wMS0wMScsICdnZW5lcmljLXVud2FudGVkLXJ1bGUnXSxcbiAgbWVtb3J5U2l6ZTogMzAwOCxcbiAgcGxhdGZvcm06ICdsaW51eC9hcm02NCcsXG59KTtcbiJdfQ==
|
package/package.json
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "image-scanner-with-trivy",
|
|
3
|
+
"description": "Scan container images with Trivy in CDK deployment",
|
|
3
4
|
"repository": {
|
|
4
5
|
"type": "git",
|
|
5
6
|
"url": "https://github.com/go-to-k/image-scanner-with-trivy"
|
|
@@ -88,7 +89,7 @@
|
|
|
88
89
|
},
|
|
89
90
|
"main": "lib/index.js",
|
|
90
91
|
"license": "Apache-2.0",
|
|
91
|
-
"version": "1.0.
|
|
92
|
+
"version": "1.0.2",
|
|
92
93
|
"types": "lib/index.d.ts",
|
|
93
94
|
"stability": "stable",
|
|
94
95
|
"jsii": {
|