token-injectable-docker-builder 0.1.0 → 0.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/lib/index.d.ts +16 -0
- package/lib/index.js +140 -0
- package/package.json +1 -1
- package/test/docker_image_asset.test.d.ts +1 -0
- package/test/docker_image_asset.test.js +30 -0
package/lib/index.d.ts
ADDED
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { DockerImageCode } from 'aws-cdk-lib/aws-lambda';
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
import { ContainerImage } from 'aws-cdk-lib/aws-ecs';
|
|
4
|
+
export interface TokenInjectableDockerBuilderProps {
|
|
5
|
+
path: string;
|
|
6
|
+
buildArgs?: {
|
|
7
|
+
[key: string]: string;
|
|
8
|
+
};
|
|
9
|
+
}
|
|
10
|
+
export declare class TokenInjectableDockerBuilder extends Construct {
|
|
11
|
+
private readonly ecrRepository;
|
|
12
|
+
private readonly buildTriggerResource;
|
|
13
|
+
constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps);
|
|
14
|
+
getContainerImage(): ContainerImage;
|
|
15
|
+
getDockerImageCode(): DockerImageCode;
|
|
16
|
+
}
|
package/lib/index.js
ADDED
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.TokenInjectableDockerBuilder = void 0;
|
|
4
|
+
const path = require("path");
|
|
5
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
6
|
+
const aws_codebuild_1 = require("aws-cdk-lib/aws-codebuild");
|
|
7
|
+
const aws_ecr_1 = require("aws-cdk-lib/aws-ecr");
|
|
8
|
+
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
|
9
|
+
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
|
|
10
|
+
const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets");
|
|
11
|
+
const custom_resources_1 = require("aws-cdk-lib/custom-resources");
|
|
12
|
+
const constructs_1 = require("constructs");
|
|
13
|
+
const aws_ecs_1 = require("aws-cdk-lib/aws-ecs");
|
|
14
|
+
const crypto = require("crypto");
|
|
15
|
+
const aws_lambda_2 = require("aws-cdk-lib/aws-lambda");
|
|
16
|
+
class TokenInjectableDockerBuilder extends constructs_1.Construct {
|
|
17
|
+
constructor(scope, id, props) {
|
|
18
|
+
super(scope, id);
|
|
19
|
+
const { path: sourcePath, buildArgs } = props; // Default to linux/amd64
|
|
20
|
+
// Define absolute paths for Lambda handlers
|
|
21
|
+
const onEventHandlerPath = path.resolve(__dirname, '../src/onEventHandler');
|
|
22
|
+
const isCompleteHandlerPath = path.resolve(__dirname, '../src/isCompleteHandler');
|
|
23
|
+
// Create an ECR repository
|
|
24
|
+
this.ecrRepository = new aws_ecr_1.Repository(this, 'ECRRepository');
|
|
25
|
+
// Package the source code as an asset
|
|
26
|
+
const sourceAsset = new aws_s3_assets_1.Asset(this, 'SourceAsset', {
|
|
27
|
+
path: sourcePath, // Path to the Dockerfile or source code
|
|
28
|
+
});
|
|
29
|
+
// Transform buildArgs into a string of --build-arg KEY=VALUE
|
|
30
|
+
const buildArgsString = buildArgs
|
|
31
|
+
? Object.entries(buildArgs)
|
|
32
|
+
.map(([key, value]) => `--build-arg ${key}=${value}`)
|
|
33
|
+
.join(' ')
|
|
34
|
+
: '';
|
|
35
|
+
// Pass the buildArgsString and platform as environment variables
|
|
36
|
+
const environmentVariables = {
|
|
37
|
+
ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },
|
|
38
|
+
BUILD_ARGS: { value: buildArgsString },
|
|
39
|
+
};
|
|
40
|
+
// Create a CodeBuild project
|
|
41
|
+
const codeBuildProject = new aws_codebuild_1.Project(this, 'UICodeBuildProject', {
|
|
42
|
+
source: aws_codebuild_1.Source.s3({
|
|
43
|
+
bucket: sourceAsset.bucket,
|
|
44
|
+
path: sourceAsset.s3ObjectKey,
|
|
45
|
+
}),
|
|
46
|
+
environment: {
|
|
47
|
+
buildImage: aws_codebuild_1.LinuxBuildImage.STANDARD_7_0,
|
|
48
|
+
privileged: true, // Required for Docker builds
|
|
49
|
+
},
|
|
50
|
+
environmentVariables: environmentVariables,
|
|
51
|
+
buildSpec: aws_codebuild_1.BuildSpec.fromObject({
|
|
52
|
+
version: '0.2',
|
|
53
|
+
phases: {
|
|
54
|
+
pre_build: {
|
|
55
|
+
commands: [
|
|
56
|
+
'echo "Retrieving AWS Account ID..."',
|
|
57
|
+
'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',
|
|
58
|
+
'echo "Logging in to Amazon ECR..."',
|
|
59
|
+
'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com',
|
|
60
|
+
],
|
|
61
|
+
},
|
|
62
|
+
build: {
|
|
63
|
+
commands: [
|
|
64
|
+
'echo Build phase: Building the Docker image...',
|
|
65
|
+
'docker build $BUILD_ARGS -t $ECR_REPO_URI:latest $CODEBUILD_SRC_DIR',
|
|
66
|
+
],
|
|
67
|
+
},
|
|
68
|
+
post_build: {
|
|
69
|
+
commands: [
|
|
70
|
+
'echo Post-build phase: Pushing the Docker image...',
|
|
71
|
+
'docker push $ECR_REPO_URI:latest',
|
|
72
|
+
],
|
|
73
|
+
},
|
|
74
|
+
},
|
|
75
|
+
}),
|
|
76
|
+
});
|
|
77
|
+
// Grant permissions to interact with ECR
|
|
78
|
+
this.ecrRepository.grantPullPush(codeBuildProject);
|
|
79
|
+
codeBuildProject.role.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
|
|
80
|
+
actions: ['ecr:GetAuthorizationToken'],
|
|
81
|
+
resources: ['*'],
|
|
82
|
+
}));
|
|
83
|
+
// Grant permissions to CodeBuild for CloudWatch Logs
|
|
84
|
+
codeBuildProject.role.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
|
|
85
|
+
actions: ['logs:PutLogEvents', 'logs:CreateLogGroup', 'logs:CreateLogStream'],
|
|
86
|
+
resources: [`arn:aws:logs:${aws_cdk_lib_1.Stack.of(this).region}:${aws_cdk_lib_1.Stack.of(this).account}:*`],
|
|
87
|
+
}));
|
|
88
|
+
// Create Node.js Lambda function for onEvent
|
|
89
|
+
const onEventHandlerFunction = new aws_lambda_2.Function(this, 'OnEventHandlerFunction', {
|
|
90
|
+
runtime: aws_lambda_1.Runtime.NODEJS_18_X, // Use Node.js runtime
|
|
91
|
+
code: aws_lambda_1.Code.fromAsset(onEventHandlerPath), // Path to handler code
|
|
92
|
+
handler: 'index.handler', // Entry point (adjust as needed)
|
|
93
|
+
timeout: aws_cdk_lib_1.Duration.minutes(15),
|
|
94
|
+
});
|
|
95
|
+
onEventHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
|
|
96
|
+
actions: ['codebuild:StartBuild'],
|
|
97
|
+
resources: [codeBuildProject.projectArn], // Restrict to specific project
|
|
98
|
+
}));
|
|
99
|
+
// Create Node.js Lambda function for isComplete
|
|
100
|
+
const isCompleteHandlerFunction = new aws_lambda_2.Function(this, 'IsCompleteHandlerFunction', {
|
|
101
|
+
runtime: aws_lambda_1.Runtime.NODEJS_18_X,
|
|
102
|
+
code: aws_lambda_1.Code.fromAsset(isCompleteHandlerPath),
|
|
103
|
+
handler: 'index.handler',
|
|
104
|
+
timeout: aws_cdk_lib_1.Duration.minutes(15),
|
|
105
|
+
});
|
|
106
|
+
isCompleteHandlerFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({
|
|
107
|
+
actions: [
|
|
108
|
+
'codebuild:BatchGetBuilds',
|
|
109
|
+
'codebuild:ListBuildsForProject',
|
|
110
|
+
'logs:GetLogEvents',
|
|
111
|
+
'logs:DescribeLogStreams',
|
|
112
|
+
'logs:DescribeLogGroups'
|
|
113
|
+
],
|
|
114
|
+
resources: ['*'],
|
|
115
|
+
}));
|
|
116
|
+
// Create a custom resource provider
|
|
117
|
+
const provider = new custom_resources_1.Provider(this, 'CustomResourceProvider', {
|
|
118
|
+
onEventHandler: onEventHandlerFunction,
|
|
119
|
+
isCompleteHandler: isCompleteHandlerFunction,
|
|
120
|
+
queryInterval: aws_cdk_lib_1.Duration.minutes(1),
|
|
121
|
+
});
|
|
122
|
+
// Define the custom resource
|
|
123
|
+
this.buildTriggerResource = new aws_cdk_lib_1.CustomResource(this, 'BuildTriggerResource', {
|
|
124
|
+
serviceToken: provider.serviceToken,
|
|
125
|
+
properties: {
|
|
126
|
+
ProjectName: codeBuildProject.projectName,
|
|
127
|
+
Trigger: crypto.randomUUID(),
|
|
128
|
+
},
|
|
129
|
+
});
|
|
130
|
+
this.buildTriggerResource.node.addDependency(codeBuildProject);
|
|
131
|
+
}
|
|
132
|
+
getContainerImage() {
|
|
133
|
+
return aws_ecs_1.ContainerImage.fromEcrRepository(this.ecrRepository, 'latest');
|
|
134
|
+
}
|
|
135
|
+
getDockerImageCode() {
|
|
136
|
+
return aws_lambda_1.DockerImageCode.fromEcr(this.ecrRepository);
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
exports.TokenInjectableDockerBuilder = TokenInjectableDockerBuilder;
|
|
140
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { Duration, CustomResource, Stack } from 'aws-cdk-lib';\nimport { Project, Source, LinuxBuildImage, BuildSpec } from 'aws-cdk-lib/aws-codebuild';\nimport { Repository } from 'aws-cdk-lib/aws-ecr';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Code, DockerImageCode, Runtime } from 'aws-cdk-lib/aws-lambda';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\nimport { ContainerImage } from 'aws-cdk-lib/aws-ecs';\nimport * as crypto from 'crypto';\nimport { Function } from 'aws-cdk-lib/aws-lambda';\n\nexport interface TokenInjectableDockerBuilderProps {\n  path: string;\n  buildArgs?: { [key: string]: string };\n}\n\nexport class TokenInjectableDockerBuilder extends Construct {\n  private readonly ecrRepository: Repository;\n  private readonly buildTriggerResource: CustomResource;\n\n  constructor(scope: Construct, id: string, props: TokenInjectableDockerBuilderProps) {\n    super(scope, id);\n\n    const { path: sourcePath, buildArgs } = props; // Default to linux/amd64\n\n    // Define absolute paths for Lambda handlers\n    const onEventHandlerPath = path.resolve(__dirname, '../src/onEventHandler');\n    const isCompleteHandlerPath = path.resolve(__dirname, '../src/isCompleteHandler');\n\n    // Create an ECR repository\n    this.ecrRepository = new Repository(this, 'ECRRepository');\n\n    // Package the source code as an asset\n    const sourceAsset = new Asset(this, 'SourceAsset', {\n      path: sourcePath, // Path to the Dockerfile or source code\n    });\n\n    // Transform buildArgs into a string of --build-arg KEY=VALUE\n    const buildArgsString = buildArgs\n      ? Object.entries(buildArgs)\n          .map(([key, value]) => `--build-arg ${key}=${value}`)\n          .join(' ')\n      : '';\n\n    // Pass the buildArgsString and platform as environment variables\n    const environmentVariables: { [name: string]: { value: string } } = {\n      ECR_REPO_URI: { value: this.ecrRepository.repositoryUri },\n      BUILD_ARGS: { value: buildArgsString },\n    };\n\n    // Create a CodeBuild project\n    const codeBuildProject = new Project(this, 'UICodeBuildProject', {\n      source: Source.s3({\n        bucket: sourceAsset.bucket,\n        path: sourceAsset.s3ObjectKey,\n      }),\n      environment: {\n        buildImage: LinuxBuildImage.STANDARD_7_0,\n        privileged: true, // Required for Docker builds\n      },\n      environmentVariables: environmentVariables,\n      buildSpec: BuildSpec.fromObject({\n        version: '0.2',\n        phases: {\n          pre_build: {\n            commands: [\n              'echo \"Retrieving AWS Account ID...\"',\n              'export ACCOUNT_ID=$(aws sts get-caller-identity --query Account --output text)',\n              'echo \"Logging in to Amazon ECR...\"',\n              'aws ecr get-login-password --region $AWS_DEFAULT_REGION | docker login --username AWS --password-stdin $ACCOUNT_ID.dkr.ecr.$AWS_DEFAULT_REGION.amazonaws.com',\n            ],\n          },\n          build: {\n            commands: [\n              'echo Build phase: Building the Docker image...',\n              'docker build $BUILD_ARGS -t $ECR_REPO_URI:latest $CODEBUILD_SRC_DIR',\n            ],\n          },\n          post_build: {\n            commands: [\n              'echo Post-build phase: Pushing the Docker image...',\n              'docker push $ECR_REPO_URI:latest',\n            ],\n          },\n        },\n      }),\n    });\n\n    // Grant permissions to interact with ECR\n    this.ecrRepository.grantPullPush(codeBuildProject);\n\n    codeBuildProject.role!.addToPrincipalPolicy(\n      new PolicyStatement({\n        actions: ['ecr:GetAuthorizationToken'],\n        resources: ['*'],\n      })\n    );\n\n    // Grant permissions to CodeBuild for CloudWatch Logs\n    codeBuildProject.role!.addToPrincipalPolicy(\n      new PolicyStatement({\n        actions: ['logs:PutLogEvents', 'logs:CreateLogGroup', 'logs:CreateLogStream'],\n        resources: [`arn:aws:logs:${Stack.of(this).region}:${Stack.of(this).account}:*`],\n      })\n    );\n\n    // Create Node.js Lambda function for onEvent\n    const onEventHandlerFunction = new Function(this, 'OnEventHandlerFunction', {\n      runtime: Runtime.NODEJS_18_X, // Use Node.js runtime\n      code: Code.fromAsset(onEventHandlerPath), // Path to handler code\n      handler: 'index.handler', // Entry point (adjust as needed)\n      timeout: Duration.minutes(15),\n    });\n\n    onEventHandlerFunction.addToRolePolicy(\n      new PolicyStatement({\n        actions: ['codebuild:StartBuild'],\n        resources: [codeBuildProject.projectArn], // Restrict to specific project\n      })\n    );\n\n    // Create Node.js Lambda function for isComplete\n    const isCompleteHandlerFunction = new Function(this, 'IsCompleteHandlerFunction', {\n      runtime: Runtime.NODEJS_18_X,\n      code: Code.fromAsset(isCompleteHandlerPath),\n      handler: 'index.handler',\n      timeout: Duration.minutes(15),\n    });\n\n    isCompleteHandlerFunction.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\n          'codebuild:BatchGetBuilds',\n          'codebuild:ListBuildsForProject',\n          'logs:GetLogEvents',\n          'logs:DescribeLogStreams',\n          'logs:DescribeLogGroups'\n        ],\n        resources: ['*'],\n      })\n    );\n\n    // Create a custom resource provider\n    const provider = new Provider(this, 'CustomResourceProvider', {\n      onEventHandler: onEventHandlerFunction,\n      isCompleteHandler: isCompleteHandlerFunction,\n      queryInterval: Duration.minutes(1),\n    });\n\n    // Define the custom resource\n    this.buildTriggerResource = new CustomResource(this, 'BuildTriggerResource', {\n      serviceToken: provider.serviceToken,\n      properties: {\n        ProjectName: codeBuildProject.projectName,\n        Trigger: crypto.randomUUID(),\n      },\n    });\n\n    this.buildTriggerResource.node.addDependency(codeBuildProject);\n  }\n\n  public getContainerImage(): ContainerImage {\n    return ContainerImage.fromEcrRepository(this.ecrRepository, 'latest');\n  }\n\n  public getDockerImageCode(): DockerImageCode {\n    return DockerImageCode.fromEcr(this.ecrRepository);\n  }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const cdk = require("aws-cdk-lib");
|
|
4
|
+
const assertions_1 = require("aws-cdk-lib/assertions");
|
|
5
|
+
const index_1 = require("../lib/index");
|
|
6
|
+
test('DockerImageAsset creates required resources', () => {
|
|
7
|
+
const app = new cdk.App();
|
|
8
|
+
const stack = new cdk.Stack(app, 'TestStack');
|
|
9
|
+
new index_1.TokenInjectableDockerBuilder(stack, 'TestDockerImageAsset', {
|
|
10
|
+
path: './src/onEventHandler', // Path to Docker context
|
|
11
|
+
buildArgs: { ENV: 'test' },
|
|
12
|
+
});
|
|
13
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
14
|
+
// Verify that an ECR repository is created
|
|
15
|
+
template.resourceCountIs('AWS::ECR::Repository', 1);
|
|
16
|
+
// Verify that a CodeBuild project is created with expected properties
|
|
17
|
+
template.hasResourceProperties('AWS::CodeBuild::Project', {
|
|
18
|
+
Environment: {
|
|
19
|
+
ComputeType: 'BUILD_GENERAL1_SMALL',
|
|
20
|
+
PrivilegedMode: true,
|
|
21
|
+
Image: 'aws/codebuild/standard:7.0',
|
|
22
|
+
},
|
|
23
|
+
Source: {
|
|
24
|
+
Type: 'S3',
|
|
25
|
+
},
|
|
26
|
+
});
|
|
27
|
+
// Verify the Custom Resource is created with the expected service token
|
|
28
|
+
template.resourceCountIs('AWS::CloudFormation::CustomResource', 1);
|
|
29
|
+
});
|
|
30
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG9ja2VyX2ltYWdlX2Fzc2V0LnRlc3QuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJkb2NrZXJfaW1hZ2VfYXNzZXQudGVzdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUFBLG1DQUFtQztBQUNuQyx1REFBa0Q7QUFDbEQsd0NBQTREO0FBRTVELElBQUksQ0FBQyw2Q0FBNkMsRUFBRSxHQUFHLEVBQUU7SUFDdkQsTUFBTSxHQUFHLEdBQUcsSUFBSSxHQUFHLENBQUMsR0FBRyxFQUFFLENBQUM7SUFDMUIsTUFBTSxLQUFLLEdBQUcsSUFBSSxHQUFHLENBQUMsS0FBSyxDQUFDLEdBQUcsRUFBRSxXQUFXLENBQUMsQ0FBQztJQUU5QyxJQUFJLG9DQUE0QixDQUFDLEtBQUssRUFBRSxzQkFBc0IsRUFBRTtRQUM5RCxJQUFJLEVBQUUsc0JBQXNCLEVBQUUseUJBQXlCO1FBQ3ZELFNBQVMsRUFBRSxFQUFFLEdBQUcsRUFBRSxNQUFNLEVBQUU7S0FDM0IsQ0FBQyxDQUFDO0lBRUgsTUFBTSxRQUFRLEdBQUcscUJBQVEsQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLENBQUM7SUFFM0MsMkNBQTJDO0lBQzNDLFFBQVEsQ0FBQyxlQUFlLENBQUMsc0JBQXNCLEVBQUUsQ0FBQyxDQUFDLENBQUM7SUFFcEQsc0VBQXNFO0lBQ3RFLFFBQVEsQ0FBQyxxQkFBcUIsQ0FBQyx5QkFBeUIsRUFBRTtRQUN4RCxXQUFXLEVBQUU7WUFDWCxXQUFXLEVBQUUsc0JBQXNCO1lBQ25DLGNBQWMsRUFBRSxJQUFJO1lBQ3BCLEtBQUssRUFBRSw0QkFBNEI7U0FDcEM7UUFDRCxNQUFNLEVBQUU7WUFDTixJQUFJLEVBQUUsSUFBSTtTQUNYO0tBQ0YsQ0FBQyxDQUFDO0lBRUgsd0VBQXdFO0lBQ3hFLFFBQVEsQ0FBQyxlQUFlLENBQUMscUNBQXFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7QUFDckUsQ0FBQyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBjZGsgZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0IHsgVGVtcGxhdGUgfSBmcm9tICdhd3MtY2RrLWxpYi9hc3NlcnRpb25zJztcbmltcG9ydCB7IFRva2VuSW5qZWN0YWJsZURvY2tlckJ1aWxkZXIgfSBmcm9tICcuLi9saWIvaW5kZXgnO1xuXG50ZXN0KCdEb2NrZXJJbWFnZUFzc2V0IGNyZWF0ZXMgcmVxdWlyZWQgcmVzb3VyY2VzJywgKCkgPT4ge1xuICBjb25zdCBhcHAgPSBuZXcgY2RrLkFwcCgpO1xuICBjb25zdCBzdGFjayA9IG5ldyBjZGsuU3RhY2soYXBwLCAnVGVzdFN0YWNrJyk7XG5cbiAgbmV3IFRva2VuSW5qZWN0YWJsZURvY2tlckJ1aWxkZXIoc3RhY2ssICdUZXN0RG9ja2VySW1hZ2VBc3NldCcsIHtcbiAgICBwYXRoOiAnLi9zcmMvb25FdmVudEhhbmRsZXInLCAvLyBQYXRoIHRvIERvY2tlciBjb250ZXh0XG4gICAgYnVpbGRBcmdzOiB7IEVOVjogJ3Rlc3QnIH0sXG4gIH0pO1xuXG4gIGNvbnN0IHRlbXBsYXRlID0gVGVtcGxhdGUuZnJvbVN0YWNrKHN0YWNrKTtcblxuICAvLyBWZXJpZnkgdGhhdCBhbiBFQ1IgcmVwb3NpdG9yeSBpcyBjcmVhdGVkXG4gIHRlbXBsYXRlLnJlc291cmNlQ291bnRJcygnQVdTOjpFQ1I6OlJlcG9zaXRvcnknLCAxKTtcblxuICAvLyBWZXJpZnkgdGhhdCBhIENvZGVCdWlsZCBwcm9qZWN0IGlzIGNyZWF0ZWQgd2l0aCBleHBlY3RlZCBwcm9wZXJ0aWVzXG4gIHRlbXBsYXRlLmhhc1Jlc291cmNlUHJvcGVydGllcygnQVdTOjpDb2RlQnVpbGQ6OlByb2plY3QnLCB7XG4gICAgRW52aXJvbm1lbnQ6IHtcbiAgICAgIENvbXB1dGVUeXBlOiAnQlVJTERfR0VORVJBTDFfU01BTEwnLFxuICAgICAgUHJpdmlsZWdlZE1vZGU6IHRydWUsXG4gICAgICBJbWFnZTogJ2F3cy9jb2RlYnVpbGQvc3RhbmRhcmQ6Ny4wJyxcbiAgICB9LFxuICAgIFNvdXJjZToge1xuICAgICAgVHlwZTogJ1MzJyxcbiAgICB9LFxuICB9KTtcblxuICAvLyBWZXJpZnkgdGhlIEN1c3RvbSBSZXNvdXJjZSBpcyBjcmVhdGVkIHdpdGggdGhlIGV4cGVjdGVkIHNlcnZpY2UgdG9rZW5cbiAgdGVtcGxhdGUucmVzb3VyY2VDb3VudElzKCdBV1M6OkNsb3VkRm9ybWF0aW9uOjpDdXN0b21SZXNvdXJjZScsIDEpO1xufSk7XG4iXX0=
|