cdk-serverless-clamscan
Version:
Serverless architecture to virus scan objects in Amazon S3.
409 lines (403 loc) • 65.3 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.ServerlessClamscan = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
// SPDX-License-Identifier: Apache-2.0
const path = require("path");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const aws_ec2_1 = require("aws-cdk-lib/aws-ec2");
const aws_efs_1 = require("aws-cdk-lib/aws-efs");
const aws_events_1 = require("aws-cdk-lib/aws-events");
const aws_events_targets_1 = require("aws-cdk-lib/aws-events-targets");
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
const aws_lambda_destinations_1 = require("aws-cdk-lib/aws-lambda-destinations");
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
const aws_s3_notifications_1 = require("aws-cdk-lib/aws-s3-notifications");
const aws_sqs_1 = require("aws-cdk-lib/aws-sqs");
const constructs_1 = require("constructs");
/**
An [aws-cdk](https://github.com/aws/aws-cdk) construct that uses [ClamAV®](https://www.clamav.net/).
to scan objects in Amazon S3 for viruses. The construct provides a flexible interface for a system
to act based on the results of a ClamAV virus scan.
The construct creates a Lambda function with EFS integration to support larger files.
A VPC with isolated subnets, a S3 Gateway endpoint will also be created.
Additionally creates an twice-daily job to download the latest ClamAV definition files to the
Virus Definitions S3 Bucket by utilizing an EventBridge rule and a Lambda function and
publishes CloudWatch Metrics to the 'serverless-clamscan' namespace.
__Important O&M__:
When ClamAV publishes updates to the scanner you will see “Your ClamAV installation is OUTDATED” in your scan results.
While the construct creates a system to keep the database definitions up to date, you must update the scanner to
detect all the latest Viruses.
Update the docker images of the Lambda functions with the latest version of ClamAV by re-running `cdk deploy`.
Successful Scan Event format
```json
{
"source": "serverless-clamscan",
"input_bucket": <input_bucket_name>,
"input_key": <object_key>,
"status": <"CLEAN"|"INFECTED"|"N/A">,
"message": <scan_summary>,
}
```
Note: The Virus Definitions bucket policy will likely cause a deletion error if you choose to delete
the stack associated in the construct. However since the bucket itself gets deleted, you can delete
the stack again to resolve the error.
*/
class ServerlessClamscan extends constructs_1.Construct {
/**
* Creates a ServerlessClamscan construct.
* @param scope The parent creating construct (usually `this`).
* @param id The construct's name.
* @param props A `ServerlessClamscanProps` interface.
*/
constructor(scope, id, props) {
super(scope, id);
this._efsRootPath = '/lambda';
this._efsMountPath = `/mnt${this._efsRootPath}`;
this._efsDefsPath = 'virus_database/';
this.useImportedBuckets = props.acceptResponsibilityForUsingImportedBucket;
if (!props.onResult) {
this.resultBus = new aws_events_1.EventBus(this, 'ScanResultBus');
this.resultDest = new aws_lambda_destinations_1.EventBridgeDestination(this.resultBus);
this.infectedRule = new aws_events_1.Rule(this, 'InfectedRule', {
eventBus: this.resultBus,
description: 'Event for when a file is marked INFECTED',
eventPattern: {
detail: {
responsePayload: {
source: ['serverless-clamscan'],
status: ['INFECTED'],
},
},
},
});
this.cleanRule = new aws_events_1.Rule(this, 'CleanRule', {
eventBus: this.resultBus,
description: 'Event for when a file is marked CLEAN',
eventPattern: {
detail: {
responsePayload: {
source: ['serverless-clamscan'],
status: ['CLEAN'],
},
},
},
});
}
else {
this.resultDest = props.onResult;
}
if (!props.onError) {
this.errorDeadLetterQueue = new aws_sqs_1.Queue(this, 'ScanErrorDeadLetterQueue', {
encryption: aws_sqs_1.QueueEncryption.KMS_MANAGED,
});
this.errorDeadLetterQueue.addToResourcePolicy(new aws_iam_1.PolicyStatement({
actions: ['sqs:*'],
effect: aws_iam_1.Effect.DENY,
principals: [new aws_iam_1.AnyPrincipal()],
conditions: { Bool: { 'aws:SecureTransport': false } },
resources: [this.errorDeadLetterQueue.queueArn],
}));
this.errorQueue = new aws_sqs_1.Queue(this, 'ScanErrorQueue', {
encryption: aws_sqs_1.QueueEncryption.KMS_MANAGED,
deadLetterQueue: {
maxReceiveCount: 3,
queue: this.errorDeadLetterQueue,
},
});
this.errorQueue.addToResourcePolicy(new aws_iam_1.PolicyStatement({
actions: ['sqs:*'],
effect: aws_iam_1.Effect.DENY,
principals: [new aws_iam_1.AnyPrincipal()],
conditions: { Bool: { 'aws:SecureTransport': false } },
resources: [this.errorQueue.queueArn],
}));
this.errorDest = new aws_lambda_destinations_1.SqsDestination(this.errorQueue);
}
else {
this.errorDest = props.onError;
}
const vpc = new aws_ec2_1.Vpc(this, 'ScanVPC', {
subnetConfiguration: [
{
subnetType: aws_ec2_1.SubnetType.PRIVATE_ISOLATED,
name: 'Isolated',
},
],
});
vpc.addFlowLog('FlowLogs');
this._s3Gw = vpc.addGatewayEndpoint('S3Endpoint', {
service: aws_ec2_1.GatewayVpcEndpointAwsService.S3,
});
const fileSystem = new aws_efs_1.FileSystem(this, 'ScanFileSystem', {
vpc: vpc,
encrypted: props.efsEncryption === false ? false : true,
lifecyclePolicy: aws_efs_1.LifecyclePolicy.AFTER_7_DAYS,
performanceMode: props.efsPerformanceMode ?? aws_efs_1.PerformanceMode.GENERAL_PURPOSE,
throughputMode: props.efsThroughputMode ?? aws_efs_1.ThroughputMode.BURSTING,
provisionedThroughputPerSecond: props.efsProvisionedThroughputPerSecond,
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
securityGroup: new aws_ec2_1.SecurityGroup(this, 'ScanFileSystemSecurityGroup', {
vpc: vpc,
allowAllOutbound: false,
}),
});
const lambda_ap = fileSystem.addAccessPoint('ScanLambdaAP', {
createAcl: {
ownerGid: '0',
ownerUid: '0',
permissions: '755',
},
posixUser: {
gid: '0',
uid: '0',
},
path: this._efsRootPath,
});
const logs_bucket = props.defsBucketAccessLogsConfig?.logsBucket;
const logs_bucket_prefix = props.defsBucketAccessLogsConfig?.logsPrefix;
if (logs_bucket === true || logs_bucket === undefined) {
this.defsAccessLogsBucket = new aws_s3_1.Bucket(this, 'VirusDefsAccessLogsBucket', {
encryption: aws_s3_1.BucketEncryption.S3_MANAGED,
removalPolicy: aws_cdk_lib_1.RemovalPolicy.RETAIN,
blockPublicAccess: {
blockPublicAcls: true,
blockPublicPolicy: true,
ignorePublicAcls: true,
restrictPublicBuckets: true,
},
objectOwnership: aws_s3_1.ObjectOwnership.OBJECT_WRITER,
});
this.defsAccessLogsBucket.addToResourcePolicy(new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.DENY,
actions: ['s3:*'],
resources: [
this.defsAccessLogsBucket.arnForObjects('*'),
this.defsAccessLogsBucket.bucketArn,
],
principals: [new aws_iam_1.AnyPrincipal()],
conditions: {
Bool: {
'aws:SecureTransport': false,
},
},
}));
}
else if (logs_bucket != false) {
this.defsAccessLogsBucket = logs_bucket;
}
const defs_bucket = new aws_s3_1.Bucket(this, 'VirusDefsBucket', {
encryption: aws_s3_1.BucketEncryption.S3_MANAGED,
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
autoDeleteObjects: true,
serverAccessLogsBucket: this.defsAccessLogsBucket,
serverAccessLogsPrefix: logs_bucket === false ? undefined : logs_bucket_prefix,
blockPublicAccess: {
blockPublicAcls: true,
blockPublicPolicy: true,
ignorePublicAcls: true,
restrictPublicBuckets: true,
},
});
defs_bucket.addToResourcePolicy(new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.DENY,
actions: ['s3:*'],
resources: [defs_bucket.arnForObjects('*'), defs_bucket.bucketArn],
principals: [new aws_iam_1.AnyPrincipal()],
conditions: {
Bool: {
'aws:SecureTransport': false,
},
},
}));
defs_bucket.addToResourcePolicy(new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.ALLOW,
actions: ['s3:GetObject', 's3:ListBucket'],
resources: [defs_bucket.arnForObjects('*'), defs_bucket.bucketArn],
principals: [new aws_iam_1.AnyPrincipal()],
conditions: {
StringEquals: {
'aws:SourceVpce': this._s3Gw.vpcEndpointId,
},
},
}));
if (props.defsBucketAllowPolicyMutation !== true) {
defs_bucket.addToResourcePolicy(new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.DENY,
actions: ['s3:PutBucketPolicy', 's3:DeleteBucketPolicy'],
resources: [defs_bucket.bucketArn],
notPrincipals: [new aws_iam_1.AccountRootPrincipal()],
}));
}
this._s3Gw.addToPolicy(new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.ALLOW,
actions: ['s3:GetObject', 's3:ListBucket'],
resources: [defs_bucket.arnForObjects('*'), defs_bucket.bucketArn],
principals: [new aws_iam_1.AnyPrincipal()],
}));
this._scanFunction = new aws_lambda_1.DockerImageFunction(this, 'ServerlessClamscan', {
code: aws_lambda_1.DockerImageCode.fromImageAsset(path.join(__dirname, '../assets/lambda/code/scan'), {
buildArgs: {
// Only force update the docker layer cache once a day
CACHE_DATE: new Date().toDateString(),
},
extraHash: Date.now().toString(),
}),
onSuccess: this.resultDest,
onFailure: this.errorDest,
filesystem: aws_lambda_1.FileSystem.fromEfsAccessPoint(lambda_ap, this._efsMountPath),
vpc: vpc,
vpcSubnets: { subnets: vpc.isolatedSubnets },
allowAllOutbound: false,
timeout: props.scanFunctionTimeout ?? aws_cdk_lib_1.Duration.minutes(15),
memorySize: props.scanFunctionMemorySize ?? 10240,
reservedConcurrentExecutions: props.reservedConcurrency,
environment: {
EFS_MOUNT_PATH: this._efsMountPath,
EFS_DEF_PATH: this._efsDefsPath,
DEFS_URL: defs_bucket.virtualHostedUrlForObject(),
POWERTOOLS_METRICS_NAMESPACE: 'serverless-clamscan',
POWERTOOLS_SERVICE_NAME: 'virus-scan',
},
});
this._scanFunction.connections.allowToAnyIpv4(aws_ec2_1.Port.tcp(443), 'Allow outbound HTTPS traffic for S3 access.');
defs_bucket.grantRead(this._scanFunction);
const download_defs = new aws_lambda_1.DockerImageFunction(this, 'DownloadDefs', {
code: aws_lambda_1.DockerImageCode.fromImageAsset(path.join(__dirname, '../assets/lambda/code/download_defs'), {
buildArgs: {
// Only force update the docker layer cache once a day
CACHE_DATE: new Date().toDateString(),
},
extraHash: Date.now().toString(),
}),
timeout: aws_cdk_lib_1.Duration.minutes(5),
memorySize: 1024,
environment: {
DEFS_BUCKET: defs_bucket.bucketName,
POWERTOOLS_SERVICE_NAME: 'freshclam-update',
},
});
const stack = aws_cdk_lib_1.Stack.of(this);
if (download_defs.role) {
const download_defs_role = `arn:${stack.partition}:sts::${stack.account}:assumed-role/${download_defs.role.roleName}/${download_defs.functionName}`;
const download_defs_assumed_principal = new aws_iam_1.ArnPrincipal(download_defs_role);
defs_bucket.addToResourcePolicy(new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.DENY,
actions: ['s3:PutObject*'],
resources: [defs_bucket.arnForObjects('*')],
notPrincipals: [download_defs.role, download_defs_assumed_principal],
}));
defs_bucket.grantReadWrite(download_defs);
}
new aws_events_1.Rule(this, 'VirusDefsUpdateRule', {
schedule: aws_events_1.Schedule.rate(aws_cdk_lib_1.Duration.hours(12)),
targets: [new aws_events_targets_1.LambdaFunction(download_defs)],
});
const init_defs_cr = new aws_lambda_1.Function(this, 'InitDefs', {
runtime: aws_lambda_1.Runtime.PYTHON_3_9,
code: aws_lambda_1.Code.fromAsset(path.join(__dirname, '../assets/lambda/code/initialize_defs_cr')),
handler: 'lambda.lambda_handler',
timeout: aws_cdk_lib_1.Duration.minutes(5),
});
download_defs.grantInvoke(init_defs_cr);
new aws_cdk_lib_1.CustomResource(this, 'InitDefsCr', {
serviceToken: init_defs_cr.functionArn,
properties: {
FnName: download_defs.functionName,
},
});
if (props.buckets) {
props.buckets.forEach((bucket) => {
this.addSourceBucket(bucket);
});
}
}
/**
* @returns ArnPrincipal the ARN of the assumed role principal for the scan function
*/
get scanAssumedPrincipal() {
if (this._scanFunction.role) {
const stack = aws_cdk_lib_1.Stack.of(this);
const scan_assumed_role = `arn:${stack.partition}:sts::${stack.account}:assumed-role/${this._scanFunction.role.roleName}/${this._scanFunction.functionName}`;
return new aws_iam_1.ArnPrincipal(scan_assumed_role);
}
else {
throw new Error('The scan function role is undefined');
}
}
/**
* Returns the statement that should be added to the bucket policy
in order to prevent objects to be accessed when they are not clean
or there have been scanning errors: this policy should be added
manually if external buckets are passed to addSourceBucket()
* @param bucket The bucket which you need to protect with the policy
* @returns PolicyStatement the policy statement if available
*/
getPolicyStatementForBucket(bucket) {
if (this._scanFunction.role) {
const scan_assumed_principal = this.scanAssumedPrincipal;
return new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.DENY,
actions: ['s3:GetObject'],
resources: [bucket.arnForObjects('*')],
principals: [new aws_iam_1.AnyPrincipal()],
conditions: {
StringEquals: {
's3:ExistingObjectTag/scan-status': [
'IN PROGRESS',
'INFECTED',
'ERROR',
],
},
ArnNotEquals: {
'aws:PrincipalArn': [this._scanFunction.role.roleArn, scan_assumed_principal.arn],
},
},
});
}
else {
throw new Error("Can't generate a valid S3 bucket policy, the scan function role is undefined");
}
}
/**
* Sets the specified S3 Bucket as a s3:ObjectCreate* for the ClamAV function.
Grants the ClamAV function permissions to get and tag objects.
Adds a bucket policy to disallow GetObject operations on files that are tagged 'IN PROGRESS', 'INFECTED', or 'ERROR'.
* @param bucket The bucket to add the scanning bucket policy and s3:ObjectCreate* trigger to.
*/
addSourceBucket(bucket) {
bucket.addEventNotification(aws_s3_1.EventType.OBJECT_CREATED, new aws_s3_notifications_1.LambdaDestination(this._scanFunction));
bucket.grantRead(this._scanFunction);
this._scanFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.ALLOW,
actions: ['s3:GetObjectTagging', 's3:GetObjectVersionTagging', 's3:PutObjectTagging', 's3:PutObjectVersionTagging'],
resources: [bucket.arnForObjects('*')],
}));
if (this._scanFunction.role) {
const scan_assumed_principal = this.scanAssumedPrincipal;
this._s3Gw.addToPolicy(new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.ALLOW,
actions: ['s3:GetObject*', 's3:GetBucket*', 's3:List*'],
resources: [bucket.bucketArn, bucket.arnForObjects('*')],
principals: [this._scanFunction.role, scan_assumed_principal],
}));
this._s3Gw.addToPolicy(new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.ALLOW,
actions: ['s3:GetObjectTagging', 's3:GetObjectVersionTagging', 's3:PutObjectTagging', 's3:PutObjectVersionTagging'],
resources: [bucket.arnForObjects('*')],
principals: [this._scanFunction.role, scan_assumed_principal],
}));
const result = bucket.addToResourcePolicy(this.getPolicyStatementForBucket(bucket));
if (!result.statementAdded && !this.useImportedBuckets) {
throw new Error('acceptResponsibilityForUsingImportedBucket must be set when adding an imported bucket. When using imported buckets the user is responsible for adding the required policy statement to the bucket policy: `getPolicyStatementForBucket()` can be used to retrieve the policy statement required by the solution');
}
}
}
}
exports.ServerlessClamscan = ServerlessClamscan;
_a = JSII_RTTI_SYMBOL_1;
ServerlessClamscan[_a] = { fqn: "cdk-serverless-clamscan.ServerlessClamscan", version: "2.12.1" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport * as path from 'path';\nimport {\n  CustomResource, Duration, RemovalPolicy,\n  Stack,\n} from 'aws-cdk-lib';\nimport {\n  GatewayVpcEndpoint,\n  GatewayVpcEndpointAwsService,\n  Port,\n  SecurityGroup, SubnetType, Vpc,\n} from 'aws-cdk-lib/aws-ec2';\nimport { FileSystem, LifecyclePolicy, PerformanceMode, ThroughputMode } from 'aws-cdk-lib/aws-efs';\nimport { EventBus, Rule, Schedule } from 'aws-cdk-lib/aws-events';\nimport { LambdaFunction } from 'aws-cdk-lib/aws-events-targets';\nimport {\n  AccountRootPrincipal,\n  AddToResourcePolicyResult, AnyPrincipal, ArnPrincipal, Effect,\n  PolicyStatement,\n} from 'aws-cdk-lib/aws-iam';\nimport {\n  Code, DockerImageCode,\n  DockerImageFunction,\n  Function,\n  IDestination,\n  FileSystem as LambdaFileSystem,\n  Runtime,\n} from 'aws-cdk-lib/aws-lambda';\nimport {\n  EventBridgeDestination,\n  SqsDestination,\n} from 'aws-cdk-lib/aws-lambda-destinations';\nimport { Bucket, BucketEncryption, EventType, IBucket, ObjectOwnership } from 'aws-cdk-lib/aws-s3';\nimport { LambdaDestination } from 'aws-cdk-lib/aws-s3-notifications';\nimport { Queue, QueueEncryption } from 'aws-cdk-lib/aws-sqs';\nimport { Size } from 'aws-cdk-lib/core';\nimport { Construct } from 'constructs';\n\n/**\n * Interface for ServerlessClamscan Virus Definitions S3 Bucket Logging.\n */\nexport interface ServerlessClamscanLoggingProps {\n  /**\n   * Destination bucket for the server access logs (Default: Creates a new S3 Bucket for access logs).\n   */\n  readonly logsBucket?: boolean | IBucket;\n  /**\n   * Optional log file prefix to use for the bucket's access logs, option is ignored if logs_bucket is set to false.\n   */\n  readonly logsPrefix?: string;\n}\n\n/**\n * Interface for creating a ServerlessClamscan.\n */\nexport interface ServerlessClamscanProps {\n  /**\n   * An optional list of S3 buckets to configure for ClamAV Virus Scanning; buckets can be added later by calling addSourceBucket.\n   */\n  readonly buckets?: IBucket[];\n  /**\n   * Optionally set a reserved concurrency for the virus scanning Lambda.\n   * @see https://docs.aws.amazon.com/lambda/latest/operatorguide/reserved-concurrency.html\n   */\n  readonly reservedConcurrency?: number;\n  /**\n   * Optionally set the memory allocation for the scan function. Note that low memory allocations may cause errors. (Default: 10240).\n   * @see https://docs.aws.amazon.com/lambda/latest/operatorguide/computing-power.html\n   */\n  readonly scanFunctionMemorySize?: number;\n  /**\n   * Optionally set the timeout for the scan function. (Default: 15 minutes).\n   */\n  readonly scanFunctionTimeout?: Duration;\n  /**\n   * The Lambda Destination for files marked 'CLEAN' or 'INFECTED' based on the ClamAV Virus scan or 'N/A' for scans triggered by S3 folder creation events marked (Default: Creates and publishes to a new Event Bridge Bus if unspecified).\n   */\n  readonly onResult?: IDestination;\n  /**\n   * The Lambda Destination for files that fail to scan and are marked 'ERROR' or stuck 'IN PROGRESS' due to a Lambda timeout (Default: Creates and publishes to a new SQS queue if unspecified).\n   */\n  readonly onError?: IDestination;\n  /**\n   * Whether or not to enable encryption on EFS filesystem (Default: enabled).\n   */\n  readonly efsEncryption?: boolean;\n  /**\n   * Set the performance mode of the EFS file system (Default: GENERAL_PURPOSE).\n   */\n  readonly efsPerformanceMode?: PerformanceMode;\n  /**\n   * Set the throughput mode of the EFS file system (Default: BURSTING).\n   */\n  readonly efsThroughputMode?: ThroughputMode;\n  /**\n   * Provisioned throughput for the EFS file system. This is a required property if the throughput mode is set to PROVISIONED. Must be at least 1MiB/s (Default: none).\n   */\n  readonly efsProvisionedThroughputPerSecond?: Size;\n  /**\n   * Whether or not to enable Access Logging for the Virus Definitions bucket, you can specify an existing bucket and prefix (Default: Creates a new S3 Bucket for access logs).\n   */\n  readonly defsBucketAccessLogsConfig?: ServerlessClamscanLoggingProps;\n  /**\n   * Allows the use of imported buckets. When using imported buckets the user is responsible for adding the required policy statement to the bucket policy: `getPolicyStatementForBucket()` can be used to retrieve the policy statement required by the solution.\n   */\n  readonly acceptResponsibilityForUsingImportedBucket?: boolean;\n  /**\n   * Allow for non-root users to modify/delete the bucket policy on the Virus Definitions bucket.\n   * Warning: changing this flag from 'false' to 'true' on existing deployments will cause updates to fail.\n   * @default false\n   */\n  readonly defsBucketAllowPolicyMutation?: boolean;\n}\n\n/**\n  An [aws-cdk](https://github.com/aws/aws-cdk) construct that uses [ClamAV®](https://www.clamav.net/).\n  to scan objects in Amazon S3 for viruses. The construct provides a flexible interface for a system\n  to act based on the results of a ClamAV virus scan.\n\n  The construct creates a Lambda function with EFS integration to support larger files.\n  A VPC with isolated subnets, a S3 Gateway endpoint will also be created.\n\n  Additionally creates an twice-daily job to download the latest ClamAV definition files to the\n  Virus Definitions S3 Bucket by utilizing an EventBridge rule and a Lambda function and\n  publishes CloudWatch Metrics to the 'serverless-clamscan' namespace.\n\n  __Important O&M__:\n  When ClamAV publishes updates to the scanner you will see “Your ClamAV installation is OUTDATED” in your scan results.\n  While the construct creates a system to keep the database definitions up to date, you must update the scanner to\n  detect all the latest Viruses.\n\n  Update the docker images of the Lambda functions with the latest version of ClamAV by re-running `cdk deploy`.\n\n  Successful Scan Event format\n  ```json\n  {\n     \"source\": \"serverless-clamscan\",\n     \"input_bucket\": <input_bucket_name>,\n     \"input_key\": <object_key>,\n     \"status\": <\"CLEAN\"|\"INFECTED\"|\"N/A\">,\n     \"message\": <scan_summary>,\n   }\n  ```\n\n  Note: The Virus Definitions bucket policy will likely cause a deletion error if you choose to delete\n  the stack associated in the construct. However since the bucket itself gets deleted, you can delete\n  the stack again to resolve the error.\n */\nexport class ServerlessClamscan extends Construct {\n  /**\n    The Lambda Destination for failed on erred scans [ERROR, IN PROGRESS (If error is due to Lambda timeout)].\n   */\n  public readonly errorDest: IDestination;\n\n  /**\n    The Lambda Destination for completed ClamAV scans [CLEAN, INFECTED].\n   */\n  public readonly resultDest: IDestination;\n\n  /**\n    Conditional: The SQS Queue for erred scans if a failure (onError) destination was not specified.\n   */\n  public readonly errorQueue?: Queue;\n\n  /**\n    Conditional: The SQS Dead Letter Queue for the errorQueue if a failure (onError) destination was not specified.\n   */\n  public readonly errorDeadLetterQueue?: Queue;\n\n  /**\n    Conditional: The Event Bridge Bus for completed ClamAV scans if a success (onResult) destination was not specified.\n   */\n  public readonly resultBus?: EventBus;\n\n  /**\n    Conditional: An Event Bridge Rule for files that are marked 'CLEAN' by ClamAV if a success destination was not specified.\n   */\n  public readonly cleanRule?: Rule;\n\n  /**\n    Conditional: An Event Bridge Rule for files that are marked 'INFECTED' by ClamAV if a success destination was not specified.\n   */\n  public readonly infectedRule?: Rule;\n\n  /**\n    Conditional: The Bucket for access logs for the virus definitions bucket if logging is enabled (defsBucketAccessLogsConfig).\n   */\n  public readonly defsAccessLogsBucket?: IBucket;\n\n  /**\n    Conditional: When true, the user accepted the responsibility for using imported buckets.\n   */\n  public readonly useImportedBuckets?: boolean;\n\n  private _scanFunction: DockerImageFunction;\n  private _s3Gw: GatewayVpcEndpoint;\n  private _efsRootPath = '/lambda';\n  private _efsMountPath = `/mnt${this._efsRootPath}`;\n  private _efsDefsPath = 'virus_database/';\n\n  /**\n   * Creates a ServerlessClamscan construct.\n   * @param scope The parent creating construct (usually `this`).\n   * @param id The construct's name.\n   * @param props A `ServerlessClamscanProps` interface.\n   */\n  constructor(scope: Construct, id: string, props: ServerlessClamscanProps) {\n    super(scope, id);\n\n    this.useImportedBuckets = props.acceptResponsibilityForUsingImportedBucket;\n\n    if (!props.onResult) {\n      this.resultBus = new EventBus(this, 'ScanResultBus');\n      this.resultDest = new EventBridgeDestination(this.resultBus);\n      this.infectedRule = new Rule(this, 'InfectedRule', {\n        eventBus: this.resultBus,\n        description: 'Event for when a file is marked INFECTED',\n        eventPattern: {\n          detail: {\n            responsePayload: {\n              source: ['serverless-clamscan'],\n              status: ['INFECTED'],\n            },\n          },\n        },\n      });\n      this.cleanRule = new Rule(this, 'CleanRule', {\n        eventBus: this.resultBus,\n        description: 'Event for when a file is marked CLEAN',\n        eventPattern: {\n          detail: {\n            responsePayload: {\n              source: ['serverless-clamscan'],\n              status: ['CLEAN'],\n            },\n          },\n        },\n      });\n    } else {\n      this.resultDest = props.onResult;\n    }\n\n    if (!props.onError) {\n      this.errorDeadLetterQueue = new Queue(this, 'ScanErrorDeadLetterQueue', {\n        encryption: QueueEncryption.KMS_MANAGED,\n      });\n      this.errorDeadLetterQueue.addToResourcePolicy(new PolicyStatement({\n        actions: ['sqs:*'],\n        effect: Effect.DENY,\n        principals: [new AnyPrincipal()],\n        conditions: { Bool: { 'aws:SecureTransport': false } },\n        resources: [this.errorDeadLetterQueue.queueArn],\n      }));\n      this.errorQueue = new Queue(this, 'ScanErrorQueue', {\n        encryption: QueueEncryption.KMS_MANAGED,\n        deadLetterQueue: {\n          maxReceiveCount: 3,\n          queue: this.errorDeadLetterQueue,\n        },\n      });\n      this.errorQueue.addToResourcePolicy(new PolicyStatement({\n        actions: ['sqs:*'],\n        effect: Effect.DENY,\n        principals: [new AnyPrincipal()],\n        conditions: { Bool: { 'aws:SecureTransport': false } },\n        resources: [this.errorQueue.queueArn],\n      }));\n      this.errorDest = new SqsDestination(this.errorQueue);\n    } else {\n      this.errorDest = props.onError;\n    }\n\n    const vpc = new Vpc(this, 'ScanVPC', {\n      subnetConfiguration: [\n        {\n          subnetType: SubnetType.PRIVATE_ISOLATED,\n          name: 'Isolated',\n        },\n      ],\n    });\n\n    vpc.addFlowLog('FlowLogs');\n\n    this._s3Gw = vpc.addGatewayEndpoint('S3Endpoint', {\n      service: GatewayVpcEndpointAwsService.S3,\n    });\n\n    const fileSystem = new FileSystem(this, 'ScanFileSystem', {\n      vpc: vpc,\n      encrypted: props.efsEncryption === false ? false : true,\n      lifecyclePolicy: LifecyclePolicy.AFTER_7_DAYS,\n      performanceMode: props.efsPerformanceMode ?? PerformanceMode.GENERAL_PURPOSE,\n      thro