@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.
Files changed (46) hide show
  1. package/README.md +18 -2
  2. package/lib/cdk-pipelines/cloud-assembly-lookup-handler.d.ts +1 -1
  3. package/lib/cdk-pipelines/cloud-assembly-lookup-handler.js +46 -35
  4. package/lib/cdk-pipelines/liflig-cdk-pipeline.js +6 -6
  5. package/lib/index.d.ts +1 -4
  6. package/lib/index.js +2 -6
  7. package/lib/ses/configurationsetsnsdestination/handler.d.ts +1 -1
  8. package/lib/ses/configurationsetsnsdestination/handler.js +15 -25
  9. package/lib/ses/configurationsetsnsdestination/index.js +7 -7
  10. package/lib/ses/sesdomain/handler.d.ts +1 -1
  11. package/lib/ses/sesdomain/handler.js +31 -31
  12. package/lib/ses/sesdomain/index.js +6 -6
  13. package/lib/ses/sesverifyemail/handler.d.ts +1 -1
  14. package/lib/ses/sesverifyemail/handler.js +12 -10
  15. package/lib/ses/sesverifyemail/index.js +6 -6
  16. package/package.json +14 -3
  17. package/lib/cdk-deploy/cdk-deploy.d.ts +0 -63
  18. package/lib/cdk-deploy/cdk-deploy.js +0 -175
  19. package/lib/cdk-deploy/index.d.ts +0 -1
  20. package/lib/cdk-deploy/index.js +0 -6
  21. package/lib/cdk-deploy/start-deploy-handler.d.ts +0 -8
  22. package/lib/cdk-deploy/start-deploy-handler.js +0 -72
  23. package/lib/cdk-deploy/status-handler.d.ts +0 -6
  24. package/lib/cdk-deploy/status-handler.js +0 -83
  25. package/lib/ecs-update-image/artifact-status.d.ts +0 -39
  26. package/lib/ecs-update-image/artifact-status.js +0 -41
  27. package/lib/ecs-update-image/ecs-update-image.d.ts +0 -41
  28. package/lib/ecs-update-image/ecs-update-image.js +0 -98
  29. package/lib/ecs-update-image/index.d.ts +0 -3
  30. package/lib/ecs-update-image/index.js +0 -10
  31. package/lib/ecs-update-image/start-deploy-handler.d.ts +0 -6
  32. package/lib/ecs-update-image/start-deploy-handler.js +0 -104
  33. package/lib/ecs-update-image/status-handler.d.ts +0 -11
  34. package/lib/ecs-update-image/status-handler.js +0 -74
  35. package/lib/ecs-update-image/tag.d.ts +0 -47
  36. package/lib/ecs-update-image/tag.js +0 -67
  37. package/lib/pipelines/conventions.d.ts +0 -14
  38. package/lib/pipelines/conventions.js +0 -24
  39. package/lib/pipelines/deploy-env.d.ts +0 -18
  40. package/lib/pipelines/deploy-env.js +0 -96
  41. package/lib/pipelines/index.d.ts +0 -2
  42. package/lib/pipelines/index.js +0 -8
  43. package/lib/pipelines/liflig-cdk-deployer-deps.d.ts +0 -13
  44. package/lib/pipelines/liflig-cdk-deployer-deps.js +0 -35
  45. package/lib/pipelines/pipeline.d.ts +0 -78
  46. 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=