aws-cdk 2.32.0 → 2.34.0
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/THIRD_PARTY_LICENSES +10 -6
- package/build-info.json +2 -2
- package/lib/api/bootstrap/bootstrap-template.yaml +1 -3
- package/lib/api/cloudformation-deployments.d.ts +1 -1
- package/lib/api/cloudformation-deployments.js +3 -3
- package/lib/api/nested-stack-helpers.d.ts +13 -13
- package/lib/api/nested-stack-helpers.js +7 -7
- package/lib/api/util/cloudformation.d.ts +3 -2
- package/lib/api/util/cloudformation.js +9 -5
- package/lib/bridge.js +25 -15
- package/lib/cdk-toolkit.d.ts +7 -0
- package/lib/cdk-toolkit.js +10 -2
- package/lib/cli.js +4 -2
- package/lib/index.js +19340 -6602
- package/lib/init-templates/app/typescript/package.json +1 -1
- package/lib/init-templates/sample-app/typescript/package.json +1 -1
- package/lib/notices.js +9 -15
- package/lib/settings.d.ts +9 -0
- package/lib/settings.js +31 -2
- package/lib/setup-sandbox.js +6 -3
- package/lib/util/validate-notification-arn.d.ts +4 -0
- package/lib/util/validate-notification-arn.js +11 -0
- package/package.json +14 -13
- package/test/cdk-toolkit.test.js +22 -2
- package/test/integ/cli/app/app.js +15 -0
- package/test/integ/cli/cli.integtest.js +7 -1
- package/test/integ/cli/sam_cdk_integ_app/src/python/Function/requirements.txt +1 -1
- package/test/integ/cli/sam_cdk_integ_app/src/python/Layer/requirements.txt +1 -1
- package/test/settings.test.js +19 -1
- package/test/util/cloudformation.test.js +31 -4
- package/test/util/validate-notification-arn.test.d.ts +1 -0
- package/test/util/validate-notification-arn.test.js +26 -0
|
@@ -9,24 +9,51 @@ const OVERRIDE = 'TheOverride';
|
|
|
9
9
|
const USE_OVERRIDE = { ParameterKey: PARAM, ParameterValue: OVERRIDE };
|
|
10
10
|
const USE_PREVIOUS = { ParameterKey: PARAM, UsePreviousValue: true };
|
|
11
11
|
let sdkProvider;
|
|
12
|
+
let describeStackMock;
|
|
13
|
+
let getTemplateMock;
|
|
12
14
|
let cfnMocks;
|
|
13
15
|
let cfn;
|
|
14
16
|
beforeEach(async () => {
|
|
15
17
|
sdkProvider = new mock_sdk_1.MockSdkProvider();
|
|
18
|
+
describeStackMock = jest.fn();
|
|
19
|
+
getTemplateMock = jest.fn();
|
|
16
20
|
cfnMocks = {
|
|
17
|
-
describeStacks:
|
|
18
|
-
|
|
19
|
-
.mockImplementation(() => ({ Stacks: [] })),
|
|
21
|
+
describeStacks: describeStackMock,
|
|
22
|
+
getTemplate: getTemplateMock,
|
|
20
23
|
};
|
|
21
24
|
sdkProvider.stubCloudFormation(cfnMocks);
|
|
22
25
|
cfn = (await sdkProvider.forEnvironment()).sdk.cloudFormation();
|
|
23
26
|
});
|
|
24
27
|
test('A non-existent stack pretends to have an empty template', async () => {
|
|
28
|
+
// GIVEN
|
|
29
|
+
describeStackMock.mockImplementation(() => ({ Stacks: [] })); // No stacks exist
|
|
25
30
|
// WHEN
|
|
26
31
|
const stack = await cloudformation_1.CloudFormationStack.lookup(cfn, 'Dummy');
|
|
27
32
|
// THEN
|
|
28
33
|
expect(await stack.template()).toEqual({});
|
|
29
34
|
});
|
|
35
|
+
test("Retrieving a processed template passes 'Processed' to CloudFormation", async () => {
|
|
36
|
+
// GIVEN
|
|
37
|
+
describeStackMock.mockImplementation(() => ({
|
|
38
|
+
Stacks: [
|
|
39
|
+
{
|
|
40
|
+
StackName: 'Dummy',
|
|
41
|
+
},
|
|
42
|
+
],
|
|
43
|
+
}));
|
|
44
|
+
getTemplateMock.mockImplementation(() => ({
|
|
45
|
+
TemplateBody: '{}',
|
|
46
|
+
}));
|
|
47
|
+
// WHEN
|
|
48
|
+
const retrieveProcessedTemplate = true;
|
|
49
|
+
const cloudFormationStack = await cloudformation_1.CloudFormationStack.lookup(cfn, 'Dummy', retrieveProcessedTemplate);
|
|
50
|
+
await cloudFormationStack.template();
|
|
51
|
+
// THEN
|
|
52
|
+
expect(getTemplateMock).toHaveBeenCalledWith({
|
|
53
|
+
StackName: 'Dummy',
|
|
54
|
+
TemplateStage: 'Processed',
|
|
55
|
+
});
|
|
56
|
+
});
|
|
30
57
|
test.each([
|
|
31
58
|
[false, false],
|
|
32
59
|
[false, true],
|
|
@@ -141,4 +168,4 @@ function makeParams(defaultValue, hasPrevValue, override) {
|
|
|
141
168
|
const stackParams = params.updateExisting({ [PARAM]: override ? OVERRIDE : undefined }, prevParams);
|
|
142
169
|
return { apiParameters: stackParams.apiParameters, changed: stackParams.hasChanges(prevParams) };
|
|
143
170
|
}
|
|
144
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudformation.test.js","sourceRoot":"","sources":["cloudformation.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { SSMPARAM_NO_INVALIDATE } from '@aws-cdk/cx-api';\nimport { CloudFormationStack, TemplateParameters } from '../../lib/api/util/cloudformation';\nimport { MockedObject, MockSdkProvider, SyncHandlerSubsetOf } from './mock-sdk';\n\nconst PARAM = 'TheParameter';\nconst DEFAULT = 'TheDefault';\nconst OVERRIDE = 'TheOverride';\n\nconst USE_OVERRIDE = { ParameterKey: PARAM, ParameterValue: OVERRIDE };\nconst USE_PREVIOUS = { ParameterKey: PARAM, UsePreviousValue: true };\n\nlet sdkProvider: MockSdkProvider;\nlet cfnMocks: MockedObject<SyncHandlerSubsetOf<AWS.CloudFormation>>;\nlet cfn: AWS.CloudFormation;\nbeforeEach(async () => {\n  sdkProvider = new MockSdkProvider();\n\n  cfnMocks = {\n    describeStacks: jest.fn()\n      // No stacks exist\n      .mockImplementation(() => ({ Stacks: [] })),\n  };\n  sdkProvider.stubCloudFormation(cfnMocks as any);\n  cfn = (await sdkProvider.forEnvironment()).sdk.cloudFormation();\n});\n\ntest('A non-existent stack pretends to have an empty template', async () => {\n  // WHEN\n  const stack = await CloudFormationStack.lookup(cfn, 'Dummy');\n\n  // THEN\n  expect(await stack.template()).toEqual({});\n});\n\ntest.each([\n  [false, false],\n  [false, true],\n  [true, false],\n  [true, true],\n])('given override, always use the override (parameter has a default: %p, parameter previously supplied: %p)',\n  (haveDefault, havePrevious) => {\n    expect(makeParams(haveDefault, havePrevious, true)).toEqual({\n      apiParameters: [USE_OVERRIDE],\n      changed: true,\n    });\n  });\n\ntest('no default, no prev, no override => error', () => {\n  expect(() => makeParams(false, false, false)).toThrow(/missing a value: TheParameter/);\n});\n\ntest('no default, yes prev, no override => use previous', () => {\n  expect(makeParams(false, true, false)).toEqual({\n    apiParameters: [USE_PREVIOUS],\n    changed: false,\n  });\n});\n\ntest('default, no prev, no override => empty param set (and obviously changes to be applied)', () => {\n  expect(makeParams(true, false, false)).toEqual({\n    apiParameters: [],\n    changed: true,\n  });\n});\n\ntest('default, prev, no override => use previous', () => {\n  expect(makeParams(true, true, false)).toEqual({\n    apiParameters: [USE_PREVIOUS],\n    changed: false,\n  });\n});\n\ntest('if a parameter is retrieved from SSM, the parameters always count as changed', () => {\n  const params = TemplateParameters.fromTemplate({\n    Parameters: {\n      Foo: {\n        Type: 'AWS::SSM::Parameter::Name',\n        Default: '/Some/Key',\n      },\n    },\n  });\n  const oldValues = { Foo: '/Some/Key' };\n\n  // If we don't pass a new value\n  expect(params.updateExisting({}, oldValues).hasChanges(oldValues)).toEqual('ssm');\n\n  // If we do pass a new value but it's the same as the old one\n  expect(params.updateExisting({ Foo: '/Some/Key' }, oldValues).hasChanges(oldValues)).toEqual('ssm');\n});\n\ntest('if a parameter is retrieved from SSM, the parameters doesnt count as changed if it has the magic marker', () => {\n  const params = TemplateParameters.fromTemplate({\n    Parameters: {\n      Foo: {\n        Type: 'AWS::SSM::Parameter::Name',\n        Default: '/Some/Key',\n        Description: `blabla ${SSMPARAM_NO_INVALIDATE}`,\n      },\n    },\n  });\n  const oldValues = { Foo: '/Some/Key' };\n\n  // If we don't pass a new value\n  expect(params.updateExisting({}, oldValues).hasChanges(oldValues)).toEqual(false);\n\n  // If we do pass a new value but it's the same as the old one\n  expect(params.updateExisting({ Foo: '/Some/Key' }, oldValues).hasChanges(oldValues)).toEqual(false);\n\n  // If we do pass a new value and it's different\n  expect(params.updateExisting({ Foo: '/OTHER/Key' }, oldValues).hasChanges(oldValues)).toEqual(true);\n});\n\ntest('empty string is a valid update value', () => {\n  const params = TemplateParameters.fromTemplate({\n    Parameters: {\n      Foo: { Type: 'String', Default: 'Foo' },\n    },\n  });\n\n  expect(params.updateExisting({ Foo: '' }, { Foo: 'ThisIsOld' }).apiParameters).toEqual([\n    { ParameterKey: 'Foo', ParameterValue: '' },\n  ]);\n});\n\ntest('unknown parameter in overrides, pass it anyway', () => {\n  // Not sure if we really want this. It seems like it would be nice\n  // to not pass parameters that aren't expected, given that CFN will\n  // just error out. But maybe we want to be warned of typos...\n  const params = TemplateParameters.fromTemplate({\n    Parameters: {\n      Foo: { Type: 'String', Default: 'Foo' },\n    },\n  });\n\n  expect(params.updateExisting({ Bar: 'Bar' }, {}).apiParameters).toEqual([\n    { ParameterKey: 'Bar', ParameterValue: 'Bar' },\n  ]);\n});\n\ntest('if an unsupplied parameter reverts to its default, it can still be dirty', () => {\n  // GIVEN\n  const templateParams = TemplateParameters.fromTemplate({\n    Parameters: {\n      Foo: { Type: 'String', Default: 'Foo' },\n    },\n  });\n\n  // WHEN\n  const stackParams = templateParams.supplyAll({});\n\n  // THEN\n  expect(stackParams.hasChanges({ Foo: 'NonStandard' })).toEqual(true);\n  expect(stackParams.hasChanges({ Foo: 'Foo' })).toEqual(false);\n});\n\nfunction makeParams(defaultValue: boolean, hasPrevValue: boolean, override: boolean) {\n  const params = TemplateParameters.fromTemplate({\n    Parameters: {\n      [PARAM]: {\n        Type: 'String',\n        Default: defaultValue ? DEFAULT : undefined,\n      },\n    },\n  });\n  const prevParams: Record<string, string> = hasPrevValue ? { [PARAM]: 'Foo' } : {};\n  const stackParams = params.updateExisting({ [PARAM]: override ? OVERRIDE : undefined }, prevParams);\n\n  return { apiParameters: stackParams.apiParameters, changed: stackParams.hasChanges(prevParams) };\n}\n"]}
|
|
171
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudformation.test.js","sourceRoot":"","sources":["cloudformation.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { SSMPARAM_NO_INVALIDATE } from '@aws-cdk/cx-api';\nimport { CloudFormationStack, TemplateParameters } from '../../lib/api/util/cloudformation';\nimport { MockedObject, MockSdkProvider, SyncHandlerSubsetOf } from './mock-sdk';\n\nconst PARAM = 'TheParameter';\nconst DEFAULT = 'TheDefault';\nconst OVERRIDE = 'TheOverride';\n\nconst USE_OVERRIDE = { ParameterKey: PARAM, ParameterValue: OVERRIDE };\nconst USE_PREVIOUS = { ParameterKey: PARAM, UsePreviousValue: true };\n\nlet sdkProvider: MockSdkProvider;\nlet describeStackMock: jest.Mock;\nlet getTemplateMock: jest.Mock;\nlet cfnMocks: MockedObject<SyncHandlerSubsetOf<AWS.CloudFormation>>;\nlet cfn: AWS.CloudFormation;\nbeforeEach(async () => {\n  sdkProvider = new MockSdkProvider();\n\n  describeStackMock = jest.fn();\n  getTemplateMock = jest.fn();\n  cfnMocks = {\n    describeStacks: describeStackMock,\n    getTemplate: getTemplateMock,\n  };\n  sdkProvider.stubCloudFormation(cfnMocks as any);\n  cfn = (await sdkProvider.forEnvironment()).sdk.cloudFormation();\n});\n\ntest('A non-existent stack pretends to have an empty template', async () => {\n  // GIVEN\n  describeStackMock.mockImplementation(() => ({ Stacks: [] })); // No stacks exist\n\n  // WHEN\n  const stack = await CloudFormationStack.lookup(cfn, 'Dummy');\n\n  // THEN\n  expect(await stack.template()).toEqual({});\n});\n\ntest(\"Retrieving a processed template passes 'Processed' to CloudFormation\", async () => {\n  // GIVEN\n  describeStackMock.mockImplementation(() => ({\n    Stacks: [\n      {\n        StackName: 'Dummy',\n      },\n    ],\n  }));\n  getTemplateMock.mockImplementation(() => ({\n    TemplateBody: '{}',\n  }));\n\n  // WHEN\n  const retrieveProcessedTemplate = true;\n  const cloudFormationStack = await CloudFormationStack.lookup(cfn, 'Dummy', retrieveProcessedTemplate);\n  await cloudFormationStack.template();\n\n  // THEN\n  expect(getTemplateMock).toHaveBeenCalledWith({\n    StackName: 'Dummy',\n    TemplateStage: 'Processed',\n  });\n});\n\ntest.each([\n  [false, false],\n  [false, true],\n  [true, false],\n  [true, true],\n])('given override, always use the override (parameter has a default: %p, parameter previously supplied: %p)',\n  (haveDefault, havePrevious) => {\n    expect(makeParams(haveDefault, havePrevious, true)).toEqual({\n      apiParameters: [USE_OVERRIDE],\n      changed: true,\n    });\n  });\n\ntest('no default, no prev, no override => error', () => {\n  expect(() => makeParams(false, false, false)).toThrow(/missing a value: TheParameter/);\n});\n\ntest('no default, yes prev, no override => use previous', () => {\n  expect(makeParams(false, true, false)).toEqual({\n    apiParameters: [USE_PREVIOUS],\n    changed: false,\n  });\n});\n\ntest('default, no prev, no override => empty param set (and obviously changes to be applied)', () => {\n  expect(makeParams(true, false, false)).toEqual({\n    apiParameters: [],\n    changed: true,\n  });\n});\n\ntest('default, prev, no override => use previous', () => {\n  expect(makeParams(true, true, false)).toEqual({\n    apiParameters: [USE_PREVIOUS],\n    changed: false,\n  });\n});\n\ntest('if a parameter is retrieved from SSM, the parameters always count as changed', () => {\n  const params = TemplateParameters.fromTemplate({\n    Parameters: {\n      Foo: {\n        Type: 'AWS::SSM::Parameter::Name',\n        Default: '/Some/Key',\n      },\n    },\n  });\n  const oldValues = { Foo: '/Some/Key' };\n\n  // If we don't pass a new value\n  expect(params.updateExisting({}, oldValues).hasChanges(oldValues)).toEqual('ssm');\n\n  // If we do pass a new value but it's the same as the old one\n  expect(params.updateExisting({ Foo: '/Some/Key' }, oldValues).hasChanges(oldValues)).toEqual('ssm');\n});\n\ntest('if a parameter is retrieved from SSM, the parameters doesnt count as changed if it has the magic marker', () => {\n  const params = TemplateParameters.fromTemplate({\n    Parameters: {\n      Foo: {\n        Type: 'AWS::SSM::Parameter::Name',\n        Default: '/Some/Key',\n        Description: `blabla ${SSMPARAM_NO_INVALIDATE}`,\n      },\n    },\n  });\n  const oldValues = { Foo: '/Some/Key' };\n\n  // If we don't pass a new value\n  expect(params.updateExisting({}, oldValues).hasChanges(oldValues)).toEqual(false);\n\n  // If we do pass a new value but it's the same as the old one\n  expect(params.updateExisting({ Foo: '/Some/Key' }, oldValues).hasChanges(oldValues)).toEqual(false);\n\n  // If we do pass a new value and it's different\n  expect(params.updateExisting({ Foo: '/OTHER/Key' }, oldValues).hasChanges(oldValues)).toEqual(true);\n});\n\ntest('empty string is a valid update value', () => {\n  const params = TemplateParameters.fromTemplate({\n    Parameters: {\n      Foo: { Type: 'String', Default: 'Foo' },\n    },\n  });\n\n  expect(params.updateExisting({ Foo: '' }, { Foo: 'ThisIsOld' }).apiParameters).toEqual([\n    { ParameterKey: 'Foo', ParameterValue: '' },\n  ]);\n});\n\ntest('unknown parameter in overrides, pass it anyway', () => {\n  // Not sure if we really want this. It seems like it would be nice\n  // to not pass parameters that aren't expected, given that CFN will\n  // just error out. But maybe we want to be warned of typos...\n  const params = TemplateParameters.fromTemplate({\n    Parameters: {\n      Foo: { Type: 'String', Default: 'Foo' },\n    },\n  });\n\n  expect(params.updateExisting({ Bar: 'Bar' }, {}).apiParameters).toEqual([\n    { ParameterKey: 'Bar', ParameterValue: 'Bar' },\n  ]);\n});\n\ntest('if an unsupplied parameter reverts to its default, it can still be dirty', () => {\n  // GIVEN\n  const templateParams = TemplateParameters.fromTemplate({\n    Parameters: {\n      Foo: { Type: 'String', Default: 'Foo' },\n    },\n  });\n\n  // WHEN\n  const stackParams = templateParams.supplyAll({});\n\n  // THEN\n  expect(stackParams.hasChanges({ Foo: 'NonStandard' })).toEqual(true);\n  expect(stackParams.hasChanges({ Foo: 'Foo' })).toEqual(false);\n});\n\nfunction makeParams(defaultValue: boolean, hasPrevValue: boolean, override: boolean) {\n  const params = TemplateParameters.fromTemplate({\n    Parameters: {\n      [PARAM]: {\n        Type: 'String',\n        Default: defaultValue ? DEFAULT : undefined,\n      },\n    },\n  });\n  const prevParams: Record<string, string> = hasPrevValue ? { [PARAM]: 'Foo' } : {};\n  const stackParams = params.updateExisting({ [PARAM]: override ? OVERRIDE : undefined }, prevParams);\n\n  return { apiParameters: stackParams.apiParameters, changed: stackParams.hasChanges(prevParams) };\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const validate_notification_arn_1 = require("../../lib/util/validate-notification-arn");
|
|
4
|
+
describe('validate sns arns', () => {
|
|
5
|
+
test('empty string', () => {
|
|
6
|
+
const arn = '';
|
|
7
|
+
expect(validate_notification_arn_1.validateSnsTopicArn(arn)).toEqual(false);
|
|
8
|
+
});
|
|
9
|
+
test('colon in topic name', () => {
|
|
10
|
+
const arn = 'arn:aws:sns:eu-west-1:abc:foo';
|
|
11
|
+
expect(validate_notification_arn_1.validateSnsTopicArn(arn)).toEqual(false);
|
|
12
|
+
});
|
|
13
|
+
test('missing :aws: in arn', () => {
|
|
14
|
+
const arn = 'arn:sns:eu-west-1:foobar';
|
|
15
|
+
expect(validate_notification_arn_1.validateSnsTopicArn(arn)).toEqual(false);
|
|
16
|
+
});
|
|
17
|
+
test('dash in topic name', () => {
|
|
18
|
+
const arn = 'arn:aws:sns:eu-west-1:123456789876:foo-bar';
|
|
19
|
+
expect(validate_notification_arn_1.validateSnsTopicArn(arn)).toEqual(true);
|
|
20
|
+
});
|
|
21
|
+
test('underscore in topic name', () => {
|
|
22
|
+
const arn = 'arn:aws:sns:eu-west-1:123456789876:foo-bar_baz';
|
|
23
|
+
expect(validate_notification_arn_1.validateSnsTopicArn(arn)).toEqual(true);
|
|
24
|
+
});
|
|
25
|
+
});
|
|
26
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmFsaWRhdGUtbm90aWZpY2F0aW9uLWFybi50ZXN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsidmFsaWRhdGUtbm90aWZpY2F0aW9uLWFybi50ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQUEsd0ZBQStFO0FBRS9FLFFBQVEsQ0FBQyxtQkFBbUIsRUFBRSxHQUFHLEVBQUU7SUFDakMsSUFBSSxDQUFDLGNBQWMsRUFBRSxHQUFHLEVBQUU7UUFDeEIsTUFBTSxHQUFHLEdBQUcsRUFBRSxDQUFDO1FBQ2YsTUFBTSxDQUFDLCtDQUFtQixDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ2xELENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLHFCQUFxQixFQUFFLEdBQUcsRUFBRTtRQUMvQixNQUFNLEdBQUcsR0FBRywrQkFBK0IsQ0FBQztRQUM1QyxNQUFNLENBQUMsK0NBQW1CLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDbEQsQ0FBQyxDQUFDLENBQUM7SUFFSCxJQUFJLENBQUMsc0JBQXNCLEVBQUUsR0FBRyxFQUFFO1FBQ2hDLE1BQU0sR0FBRyxHQUFHLDBCQUEwQixDQUFDO1FBQ3ZDLE1BQU0sQ0FBQywrQ0FBbUIsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUNsRCxDQUFDLENBQUMsQ0FBQztJQUVILElBQUksQ0FBQyxvQkFBb0IsRUFBRSxHQUFHLEVBQUU7UUFDOUIsTUFBTSxHQUFHLEdBQUcsNENBQTRDLENBQUM7UUFDekQsTUFBTSxDQUFDLCtDQUFtQixDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQ2pELENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLDBCQUEwQixFQUFFLEdBQUcsRUFBRTtRQUNwQyxNQUFNLEdBQUcsR0FBRyxnREFBZ0QsQ0FBQztRQUM3RCxNQUFNLENBQUMsK0NBQW1CLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDakQsQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHZhbGlkYXRlU25zVG9waWNBcm4gfSBmcm9tICcuLi8uLi9saWIvdXRpbC92YWxpZGF0ZS1ub3RpZmljYXRpb24tYXJuJztcblxuZGVzY3JpYmUoJ3ZhbGlkYXRlIHNucyBhcm5zJywgKCkgPT4ge1xuICB0ZXN0KCdlbXB0eSBzdHJpbmcnLCAoKSA9PiB7XG4gICAgY29uc3QgYXJuID0gJyc7XG4gICAgZXhwZWN0KHZhbGlkYXRlU25zVG9waWNBcm4oYXJuKSkudG9FcXVhbChmYWxzZSk7XG4gIH0pO1xuXG4gIHRlc3QoJ2NvbG9uIGluIHRvcGljIG5hbWUnLCAoKSA9PiB7XG4gICAgY29uc3QgYXJuID0gJ2Fybjphd3M6c25zOmV1LXdlc3QtMTphYmM6Zm9vJztcbiAgICBleHBlY3QodmFsaWRhdGVTbnNUb3BpY0Fybihhcm4pKS50b0VxdWFsKGZhbHNlKTtcbiAgfSk7XG5cbiAgdGVzdCgnbWlzc2luZyA6YXdzOiBpbiBhcm4nLCAoKSA9PiB7XG4gICAgY29uc3QgYXJuID0gJ2FybjpzbnM6ZXUtd2VzdC0xOmZvb2Jhcic7XG4gICAgZXhwZWN0KHZhbGlkYXRlU25zVG9waWNBcm4oYXJuKSkudG9FcXVhbChmYWxzZSk7XG4gIH0pO1xuXG4gIHRlc3QoJ2Rhc2ggaW4gdG9waWMgbmFtZScsICgpID0+IHtcbiAgICBjb25zdCBhcm4gPSAnYXJuOmF3czpzbnM6ZXUtd2VzdC0xOjEyMzQ1Njc4OTg3Njpmb28tYmFyJztcbiAgICBleHBlY3QodmFsaWRhdGVTbnNUb3BpY0Fybihhcm4pKS50b0VxdWFsKHRydWUpO1xuICB9KTtcblxuICB0ZXN0KCd1bmRlcnNjb3JlIGluIHRvcGljIG5hbWUnLCAoKSA9PiB7XG4gICAgY29uc3QgYXJuID0gJ2Fybjphd3M6c25zOmV1LXdlc3QtMToxMjM0NTY3ODk4NzY6Zm9vLWJhcl9iYXonO1xuICAgIGV4cGVjdCh2YWxpZGF0ZVNuc1RvcGljQXJuKGFybikpLnRvRXF1YWwodHJ1ZSk7XG4gIH0pO1xufSk7XG5cbiJdfQ==
|