@liflig/cdk 2.21.2 → 2.21.4
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/README.md +18 -2
- package/lib/cdk-pipelines/cloud-assembly-lookup-handler.d.ts +1 -1
- package/lib/cdk-pipelines/cloud-assembly-lookup-handler.js +46 -35
- package/lib/cdk-pipelines/liflig-cdk-pipeline.js +6 -6
- package/lib/index.d.ts +1 -4
- package/lib/index.js +2 -6
- package/lib/ses/configurationsetsnsdestination/handler.d.ts +1 -1
- package/lib/ses/configurationsetsnsdestination/handler.js +15 -25
- package/lib/ses/configurationsetsnsdestination/index.js +7 -7
- package/lib/ses/sesdomain/handler.d.ts +1 -1
- package/lib/ses/sesdomain/handler.js +31 -31
- package/lib/ses/sesdomain/index.js +6 -6
- package/lib/ses/sesverifyemail/handler.d.ts +1 -1
- package/lib/ses/sesverifyemail/handler.js +12 -10
- package/lib/ses/sesverifyemail/index.js +6 -6
- package/package.json +14 -3
- package/lib/cdk-deploy/cdk-deploy.d.ts +0 -63
- package/lib/cdk-deploy/cdk-deploy.js +0 -175
- package/lib/cdk-deploy/index.d.ts +0 -1
- package/lib/cdk-deploy/index.js +0 -6
- package/lib/cdk-deploy/start-deploy-handler.d.ts +0 -8
- package/lib/cdk-deploy/start-deploy-handler.js +0 -72
- package/lib/cdk-deploy/status-handler.d.ts +0 -6
- package/lib/cdk-deploy/status-handler.js +0 -83
- package/lib/ecs-update-image/artifact-status.d.ts +0 -39
- package/lib/ecs-update-image/artifact-status.js +0 -41
- package/lib/ecs-update-image/ecs-update-image.d.ts +0 -41
- package/lib/ecs-update-image/ecs-update-image.js +0 -98
- package/lib/ecs-update-image/index.d.ts +0 -3
- package/lib/ecs-update-image/index.js +0 -10
- package/lib/ecs-update-image/start-deploy-handler.d.ts +0 -6
- package/lib/ecs-update-image/start-deploy-handler.js +0 -104
- package/lib/ecs-update-image/status-handler.d.ts +0 -11
- package/lib/ecs-update-image/status-handler.js +0 -74
- package/lib/ecs-update-image/tag.d.ts +0 -47
- package/lib/ecs-update-image/tag.js +0 -67
- package/lib/pipelines/conventions.d.ts +0 -14
- package/lib/pipelines/conventions.js +0 -24
- package/lib/pipelines/deploy-env.d.ts +0 -18
- package/lib/pipelines/deploy-env.js +0 -96
- package/lib/pipelines/index.d.ts +0 -2
- package/lib/pipelines/index.js +0 -8
- package/lib/pipelines/liflig-cdk-deployer-deps.d.ts +0 -13
- package/lib/pipelines/liflig-cdk-deployer-deps.js +0 -35
- package/lib/pipelines/pipeline.d.ts +0 -78
- package/lib/pipelines/pipeline.js +0 -224
|
@@ -1,224 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.Pipeline = void 0;
|
|
4
|
-
const constructs = require("constructs");
|
|
5
|
-
const events = require("aws-cdk-lib/aws-events");
|
|
6
|
-
const eventsTargets = require("aws-cdk-lib/aws-events-targets");
|
|
7
|
-
const iam = require("aws-cdk-lib/aws-iam");
|
|
8
|
-
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
9
|
-
const sfn = require("aws-cdk-lib/aws-stepfunctions");
|
|
10
|
-
const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
|
|
11
|
-
const tasks = require("aws-cdk-lib/aws-stepfunctions-tasks");
|
|
12
|
-
const cdk = require("aws-cdk-lib");
|
|
13
|
-
const griid_1 = require("../griid");
|
|
14
|
-
const conventions_1 = require("./conventions");
|
|
15
|
-
const deploy_env_1 = require("./deploy-env");
|
|
16
|
-
/**
|
|
17
|
-
* Pipeline for doing a multi-account CDK deployment based
|
|
18
|
-
* on a built CDK Cloud Assembly and parameters stored in S3.
|
|
19
|
-
*
|
|
20
|
-
* The accounts being deployed to must be provisioned with
|
|
21
|
-
* the LifligCdkDeployerDeps construct so expected IAM
|
|
22
|
-
* roles is present.
|
|
23
|
-
*
|
|
24
|
-
* The pipeline starts by writing an empty file to
|
|
25
|
-
* s3://<artifacts-bucket>/pipelines/<pipeline-name>/trigger
|
|
26
|
-
*
|
|
27
|
-
* The CDK deploy process is handled by liflig-cdk-deployer.
|
|
28
|
-
* See https://github.com/capralifecycle/liflig-cdk-deployer
|
|
29
|
-
*
|
|
30
|
-
* Configuration files are read from S3 at the path
|
|
31
|
-
* s3://<artifacts-bucket>/pipelines/<pipeline-name>/
|
|
32
|
-
*
|
|
33
|
-
* - cloud-assembly.json which has the format described as
|
|
34
|
-
* CDK_CLOUD_ASSEMBLY in liflig-cdk-deployer
|
|
35
|
-
*
|
|
36
|
-
* - variables*.json which can be zero or more files
|
|
37
|
-
* with string-string map that will be concatenated to
|
|
38
|
-
* form the format described as CDK_VARIABLES in
|
|
39
|
-
* liflig-cdk-deployer
|
|
40
|
-
*
|
|
41
|
-
* The separation of Cloud Assembly details and variables enables
|
|
42
|
-
* separation of IaC code and application code if they are not
|
|
43
|
-
* colocated in the same repository.
|
|
44
|
-
*/
|
|
45
|
-
class Pipeline extends constructs.Construct {
|
|
46
|
-
constructor(scope, id, props) {
|
|
47
|
-
var _a, _b;
|
|
48
|
-
super(scope, id);
|
|
49
|
-
const s3Prefix = (0, conventions_1.pipelineS3Prefix)(props.pipelineName);
|
|
50
|
-
const s3TriggerKey = (0, conventions_1.pipelineS3TriggerKey)(props.pipelineName);
|
|
51
|
-
const artifactsBucket = (_a = props.artifactsBucket) !== null && _a !== void 0 ? _a : (0, griid_1.getGriidArtefactBucket)(this);
|
|
52
|
-
const checkCanRunFn = new lambda.SingletonFunction(this, "CheckCanRunFn", {
|
|
53
|
-
uuid: "30ad3abb-f774-4804-a6ef-2c2f4a247362",
|
|
54
|
-
code: new lambda.InlineCode(`exports.handler = ${checkCanRunHandler.toString()};`),
|
|
55
|
-
runtime: lambda.Runtime.NODEJS_16_X,
|
|
56
|
-
handler: "index.handler",
|
|
57
|
-
timeout: cdk.Duration.seconds(10),
|
|
58
|
-
});
|
|
59
|
-
const checkCanRunTask = new tasks.LambdaInvoke(this, "Check if the process can run", {
|
|
60
|
-
lambdaFunction: checkCanRunFn,
|
|
61
|
-
outputPath: "$.Payload",
|
|
62
|
-
payload: sfn.TaskInput.fromObject({
|
|
63
|
-
"stateMachineId.$": "$$.StateMachine.Id",
|
|
64
|
-
"executionId.$": "$$.Execution.Id",
|
|
65
|
-
}),
|
|
66
|
-
});
|
|
67
|
-
const wait = new sfn.Wait(this, "Wait before rechecking status", {
|
|
68
|
-
time: sfn.WaitTime.duration(cdk.Duration.seconds(15)),
|
|
69
|
-
});
|
|
70
|
-
const skip = new sfn.Succeed(this, "Skip");
|
|
71
|
-
const collectFilesFn = new lambda.SingletonFunction(this, "CollectFilesFn", {
|
|
72
|
-
uuid: "c49cbfe1-50e0-4721-8964-fb20f4e5a7ad",
|
|
73
|
-
code: new lambda.InlineCode(`exports.handler = ${collectFilesHandler.toString()};`),
|
|
74
|
-
runtime: lambda.Runtime.NODEJS_16_X,
|
|
75
|
-
handler: "index.handler",
|
|
76
|
-
timeout: cdk.Duration.seconds(30),
|
|
77
|
-
});
|
|
78
|
-
artifactsBucket.grantRead(collectFilesFn);
|
|
79
|
-
const collectFilesTask = new tasks.LambdaInvoke(this, "Collect files from S3", {
|
|
80
|
-
lambdaFunction: collectFilesFn,
|
|
81
|
-
outputPath: "$.Payload",
|
|
82
|
-
payload: sfn.TaskInput.fromObject({
|
|
83
|
-
bucketName: artifactsBucket.bucketName,
|
|
84
|
-
bucketPrefix: s3Prefix,
|
|
85
|
-
envNames: props.environments.map((it) => it.name),
|
|
86
|
-
}),
|
|
87
|
-
});
|
|
88
|
-
let run = sfn.Chain.start(collectFilesTask);
|
|
89
|
-
const ifHavingStacks = (name, work) => new sfn.Choice(this, `Check if ${name} has stacks`)
|
|
90
|
-
.when(aws_stepfunctions_1.Condition.or(aws_stepfunctions_1.Condition.isNull(`$.StackCountPerEnv.${name}`), aws_stepfunctions_1.Condition.numberEquals(`$.StackCountPerEnv.${name}`, 0)), new sfn.Pass(this, `Skip ${name}`))
|
|
91
|
-
.otherwise(work)
|
|
92
|
-
.afterwards();
|
|
93
|
-
for (const env of props.environments) {
|
|
94
|
-
const it = new deploy_env_1.DeployEnv(this, env.name, {
|
|
95
|
-
accountId: env.accountId,
|
|
96
|
-
afterSuccessfulDeploy: env.afterSuccessfulDeploy,
|
|
97
|
-
artefactBucket: artifactsBucket,
|
|
98
|
-
envName: env.name,
|
|
99
|
-
vpc: props.vpc,
|
|
100
|
-
});
|
|
101
|
-
run = run.next(ifHavingStacks(env.name, it.chain));
|
|
102
|
-
}
|
|
103
|
-
const definition = sfn.Chain.start(checkCanRunTask).next(new sfn.Choice(this, "Can run?")
|
|
104
|
-
.when(aws_stepfunctions_1.Condition.stringEquals("$.CanRunState", "CONTINUE"), run)
|
|
105
|
-
.when(aws_stepfunctions_1.Condition.stringEquals("$.CanRunState", "SKIP"), skip)
|
|
106
|
-
.otherwise(wait.next(checkCanRunTask)));
|
|
107
|
-
const machine = new sfn.StateMachine(this, "StateMachine", {
|
|
108
|
-
definition,
|
|
109
|
-
// https://docs.aws.amazon.com/step-functions/latest/dg/sfn-stuck-execution.html
|
|
110
|
-
timeout: cdk.Duration.hours(3),
|
|
111
|
-
});
|
|
112
|
-
new iam.Policy(this, "CheckCanRunPolicy", {
|
|
113
|
-
roles: [checkCanRunFn.role],
|
|
114
|
-
statements: [
|
|
115
|
-
new iam.PolicyStatement({
|
|
116
|
-
actions: ["states:ListExecutions"],
|
|
117
|
-
resources: [machine.stateMachineArn],
|
|
118
|
-
}),
|
|
119
|
-
],
|
|
120
|
-
});
|
|
121
|
-
if ((_b = props.triggerEnabled) !== null && _b !== void 0 ? _b : true) {
|
|
122
|
-
artifactsBucket.onCloudTrailWriteObject("Trigger", {
|
|
123
|
-
paths: [s3TriggerKey],
|
|
124
|
-
target: new eventsTargets.SfnStateMachine(machine, {
|
|
125
|
-
input: events.RuleTargetInput.fromObject({}),
|
|
126
|
-
}),
|
|
127
|
-
});
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
exports.Pipeline = Pipeline;
|
|
132
|
-
// This is a self-contained function that will be serialized as a lambda.
|
|
133
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
134
|
-
const collectFilesHandler = async (event) => {
|
|
135
|
-
var _a;
|
|
136
|
-
// eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment
|
|
137
|
-
const AWS = require("aws-sdk");
|
|
138
|
-
// eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access
|
|
139
|
-
const s3 = new AWS.S3();
|
|
140
|
-
console.log("Event received: ", event);
|
|
141
|
-
const bucketName = event.bucketName;
|
|
142
|
-
const bucketPrefix = event.bucketPrefix;
|
|
143
|
-
const envNames = event.envNames;
|
|
144
|
-
if (bucketPrefix.slice(-1) !== "/") {
|
|
145
|
-
throw new Error(`Expected bucket prefix to end with '/' but its value is '${bucketPrefix}'`);
|
|
146
|
-
}
|
|
147
|
-
const files = await s3
|
|
148
|
-
.listObjectsV2({
|
|
149
|
-
Bucket: bucketName,
|
|
150
|
-
Prefix: bucketPrefix,
|
|
151
|
-
})
|
|
152
|
-
.promise();
|
|
153
|
-
async function getData(key) {
|
|
154
|
-
const result = await s3
|
|
155
|
-
.getObject({
|
|
156
|
-
Bucket: bucketName,
|
|
157
|
-
Key: key,
|
|
158
|
-
})
|
|
159
|
-
.promise();
|
|
160
|
-
return result.Body.toString();
|
|
161
|
-
}
|
|
162
|
-
let cloudAssembly = null;
|
|
163
|
-
let variables = {};
|
|
164
|
-
for (const file of (_a = files.Contents) !== null && _a !== void 0 ? _a : []) {
|
|
165
|
-
const key = file.Key;
|
|
166
|
-
const filename = key.slice(bucketPrefix.length);
|
|
167
|
-
console.log(`File: ${filename}`);
|
|
168
|
-
if (filename === "cloud-assembly.json") {
|
|
169
|
-
console.log("Found Cloud Assembly");
|
|
170
|
-
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
|
|
171
|
-
cloudAssembly = JSON.parse(await getData(key));
|
|
172
|
-
}
|
|
173
|
-
else if (/^variables.*\.json$/.test(filename)) {
|
|
174
|
-
console.log("Found variables file");
|
|
175
|
-
variables = {
|
|
176
|
-
...variables,
|
|
177
|
-
...JSON.parse(await getData(key)),
|
|
178
|
-
};
|
|
179
|
-
}
|
|
180
|
-
else {
|
|
181
|
-
console.log("Ignoring unknown file");
|
|
182
|
-
}
|
|
183
|
-
}
|
|
184
|
-
if (cloudAssembly === null) {
|
|
185
|
-
throw new Error("cloud-assembly.json not found");
|
|
186
|
-
}
|
|
187
|
-
return {
|
|
188
|
-
CloudAssembly: JSON.stringify(cloudAssembly),
|
|
189
|
-
Variables: JSON.stringify(variables),
|
|
190
|
-
StackCountPerEnv: Object.fromEntries(envNames.map((name) => {
|
|
191
|
-
var _a, _b;
|
|
192
|
-
return [
|
|
193
|
-
name,
|
|
194
|
-
(_b = (_a = cloudAssembly.environments.find((it) => it.name === name)) === null || _a === void 0 ? void 0 : _a.stackNames.length) !== null && _b !== void 0 ? _b : 0,
|
|
195
|
-
];
|
|
196
|
-
})),
|
|
197
|
-
};
|
|
198
|
-
};
|
|
199
|
-
// This is a self-contained function that will be serialized as a lambda.
|
|
200
|
-
const checkCanRunHandler = async (event) => {
|
|
201
|
-
// eslint-disable-next-line @typescript-eslint/no-var-requires,@typescript-eslint/no-unsafe-assignment
|
|
202
|
-
const AWS = require("aws-sdk");
|
|
203
|
-
// eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access
|
|
204
|
-
const sf = new AWS.StepFunctions();
|
|
205
|
-
console.log("Event received: ", event);
|
|
206
|
-
const stateMachineArn = event["stateMachineId"];
|
|
207
|
-
const currentExecutionArn = event["executionId"];
|
|
208
|
-
const executions = (await sf
|
|
209
|
-
.listExecutions({
|
|
210
|
-
stateMachineArn,
|
|
211
|
-
statusFilter: "RUNNING",
|
|
212
|
-
})
|
|
213
|
-
.promise()).executions;
|
|
214
|
-
console.log("Executions: ", executions);
|
|
215
|
-
const currentExecution = executions.find((it) => it.executionArn == currentExecutionArn);
|
|
216
|
-
if (!currentExecution) {
|
|
217
|
-
throw new Error("Could not find current execution");
|
|
218
|
-
}
|
|
219
|
-
const newer = executions.filter((it) => it.startDate > currentExecution.startDate).length;
|
|
220
|
-
return {
|
|
221
|
-
CanRunState: newer > 0 ? "SKIP" : executions.length == 1 ? "CONTINUE" : "WAIT",
|
|
222
|
-
};
|
|
223
|
-
};
|
|
224
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGlwZWxpbmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcGlwZWxpbmVzL3BpcGVsaW5lLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHlDQUF3QztBQUV4QyxpREFBZ0Q7QUFDaEQsZ0VBQStEO0FBQy9ELDJDQUEwQztBQUMxQyxpREFBZ0Q7QUFFaEQscURBQW9EO0FBQ3BELHFFQUF5RDtBQUN6RCw2REFBNEQ7QUFDNUQsbUNBQWtDO0FBR2xDLG9DQUFpRDtBQUNqRCwrQ0FBc0U7QUFDdEUsNkNBQXdDO0FBOEN4Qzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQTRCRztBQUNILE1BQWEsUUFBUyxTQUFRLFVBQVUsQ0FBQyxTQUFTO0lBQ2hELFlBQVksS0FBMkIsRUFBRSxFQUFVLEVBQUUsS0FBb0I7O1FBQ3ZFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFFaEIsTUFBTSxRQUFRLEdBQUcsSUFBQSw4QkFBZ0IsRUFBQyxLQUFLLENBQUMsWUFBWSxDQUFDLENBQUE7UUFDckQsTUFBTSxZQUFZLEdBQUcsSUFBQSxrQ0FBb0IsRUFBQyxLQUFLLENBQUMsWUFBWSxDQUFDLENBQUE7UUFFN0QsTUFBTSxlQUFlLEdBQ25CLE1BQUEsS0FBSyxDQUFDLGVBQWUsbUNBQUksSUFBQSw4QkFBc0IsRUFBQyxJQUFJLENBQUMsQ0FBQTtRQUV2RCxNQUFNLGFBQWEsR0FBRyxJQUFJLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxJQUFJLEVBQUUsZUFBZSxFQUFFO1lBQ3hFLElBQUksRUFBRSxzQ0FBc0M7WUFDNUMsSUFBSSxFQUFFLElBQUksTUFBTSxDQUFDLFVBQVUsQ0FDekIscUJBQXFCLGtCQUFrQixDQUFDLFFBQVEsRUFBRSxHQUFHLENBQ3REO1lBQ0QsT0FBTyxFQUFFLE1BQU0sQ0FBQyxPQUFPLENBQUMsV0FBVztZQUNuQyxPQUFPLEVBQUUsZUFBZTtZQUN4QixPQUFPLEVBQUUsR0FBRyxDQUFDLFFB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QyxNQUFNLG1CQUFtQixHQUFHLEtBQUssQ0FBQyxhQUFhLENBQUMsQ0FBQTtJQUVoRCxNQUFNLFVBQVUsR0FBRyxDQUNqQixNQUFNLEVBQUU7U0FDTCxjQUFjLENBQUM7UUFDZCxlQUFlO1FBQ2YsWUFBWSxFQUFFLFNBQVM7S0FDeEIsQ0FBQztTQUNELE9BQU8sRUFBRSxDQUNiLENBQUMsVUFBVSxDQUFBO0lBRVosT0FBTyxDQUFDLEdBQUcsQ0FBQyxjQUFjLEVBQUUsVUFBVSxDQUFDLENBQUE7SUFFdkMsTUFBTSxnQkFBZ0IsR0FBRyxVQUFVLENBQUMsSUFBSSxDQUN0QyxDQUFDLEVBQUUsRUFBRSxFQUFFLENBQUMsRUFBRSxDQUFDLFlBQVksSUFBSSxtQkFBbUIsQ0FDL0MsQ0FBQTtJQUVELElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO1FBQ3RCLE1BQU0sSUFBSSxLQUFLLENBQUMsa0NBQWtDLENBQUMsQ0FBQTtJQUNyRCxDQUFDO0lBRUQsTUFBTSxLQUFLLEdBQUcsVUFBVSxDQUFDLE1BQU0sQ0FDN0IsQ0FBQyxFQUFFLEVBQUUsRUFBRSxDQUFDLEVBQUUsQ0FBQyxTQUFTLEdBQUcsZ0JBQWdCLENBQUMsU0FBUyxDQUNsRCxDQUFDLE1BQU0sQ0FBQTtJQUVSLE9BQU87UUFDTCxXQUFXLEVBQ1QsS0FBSyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxVQUFVLENBQUMsTUFBTSxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQyxNQUFNO0tBQ3BFLENBQUE7QUFDSCxDQUFDLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBjb25zdHJ1Y3RzIGZyb20gXCJjb25zdHJ1Y3RzXCJcbmltcG9ydCAqIGFzIGVjMiBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWVjMlwiXG5pbXBvcnQgKiBhcyBldmVudHMgZnJvbSBcImF3cy1jZGstbGliL2F3cy1ldmVudHNcIlxuaW1wb3J0ICogYXMgZXZlbnRzVGFyZ2V0cyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWV2ZW50cy10YXJnZXRzXCJcbmltcG9ydCAqIGFzIGlhbSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiXG5pbXBvcnQgKiBhcyBsYW1iZGEgZnJvbSBcImF3cy1jZGstbGliL2F3cy1sYW1iZGFcIlxuaW1wb3J0ICogYXMgczMgZnJvbSBcImF3cy1jZGstbGliL2F3cy1zM1wiXG5pbXBvcnQgKiBhcyBzZm4gZnJvbSBcImF3cy1jZGstbGliL2F3cy1zdGVwZnVuY3Rpb25zXCJcbmltcG9ydCB7IENvbmRpdGlvbiB9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3Mtc3RlcGZ1bmN0aW9uc1wiXG5pbXBvcnQgKiBhcyB0YXNrcyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXN0ZXBmdW5jdGlvbnMtdGFza3NcIlxuaW1wb3J0ICogYXMgY2RrIGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgdHlwZSB7IEhhbmRsZXIgfSBmcm9tIFwiYXdzLWxhbWJkYVwiXG5pbXBvcnQgdHlwZSAqIGFzIF9BV1MgZnJvbSBcImF3cy1zZGtcIlxuaW1wb3J0IHsgZ2V0R3JpaWRBcnRlZmFjdEJ1Y2tldCB9IGZyb20gXCIuLi9ncmlpZFwiXG5pbXBvcnQgeyBwaXBlbGluZVMzUHJlZml4LCBwaXBlbGluZVMzVHJpZ2dlcktleSB9IGZyb20gXCIuL2NvbnZlbnRpb25zXCJcbmltcG9ydCB7IERlcGxveUVudiB9IGZyb20gXCIuL2RlcGxveS1lbnZcIlxuXG5leHBvcnQgaW50ZXJmYWNlIFBpcGVsaW5lUHJvcHMge1xuICAvKipcbiAgICogQnVja2V0IGhvbGRpbmcgcGlwZWxpbmUgY29uZmlndXJhdGlvbiBhbmQgdHJpZ2dlciBmaWxlLlxuICAgKlxuICAgKiBAZGVmYXVsdCAtIHVzZSBleGlzdGluZyBidWNrZXQgYmFzZWQgb24gR3JpaWQgY29udmVudGlvbnNcbiAgICovXG4gIGFydGlmYWN0c0J1Y2tldD86IHMzLklCdWNrZXRcbiAgLyoqXG4gICAqIEVudmlyb25tZW50cyBmb3IgdGhpcyBwaXBlbGluZS4gRWFjaCBlbnZpcm9ubWVudCBpcyBkZXBsb3llZCBzZXF1ZW50aWFsbHlcbiAgICogaW4gdGhlIG9yZGVyIGdpdmVuLlxuICAgKi9cbiAgZW52aXJvbm1lbnRzOiBQaXBlbGluZUVudmlyb25tZW50W11cbiAgLyoqXG4gICAqIE5hbWUgb2YgcGlwZWxpbmUuIFRoaXMgaXMgdXNlZCBmb3IgdGhlIHBhdGggd2hlcmUgY29uZmlndXJhdGlvblxuICAgKiBpcyBzdG9yZWQgaW4gUzMuXG4gICAqL1xuICBwaXBlbGluZU5hbWU6IHN0cmluZ1xuICAvKipcbiAgICogVHJpZ2dlciB0aGUgcGlwZWxpbmUgd2hlbiB0aGUgdHJpZ2dlciBmaWxlIGlzIHdyaXR0ZW4uXG4gICAqXG4gICAqIEBkZWZhdWx0IC0gdHJ1ZVxuICAgKi9cbiAgdHJpZ2dlckVuYWJsZWQ/OiBib29sZWFuXG4gIC8qKlxuICAgKiBWUEMgdXNlZCBmb3IgRmFyZ2F0ZSByZXNvdXJjZXMuXG4gICAqL1xuICB2cGM6IGVjMi5JVnBjXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUGlwZWxpbmVFbnZpcm9ubWVudCB7XG4gIC8qKlxuICAgKiBBY2NvdW50IG51bWJlciBob3N0aW5nIHRoZSBlbnZpcm9ubWVudC5cbiAgICovXG4gIGFjY291bnRJZDogc3RyaW5nXG4gIC8qKlxuICAgKiBBZGRpdGlvbmFsIHRhc2tzIHRvIHJ1biBhZnRlciB0aGUgZW52aXJvbm1lbnQgaGFzIGJlZW4gZGVwbG95ZWQuXG4gICAqL1xuICBhZnRlclN1Y2Nlc3NmdWxEZXBsb3k/OiBzZm4uQ2hhaW5cbiAgLyoqXG4gICAqIE5hbWUgb2YgZW52aXJvbm1lbnQuXG4gICAqL1xuICBuYW1lOiBzdHJpbmdcbn1cblxuLyoqXG4gKiBQaXBlbGluZSBmb3IgZG9pbmcgYSBtdWx0aS1hY2NvdW50IENESyBkZXBsb3ltZW50IGJhc2VkXG4gKiBvbiBhIGJ1aWx0IENESyBDbG91ZCBBc3NlbWJseSBhbmQgcGFyYW1ldGVycyBzdG9yZWQgaW4gUzMuXG4gKlxuICogVGhlIGFjY291bnRzIGJlaW5nIGRlcGxveWVkIHRvIG11c3QgYmUgcHJvdmlzaW9uZWQgd2l0aFxuICogdGhlIExpZmxpZ0Nka0RlcGxveWVyRGVwcyBjb25zdHJ1Y3Qgc28gZXhwZWN0ZWQgSUFNXG4gKiByb2xlcyBpcyBwcmVzZW50LlxuICpcbiAqIFRoZSBwaXBlbGluZSBzdGFydHMgYnkgd3JpdGluZyBhbiBlbXB0eSBmaWxlIHRvXG4gKiBzMzovLzxhcnRpZmFjdHMtYnVja2V0Pi9waXBlbGluZXMvPHBpcGVsaW5lLW5hbWU+L3RyaWdnZXJcbiAqXG4gKiBUaGUgQ0RLIGRlcGxveSBwcm9jZXNzIGlzIGhhbmRsZWQgYnkgbGlmbGlnLWNkay1kZXBsb3llci5cbiAqIFNlZSBodHRwczovL2dpdGh1Yi5jb20vY2FwcmFsaWZlY3ljbGUvbGlmbGlnLWNkay1kZXBsb3llclxuICpcbiAqIENvbmZpZ3VyYXRpb24gZmlsZXMgYXJlIHJlYWQgZnJvbSBTMyBhdCB0aGUgcGF0aFxuICogczM6Ly88YXJ0aWZhY3RzLWJ1Y2tldD4vcGlwZWxpbmVzLzxwaXBlbGluZS1uYW1lPi9cbiAqXG4gKiAgLSBjbG91ZC1hc3NlbWJseS5qc29uIHdoaWNoIGhhcyB0aGUgZm9ybWF0IGRlc2NyaWJlZCBhc1xuICogICAgQ0RLX0NMT1VEX0FTU0VNQkxZIGluIGxpZmxpZy1jZGstZGVwbG95ZXJcbiAqXG4gKiAgLSB2YXJpYWJsZXMqLmpzb24gd2hpY2ggY2FuIGJlIHplcm8gb3IgbW9yZSBmaWxlc1xuICogICAgd2l0aCBzdHJpbmctc3RyaW5nIG1hcCB0aGF0IHdpbGwgYmUgY29uY2F0ZW5hdGVkIHRvXG4gKiAgICBmb3JtIHRoZSBmb3JtYXQgZGVzY3JpYmVkIGFzIENES19WQVJJQUJMRVMgaW5cbiAqICAgIGxpZmxpZy1jZGstZGVwbG95ZXJcbiAqXG4gKiBUaGUgc2VwYXJhdGlvbiBvZiBDbG91ZCBBc3NlbWJseSBkZXRhaWxzIGFuZCB2YXJpYWJsZXMgZW5hYmxlc1xuICogc2VwYXJhdGlvbiBvZiBJYUMgY29kZSBhbmQgYXBwbGljYXRpb24gY29kZSBpZiB0aGV5IGFyZSBub3RcbiAqIGNvbG9jYXRlZCBpbiB0aGUgc2FtZSByZXBvc2l0b3J5LlxuICovXG5leHBvcnQgY2xhc3MgUGlwZWxpbmUgZXh0ZW5kcyBjb25zdHJ1Y3RzLkNvbnN0cnVjdCB7XG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBjb25zdHJ1Y3RzLkNvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IFBpcGVsaW5lUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICBjb25zdCBzM1ByZWZpeCA9IHBpcGVsaW5lUzNQcmVmaXgocHJvcHMucGlwZWxpbmVOYW1lKVxuICAgIGNvbnN0IHMzVHJpZ2dlcktleSA9IHBpcGVsaW5lUzNUcmlnZ2VyS2V5KHByb3BzLnBpcGVsaW5lTmFtZSlcblxuICAgIGNvbnN0IGFydGlmYWN0c0J1Y2tldCA9XG4gICAgICBwcm9wcy5hcnRpZmFjdHNCdWNrZXQgPz8gZ2V0R3JpaWRBcnRlZmFjdEJ1Y2tldCh0aGlzKVxuXG4gICAgY29uc3QgY2hlY2tDYW5SdW5GbiA9IG5ldyBsYW1iZGEuU2luZ2xldG9uRnVuY3Rpb24odGhpcywgXCJDaGVja0NhblJ1bkZuXCIsIHtcbiAgICAgIHV1aWQ6IFwiMzBhZDNhYmItZjc3NC00ODA0LWE2ZWYtMmMyZjRhMjQ3MzYyXCIsXG4gICAgICBjb2RlOiBuZXcgbGFtYmRhLklubGluZUNvZGUoXG4gICAgICAgIGBleHBvcnRzLmhhbmRsZXIgPSAke2NoZWNrQ2FuUnVuSGFuZGxlci50b1N0cmluZygpfTtgLFxuICAgICAgKSxcbiAgICAgIHJ1bnRpbWU6IGxhbWJkYS5SdW50aW1lLk5PREVKU18xNl9YLFxuICAgICAgaGFuZGxlcjogXCJpbmRleC5oYW5kbGVyXCIsXG4gICAgICB0aW1lb3V0OiBjZGsuRHVyYXRpb24uc2Vjb25kcygxMCksXG4gICAgfSlcblxuICAgIGNvbnN0IGNoZWNrQ2FuUnVuVGFzayA9IG5ldyB0YXNrcy5MYW1iZGFJbnZva2UoXG4gICAgICB0aGlzLFxuICAgICAgXCJDaGVjayBpZiB0aGUgcHJvY2VzcyBjYW4gcnVuXCIsXG4gICAgICB7XG4gICAgICAgIGxhbWJkYUZ1bmN0aW9uOiBjaGVja0NhblJ1bkZuLFxuICAgICAgICBvdXRwdXRQYXRoOiBcIiQuUGF5bG9hZFwiLFxuICAgICAgICBwYXlsb2FkOiBzZm4uVGFza0lucHV0LmZyb21PYmplY3Qoe1xuICAgICAgICAgIFwic3RhdGVNYWNoaW5lSWQuJFwiOiBcIiQkLlN0YXRlTWFjaGluZS5JZFwiLFxuICAgICAgICAgIFwiZXhlY3V0aW9uSWQuJFwiOiBcIiQkLkV4ZWN1dGlvbi5JZFwiLFxuICAgICAgICB9KSxcbiAgICAgIH0sXG4gICAgKVxuXG4gICAgY29uc3Qgd2FpdCA9IG5ldyBzZm4uV2FpdCh0aGlzLCBcIldhaXQgYmVmb3JlIHJlY2hlY2tpbmcgc3RhdHVzXCIsIHtcbiAgICAgIHRpbWU6IHNmbi5XYWl0VGltZS5kdXJhdGlvbihjZGsuRHVyYXRpb24uc2Vjb25kcygxNSkpLFxuICAgIH0pXG5cbiAgICBjb25zdCBza2lwID0gbmV3IHNmbi5TdWNjZWVkKHRoaXMsIFwiU2tpcFwiKVxuXG4gICAgY29uc3QgY29sbGVjdEZpbGVzRm4gPSBuZXcgbGFtYmRhLlNpbmdsZXRvbkZ1bmN0aW9uKFxuICAgICAgdGhpcyxcbiAgICAgIFwiQ29sbGVjdEZpbGVzRm5cIixcbiAgICAgIHtcbiAgICAgICAgdXVpZDogXCJjNDljYmZlMS01MGUwLTQ3MjEtODk2NC1mYjIwZjRlNWE3YWRcIixcbiAgICAgICAgY29kZTogbmV3IGxhbWJkYS5JbmxpbmVDb2RlKFxuICAgICAgICAgIGBleHBvcnRzLmhhbmRsZXIgPSAke2NvbGxlY3RGaWxlc0hhbmRsZXIudG9TdHJpbmcoKX07YCxcbiAgICAgICAgKSxcbiAgICAgICAgcnVudGltZTogbGFtYmRhLlJ1bnRpbWUuTk9ERUpTXzE2X1gsXG4gICAgICAgIGhhbmRsZXI6IFwiaW5kZXguaGFuZGxlclwiLFxuICAgICAgICB0aW1lb3V0OiBjZGsuRHVyYXRpb24uc2Vjb25kcygzMCksXG4gICAgICB9LFxuICAgIClcblxuICAgIGFydGlmYWN0c0J1Y2tldC5ncmFudFJlYWQoY29sbGVjdEZpbGVzRm4pXG5cbiAgICBjb25zdCBjb2xsZWN0RmlsZXNUYXNrID0gbmV3IHRhc2tzLkxhbWJkYUludm9rZShcbiAgICAgIHRoaXMsXG4gICAgICBcIkNvbGxlY3QgZmlsZXMgZnJvbSBTM1wiLFxuICAgICAge1xuICAgICAgICBsYW1iZGFGdW5jdGlvbjogY29sbGVjdEZpbGVzRm4sXG4gICAgICAgIG91dHB1dFBhdGg6IFwiJC5QYXlsb2FkXCIsXG4gICAgICAgIHBheWxvYWQ6IHNmbi5UYXNrSW5wdXQuZnJvbU9iamVjdCh7XG4gICAgICAgICAgYnVja2V0TmFtZTogYXJ0aWZhY3RzQnVja2V0LmJ1Y2tldE5hbWUsXG4gICAgICAgICAgYnVja2V0UHJlZml4OiBzM1ByZWZpeCxcbiAgICAgICAgICBlbnZOYW1lczogcHJvcHMuZW52aXJvbm1lbnRzLm1hcCgoaXQpID0+IGl0Lm5hbWUpLFxuICAgICAgICB9KSxcbiAgICAgIH0sXG4gICAgKVxuXG4gICAgbGV0IHJ1bjogc2ZuLkNoYWluID0gc2ZuLkNoYWluLnN0YXJ0KGNvbGxlY3RGaWxlc1Rhc2spXG5cbiAgICBjb25zdCBpZkhhdmluZ1N0YWNrcyA9IChuYW1lOiBzdHJpbmcsIHdvcms6IHNmbi5DaGFpbikgPT5cbiAgICAgIG5ldyBzZm4uQ2hvaWNlKHRoaXMsIGBDaGVjayBpZiAke25hbWV9IGhhcyBzdGFja3NgKVxuICAgICAgICAud2hlbihcbiAgICAgICAgICBDb25kaXRpb24ub3IoXG4gICAgICAgICAgICBDb25kaXRpb24uaXNOdWxsKGAkLlN0YWNrQ291bnRQZXJFbnYuJHtuYW1lfWApLFxuICAgICAgICAgICAgQ29uZGl0aW9uLm51bWJlckVxdWFscyhgJC5TdGFja0NvdW50UGVyRW52LiR7bmFtZX1gLCAwKSxcbiAgICAgICAgICApLFxuICAgICAgICAgIG5ldyBzZm4uUGFzcyh0aGlzLCBgU2tpcCAke25hbWV9YCksXG4gICAgICAgIClcbiAgICAgICAgLm90aGVyd2lzZSh3b3JrKVxuICAgICAgICAuYWZ0ZXJ3YXJkcygpXG5cbiAgICBmb3IgKGNvbnN0IGVudiBvZiBwcm9wcy5lbnZpcm9ubWVudHMpIHtcbiAgICAgIGNvbnN0IGl0ID0gbmV3IERlcGxveUVudih0aGlzLCBlbnYubmFtZSwge1xuICAgICAgICBhY2NvdW50SWQ6IGVudi5hY2NvdW50SWQsXG4gICAgICAgIGFmdGVyU3VjY2Vzc2Z1bERlcGxveTogZW52LmFmdGVyU3VjY2Vzc2Z1bERlcGxveSxcbiAgICAgICAgYXJ0ZWZhY3RCdWNrZXQ6IGFydGlmYWN0c0J1Y2tldCxcbiAgICAgICAgZW52TmFtZTogZW52Lm5hbWUsXG4gICAgICAgIHZwYzogcHJvcHMudnBjLFxuICAgICAgfSlcblxuICAgICAgcnVuID0gcnVuLm5leHQoaWZIYXZpbmdTdGFja3MoZW52Lm5hbWUsIGl0LmNoYWluKSlcbiAgICB9XG5cbiAgICBjb25zdCBkZWZpbml0aW9uOiBzZm4uQ2hhaW4gPSBzZm4uQ2hhaW4uc3RhcnQoY2hlY2tDYW5SdW5UYXNrKS5uZXh0KFxuICAgICAgbmV3IHNmbi5DaG9pY2UodGhpcywgXCJDYW4gcnVuP1wiKVxuICAgICAgICAud2hlbihDb25kaXRpb24uc3RyaW5nRXF1YWxzKFwiJC5DYW5SdW5TdGF0ZVwiLCBcIkNPTlRJTlVFXCIpLCBydW4pXG4gICAgICAgIC53aGVuKENvbmRpdGlvbi5zdHJpbmdFcXVhbHMoXCIkLkNhblJ1blN0YXRlXCIsIFwiU0tJUFwiKSwgc2tpcClcbiAgICAgICAgLm90aGVyd2lzZSh3YWl0Lm5leHQoY2hlY2tDYW5SdW5UYXNrKSksXG4gICAgKVxuXG4gICAgY29uc3QgbWFjaGluZSA9IG5ldyBzZm4uU3RhdGVNYWNoaW5lKHRoaXMsIFwiU3RhdGVNYWNoaW5lXCIsIHtcbiAgICAgIGRlZmluaXRpb24sXG4gICAgICAvLyBodHRwczovL2RvY3MuYXdzLmFtYXpvbi5jb20vc3RlcC1mdW5jdGlvbnMvbGF0ZXN0L2RnL3Nmbi1zdHVjay1leGVjdXRpb24uaHRtbFxuICAgICAgdGltZW91dDogY2RrLkR1cmF0aW9uLmhvdXJzKDMpLFxuICAgIH0pXG5cbiAgICBuZXcgaWFtLlBvbGljeSh0aGlzLCBcIkNoZWNrQ2FuUnVuUG9saWN5XCIsIHtcbiAgICAgIHJvbGVzOiBbY2hlY2tDYW5SdW5Gbi5yb2xlIV0sXG4gICAgICBzdGF0ZW1lbnRzOiBbXG4gICAgICAgIG5ldyBpYW0uUG9saWN5U3RhdGVtZW50KHtcbiAgICAgICAgICBhY3Rpb25zOiBbXCJzdGF0ZXM6TGlzdEV4ZWN1dGlvbnNcIl0sXG4gICAgICAgICAgcmVzb3VyY2VzOiBbbWFjaGluZS5zdGF0ZU1hY2hpbmVBcm5dLFxuICAgICAgICB9KSxcbiAgICAgIF0sXG4gICAgfSlcblxuICAgIGlmIChwcm9wcy50cmlnZ2VyRW5hYmxlZCA/PyB0cnVlKSB7XG4gICAgICBhcnRpZmFjdHNCdWNrZXQub25DbG91ZFRyYWlsV3JpdGVPYmplY3QoXCJUcmlnZ2VyXCIsIHtcbiAgICAgICAgcGF0aHM6IFtzM1RyaWdnZXJLZXldLFxuICAgICAgICB0YXJnZXQ6IG5ldyBldmVudHNUYXJnZXRzLlNmblN0YXRlTWFjaGluZShtYWNoaW5lLCB7XG4gICAgICAgICAgaW5wdXQ6IGV2ZW50cy5SdWxlVGFyZ2V0SW5wdXQuZnJvbU9iamVjdCh7fSksXG4gICAgICAgIH0pLFxuICAgICAgfSlcbiAgICB9XG4gIH1cbn1cblxuaW50ZXJmYWNlIENsb3VkQXNzZW1ibHkge1xuICBjbG91ZEFzc2VtYmx5QnVja2V0TmFtZTogc3RyaW5nXG4gIGNsb3VkQXNzZW1ibHlCdWNrZXRLZXk6IHN0cmluZ1xuICBlbnZpcm9ubWVudHM6IHtcbiAgICBuYW1lOiBzdHJpbmdcbiAgICBzdGFja05hbWVzOiBzdHJpbmdbXVxuICB9W11cbiAgcGFyYW1ldGVyczoge1xuICAgIG5hbWU6IHN0cmluZ1xuICAgIHZhbHVlOiB1bmtub3duIHwgeyB0eXBlOiBcInZhcmlhYmxlXCI7IHZhcmlhYmxlOiBzdHJpbmcgfVxuICB9W11cbn1cblxuLy8gVGhpcyBpcyBhIHNlbGYtY29udGFpbmVkIGZ1bmN0aW9uIHRoYXQgd2lsbCBiZSBzZXJpYWxpemVkIGFzIGEgbGFtYmRhLlxuLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby1leHBsaWNpdC1hbnlcbmNvbnN0IGNvbGxlY3RGaWxlc0hhbmRsZXI6IEhhbmRsZXIgPSBhc3luYyAoZXZlbnQ6IFJlY29yZDxzdHJpbmcsIGFueT4pID0+IHtcbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby12YXItcmVxdWlyZXMsQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1hc3NpZ25tZW50XG4gIGNvbnN0IEFXUyA9IHJlcXVpcmUoXCJhd3Mtc2RrXCIpXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLWNhbGwsQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1tZW1iZXItYWNjZXNzXG4gIGNvbnN0IHMzID0gbmV3IEFXUy5TMygpIGFzIF9BV1MuUzNcblxuICBjb25zb2xlLmxvZyhcIkV2ZW50IHJlY2VpdmVkOiBcIiwgZXZlbnQpXG5cbiAgY29uc3QgYnVja2V0TmFtZSA9IGV2ZW50LmJ1Y2tldE5hbWUgYXMgc3RyaW5nXG4gIGNvbnN0IGJ1Y2tldFByZWZpeCA9IGV2ZW50LmJ1Y2tldFByZWZpeCBhcyBzdHJpbmdcbiAgY29uc3QgZW52TmFtZXMgPSBldmVudC5lbnZOYW1lcyBhcyBzdHJpbmdbXVxuXG4gIGlmIChidWNrZXRQcmVmaXguc2xpY2UoLTEpICE9PSBcIi9cIikge1xuICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgIGBFeHBlY3RlZCBidWNrZXQgcHJlZml4IHRvIGVuZCB3aXRoICcvJyBidXQgaXRzIHZhbHVlIGlzICcke2J1Y2tldFByZWZpeH0nYCxcbiAgICApXG4gIH1cblxuICBjb25zdCBmaWxlcyA9IGF3YWl0IHMzXG4gICAgLmxpc3RPYmplY3RzVjIoe1xuICAgICAgQnVja2V0OiBidWNrZXROYW1lLFxuICAgICAgUHJlZml4OiBidWNrZXRQcmVmaXgsXG4gICAgfSlcbiAgICAucHJvbWlzZSgpXG5cbiAgYXN5bmMgZnVuY3Rpb24gZ2V0RGF0YShrZXk6IHN0cmluZyk6IFByb21pc2U8c3RyaW5nPiB7XG4gICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgczNcbiAgICAgIC5nZXRPYmplY3Qoe1xuICAgICAgICBCdWNrZXQ6IGJ1Y2tldE5hbWUsXG4gICAgICAgIEtleToga2V5LFxuICAgICAgfSlcbiAgICAgIC5wcm9taXNlKClcbiAgICByZXR1cm4gcmVzdWx0LkJvZHkhLnRvU3RyaW5nKClcbiAgfVxuXG4gIGxldCBjbG91ZEFzc2VtYmx5OiBDbG91ZEFzc2VtYmx5IHwgbnVsbCA9IG51bGxcbiAgbGV0IHZhcmlhYmxlczogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHt9XG5cbiAgZm9yIChjb25zdCBmaWxlIG9mIGZpbGVzLkNvbnRlbnRzID8/IFtdKSB7XG4gICAgY29uc3Qga2V5ID0gZmlsZS5LZXkhXG4gICAgY29uc3QgZmlsZW5hbWUgPSBrZXkuc2xpY2UoYnVja2V0UHJlZml4Lmxlbmd0aClcblxuICAgIGNvbnNvbGUubG9nKGBGaWxlOiAke2ZpbGVuYW1lfWApXG5cbiAgICBpZiAoZmlsZW5hbWUgPT09IFwiY2xvdWQtYXNzZW1ibHkuanNvblwiKSB7XG4gICAgICBjb25zb2xlLmxvZyhcIkZvdW5kIENsb3VkIEFzc2VtYmx5XCIpXG4gICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1hc3NpZ25tZW50XG4gICAgICBjbG91ZEFzc2VtYmx5ID0gSlNPTi5wYXJzZShhd2FpdCBnZXREYXRhKGtleSkpXG4gICAgfSBlbHNlIGlmICgvXnZhcmlhYmxlcy4qXFwuanNvbiQvLnRlc3QoZmlsZW5hbWUpKSB7XG4gICAgICBjb25zb2xlLmxvZyhcIkZvdW5kIHZhcmlhYmxlcyBmaWxlXCIpXG4gICAgICB2YXJpYWJsZXMgPSB7XG4gICAgICAgIC4uLnZhcmlhYmxlcyxcbiAgICAgICAgLi4uKEpTT04ucGFyc2UoYXdhaXQgZ2V0RGF0YShrZXkpKSBhcyBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+KSxcbiAgICAgIH1cbiAgICB9IGVsc2Uge1xuICAgICAgY29uc29sZS5sb2coXCJJZ25vcmluZyB1bmtub3duIGZpbGVcIilcbiAgICB9XG4gIH1cblxuICBpZiAoY2xvdWRBc3NlbWJseSA9PT0gbnVsbCkge1xuICAgIHRocm93IG5ldyBFcnJvcihcImNsb3VkLWFzc2VtYmx5Lmpzb24gbm90IGZvdW5kXCIpXG4gIH1cblxuICByZXR1cm4ge1xuICAgIENsb3VkQXNzZW1ibHk6IEpTT04uc3RyaW5naWZ5KGNsb3VkQXNzZW1ibHkpLFxuICAgIFZhcmlhYmxlczogSlNPTi5zdHJpbmdpZnkodmFyaWFibGVzKSxcbiAgICBTdGFja0NvdW50UGVyRW52OiBPYmplY3QuZnJvbUVudHJpZXMoXG4gICAgICBlbnZOYW1lcy5tYXAoKG5hbWUpID0+IFtcbiAgICAgICAgbmFtZSxcbiAgICAgICAgY2xvdWRBc3NlbWJseS5lbnZpcm9ubWVudHMuZmluZCgoaXQpID0+IGl0Lm5hbWUgPT09IG5hbWUpPy5zdGFja05hbWVzXG4gICAgICAgICAgLmxlbmd0aCA/PyAwLFxuICAgICAgXSksXG4gICAgKSxcbiAgfVxufVxuXG4vLyBUaGlzIGlzIGEgc2VsZi1jb250YWluZWQgZnVuY3Rpb24gdGhhdCB3aWxsIGJlIHNlcmlhbGl6ZWQgYXMgYSBsYW1iZGEuXG5jb25zdCBjaGVja0NhblJ1bkhhbmRsZXI6IEhhbmRsZXIgPSBhc3luYyAoZXZlbnQ6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4pID0+IHtcbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby12YXItcmVxdWlyZXMsQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1hc3NpZ25tZW50XG4gIGNvbnN0IEFXUyA9IHJlcXVpcmUoXCJhd3Mtc2RrXCIpXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLWNhbGwsQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1tZW1iZXItYWNjZXNzXG4gIGNvbnN0IHNmID0gbmV3IEFXUy5TdGVwRnVuY3Rpb25zKCkgYXMgX0FXUy5TdGVwRnVuY3Rpb25zXG5cbiAgY29uc29sZS5sb2coXCJFdmVudCByZWNlaXZlZDogXCIsIGV2ZW50KVxuXG4gIGNvbnN0IHN0YXRlTWFjaGluZUFybiA9IGV2ZW50W1wic3RhdGVNYWNoaW5lSWRcIl1cbiAgY29uc3QgY3VycmVudEV4ZWN1dGlvbkFybiA9IGV2ZW50W1wiZXhlY3V0aW9uSWRcIl1cblxuICBjb25zdCBleGVjdXRpb25zID0gKFxuICAgIGF3YWl0IHNmXG4gICAgICAubGlzdEV4ZWN1dGlvbnMoe1xuICAgICAgICBzdGF0ZU1hY2hpbmVBcm4sXG4gICAgICAgIHN0YXR1c0ZpbHRlcjogXCJSVU5OSU5HXCIsXG4gICAgICB9KVxuICAgICAgLnByb21pc2UoKVxuICApLmV4ZWN1dGlvbnNcblxuICBjb25zb2xlLmxvZyhcIkV4ZWN1dGlvbnM6IFwiLCBleGVjdXRpb25zKVxuXG4gIGNvbnN0IGN1cnJlbnRFeGVjdXRpb24gPSBleGVjdXRpb25zLmZpbmQoXG4gICAgKGl0KSA9PiBpdC5leGVjdXRpb25Bcm4gPT0gY3VycmVudEV4ZWN1dGlvbkFybixcbiAgKVxuXG4gIGlmICghY3VycmVudEV4ZWN1dGlvbikge1xuICAgIHRocm93IG5ldyBFcnJvcihcIkNvdWxkIG5vdCBmaW5kIGN1cnJlbnQgZXhlY3V0aW9uXCIpXG4gIH1cblxuICBjb25zdCBuZXdlciA9IGV4ZWN1dGlvbnMuZmlsdGVyKFxuICAgIChpdCkgPT4gaXQuc3RhcnREYXRlID4gY3VycmVudEV4ZWN1dGlvbi5zdGFydERhdGUsXG4gICkubGVuZ3RoXG5cbiAgcmV0dXJuIHtcbiAgICBDYW5SdW5TdGF0ZTpcbiAgICAgIG5ld2VyID4gMCA/IFwiU0tJUFwiIDogZXhlY3V0aW9ucy5sZW5ndGggPT0gMSA/IFwiQ09OVElOVUVcIiA6IFwiV0FJVFwiLFxuICB9XG59XG4iXX0=
|