aws-cdk 2.3.0 → 2.7.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/LICENSE +1 -1
- package/NOTICE +1 -1
- package/README.md +4 -1
- package/bin/cdk.js +7 -6
- package/build-info.json +2 -2
- package/lib/api/aws-auth/sdk-provider.d.ts +26 -1
- package/lib/api/aws-auth/sdk-provider.js +14 -24
- package/lib/api/aws-auth/sdk.d.ts +2 -0
- package/lib/api/aws-auth/sdk.js +4 -1
- package/lib/api/bootstrap/bootstrap-template.yaml +3 -1
- package/lib/api/bootstrap/deploy-bootstrap.js +2 -2
- package/lib/api/cloudformation-deployments.d.ts +63 -1
- package/lib/api/cloudformation-deployments.js +66 -3
- package/lib/api/cxapp/cloud-assembly.js +5 -5
- package/lib/api/deploy-stack.js +10 -4
- package/lib/api/hotswap/code-build-projects.d.ts +3 -0
- package/lib/api/hotswap/code-build-projects.js +53 -0
- package/lib/api/hotswap/common.d.ts +17 -1
- package/lib/api/hotswap/common.js +30 -6
- package/lib/api/hotswap/ecs-services.js +4 -18
- package/lib/api/hotswap/lambda-functions.js +169 -44
- package/lib/api/hotswap/s3-bucket-deployments.js +3 -10
- package/lib/api/hotswap/stepfunctions-state-machines.js +2 -1
- package/lib/api/hotswap-deployments.d.ts +0 -7
- package/lib/api/hotswap-deployments.js +85 -13
- package/lib/api/toolkit-info.d.ts +5 -5
- package/lib/api/toolkit-info.js +10 -10
- package/lib/api/util/cloudformation/stack-activity-monitor.js +22 -22
- package/lib/assets.js +3 -3
- package/lib/cdk-toolkit.js +66 -32
- package/lib/commands/context.js +7 -7
- package/lib/commands/docs.js +5 -5
- package/lib/commands/doctor.js +6 -6
- package/lib/context-providers/ami.js +2 -2
- package/lib/context-providers/availability-zones.js +2 -2
- package/lib/context-providers/endpoint-service-availability-zones.js +2 -2
- package/lib/context-providers/hosted-zones.js +2 -2
- package/lib/context-providers/keys.js +2 -2
- package/lib/context-providers/load-balancers.js +3 -3
- package/lib/context-providers/security-groups.js +2 -2
- package/lib/context-providers/ssm-parameters.js +2 -2
- package/lib/context-providers/vpcs.js +2 -2
- package/lib/diff.js +3 -3
- package/lib/init.js +15 -14
- package/lib/logging.js +7 -7
- package/lib/os.js +3 -3
- package/lib/plugin.js +4 -4
- package/lib/util/asset-publishing.js +3 -3
- package/lib/util/console-formatters.js +4 -3
- package/lib/util/npm.d.ts +1 -0
- package/lib/util/npm.js +21 -0
- package/lib/version.d.ts +1 -1
- package/lib/version.js +23 -20
- package/npm-shrinkwrap.json +229 -101
- package/package.json +19 -18
- package/test/api/cloudformation-deployments.test.js +2 -2
- package/test/api/hotswap/{lambda-hotswap-deployments.test.d.ts → code-build-projects-hotswap-deployments.test.d.ts} +0 -0
- package/test/api/hotswap/code-build-projects-hotswap-deployments.test.js +576 -0
- package/test/api/hotswap/hotswap-deployments.test.js +4 -2
- package/test/api/hotswap/hotswap-test-setup.d.ts +3 -1
- package/test/api/hotswap/hotswap-test-setup.js +7 -4
- package/test/api/hotswap/lambda-functions-hotswap-deployments.test.d.ts +1 -0
- package/test/api/hotswap/lambda-functions-hotswap-deployments.test.js +502 -0
- package/test/api/hotswap/lambda-versions-aliases-hotswap-deployments.test.d.ts +1 -0
- package/test/api/hotswap/lambda-versions-aliases-hotswap-deployments.test.js +197 -0
- package/test/api/sdk-provider.test.js +11 -11
- package/test/api/stack-activity-monitor.test.js +13 -13
- package/test/cdk-toolkit.test.js +240 -13
- package/test/context-providers/load-balancers.test.js +4 -4
- package/test/init.test.js +5 -4
- package/test/util/cloudformation.test.js +2 -2
- package/test/util/console-formatters.test.js +6 -6
- package/test/util/mock-sdk.d.ts +4 -2
- package/test/util/mock-sdk.js +6 -2
- package/test/util/mock-toolkitinfo.js +2 -2
- package/test/version.test.js +45 -3
- package/test/api/hotswap/lambda-hotswap-deployments.test.js +0 -418
|
@@ -0,0 +1,197 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const setup = require("./hotswap-test-setup");
|
|
4
|
+
let mockUpdateLambdaCode;
|
|
5
|
+
let mockPublishVersion;
|
|
6
|
+
let mockUpdateAlias;
|
|
7
|
+
let hotswapMockSdkProvider;
|
|
8
|
+
beforeEach(() => {
|
|
9
|
+
hotswapMockSdkProvider = setup.setupHotswapTests();
|
|
10
|
+
mockUpdateLambdaCode = jest.fn();
|
|
11
|
+
mockPublishVersion = jest.fn();
|
|
12
|
+
mockUpdateAlias = jest.fn();
|
|
13
|
+
hotswapMockSdkProvider.stubLambda({
|
|
14
|
+
updateFunctionCode: mockUpdateLambdaCode,
|
|
15
|
+
publishVersion: mockPublishVersion,
|
|
16
|
+
updateAlias: mockUpdateAlias,
|
|
17
|
+
waitFor: jest.fn(),
|
|
18
|
+
});
|
|
19
|
+
});
|
|
20
|
+
test('hotswaps a Version if it points to a changed Function, even if it itself is unchanged', async () => {
|
|
21
|
+
// GIVEN
|
|
22
|
+
setup.setCurrentCfnStackTemplate({
|
|
23
|
+
Resources: {
|
|
24
|
+
Func: {
|
|
25
|
+
Type: 'AWS::Lambda::Function',
|
|
26
|
+
Properties: {
|
|
27
|
+
Code: {
|
|
28
|
+
S3Bucket: 'current-bucket',
|
|
29
|
+
S3Key: 'current-key',
|
|
30
|
+
},
|
|
31
|
+
FunctionName: 'my-function',
|
|
32
|
+
},
|
|
33
|
+
},
|
|
34
|
+
Version: {
|
|
35
|
+
Type: 'AWS::Lambda::Version',
|
|
36
|
+
Properties: {
|
|
37
|
+
FunctionName: { Ref: 'Func' },
|
|
38
|
+
},
|
|
39
|
+
},
|
|
40
|
+
},
|
|
41
|
+
});
|
|
42
|
+
const cdkStackArtifact = setup.cdkStackArtifactOf({
|
|
43
|
+
template: {
|
|
44
|
+
Resources: {
|
|
45
|
+
Func: {
|
|
46
|
+
Type: 'AWS::Lambda::Function',
|
|
47
|
+
Properties: {
|
|
48
|
+
Code: {
|
|
49
|
+
S3Bucket: 'current-bucket',
|
|
50
|
+
S3Key: 'new-key',
|
|
51
|
+
},
|
|
52
|
+
FunctionName: 'my-function',
|
|
53
|
+
},
|
|
54
|
+
},
|
|
55
|
+
Version: {
|
|
56
|
+
Type: 'AWS::Lambda::Version',
|
|
57
|
+
Properties: {
|
|
58
|
+
FunctionName: { Ref: 'Func' },
|
|
59
|
+
},
|
|
60
|
+
},
|
|
61
|
+
},
|
|
62
|
+
},
|
|
63
|
+
});
|
|
64
|
+
// WHEN
|
|
65
|
+
const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);
|
|
66
|
+
// THEN
|
|
67
|
+
expect(deployStackResult).not.toBeUndefined();
|
|
68
|
+
expect(mockPublishVersion).toHaveBeenCalledWith({
|
|
69
|
+
FunctionName: 'my-function',
|
|
70
|
+
});
|
|
71
|
+
});
|
|
72
|
+
test('hotswaps a Version if it points to a changed Function, even if it itself is replaced', async () => {
|
|
73
|
+
// GIVEN
|
|
74
|
+
setup.setCurrentCfnStackTemplate({
|
|
75
|
+
Resources: {
|
|
76
|
+
Func: {
|
|
77
|
+
Type: 'AWS::Lambda::Function',
|
|
78
|
+
Properties: {
|
|
79
|
+
Code: {
|
|
80
|
+
S3Bucket: 'current-bucket',
|
|
81
|
+
S3Key: 'current-key',
|
|
82
|
+
},
|
|
83
|
+
FunctionName: 'my-function',
|
|
84
|
+
},
|
|
85
|
+
},
|
|
86
|
+
Version1: {
|
|
87
|
+
Type: 'AWS::Lambda::Version',
|
|
88
|
+
Properties: {
|
|
89
|
+
FunctionName: { Ref: 'Func' },
|
|
90
|
+
},
|
|
91
|
+
},
|
|
92
|
+
},
|
|
93
|
+
});
|
|
94
|
+
const cdkStackArtifact = setup.cdkStackArtifactOf({
|
|
95
|
+
template: {
|
|
96
|
+
Resources: {
|
|
97
|
+
Func: {
|
|
98
|
+
Type: 'AWS::Lambda::Function',
|
|
99
|
+
Properties: {
|
|
100
|
+
Code: {
|
|
101
|
+
S3Bucket: 'current-bucket',
|
|
102
|
+
S3Key: 'new-key',
|
|
103
|
+
},
|
|
104
|
+
FunctionName: 'my-function',
|
|
105
|
+
},
|
|
106
|
+
},
|
|
107
|
+
Version2: {
|
|
108
|
+
Type: 'AWS::Lambda::Version',
|
|
109
|
+
Properties: {
|
|
110
|
+
FunctionName: { Ref: 'Func' },
|
|
111
|
+
},
|
|
112
|
+
},
|
|
113
|
+
},
|
|
114
|
+
},
|
|
115
|
+
});
|
|
116
|
+
// WHEN
|
|
117
|
+
const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);
|
|
118
|
+
// THEN
|
|
119
|
+
expect(deployStackResult).not.toBeUndefined();
|
|
120
|
+
expect(mockPublishVersion).toHaveBeenCalledWith({
|
|
121
|
+
FunctionName: 'my-function',
|
|
122
|
+
});
|
|
123
|
+
});
|
|
124
|
+
test('hotswaps a Version and an Alias if the Function they point to changed', async () => {
|
|
125
|
+
// GIVEN
|
|
126
|
+
setup.setCurrentCfnStackTemplate({
|
|
127
|
+
Resources: {
|
|
128
|
+
Func: {
|
|
129
|
+
Type: 'AWS::Lambda::Function',
|
|
130
|
+
Properties: {
|
|
131
|
+
Code: {
|
|
132
|
+
S3Bucket: 'current-bucket',
|
|
133
|
+
S3Key: 'current-key',
|
|
134
|
+
},
|
|
135
|
+
FunctionName: 'my-function',
|
|
136
|
+
},
|
|
137
|
+
},
|
|
138
|
+
Version1: {
|
|
139
|
+
Type: 'AWS::Lambda::Version',
|
|
140
|
+
Properties: {
|
|
141
|
+
FunctionName: { Ref: 'Func' },
|
|
142
|
+
},
|
|
143
|
+
},
|
|
144
|
+
Alias: {
|
|
145
|
+
Type: 'AWS::Lambda::Alias',
|
|
146
|
+
Properties: {
|
|
147
|
+
FunctionName: { Ref: 'Func' },
|
|
148
|
+
FunctionVersion: { 'Fn::GetAtt': ['Version1', 'Version'] },
|
|
149
|
+
Name: 'dev',
|
|
150
|
+
},
|
|
151
|
+
},
|
|
152
|
+
},
|
|
153
|
+
});
|
|
154
|
+
const cdkStackArtifact = setup.cdkStackArtifactOf({
|
|
155
|
+
template: {
|
|
156
|
+
Resources: {
|
|
157
|
+
Func: {
|
|
158
|
+
Type: 'AWS::Lambda::Function',
|
|
159
|
+
Properties: {
|
|
160
|
+
Code: {
|
|
161
|
+
S3Bucket: 'current-bucket',
|
|
162
|
+
S3Key: 'new-key',
|
|
163
|
+
},
|
|
164
|
+
FunctionName: 'my-function',
|
|
165
|
+
},
|
|
166
|
+
},
|
|
167
|
+
Version2: {
|
|
168
|
+
Type: 'AWS::Lambda::Version',
|
|
169
|
+
Properties: {
|
|
170
|
+
FunctionName: { Ref: 'Func' },
|
|
171
|
+
},
|
|
172
|
+
},
|
|
173
|
+
Alias: {
|
|
174
|
+
Type: 'AWS::Lambda::Alias',
|
|
175
|
+
Properties: {
|
|
176
|
+
FunctionName: { Ref: 'Func' },
|
|
177
|
+
FunctionVersion: { 'Fn::GetAtt': ['Version2', 'Version'] },
|
|
178
|
+
Name: 'dev',
|
|
179
|
+
},
|
|
180
|
+
},
|
|
181
|
+
},
|
|
182
|
+
},
|
|
183
|
+
});
|
|
184
|
+
mockPublishVersion.mockReturnValue({
|
|
185
|
+
Version: 'v2',
|
|
186
|
+
});
|
|
187
|
+
// WHEN
|
|
188
|
+
const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);
|
|
189
|
+
// THEN
|
|
190
|
+
expect(deployStackResult).not.toBeUndefined();
|
|
191
|
+
expect(mockUpdateAlias).toHaveBeenCalledWith({
|
|
192
|
+
FunctionName: 'my-function',
|
|
193
|
+
FunctionVersion: 'v2',
|
|
194
|
+
Name: 'dev',
|
|
195
|
+
});
|
|
196
|
+
});
|
|
197
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-versions-aliases-hotswap-deployments.test.js","sourceRoot":"","sources":["lambda-versions-aliases-hotswap-deployments.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { Lambda } from 'aws-sdk';\nimport * as setup from './hotswap-test-setup';\n\nlet mockUpdateLambdaCode: (params: Lambda.Types.UpdateFunctionCodeRequest) => Lambda.Types.FunctionConfiguration;\nlet mockPublishVersion: jest.Mock<Lambda.FunctionConfiguration, Lambda.PublishVersionRequest[]>;\nlet mockUpdateAlias: (params: Lambda.UpdateAliasRequest) => Lambda.AliasConfiguration;\nlet hotswapMockSdkProvider: setup.HotswapMockSdkProvider;\n\nbeforeEach(() => {\n  hotswapMockSdkProvider = setup.setupHotswapTests();\n  mockUpdateLambdaCode = jest.fn();\n  mockPublishVersion = jest.fn();\n  mockUpdateAlias = jest.fn();\n  hotswapMockSdkProvider.stubLambda({\n    updateFunctionCode: mockUpdateLambdaCode,\n    publishVersion: mockPublishVersion,\n    updateAlias: mockUpdateAlias,\n    waitFor: jest.fn(),\n  });\n});\n\ntest('hotswaps a Version if it points to a changed Function, even if it itself is unchanged', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      Func: {\n        Type: 'AWS::Lambda::Function',\n        Properties: {\n          Code: {\n            S3Bucket: 'current-bucket',\n            S3Key: 'current-key',\n          },\n          FunctionName: 'my-function',\n        },\n      },\n      Version: {\n        Type: 'AWS::Lambda::Version',\n        Properties: {\n          FunctionName: { Ref: 'Func' },\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        Func: {\n          Type: 'AWS::Lambda::Function',\n          Properties: {\n            Code: {\n              S3Bucket: 'current-bucket',\n              S3Key: 'new-key',\n            },\n            FunctionName: 'my-function',\n          },\n        },\n        Version: {\n          Type: 'AWS::Lambda::Version',\n          Properties: {\n            FunctionName: { Ref: 'Func' },\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).not.toBeUndefined();\n  expect(mockPublishVersion).toHaveBeenCalledWith({\n    FunctionName: 'my-function',\n  });\n});\n\ntest('hotswaps a Version if it points to a changed Function, even if it itself is replaced', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      Func: {\n        Type: 'AWS::Lambda::Function',\n        Properties: {\n          Code: {\n            S3Bucket: 'current-bucket',\n            S3Key: 'current-key',\n          },\n          FunctionName: 'my-function',\n        },\n      },\n      Version1: {\n        Type: 'AWS::Lambda::Version',\n        Properties: {\n          FunctionName: { Ref: 'Func' },\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        Func: {\n          Type: 'AWS::Lambda::Function',\n          Properties: {\n            Code: {\n              S3Bucket: 'current-bucket',\n              S3Key: 'new-key',\n            },\n            FunctionName: 'my-function',\n          },\n        },\n        Version2: {\n          Type: 'AWS::Lambda::Version',\n          Properties: {\n            FunctionName: { Ref: 'Func' },\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).not.toBeUndefined();\n  expect(mockPublishVersion).toHaveBeenCalledWith({\n    FunctionName: 'my-function',\n  });\n});\n\ntest('hotswaps a Version and an Alias if the Function they point to changed', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      Func: {\n        Type: 'AWS::Lambda::Function',\n        Properties: {\n          Code: {\n            S3Bucket: 'current-bucket',\n            S3Key: 'current-key',\n          },\n          FunctionName: 'my-function',\n        },\n      },\n      Version1: {\n        Type: 'AWS::Lambda::Version',\n        Properties: {\n          FunctionName: { Ref: 'Func' },\n        },\n      },\n      Alias: {\n        Type: 'AWS::Lambda::Alias',\n        Properties: {\n          FunctionName: { Ref: 'Func' },\n          FunctionVersion: { 'Fn::GetAtt': ['Version1', 'Version'] },\n          Name: 'dev',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        Func: {\n          Type: 'AWS::Lambda::Function',\n          Properties: {\n            Code: {\n              S3Bucket: 'current-bucket',\n              S3Key: 'new-key',\n            },\n            FunctionName: 'my-function',\n          },\n        },\n        Version2: {\n          Type: 'AWS::Lambda::Version',\n          Properties: {\n            FunctionName: { Ref: 'Func' },\n          },\n        },\n        Alias: {\n          Type: 'AWS::Lambda::Alias',\n          Properties: {\n            FunctionName: { Ref: 'Func' },\n            FunctionVersion: { 'Fn::GetAtt': ['Version2', 'Version'] },\n            Name: 'dev',\n          },\n        },\n      },\n    },\n  });\n  mockPublishVersion.mockReturnValue({\n    Version: 'v2',\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).not.toBeUndefined();\n  expect(mockUpdateAlias).toHaveBeenCalledWith({\n    FunctionName: 'my-function',\n    FunctionVersion: 'v2',\n    Name: 'dev',\n  });\n});\n"]}
|
|
@@ -88,7 +88,7 @@ describe('with intercepted network calls', () => {
|
|
|
88
88
|
expect(provider.defaultRegion).toEqual('eu-bla-5');
|
|
89
89
|
await expect(provider.defaultAccount()).resolves.toEqual({ accountId: uniq('11111'), partition: 'aws-here' });
|
|
90
90
|
// Ask for a different region
|
|
91
|
-
const sdk = await provider.forEnvironment({ ...env(uniq('11111')), region: 'rgn' }, aws_auth_1.Mode.ForReading);
|
|
91
|
+
const sdk = (await provider.forEnvironment({ ...env(uniq('11111')), region: 'rgn' }, aws_auth_1.Mode.ForReading)).sdk;
|
|
92
92
|
expect(sdkConfig(sdk).credentials.accessKeyId).toEqual(uniq('access'));
|
|
93
93
|
expect(sdk.currentRegion).toEqual('rgn');
|
|
94
94
|
});
|
|
@@ -116,7 +116,7 @@ describe('with intercepted network calls', () => {
|
|
|
116
116
|
});
|
|
117
117
|
const provider = await providerFromProfile(undefined);
|
|
118
118
|
// THEN
|
|
119
|
-
const sdk = await provider.forEnvironment(cxapi.EnvironmentUtils.make(cxapi.UNKNOWN_ACCOUNT, cxapi.UNKNOWN_REGION), aws_auth_1.Mode.ForReading);
|
|
119
|
+
const sdk = (await provider.forEnvironment(cxapi.EnvironmentUtils.make(cxapi.UNKNOWN_ACCOUNT, cxapi.UNKNOWN_REGION), aws_auth_1.Mode.ForReading)).sdk;
|
|
120
120
|
expect(sdkConfig(sdk).credentials.accessKeyId).toEqual(uniq('access'));
|
|
121
121
|
expect((await sdk.currentAccount()).accountId).toEqual(uniq('11111'));
|
|
122
122
|
expect(sdk.currentRegion).toEqual('eu-bla-5');
|
|
@@ -155,7 +155,7 @@ describe('with intercepted network calls', () => {
|
|
|
155
155
|
// THEN
|
|
156
156
|
expect(provider.defaultRegion).toEqual('eu-west-1');
|
|
157
157
|
await expect(provider.defaultAccount()).resolves.toEqual({ accountId: uniq('22222'), partition: 'aws' });
|
|
158
|
-
const sdk = await provider.forEnvironment(env(uniq('22222')), aws_auth_1.Mode.ForReading);
|
|
158
|
+
const sdk = (await provider.forEnvironment(env(uniq('22222')), aws_auth_1.Mode.ForReading)).sdk;
|
|
159
159
|
expect(sdkConfig(sdk).credentials.accessKeyId).toEqual(uniq('fooccess'));
|
|
160
160
|
});
|
|
161
161
|
test('supports profile only in config_file', async () => {
|
|
@@ -171,7 +171,7 @@ describe('with intercepted network calls', () => {
|
|
|
171
171
|
// THEN
|
|
172
172
|
expect(provider.defaultRegion).toEqual('eu-bla-5'); // Fall back to default config
|
|
173
173
|
await expect(provider.defaultAccount()).resolves.toEqual({ accountId: uniq('22222'), partition: 'aws' });
|
|
174
|
-
const sdk = await provider.forEnvironment(env(uniq('22222')), aws_auth_1.Mode.ForReading);
|
|
174
|
+
const sdk = (await provider.forEnvironment(env(uniq('22222')), aws_auth_1.Mode.ForReading)).sdk;
|
|
175
175
|
expect(sdkConfig(sdk).credentials.accessKeyId).toEqual(uniq('fooccess'));
|
|
176
176
|
});
|
|
177
177
|
test('can assume-role configured in config', async () => {
|
|
@@ -194,7 +194,7 @@ describe('with intercepted network calls', () => {
|
|
|
194
194
|
});
|
|
195
195
|
const provider = await providerFromProfile('assumable');
|
|
196
196
|
// WHEN
|
|
197
|
-
const sdk = await provider.forEnvironment(env(uniq('66666')), aws_auth_1.Mode.ForReading);
|
|
197
|
+
const sdk = (await provider.forEnvironment(env(uniq('66666')), aws_auth_1.Mode.ForReading)).sdk;
|
|
198
198
|
// THEN
|
|
199
199
|
expect((await sdk.currentAccount()).accountId).toEqual(uniq('66666'));
|
|
200
200
|
});
|
|
@@ -237,7 +237,7 @@ describe('with intercepted network calls', () => {
|
|
|
237
237
|
const provider = await providerFromProfile('mfa-role');
|
|
238
238
|
const promptlyMockCalls = promptly.prompt.mock.calls.length;
|
|
239
239
|
// THEN
|
|
240
|
-
const sdk = await provider.forEnvironment(env(uniq('66666')), aws_auth_1.Mode.ForReading);
|
|
240
|
+
const sdk = (await provider.forEnvironment(env(uniq('66666')), aws_auth_1.Mode.ForReading)).sdk;
|
|
241
241
|
expect((await sdk.currentAccount()).accountId).toEqual(uniq('66666'));
|
|
242
242
|
expect(fakeSts.assumedRoles[0]).toEqual(expect.objectContaining({
|
|
243
243
|
roleArn: 'arn:aws:iam::66666:role/Assumable',
|
|
@@ -284,7 +284,7 @@ describe('with intercepted network calls', () => {
|
|
|
284
284
|
userInfo.mockReturnValue({ username: 'skål', uid: 1, gid: 1, homedir: '/here', shell: '/bin/sh' });
|
|
285
285
|
// WHEN
|
|
286
286
|
const provider = await providerFromProfile(undefined);
|
|
287
|
-
const sdk = await provider.forEnvironment(env(uniq('88888')), aws_auth_1.Mode.ForReading, { assumeRoleArn: 'arn:aws:role' });
|
|
287
|
+
const sdk = (await provider.forEnvironment(env(uniq('88888')), aws_auth_1.Mode.ForReading, { assumeRoleArn: 'arn:aws:role' })).sdk;
|
|
288
288
|
await sdk.currentAccount();
|
|
289
289
|
// THEN
|
|
290
290
|
expect(fakeSts.assumedRoles[0]).toEqual(expect.objectContaining({
|
|
@@ -302,7 +302,7 @@ describe('with intercepted network calls', () => {
|
|
|
302
302
|
});
|
|
303
303
|
const provider = await providerFromProfile(undefined);
|
|
304
304
|
// WHEN
|
|
305
|
-
const sdk = await provider.forEnvironment(env(uniq('88888')), aws_auth_1.Mode.ForReading, { assumeRoleArn: 'arn:aws:role' });
|
|
305
|
+
const sdk = (await provider.forEnvironment(env(uniq('88888')), aws_auth_1.Mode.ForReading, { assumeRoleArn: 'arn:aws:role' })).sdk;
|
|
306
306
|
// THEN
|
|
307
307
|
expect((await sdk.currentAccount()).accountId).toEqual(uniq('88888'));
|
|
308
308
|
});
|
|
@@ -316,7 +316,7 @@ describe('with intercepted network calls', () => {
|
|
|
316
316
|
});
|
|
317
317
|
const provider = await providerFromProfile(undefined);
|
|
318
318
|
// WHEN - assumeRole fails because the role can only be assumed from account 11111
|
|
319
|
-
const sdk = await provider.forEnvironment(env(uniq('88888')), aws_auth_1.Mode.ForReading, { assumeRoleArn: 'arn:aws:role' });
|
|
319
|
+
const sdk = (await provider.forEnvironment(env(uniq('88888')), aws_auth_1.Mode.ForReading, { assumeRoleArn: 'arn:aws:role' })).sdk;
|
|
320
320
|
// THEN
|
|
321
321
|
expect((await sdk.currentAccount()).accountId).toEqual(uniq('88888'));
|
|
322
322
|
});
|
|
@@ -351,7 +351,7 @@ describe('with intercepted network calls', () => {
|
|
|
351
351
|
});
|
|
352
352
|
test('even if AssumeRole fails but current credentials are from a plugin, we will still use them', async () => {
|
|
353
353
|
const provider = await providerFromProfile(undefined);
|
|
354
|
-
const sdk = await provider.forEnvironment(env(uniq('99999')), aws_auth_1.Mode.ForReading, { assumeRoleArn: 'does:not:exist' });
|
|
354
|
+
const sdk = (await provider.forEnvironment(env(uniq('99999')), aws_auth_1.Mode.ForReading, { assumeRoleArn: 'does:not:exist' })).sdk;
|
|
355
355
|
// THEN
|
|
356
356
|
expect((await sdk.currentAccount()).accountId).toEqual(uniq('99999'));
|
|
357
357
|
});
|
|
@@ -540,4 +540,4 @@ function isProfileRole(x) {
|
|
|
540
540
|
function providerFromProfile(profile) {
|
|
541
541
|
return aws_auth_1.SdkProvider.withAwsCliCompatibleDefaults({ ...defaultCredOptions, profile });
|
|
542
542
|
}
|
|
543
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sdk-provider.test.js","sourceRoot":"","sources":["sdk-provider.test.ts"],"names":[],"mappings":"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","sourcesContent":["import * as os from 'os';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as AWS from 'aws-sdk';\nimport type { ConfigurationOptions } from 'aws-sdk/lib/config-base';\nimport * as promptly from 'promptly';\nimport * as uuid from 'uuid';\nimport { ISDK, Mode, SDK, SdkProvider } from '../../lib/api/aws-auth';\nimport * as logging from '../../lib/logging';\nimport { PluginHost } from '../../lib/plugin';\nimport * as bockfs from '../bockfs';\nimport { withMocked } from '../util';\nimport { FakeSts, RegisterRoleOptions, RegisterUserOptions } from './fake-sts';\n\njest.mock('promptly', () => ({\n  prompt: jest.fn().mockResolvedValue('1234'),\n}));\n\nconst defaultCredOptions = {\n  ec2creds: false,\n  containerCreds: false,\n};\n\nlet uid: string;\nlet pluginQueried = false;\n\nbeforeEach(() => {\n  // Cache busters!\n  // We prefix everything with UUIDs because:\n  //\n  // - We have a cache from account# -> credentials\n  // - We have a cache from access key -> account\n  uid = `(${uuid.v4()})`;\n\n  logging.setLogLevel(logging.LogLevel.TRACE);\n\n  PluginHost.instance.credentialProviderSources.splice(0);\n  PluginHost.instance.credentialProviderSources.push({\n    isAvailable() { return Promise.resolve(true); },\n    canProvideCredentials(account) { return Promise.resolve(account === uniq('99999')); },\n    getProvider() {\n      pluginQueried = true;\n      return Promise.resolve(new AWS.Credentials({\n        accessKeyId: `${uid}plugin_key`,\n        secretAccessKey: 'plugin_secret',\n        sessionToken: 'plugin_token',\n      }));\n    },\n    name: 'test plugin',\n  });\n\n  // Make sure these point to nonexistant files to start, if we don't call\n  // prepare() then we don't accidentally want to fall back to system config.\n  process.env.AWS_CONFIG_FILE = '/dev/null';\n  process.env.AWS_SHARED_CREDENTIALS_FILE = '/dev/null';\n});\n\nafterEach(() => {\n  bockfs.restore();\n});\n\nfunction uniq(account: string) {\n  return `${uid}${account}`;\n}\n\nfunction env(account: string) {\n  return cxapi.EnvironmentUtils.make(account, 'def');\n}\n\ndescribe('with intercepted network calls', () => {\n  // Most tests will use intercepted network calls, except one test that tests\n  // that the right HTTP `Agent` is used.\n\n  let fakeSts: FakeSts;\n  beforeEach(() => {\n    fakeSts = new FakeSts();\n    fakeSts.begin();\n\n    // Make sure the KeyID returned by the plugin is recognized\n    fakeSts.registerUser(uniq('99999'), uniq('plugin_key'));\n  });\n\n  afterEach(() => {\n    fakeSts.restore();\n  });\n\n  // Set of tests where the CDK will not trigger assume-role\n  // (the INI file might still do assume-role)\n  describe('when CDK does not AssumeRole', () => {\n    test('uses default credentials by default', async () => {\n      // WHEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          default: { aws_access_key_id: 'access', $account: '11111', $fakeStsOptions: { partition: 'aws-here' } },\n        },\n        config: {\n          default: { region: 'eu-bla-5' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n\n      // THEN\n      expect(provider.defaultRegion).toEqual('eu-bla-5');\n      await expect(provider.defaultAccount()).resolves.toEqual({ accountId: uniq('11111'), partition: 'aws-here' });\n\n      // Ask for a different region\n      const sdk = await provider.forEnvironment({ ...env(uniq('11111')), region: 'rgn' }, Mode.ForReading);\n      expect(sdkConfig(sdk).credentials!.accessKeyId).toEqual(uniq('access'));\n      expect(sdk.currentRegion).toEqual('rgn');\n    });\n\n    test('throws if profile credentials are not for the right account', async () => {\n      // WHEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          'profile boo': { aws_access_key_id: 'access', $account: '11111' },\n        },\n      });\n      const provider = await providerFromProfile('boo');\n\n      await expect(provider.forEnvironment(env(uniq('some_account_#')), Mode.ForReading)).rejects.toThrow('Need to perform AWS calls');\n    });\n\n    test('use profile acct/region if agnostic env requested', async () => {\n      // WHEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          default: { aws_access_key_id: 'access', $account: '11111' },\n        },\n        config: {\n          default: { region: 'eu-bla-5' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n\n      // THEN\n      const sdk = await provider.forEnvironment(cxapi.EnvironmentUtils.make(cxapi.UNKNOWN_ACCOUNT, cxapi.UNKNOWN_REGION), Mode.ForReading);\n      expect(sdkConfig(sdk).credentials!.accessKeyId).toEqual(uniq('access'));\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('11111'));\n      expect(sdk.currentRegion).toEqual('eu-bla-5');\n    });\n\n    test('passing profile skips EnvironmentCredentials', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          foo: { aws_access_key_id: 'access', $account: '11111' },\n        },\n      });\n      const provider = await providerFromProfile('foo');\n\n      const environmentCredentialsPrototype = (new AWS.EnvironmentCredentials('AWS')).constructor.prototype;\n\n      await withMocked(environmentCredentialsPrototype, 'refresh', async (refresh) => {\n        refresh.mockImplementation((callback: (err?: Error) => void) => callback(new Error('This function should not have been called')));\n\n        // WHEN\n        expect((await provider.defaultAccount())?.accountId).toEqual(uniq('11111'));\n\n        expect(refresh).not.toHaveBeenCalled();\n      });\n    });\n\n    test('supports profile spread over config_file and credentials_file', async () => {\n      // WHEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          foo: { aws_access_key_id: 'fooccess', $account: '22222' },\n        },\n        config: {\n          'default': { region: 'eu-bla-5' },\n          'profile foo': { region: 'eu-west-1' },\n        },\n      });\n      const provider = await SdkProvider.withAwsCliCompatibleDefaults({ ...defaultCredOptions, profile: 'foo' });\n\n      // THEN\n      expect(provider.defaultRegion).toEqual('eu-west-1');\n      await expect(provider.defaultAccount()).resolves.toEqual({ accountId: uniq('22222'), partition: 'aws' });\n\n      const sdk = await provider.forEnvironment(env(uniq('22222')), Mode.ForReading);\n      expect(sdkConfig(sdk).credentials!.accessKeyId).toEqual(uniq('fooccess'));\n    });\n\n    test('supports profile only in config_file', async () => {\n      // WHEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          'default': { region: 'eu-bla-5' },\n          'profile foo': { aws_access_key_id: 'fooccess', $account: '22222' },\n        },\n      });\n      const provider = await providerFromProfile('foo');\n\n      // THEN\n      expect(provider.defaultRegion).toEqual('eu-bla-5'); // Fall back to default config\n      await expect(provider.defaultAccount()).resolves.toEqual({ accountId: uniq('22222'), partition: 'aws' });\n\n      const sdk = await provider.forEnvironment(env(uniq('22222')), Mode.ForReading);\n      expect(sdkConfig(sdk).credentials!.accessKeyId).toEqual(uniq('fooccess'));\n    });\n\n    test('can assume-role configured in config', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          assumer: { aws_access_key_id: 'assumer', $account: '11111' },\n        },\n        config: {\n          'default': { region: 'eu-bla-5' },\n          'profile assumer': { region: 'us-east-2' },\n          'profile assumable': {\n            role_arn: 'arn:aws:iam::66666:role/Assumable',\n            source_profile: 'assumer',\n            $account: '66666',\n            $fakeStsOptions: { allowedAccounts: ['11111'] },\n          },\n        },\n      });\n      const provider = await providerFromProfile('assumable');\n\n      // WHEN\n      const sdk = await provider.forEnvironment(env(uniq('66666')), Mode.ForReading);\n\n      // THEN\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('66666'));\n    });\n\n    test('can assume role even if [default] profile is missing', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          assumer: { aws_access_key_id: 'assumer', $account: '22222' },\n          assumable: { role_arn: 'arn:aws:iam::12356789012:role/Assumable', source_profile: 'assumer', $account: '22222' },\n        },\n        config: {\n          'profile assumable': { region: 'eu-bla-5' },\n        },\n      });\n\n      // WHEN\n      const provider = await providerFromProfile('assumable');\n\n      // THEN\n      expect((await provider.defaultAccount())?.accountId).toEqual(uniq('22222'));\n    });\n\n    test('mfa_serial in profile will ask user for token', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          assumer: { aws_access_key_id: 'assumer', $account: '66666' },\n        },\n        config: {\n          'default': { region: 'eu-bla-5' },\n          'profile assumer': { region: 'us-east-2' },\n          'profile mfa-role': {\n            role_arn: 'arn:aws:iam::66666:role/Assumable',\n            source_profile: 'assumer',\n            mfa_serial: 'arn:aws:iam::account:mfa/user',\n            $account: '66666',\n          },\n        },\n      });\n      const provider = await providerFromProfile('mfa-role');\n\n      const promptlyMockCalls = (promptly.prompt as jest.Mock).mock.calls.length;\n\n      // THEN\n      const sdk = await provider.forEnvironment(env(uniq('66666')), Mode.ForReading);\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('66666'));\n      expect(fakeSts.assumedRoles[0]).toEqual(expect.objectContaining({\n        roleArn: 'arn:aws:iam::66666:role/Assumable',\n        serialNumber: 'arn:aws:iam::account:mfa/user',\n        tokenCode: '1234',\n      }));\n\n      // Mock response was set to fail to make sure we don't call STS\n      // Make sure the MFA mock was called during this test\n      expect((promptly.prompt as jest.Mock).mock.calls.length).toBe(promptlyMockCalls + 1);\n    });\n  });\n\n  // For DefaultSynthesis we will do an assume-role after having gotten base credentials\n  describe('when CDK AssumeRoles', () => {\n    beforeEach(() => {\n      // All these tests share that 'arn:aws:role' is a role into account 88888 which can be assumed from 11111\n      fakeSts.registerRole(uniq('88888'), 'arn:aws:role', { allowedAccounts: [uniq('11111')] });\n    });\n\n    test('error we get from assuming a role is useful', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          default: { aws_access_key_id: 'foo' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n\n      // WHEN\n      const promise = provider.forEnvironment(env(uniq('88888')), Mode.ForReading, {\n        assumeRoleArn: 'doesnotexist.role.arn',\n      });\n\n      // THEN - error message contains both a helpful hint and the underlying AssumeRole message\n      await expect(promise).rejects.toThrow('(re)-bootstrap the environment');\n      await expect(promise).rejects.toThrow('doesnotexist.role.arn');\n    });\n\n    test('assuming a role sanitizes the username into the session name', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          default: { aws_access_key_id: 'foo', $account: '11111' },\n        },\n      });\n\n      await withMocked(os, 'userInfo', async (userInfo) => {\n        userInfo.mockReturnValue({ username: 'skål', uid: 1, gid: 1, homedir: '/here', shell: '/bin/sh' });\n\n        // WHEN\n        const provider = await providerFromProfile(undefined);\n\n        const sdk = await provider.forEnvironment(env(uniq('88888')), Mode.ForReading, { assumeRoleArn: 'arn:aws:role' }) as SDK;\n        await sdk.currentAccount();\n\n        // THEN\n        expect(fakeSts.assumedRoles[0]).toEqual(expect.objectContaining({\n          roleSessionName: 'aws-cdk-sk@l',\n        }));\n      });\n    });\n\n    test('even if current credentials are for the wrong account, we will still use them to AssumeRole', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          default: { aws_access_key_id: 'foo', $account: '11111' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n\n      // WHEN\n      const sdk = await provider.forEnvironment(env(uniq('88888')), Mode.ForReading, { assumeRoleArn: 'arn:aws:role' }) as SDK;\n\n      // THEN\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('88888'));\n    });\n\n    test('if AssumeRole fails but current credentials are for the right account, we will still use them', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          default: { aws_access_key_id: 'foo', $account: '88888' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n\n      // WHEN - assumeRole fails because the role can only be assumed from account 11111\n      const sdk = await provider.forEnvironment(env(uniq('88888')), Mode.ForReading, { assumeRoleArn: 'arn:aws:role' }) as SDK;\n\n      // THEN\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('88888'));\n    });\n  });\n\n  describe('Plugins', () => {\n    test('does not use plugins if current credentials are for expected account', async () => {\n      prepareCreds({\n        fakeSts,\n        config: {\n          default: { aws_access_key_id: 'foo', $account: '11111' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n      await provider.forEnvironment(env(uniq('11111')), Mode.ForReading);\n      expect(pluginQueried).toEqual(false);\n    });\n\n    test('uses plugin for account 99999', async () => {\n      const provider = await providerFromProfile(undefined);\n      await provider.forEnvironment(env(uniq('99999')), Mode.ForReading);\n      expect(pluginQueried).toEqual(true);\n    });\n\n    test('can assume role with credentials from plugin', async () => {\n      fakeSts.registerRole(uniq('99999'), 'arn:aws:iam::99999:role/Assumable');\n\n      const provider = await providerFromProfile(undefined);\n      await provider.forEnvironment(env(uniq('99999')), Mode.ForReading, {\n        assumeRoleArn: 'arn:aws:iam::99999:role/Assumable',\n      });\n\n      expect(fakeSts.assumedRoles[0]).toEqual(expect.objectContaining({\n        roleArn: 'arn:aws:iam::99999:role/Assumable',\n      }));\n      expect(pluginQueried).toEqual(true);\n    });\n\n    test('even if AssumeRole fails but current credentials are from a plugin, we will still use them', async () => {\n      const provider = await providerFromProfile(undefined);\n      const sdk = await provider.forEnvironment(env(uniq('99999')), Mode.ForReading, { assumeRoleArn: 'does:not:exist' });\n\n      // THEN\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('99999'));\n    });\n\n    test('plugins are still queried even if current credentials are expired (or otherwise invalid)', async () => {\n      // GIVEN\n      process.env.AWS_ACCESS_KEY_ID = `${uid}akid`;\n      process.env.AWS_SECRET_ACCESS_KEY = 'sekrit';\n      const provider = await providerFromProfile(undefined);\n\n      // WHEN\n      await provider.forEnvironment(env(uniq('99999')), Mode.ForReading);\n\n      // THEN\n      expect(pluginQueried).toEqual(true);\n    });\n  });\n\n  describe('support for credential_source', () => {\n    test('can assume role with ecs credentials', async () => {\n      return withMocked(AWS.ECSCredentials.prototype, 'needsRefresh', async (needsRefresh) => {\n        // GIVEN\n        prepareCreds({\n          config: {\n            'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'EcsContainer', $account: '22222' },\n          },\n        });\n        const provider = await providerFromProfile('ecs');\n\n        // WHEN\n        await provider.defaultAccount();\n\n        // THEN\n        expect(needsRefresh).toHaveBeenCalled();\n      });\n\n    });\n\n    test('can assume role with ec2 credentials', async () => {\n      return withMocked(AWS.EC2MetadataCredentials.prototype, 'needsRefresh', async (needsRefresh) => {\n        // GIVEN\n        prepareCreds({\n          config: {\n            'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'Ec2InstanceMetadata', $account: '22222' },\n          },\n        });\n        const provider = await providerFromProfile('ecs');\n\n        // WHEN\n        await provider.defaultAccount();\n\n        // THEN\n        expect(needsRefresh).toHaveBeenCalled();\n\n      });\n\n    });\n\n    test('can assume role with env credentials', async () => {\n      return withMocked(AWS.EnvironmentCredentials.prototype, 'needsRefresh', async (needsRefresh) => {\n        // GIVEN\n        prepareCreds({\n          config: {\n            'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'Environment', $account: '22222' },\n          },\n        });\n        const provider = await providerFromProfile('ecs');\n\n        // WHEN\n        await provider.defaultAccount();\n\n        // THEN\n        expect(needsRefresh).toHaveBeenCalled();\n      });\n    });\n\n    test('assume fails with unsupported credential_source', async () => {\n      // GIVEN\n      prepareCreds({\n        config: {\n          'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'unsupported', $account: '22222' },\n        },\n      });\n      const provider = await providerFromProfile('ecs');\n\n      // WHEN\n      const account = await provider.defaultAccount();\n\n      // THEN\n      expect(account?.accountId).toEqual(undefined);\n    });\n  });\n\n  test('defaultAccount returns undefined if STS call fails', async () => {\n    // GIVEN\n    process.env.AWS_ACCESS_KEY_ID = `${uid}akid`;\n    process.env.AWS_SECRET_ACCESS_KEY = 'sekrit';\n\n    // WHEN\n    const provider = await providerFromProfile(undefined);\n\n    // THEN\n    await expect(provider.defaultAccount()).resolves.toBe(undefined);\n  });\n});\n\ntest('even when using a profile to assume another profile, STS calls goes through the proxy', async () => {\n  prepareCreds({\n    credentials: {\n      assumer: { aws_access_key_id: 'assumer' },\n    },\n    config: {\n      'default': { region: 'eu-bla-5' },\n      'profile assumable': { role_arn: 'arn:aws:iam::66666:role/Assumable', source_profile: 'assumer', $account: '66666' },\n      'profile assumer': { region: 'us-east-2' },\n    },\n  });\n\n  // Messy mocking\n  let called = false;\n  jest.mock('proxy-agent', () => {\n    // eslint-disable-next-line @typescript-eslint/no-require-imports\n    class FakeAgent extends require('https').Agent {\n      public addRequest(_: any, __: any) {\n        // FIXME: this error takes 6 seconds to be completely handled. It\n        // might be retries in the SDK somewhere, or something about the Node\n        // event loop. I've spent an hour trying to figure it out and I can't,\n        // and I gave up. We'll just have to live with this until someone gets\n        // inspired.\n        const error = new Error('ABORTED BY TEST');\n        (error as any).code = 'RequestAbortedError';\n        (error as any).retryable = false;\n        called = true;\n        throw error;\n      }\n    }\n    return FakeAgent;\n  });\n\n  // WHEN\n  const provider = await SdkProvider.withAwsCliCompatibleDefaults({\n    ...defaultCredOptions,\n    profile: 'assumable',\n    httpOptions: {\n      proxyAddress: 'http://DOESNTMATTER/',\n    },\n  });\n\n  await provider.defaultAccount();\n\n  // THEN -- the fake proxy agent got called, we don't care about the result\n  expect(called).toEqual(true);\n});\n\n/**\n * Use object hackery to get the credentials out of the SDK object\n */\nfunction sdkConfig(sdk: ISDK): ConfigurationOptions {\n  return (sdk as any).config;\n}\n\n/**\n * Fixture for SDK auth for this test suite\n *\n * Has knowledge of the cache buster, will write proper fake config files and\n * register users and roles in FakeSts at the same time.\n */\nfunction prepareCreds(options: PrepareCredsOptions) {\n  function convertSections(sections?: Record<string, ProfileUser | ProfileRole>) {\n    const ret = [];\n    for (const [profile, user] of Object.entries(sections ?? {})) {\n      ret.push(`[${profile}]`);\n\n      if (isProfileRole(user)) {\n        ret.push(`role_arn=${user.role_arn}`);\n        if ('source_profile' in user) {\n          ret.push(`source_profile=${user.source_profile}`);\n        }\n        if ('credential_source' in user) {\n          ret.push(`credential_source=${user.credential_source}`);\n        }\n        if (user.mfa_serial) {\n          ret.push(`mfa_serial=${user.mfa_serial}`);\n        }\n        options.fakeSts?.registerRole(uniq(user.$account ?? '00000'), user.role_arn, {\n          ...user.$fakeStsOptions,\n          allowedAccounts: user.$fakeStsOptions?.allowedAccounts?.map(uniq),\n        });\n      } else {\n        if (user.aws_access_key_id) {\n          ret.push(`aws_access_key_id=${uniq(user.aws_access_key_id)}`);\n          ret.push('aws_secret_access_key=secret');\n          options.fakeSts?.registerUser(uniq(user.$account ?? '00000'), uniq(user.aws_access_key_id), user.$fakeStsOptions);\n        }\n      }\n\n      if (user.region) {\n        ret.push(`region=${user.region}`);\n      }\n    }\n    return ret.join('\\n');\n  }\n\n  bockfs({\n    '/home/me/.bxt/credentials': convertSections(options.credentials),\n    '/home/me/.bxt/config': convertSections(options.config),\n  });\n\n  // Set environment variables that we want\n  process.env.AWS_CONFIG_FILE = bockfs.path('/home/me/.bxt/config');\n  process.env.AWS_SHARED_CREDENTIALS_FILE = bockfs.path('/home/me/.bxt/credentials');\n}\n\ninterface PrepareCredsOptions {\n  /**\n   * Write the aws/credentials file\n   */\n  readonly credentials?: Record<string, ProfileUser | ProfileRole>;\n\n  /**\n   * Write the aws/config file\n   */\n  readonly config?: Record<string, ProfileUser | ProfileRole>;\n\n  /**\n   * If given, add users to FakeSTS\n   */\n  readonly fakeSts?: FakeSts;\n}\n\ninterface ProfileUser {\n  readonly aws_access_key_id?: string;\n  readonly $account?: string;\n  readonly region?: string;\n  readonly $fakeStsOptions?: RegisterUserOptions;\n}\n\ntype ProfileRole = {\n  readonly role_arn: string;\n  readonly mfa_serial?: string;\n  readonly $account: string;\n  readonly region?: string;\n  readonly $fakeStsOptions?: RegisterRoleOptions;\n} & ({ readonly source_profile: string } | { readonly credential_source: string });\n\nfunction isProfileRole(x: ProfileUser | ProfileRole): x is ProfileRole {\n  return 'role_arn' in x;\n}\n\nfunction providerFromProfile(profile: string | undefined) {\n  return SdkProvider.withAwsCliCompatibleDefaults({ ...defaultCredOptions, profile });\n}"]}
|
|
543
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sdk-provider.test.js","sourceRoot":"","sources":["sdk-provider.test.ts"],"names":[],"mappings":";;AAAA,yBAAyB;AACzB,yCAAyC;AACzC,+BAA+B;AAE/B,qCAAqC;AACrC,6BAA6B;AAC7B,qDAAsE;AACtE,6CAA6C;AAC7C,6CAA8C;AAC9C,oCAAoC;AACpC,kCAAqC;AACrC,yCAA+E;AAE/E,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,GAAG,EAAE,CAAC,CAAC;IAC3B,MAAM,EAAE,IAAI,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,MAAM,CAAC;CAC5C,CAAC,CAAC,CAAC;AAEJ,MAAM,kBAAkB,GAAG;IACzB,QAAQ,EAAE,KAAK;IACf,cAAc,EAAE,KAAK;CACtB,CAAC;AAEF,IAAI,GAAW,CAAC;AAChB,IAAI,aAAa,GAAG,KAAK,CAAC;AAE1B,UAAU,CAAC,GAAG,EAAE;IACd,iBAAiB;IACjB,2CAA2C;IAC3C,EAAE;IACF,iDAAiD;IACjD,+CAA+C;IAC/C,GAAG,GAAG,IAAI,IAAI,CAAC,EAAE,EAAE,GAAG,CAAC;IAEvB,OAAO,CAAC,WAAW,eAAwB,CAAC;IAE5C,mBAAU,CAAC,QAAQ,CAAC,yBAAyB,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;IACxD,mBAAU,CAAC,QAAQ,CAAC,yBAAyB,CAAC,IAAI,CAAC;QACjD,WAAW,KAAK,OAAO,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAC/C,qBAAqB,CAAC,OAAO,IAAI,OAAO,OAAO,CAAC,OAAO,CAAC,OAAO,KAAK,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;QACrF,WAAW;YACT,aAAa,GAAG,IAAI,CAAC;YACrB,OAAO,OAAO,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC,WAAW,CAAC;gBACzC,WAAW,EAAE,GAAG,GAAG,YAAY;gBAC/B,eAAe,EAAE,eAAe;gBAChC,YAAY,EAAE,cAAc;aAC7B,CAAC,CAAC,CAAC;QACN,CAAC;QACD,IAAI,EAAE,aAAa;KACpB,CAAC,CAAC;IAEH,wEAAwE;IACxE,2EAA2E;IAC3E,OAAO,CAAC,GAAG,CAAC,eAAe,GAAG,WAAW,CAAC;IAC1C,OAAO,CAAC,GAAG,CAAC,2BAA2B,GAAG,WAAW,CAAC;AACxD,CAAC,CAAC,CAAC;AAEH,SAAS,CAAC,GAAG,EAAE;IACb,MAAM,CAAC,OAAO,EAAE,CAAC;AACnB,CAAC,CAAC,CAAC;AAEH,SAAS,IAAI,CAAC,OAAe;IAC3B,OAAO,GAAG,GAAG,GAAG,OAAO,EAAE,CAAC;AAC5B,CAAC;AAED,SAAS,GAAG,CAAC,OAAe;IAC1B,OAAO,KAAK,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;AACrD,CAAC;AAED,QAAQ,CAAC,gCAAgC,EAAE,GAAG,EAAE;IAC9C,4EAA4E;IAC5E,uCAAuC;IAEvC,IAAI,OAAgB,CAAC;IACrB,UAAU,CAAC,GAAG,EAAE;QACd,OAAO,GAAG,IAAI,kBAAO,EAAE,CAAC;QACxB,OAAO,CAAC,KAAK,EAAE,CAAC;QAEhB,2DAA2D;QAC3D,OAAO,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;IAC1D,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE;QACb,OAAO,CAAC,OAAO,EAAE,CAAC;IACpB,CAAC,CAAC,CAAC;IAEH,0DAA0D;IAC1D,4CAA4C;IAC5C,QAAQ,CAAC,8BAA8B,EAAE,GAAG,EAAE;QAC5C,IAAI,CAAC,qCAAqC,EAAE,KAAK,IAAI,EAAE;YACrD,OAAO;YACP,YAAY,CAAC;gBACX,OAAO;gBACP,WAAW,EAAE;oBACX,OAAO,EAAE,EAAE,iBAAiB,EAAE,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,eAAe,EAAE,EAAE,SAAS,EAAE,UAAU,EAAE,EAAE;iBACxG;gBACD,MAAM,EAAE;oBACN,OAAO,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE;iBAChC;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;YAEtD,OAAO;YACP,MAAM,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;YACnD,MAAM,MAAM,CAAC,QAAQ,CAAC,cAAc,EAAE,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,UAAU,EAAE,CAAC,CAAC;YAE9G,6BAA6B;YAC7B,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,cAAc,CAAC,EAAE,GAAG,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,MAAM,EAAE,KAAK,EAAE,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC;YAC3G,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,WAAY,CAAC,WAAW,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;YACxE,MAAM,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QAC3C,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,6DAA6D,EAAE,KAAK,IAAI,EAAE;YAC7E,OAAO;YACP,YAAY,CAAC;gBACX,OAAO;gBACP,MAAM,EAAE;oBACN,aAAa,EAAE,EAAE,iBAAiB,EAAE,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE;iBAClE;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,KAAK,CAAC,CAAC;YAElD,MAAM,MAAM,CAAC,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,2BAA2B,CAAC,CAAC;QACnI,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,mDAAmD,EAAE,KAAK,IAAI,EAAE;YACnE,OAAO;YACP,YAAY,CAAC;gBACX,OAAO;gBACP,WAAW,EAAE;oBACX,OAAO,EAAE,EAAE,iBAAiB,EAAE,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE;iBAC5D;gBACD,MAAM,EAAE;oBACN,OAAO,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE;iBAChC;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;YAEtD,OAAO;YACP,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,cAAc,CAAC,KAAK,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,eAAe,EAAE,KAAK,CAAC,cAAc,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC;YAC3I,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,WAAY,CAAC,WAAW,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;YACxE,MAAM,CAAC,CAAC,MAAM,GAAG,CAAC,cAAc,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;YACtE,MAAM,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,8CAA8C,EAAE,KAAK,IAAI,EAAE;YAC9D,QAAQ;YACR,YAAY,CAAC;gBACX,OAAO;gBACP,WAAW,EAAE;oBACX,GAAG,EAAE,EAAE,iBAAiB,EAAE,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE;iBACxD;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,KAAK,CAAC,CAAC;YAElD,MAAM,+BAA+B,GAAG,CAAC,IAAI,GAAG,CAAC,sBAAsB,CAAC,KAAK,CAAC,CAAC,CAAC,WAAW,CAAC,SAAS,CAAC;YAEtG,MAAM,iBAAU,CAAC,+BAA+B,EAAE,SAAS,EAAE,KAAK,EAAE,OAAO,EAAE,EAAE;;gBAC7E,OAAO,CAAC,kBAAkB,CAAC,CAAC,QAA+B,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI,KAAK,CAAC,2CAA2C,CAAC,CAAC,CAAC,CAAC;gBAElI,OAAO;gBACP,MAAM,OAAC,CAAC,MAAM,QAAQ,CAAC,cAAc,EAAE,CAAC,0CAAE,SAAS,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;gBAE5E,MAAM,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;YACzC,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,+DAA+D,EAAE,KAAK,IAAI,EAAE;YAC/E,OAAO;YACP,YAAY,CAAC;gBACX,OAAO;gBACP,WAAW,EAAE;oBACX,GAAG,EAAE,EAAE,iBAAiB,EAAE,UAAU,EAAE,QAAQ,EAAE,OAAO,EAAE;iBAC1D;gBACD,MAAM,EAAE;oBACN,SAAS,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE;oBACjC,aAAa,EAAE,EAAE,MAAM,EAAE,WAAW,EAAE;iBACvC;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,sBAAW,CAAC,4BAA4B,CAAC,EAAE,GAAG,kBAAkB,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;YAE3G,OAAO;YACP,MAAM,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;YACpD,MAAM,MAAM,CAAC,QAAQ,CAAC,cAAc,EAAE,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;YAEzG,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC;YACrF,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,WAAY,CAAC,WAAW,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC;QAC5E,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,sCAAsC,EAAE,KAAK,IAAI,EAAE;YACtD,OAAO;YACP,YAAY,CAAC;gBACX,OAAO;gBACP,MAAM,EAAE;oBACN,SAAS,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE;oBACjC,aAAa,EAAE,EAAE,iBAAiB,EAAE,UAAU,EAAE,QAAQ,EAAE,OAAO,EAAE;iBACpE;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,KAAK,CAAC,CAAC;YAElD,OAAO;YACP,MAAM,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC,8BAA8B;YAClF,MAAM,MAAM,CAAC,QAAQ,CAAC,cAAc,EAAE,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;YAEzG,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC;YACrF,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,WAAY,CAAC,WAAW,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC;QAC5E,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,sCAAsC,EAAE,KAAK,IAAI,EAAE;YACtD,QAAQ;YACR,YAAY,CAAC;gBACX,OAAO;gBACP,WAAW,EAAE;oBACX,OAAO,EAAE,EAAE,iBAAiB,EAAE,SAAS,EAAE,QAAQ,EAAE,OAAO,EAAE;iBAC7D;gBACD,MAAM,EAAE;oBACN,SAAS,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE;oBACjC,iBAAiB,EAAE,EAAE,MAAM,EAAE,WAAW,EAAE;oBAC1C,mBAAmB,EAAE;wBACnB,QAAQ,EAAE,mCAAmC;wBAC7C,cAAc,EAAE,SAAS;wBACzB,QAAQ,EAAE,OAAO;wBACjB,eAAe,EAAE,EAAE,eAAe,EAAE,CAAC,OAAO,CAAC,EAAE;qBAChD;iBACF;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,WAAW,CAAC,CAAC;YAExD,OAAO;YACP,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC;YAErF,OAAO;YACP,MAAM,CAAC,CAAC,MAAM,GAAG,CAAC,cAAc,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;QACxE,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,sDAAsD,EAAE,KAAK,IAAI,EAAE;;YACtE,QAAQ;YACR,YAAY,CAAC;gBACX,OAAO;gBACP,WAAW,EAAE;oBACX,OAAO,EAAE,EAAE,iBAAiB,EAAE,SAAS,EAAE,QAAQ,EAAE,OAAO,EAAE;oBAC5D,SAAS,EAAE,EAAE,QAAQ,EAAE,yCAAyC,EAAE,cAAc,EAAE,SAAS,EAAE,QAAQ,EAAE,OAAO,EAAE;iBACjH;gBACD,MAAM,EAAE;oBACN,mBAAmB,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE;iBAC5C;aACF,CAAC,CAAC;YAEH,OAAO;YACP,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,WAAW,CAAC,CAAC;YAExD,OAAO;YACP,MAAM,OAAC,CAAC,MAAM,QAAQ,CAAC,cAAc,EAAE,CAAC,0CAAE,SAAS,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;QAC9E,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,+CAA+C,EAAE,KAAK,IAAI,EAAE;YAC/D,QAAQ;YACR,YAAY,CAAC;gBACX,OAAO;gBACP,WAAW,EAAE;oBACX,OAAO,EAAE,EAAE,iBAAiB,EAAE,SAAS,EAAE,QAAQ,EAAE,OAAO,EAAE;iBAC7D;gBACD,MAAM,EAAE;oBACN,SAAS,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE;oBACjC,iBAAiB,EAAE,EAAE,MAAM,EAAE,WAAW,EAAE;oBAC1C,kBAAkB,EAAE;wBAClB,QAAQ,EAAE,mCAAmC;wBAC7C,cAAc,EAAE,SAAS;wBACzB,UAAU,EAAE,+BAA+B;wBAC3C,QAAQ,EAAE,OAAO;qBAClB;iBACF;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,UAAU,CAAC,CAAC;YAEvD,MAAM,iBAAiB,GAAI,QAAQ,CAAC,MAAoB,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC;YAE3E,OAAO;YACP,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC;YACrF,MAAM,CAAC,CAAC,MAAM,GAAG,CAAC,cAAc,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;YACtE,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,gBAAgB,CAAC;gBAC9D,OAAO,EAAE,mCAAmC;gBAC5C,YAAY,EAAE,+BAA+B;gBAC7C,SAAS,EAAE,MAAM;aAClB,CAAC,CAAC,CAAC;YAEJ,+DAA+D;YAC/D,qDAAqD;YACrD,MAAM,CAAE,QAAQ,CAAC,MAAoB,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,iBAAiB,GAAG,CAAC,CAAC,CAAC;QACvF,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,sFAAsF;IACtF,QAAQ,CAAC,sBAAsB,EAAE,GAAG,EAAE;QACpC,UAAU,CAAC,GAAG,EAAE;YACd,yGAAyG;YACzG,OAAO,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,cAAc,EAAE,EAAE,eAAe,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,CAAC,CAAC;QAC5F,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,6CAA6C,EAAE,KAAK,IAAI,EAAE;YAC7D,QAAQ;YACR,YAAY,CAAC;gBACX,OAAO;gBACP,MAAM,EAAE;oBACN,OAAO,EAAE,EAAE,iBAAiB,EAAE,KAAK,EAAE;iBACtC;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;YAEtD,OAAO;YACP,MAAM,OAAO,GAAG,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,EAAE;gBAC3E,aAAa,EAAE,uBAAuB;aACvC,CAAC,CAAC;YAEH,0FAA0F;YAC1F,MAAM,MAAM,CAAC,OAAO,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,gCAAgC,CAAC,CAAC;YACxE,MAAM,MAAM,CAAC,OAAO,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,uBAAuB,CAAC,CAAC;QACjE,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,8DAA8D,EAAE,KAAK,IAAI,EAAE;YAC9E,QAAQ;YACR,YAAY,CAAC;gBACX,OAAO;gBACP,MAAM,EAAE;oBACN,OAAO,EAAE,EAAE,iBAAiB,EAAE,KAAK,EAAE,QAAQ,EAAE,OAAO,EAAE;iBACzD;aACF,CAAC,CAAC;YAEH,MAAM,iBAAU,CAAC,EAAE,EAAE,UAAU,EAAE,KAAK,EAAE,QAAQ,EAAE,EAAE;gBAClD,QAAQ,CAAC,eAAe,CAAC,EAAE,QAAQ,EAAE,MAAM,EAAE,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAE,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,CAAC,CAAC;gBAEnG,OAAO;gBACP,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;gBAEtD,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,EAAE,EAAE,aAAa,EAAE,cAAc,EAAE,CAAC,CAAC,CAAC,GAAU,CAAC;gBAC/H,MAAM,GAAG,CAAC,cAAc,EAAE,CAAC;gBAE3B,OAAO;gBACP,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,gBAAgB,CAAC;oBAC9D,eAAe,EAAE,cAAc;iBAChC,CAAC,CAAC,CAAC;YACN,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,6FAA6F,EAAE,KAAK,IAAI,EAAE;YAC7G,QAAQ;YACR,YAAY,CAAC;gBACX,OAAO;gBACP,MAAM,EAAE;oBACN,OAAO,EAAE,EAAE,iBAAiB,EAAE,KAAK,EAAE,QAAQ,EAAE,OAAO,EAAE;iBACzD;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;YAEtD,OAAO;YACP,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,EAAE,EAAE,aAAa,EAAE,cAAc,EAAE,CAAC,CAAC,CAAC,GAAU,CAAC;YAE/H,OAAO;YACP,MAAM,CAAC,CAAC,MAAM,GAAG,CAAC,cAAc,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;QACxE,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,+FAA+F,EAAE,KAAK,IAAI,EAAE;YAC/G,QAAQ;YACR,YAAY,CAAC;gBACX,OAAO;gBACP,MAAM,EAAE;oBACN,OAAO,EAAE,EAAE,iBAAiB,EAAE,KAAK,EAAE,QAAQ,EAAE,OAAO,EAAE;iBACzD;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;YAEtD,kFAAkF;YAClF,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,EAAE,EAAE,aAAa,EAAE,cAAc,EAAE,CAAC,CAAC,CAAC,GAAU,CAAC;YAE/H,OAAO;YACP,MAAM,CAAC,CAAC,MAAM,GAAG,CAAC,cAAc,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;QACxE,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,SAAS,EAAE,GAAG,EAAE;QACvB,IAAI,CAAC,sEAAsE,EAAE,KAAK,IAAI,EAAE;YACtF,YAAY,CAAC;gBACX,OAAO;gBACP,MAAM,EAAE;oBACN,OAAO,EAAE,EAAE,iBAAiB,EAAE,KAAK,EAAE,QAAQ,EAAE,OAAO,EAAE;iBACzD;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;YACtD,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC;YACnE,MAAM,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QACvC,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,+BAA+B,EAAE,KAAK,IAAI,EAAE;YAC/C,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;YACtD,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC;YACnE,MAAM,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,8CAA8C,EAAE,KAAK,IAAI,EAAE;YAC9D,OAAO,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,mCAAmC,CAAC,CAAC;YAEzE,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;YACtD,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,EAAE;gBACjE,aAAa,EAAE,mCAAmC;aACnD,CAAC,CAAC;YAEH,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,gBAAgB,CAAC;gBAC9D,OAAO,EAAE,mCAAmC;aAC7C,CAAC,CAAC,CAAC;YACJ,MAAM,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,4FAA4F,EAAE,KAAK,IAAI,EAAE;YAC5G,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;YACtD,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,EAAE,EAAE,aAAa,EAAE,gBAAgB,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC;YAE1H,OAAO;YACP,MAAM,CAAC,CAAC,MAAM,GAAG,CAAC,cAAc,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;QACxE,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,0FAA0F,EAAE,KAAK,IAAI,EAAE;YAC1G,QAAQ;YACR,OAAO,CAAC,GAAG,CAAC,iBAAiB,GAAG,GAAG,GAAG,MAAM,CAAC;YAC7C,OAAO,CAAC,GAAG,CAAC,qBAAqB,GAAG,QAAQ,CAAC;YAC7C,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;YAEtD,OAAO;YACP,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC;YAEnE,OAAO;YACP,MAAM,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,+BAA+B,EAAE,GAAG,EAAE;QAC7C,IAAI,CAAC,sCAAsC,EAAE,KAAK,IAAI,EAAE;YACtD,OAAO,iBAAU,CAAC,GAAG,CAAC,cAAc,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,EAAE,YAAY,EAAE,EAAE;gBACrF,QAAQ;gBACR,YAAY,CAAC;oBACX,MAAM,EAAE;wBACN,aAAa,EAAE,EAAE,QAAQ,EAAE,yCAAyC,EAAE,iBAAiB,EAAE,cAAc,EAAE,QAAQ,EAAE,OAAO,EAAE;qBAC7H;iBACF,CAAC,CAAC;gBACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,KAAK,CAAC,CAAC;gBAElD,OAAO;gBACP,MAAM,QAAQ,CAAC,cAAc,EAAE,CAAC;gBAEhC,OAAO;gBACP,MAAM,CAAC,YAAY,CAAC,CAAC,gBAAgB,EAAE,CAAC;YAC1C,CAAC,CAAC,CAAC;QAEL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,sCAAsC,EAAE,KAAK,IAAI,EAAE;YACtD,OAAO,iBAAU,CAAC,GAAG,CAAC,sBAAsB,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,EAAE,YAAY,EAAE,EAAE;gBAC7F,QAAQ;gBACR,YAAY,CAAC;oBACX,MAAM,EAAE;wBACN,aAAa,EAAE,EAAE,QAAQ,EAAE,yCAAyC,EAAE,iBAAiB,EAAE,qBAAqB,EAAE,QAAQ,EAAE,OAAO,EAAE;qBACpI;iBACF,CAAC,CAAC;gBACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,KAAK,CAAC,CAAC;gBAElD,OAAO;gBACP,MAAM,QAAQ,CAAC,cAAc,EAAE,CAAC;gBAEhC,OAAO;gBACP,MAAM,CAAC,YAAY,CAAC,CAAC,gBAAgB,EAAE,CAAC;YAE1C,CAAC,CAAC,CAAC;QAEL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,sCAAsC,EAAE,KAAK,IAAI,EAAE;YACtD,OAAO,iBAAU,CAAC,GAAG,CAAC,sBAAsB,CAAC,SAAS,EAAE,cAAc,EAAE,KAAK,EAAE,YAAY,EAAE,EAAE;gBAC7F,QAAQ;gBACR,YAAY,CAAC;oBACX,MAAM,EAAE;wBACN,aAAa,EAAE,EAAE,QAAQ,EAAE,yCAAyC,EAAE,iBAAiB,EAAE,aAAa,EAAE,QAAQ,EAAE,OAAO,EAAE;qBAC5H;iBACF,CAAC,CAAC;gBACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,KAAK,CAAC,CAAC;gBAElD,OAAO;gBACP,MAAM,QAAQ,CAAC,cAAc,EAAE,CAAC;gBAEhC,OAAO;gBACP,MAAM,CAAC,YAAY,CAAC,CAAC,gBAAgB,EAAE,CAAC;YAC1C,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,iDAAiD,EAAE,KAAK,IAAI,EAAE;YACjE,QAAQ;YACR,YAAY,CAAC;gBACX,MAAM,EAAE;oBACN,aAAa,EAAE,EAAE,QAAQ,EAAE,yCAAyC,EAAE,iBAAiB,EAAE,aAAa,EAAE,QAAQ,EAAE,OAAO,EAAE;iBAC5H;aACF,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,KAAK,CAAC,CAAC;YAElD,OAAO;YACP,MAAM,OAAO,GAAG,MAAM,QAAQ,CAAC,cAAc,EAAE,CAAC;YAEhD,OAAO;YACP,MAAM,CAAC,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,SAAS,CAAC,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,oDAAoD,EAAE,KAAK,IAAI,EAAE;QACpE,QAAQ;QACR,OAAO,CAAC,GAAG,CAAC,iBAAiB,GAAG,GAAG,GAAG,MAAM,CAAC;QAC7C,OAAO,CAAC,GAAG,CAAC,qBAAqB,GAAG,QAAQ,CAAC;QAE7C,OAAO;QACP,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC,SAAS,CAAC,CAAC;QAEtD,OAAO;QACP,MAAM,MAAM,CAAC,QAAQ,CAAC,cAAc,EAAE,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IACnE,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,uFAAuF,EAAE,KAAK,IAAI,EAAE;IACvG,YAAY,CAAC;QACX,WAAW,EAAE;YACX,OAAO,EAAE,EAAE,iBAAiB,EAAE,SAAS,EAAE;SAC1C;QACD,MAAM,EAAE;YACN,SAAS,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE;YACjC,mBAAmB,EAAE,EAAE,QAAQ,EAAE,mCAAmC,EAAE,cAAc,EAAE,SAAS,EAAE,QAAQ,EAAE,OAAO,EAAE;YACpH,iBAAiB,EAAE,EAAE,MAAM,EAAE,WAAW,EAAE;SAC3C;KACF,CAAC,CAAC;IAEH,gBAAgB;IAChB,IAAI,MAAM,GAAG,KAAK,CAAC;IACnB,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE;QAC5B,iEAAiE;QACjE,MAAM,SAAU,SAAQ,OAAO,CAAC,OAAO,CAAC,CAAC,KAAK;YACrC,UAAU,CAAC,CAAM,EAAE,EAAO;gBAC/B,iEAAiE;gBACjE,qEAAqE;gBACrE,sEAAsE;gBACtE,sEAAsE;gBACtE,YAAY;gBACZ,MAAM,KAAK,GAAG,IAAI,KAAK,CAAC,iBAAiB,CAAC,CAAC;gBAC1C,KAAa,CAAC,IAAI,GAAG,qBAAqB,CAAC;gBAC3C,KAAa,CAAC,SAAS,GAAG,KAAK,CAAC;gBACjC,MAAM,GAAG,IAAI,CAAC;gBACd,MAAM,KAAK,CAAC;YACd,CAAC;SACF;QACD,OAAO,SAAS,CAAC;IACnB,CAAC,CAAC,CAAC;IAEH,OAAO;IACP,MAAM,QAAQ,GAAG,MAAM,sBAAW,CAAC,4BAA4B,CAAC;QAC9D,GAAG,kBAAkB;QACrB,OAAO,EAAE,WAAW;QACpB,WAAW,EAAE;YACX,YAAY,EAAE,sBAAsB;SACrC;KACF,CAAC,CAAC;IAEH,MAAM,QAAQ,CAAC,cAAc,EAAE,CAAC;IAEhC,0EAA0E;IAC1E,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;AAC/B,CAAC,CAAC,CAAC;AAEH;;GAEG;AACH,SAAS,SAAS,CAAC,GAAS;IAC1B,OAAQ,GAAW,CAAC,MAAM,CAAC;AAC7B,CAAC;AAED;;;;;GAKG;AACH,SAAS,YAAY,CAAC,OAA4B;IAChD,SAAS,eAAe,CAAC,QAAoD;;QAC3E,MAAM,GAAG,GAAG,EAAE,CAAC;QACf,KAAK,MAAM,CAAC,OAAO,EAAE,IAAI,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,QAAQ,aAAR,QAAQ,cAAR,QAAQ,GAAI,EAAE,CAAC,EAAE;YAC5D,GAAG,CAAC,IAAI,CAAC,IAAI,OAAO,GAAG,CAAC,CAAC;YAEzB,IAAI,aAAa,CAAC,IAAI,CAAC,EAAE;gBACvB,GAAG,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;gBACtC,IAAI,gBAAgB,IAAI,IAAI,EAAE;oBAC5B,GAAG,CAAC,IAAI,CAAC,kBAAkB,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;iBACnD;gBACD,IAAI,mBAAmB,IAAI,IAAI,EAAE;oBAC/B,GAAG,CAAC,IAAI,CAAC,qBAAqB,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC;iBACzD;gBACD,IAAI,IAAI,CAAC,UAAU,EAAE;oBACnB,GAAG,CAAC,IAAI,CAAC,cAAc,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC;iBAC3C;gBACD,MAAA,OAAO,CAAC,OAAO,0CAAE,YAAY,CAAC,IAAI,OAAC,IAAI,CAAC,QAAQ,mCAAI,OAAO,CAAC,EAAE,IAAI,CAAC,QAAQ,EAAE;oBAC3E,GAAG,IAAI,CAAC,eAAe;oBACvB,eAAe,cAAE,IAAI,CAAC,eAAe,0CAAE,eAAe,0CAAE,GAAG,CAAC,IAAI,CAAC;iBAClE,EAAE;aACJ;iBAAM;gBACL,IAAI,IAAI,CAAC,iBAAiB,EAAE;oBAC1B,GAAG,CAAC,IAAI,CAAC,qBAAqB,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC;oBAC9D,GAAG,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;oBACzC,MAAA,OAAO,CAAC,OAAO,0CAAE,YAAY,CAAC,IAAI,OAAC,IAAI,CAAC,QAAQ,mCAAI,OAAO,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,EAAE,IAAI,CAAC,eAAe,EAAE;iBACnH;aACF;YAED,IAAI,IAAI,CAAC,MAAM,EAAE;gBACf,GAAG,CAAC,IAAI,CAAC,UAAU,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;aACnC;SACF;QACD,OAAO,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACxB,CAAC;IAED,MAAM,CAAC;QACL,2BAA2B,EAAE,eAAe,CAAC,OAAO,CAAC,WAAW,CAAC;QACjE,sBAAsB,EAAE,eAAe,CAAC,OAAO,CAAC,MAAM,CAAC;KACxD,CAAC,CAAC;IAEH,yCAAyC;IACzC,OAAO,CAAC,GAAG,CAAC,eAAe,GAAG,MAAM,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC;IAClE,OAAO,CAAC,GAAG,CAAC,2BAA2B,GAAG,MAAM,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;AACrF,CAAC;AAkCD,SAAS,aAAa,CAAC,CAA4B;IACjD,OAAO,UAAU,IAAI,CAAC,CAAC;AACzB,CAAC;AAED,SAAS,mBAAmB,CAAC,OAA2B;IACtD,OAAO,sBAAW,CAAC,4BAA4B,CAAC,EAAE,GAAG,kBAAkB,EAAE,OAAO,EAAE,CAAC,CAAC;AACtF,CAAC","sourcesContent":["import * as os from 'os';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as AWS from 'aws-sdk';\nimport type { ConfigurationOptions } from 'aws-sdk/lib/config-base';\nimport * as promptly from 'promptly';\nimport * as uuid from 'uuid';\nimport { ISDK, Mode, SDK, SdkProvider } from '../../lib/api/aws-auth';\nimport * as logging from '../../lib/logging';\nimport { PluginHost } from '../../lib/plugin';\nimport * as bockfs from '../bockfs';\nimport { withMocked } from '../util';\nimport { FakeSts, RegisterRoleOptions, RegisterUserOptions } from './fake-sts';\n\njest.mock('promptly', () => ({\n  prompt: jest.fn().mockResolvedValue('1234'),\n}));\n\nconst defaultCredOptions = {\n  ec2creds: false,\n  containerCreds: false,\n};\n\nlet uid: string;\nlet pluginQueried = false;\n\nbeforeEach(() => {\n  // Cache busters!\n  // We prefix everything with UUIDs because:\n  //\n  // - We have a cache from account# -> credentials\n  // - We have a cache from access key -> account\n  uid = `(${uuid.v4()})`;\n\n  logging.setLogLevel(logging.LogLevel.TRACE);\n\n  PluginHost.instance.credentialProviderSources.splice(0);\n  PluginHost.instance.credentialProviderSources.push({\n    isAvailable() { return Promise.resolve(true); },\n    canProvideCredentials(account) { return Promise.resolve(account === uniq('99999')); },\n    getProvider() {\n      pluginQueried = true;\n      return Promise.resolve(new AWS.Credentials({\n        accessKeyId: `${uid}plugin_key`,\n        secretAccessKey: 'plugin_secret',\n        sessionToken: 'plugin_token',\n      }));\n    },\n    name: 'test plugin',\n  });\n\n  // Make sure these point to nonexistant files to start, if we don't call\n  // prepare() then we don't accidentally want to fall back to system config.\n  process.env.AWS_CONFIG_FILE = '/dev/null';\n  process.env.AWS_SHARED_CREDENTIALS_FILE = '/dev/null';\n});\n\nafterEach(() => {\n  bockfs.restore();\n});\n\nfunction uniq(account: string) {\n  return `${uid}${account}`;\n}\n\nfunction env(account: string) {\n  return cxapi.EnvironmentUtils.make(account, 'def');\n}\n\ndescribe('with intercepted network calls', () => {\n  // Most tests will use intercepted network calls, except one test that tests\n  // that the right HTTP `Agent` is used.\n\n  let fakeSts: FakeSts;\n  beforeEach(() => {\n    fakeSts = new FakeSts();\n    fakeSts.begin();\n\n    // Make sure the KeyID returned by the plugin is recognized\n    fakeSts.registerUser(uniq('99999'), uniq('plugin_key'));\n  });\n\n  afterEach(() => {\n    fakeSts.restore();\n  });\n\n  // Set of tests where the CDK will not trigger assume-role\n  // (the INI file might still do assume-role)\n  describe('when CDK does not AssumeRole', () => {\n    test('uses default credentials by default', async () => {\n      // WHEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          default: { aws_access_key_id: 'access', $account: '11111', $fakeStsOptions: { partition: 'aws-here' } },\n        },\n        config: {\n          default: { region: 'eu-bla-5' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n\n      // THEN\n      expect(provider.defaultRegion).toEqual('eu-bla-5');\n      await expect(provider.defaultAccount()).resolves.toEqual({ accountId: uniq('11111'), partition: 'aws-here' });\n\n      // Ask for a different region\n      const sdk = (await provider.forEnvironment({ ...env(uniq('11111')), region: 'rgn' }, Mode.ForReading)).sdk;\n      expect(sdkConfig(sdk).credentials!.accessKeyId).toEqual(uniq('access'));\n      expect(sdk.currentRegion).toEqual('rgn');\n    });\n\n    test('throws if profile credentials are not for the right account', async () => {\n      // WHEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          'profile boo': { aws_access_key_id: 'access', $account: '11111' },\n        },\n      });\n      const provider = await providerFromProfile('boo');\n\n      await expect(provider.forEnvironment(env(uniq('some_account_#')), Mode.ForReading)).rejects.toThrow('Need to perform AWS calls');\n    });\n\n    test('use profile acct/region if agnostic env requested', async () => {\n      // WHEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          default: { aws_access_key_id: 'access', $account: '11111' },\n        },\n        config: {\n          default: { region: 'eu-bla-5' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n\n      // THEN\n      const sdk = (await provider.forEnvironment(cxapi.EnvironmentUtils.make(cxapi.UNKNOWN_ACCOUNT, cxapi.UNKNOWN_REGION), Mode.ForReading)).sdk;\n      expect(sdkConfig(sdk).credentials!.accessKeyId).toEqual(uniq('access'));\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('11111'));\n      expect(sdk.currentRegion).toEqual('eu-bla-5');\n    });\n\n    test('passing profile skips EnvironmentCredentials', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          foo: { aws_access_key_id: 'access', $account: '11111' },\n        },\n      });\n      const provider = await providerFromProfile('foo');\n\n      const environmentCredentialsPrototype = (new AWS.EnvironmentCredentials('AWS')).constructor.prototype;\n\n      await withMocked(environmentCredentialsPrototype, 'refresh', async (refresh) => {\n        refresh.mockImplementation((callback: (err?: Error) => void) => callback(new Error('This function should not have been called')));\n\n        // WHEN\n        expect((await provider.defaultAccount())?.accountId).toEqual(uniq('11111'));\n\n        expect(refresh).not.toHaveBeenCalled();\n      });\n    });\n\n    test('supports profile spread over config_file and credentials_file', async () => {\n      // WHEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          foo: { aws_access_key_id: 'fooccess', $account: '22222' },\n        },\n        config: {\n          'default': { region: 'eu-bla-5' },\n          'profile foo': { region: 'eu-west-1' },\n        },\n      });\n      const provider = await SdkProvider.withAwsCliCompatibleDefaults({ ...defaultCredOptions, profile: 'foo' });\n\n      // THEN\n      expect(provider.defaultRegion).toEqual('eu-west-1');\n      await expect(provider.defaultAccount()).resolves.toEqual({ accountId: uniq('22222'), partition: 'aws' });\n\n      const sdk = (await provider.forEnvironment(env(uniq('22222')), Mode.ForReading)).sdk;\n      expect(sdkConfig(sdk).credentials!.accessKeyId).toEqual(uniq('fooccess'));\n    });\n\n    test('supports profile only in config_file', async () => {\n      // WHEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          'default': { region: 'eu-bla-5' },\n          'profile foo': { aws_access_key_id: 'fooccess', $account: '22222' },\n        },\n      });\n      const provider = await providerFromProfile('foo');\n\n      // THEN\n      expect(provider.defaultRegion).toEqual('eu-bla-5'); // Fall back to default config\n      await expect(provider.defaultAccount()).resolves.toEqual({ accountId: uniq('22222'), partition: 'aws' });\n\n      const sdk = (await provider.forEnvironment(env(uniq('22222')), Mode.ForReading)).sdk;\n      expect(sdkConfig(sdk).credentials!.accessKeyId).toEqual(uniq('fooccess'));\n    });\n\n    test('can assume-role configured in config', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          assumer: { aws_access_key_id: 'assumer', $account: '11111' },\n        },\n        config: {\n          'default': { region: 'eu-bla-5' },\n          'profile assumer': { region: 'us-east-2' },\n          'profile assumable': {\n            role_arn: 'arn:aws:iam::66666:role/Assumable',\n            source_profile: 'assumer',\n            $account: '66666',\n            $fakeStsOptions: { allowedAccounts: ['11111'] },\n          },\n        },\n      });\n      const provider = await providerFromProfile('assumable');\n\n      // WHEN\n      const sdk = (await provider.forEnvironment(env(uniq('66666')), Mode.ForReading)).sdk;\n\n      // THEN\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('66666'));\n    });\n\n    test('can assume role even if [default] profile is missing', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          assumer: { aws_access_key_id: 'assumer', $account: '22222' },\n          assumable: { role_arn: 'arn:aws:iam::12356789012:role/Assumable', source_profile: 'assumer', $account: '22222' },\n        },\n        config: {\n          'profile assumable': { region: 'eu-bla-5' },\n        },\n      });\n\n      // WHEN\n      const provider = await providerFromProfile('assumable');\n\n      // THEN\n      expect((await provider.defaultAccount())?.accountId).toEqual(uniq('22222'));\n    });\n\n    test('mfa_serial in profile will ask user for token', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        credentials: {\n          assumer: { aws_access_key_id: 'assumer', $account: '66666' },\n        },\n        config: {\n          'default': { region: 'eu-bla-5' },\n          'profile assumer': { region: 'us-east-2' },\n          'profile mfa-role': {\n            role_arn: 'arn:aws:iam::66666:role/Assumable',\n            source_profile: 'assumer',\n            mfa_serial: 'arn:aws:iam::account:mfa/user',\n            $account: '66666',\n          },\n        },\n      });\n      const provider = await providerFromProfile('mfa-role');\n\n      const promptlyMockCalls = (promptly.prompt as jest.Mock).mock.calls.length;\n\n      // THEN\n      const sdk = (await provider.forEnvironment(env(uniq('66666')), Mode.ForReading)).sdk;\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('66666'));\n      expect(fakeSts.assumedRoles[0]).toEqual(expect.objectContaining({\n        roleArn: 'arn:aws:iam::66666:role/Assumable',\n        serialNumber: 'arn:aws:iam::account:mfa/user',\n        tokenCode: '1234',\n      }));\n\n      // Mock response was set to fail to make sure we don't call STS\n      // Make sure the MFA mock was called during this test\n      expect((promptly.prompt as jest.Mock).mock.calls.length).toBe(promptlyMockCalls + 1);\n    });\n  });\n\n  // For DefaultSynthesis we will do an assume-role after having gotten base credentials\n  describe('when CDK AssumeRoles', () => {\n    beforeEach(() => {\n      // All these tests share that 'arn:aws:role' is a role into account 88888 which can be assumed from 11111\n      fakeSts.registerRole(uniq('88888'), 'arn:aws:role', { allowedAccounts: [uniq('11111')] });\n    });\n\n    test('error we get from assuming a role is useful', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          default: { aws_access_key_id: 'foo' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n\n      // WHEN\n      const promise = provider.forEnvironment(env(uniq('88888')), Mode.ForReading, {\n        assumeRoleArn: 'doesnotexist.role.arn',\n      });\n\n      // THEN - error message contains both a helpful hint and the underlying AssumeRole message\n      await expect(promise).rejects.toThrow('(re)-bootstrap the environment');\n      await expect(promise).rejects.toThrow('doesnotexist.role.arn');\n    });\n\n    test('assuming a role sanitizes the username into the session name', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          default: { aws_access_key_id: 'foo', $account: '11111' },\n        },\n      });\n\n      await withMocked(os, 'userInfo', async (userInfo) => {\n        userInfo.mockReturnValue({ username: 'skål', uid: 1, gid: 1, homedir: '/here', shell: '/bin/sh' });\n\n        // WHEN\n        const provider = await providerFromProfile(undefined);\n\n        const sdk = (await provider.forEnvironment(env(uniq('88888')), Mode.ForReading, { assumeRoleArn: 'arn:aws:role' })).sdk as SDK;\n        await sdk.currentAccount();\n\n        // THEN\n        expect(fakeSts.assumedRoles[0]).toEqual(expect.objectContaining({\n          roleSessionName: 'aws-cdk-sk@l',\n        }));\n      });\n    });\n\n    test('even if current credentials are for the wrong account, we will still use them to AssumeRole', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          default: { aws_access_key_id: 'foo', $account: '11111' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n\n      // WHEN\n      const sdk = (await provider.forEnvironment(env(uniq('88888')), Mode.ForReading, { assumeRoleArn: 'arn:aws:role' })).sdk as SDK;\n\n      // THEN\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('88888'));\n    });\n\n    test('if AssumeRole fails but current credentials are for the right account, we will still use them', async () => {\n      // GIVEN\n      prepareCreds({\n        fakeSts,\n        config: {\n          default: { aws_access_key_id: 'foo', $account: '88888' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n\n      // WHEN - assumeRole fails because the role can only be assumed from account 11111\n      const sdk = (await provider.forEnvironment(env(uniq('88888')), Mode.ForReading, { assumeRoleArn: 'arn:aws:role' })).sdk as SDK;\n\n      // THEN\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('88888'));\n    });\n  });\n\n  describe('Plugins', () => {\n    test('does not use plugins if current credentials are for expected account', async () => {\n      prepareCreds({\n        fakeSts,\n        config: {\n          default: { aws_access_key_id: 'foo', $account: '11111' },\n        },\n      });\n      const provider = await providerFromProfile(undefined);\n      await provider.forEnvironment(env(uniq('11111')), Mode.ForReading);\n      expect(pluginQueried).toEqual(false);\n    });\n\n    test('uses plugin for account 99999', async () => {\n      const provider = await providerFromProfile(undefined);\n      await provider.forEnvironment(env(uniq('99999')), Mode.ForReading);\n      expect(pluginQueried).toEqual(true);\n    });\n\n    test('can assume role with credentials from plugin', async () => {\n      fakeSts.registerRole(uniq('99999'), 'arn:aws:iam::99999:role/Assumable');\n\n      const provider = await providerFromProfile(undefined);\n      await provider.forEnvironment(env(uniq('99999')), Mode.ForReading, {\n        assumeRoleArn: 'arn:aws:iam::99999:role/Assumable',\n      });\n\n      expect(fakeSts.assumedRoles[0]).toEqual(expect.objectContaining({\n        roleArn: 'arn:aws:iam::99999:role/Assumable',\n      }));\n      expect(pluginQueried).toEqual(true);\n    });\n\n    test('even if AssumeRole fails but current credentials are from a plugin, we will still use them', async () => {\n      const provider = await providerFromProfile(undefined);\n      const sdk = (await provider.forEnvironment(env(uniq('99999')), Mode.ForReading, { assumeRoleArn: 'does:not:exist' })).sdk;\n\n      // THEN\n      expect((await sdk.currentAccount()).accountId).toEqual(uniq('99999'));\n    });\n\n    test('plugins are still queried even if current credentials are expired (or otherwise invalid)', async () => {\n      // GIVEN\n      process.env.AWS_ACCESS_KEY_ID = `${uid}akid`;\n      process.env.AWS_SECRET_ACCESS_KEY = 'sekrit';\n      const provider = await providerFromProfile(undefined);\n\n      // WHEN\n      await provider.forEnvironment(env(uniq('99999')), Mode.ForReading);\n\n      // THEN\n      expect(pluginQueried).toEqual(true);\n    });\n  });\n\n  describe('support for credential_source', () => {\n    test('can assume role with ecs credentials', async () => {\n      return withMocked(AWS.ECSCredentials.prototype, 'needsRefresh', async (needsRefresh) => {\n        // GIVEN\n        prepareCreds({\n          config: {\n            'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'EcsContainer', $account: '22222' },\n          },\n        });\n        const provider = await providerFromProfile('ecs');\n\n        // WHEN\n        await provider.defaultAccount();\n\n        // THEN\n        expect(needsRefresh).toHaveBeenCalled();\n      });\n\n    });\n\n    test('can assume role with ec2 credentials', async () => {\n      return withMocked(AWS.EC2MetadataCredentials.prototype, 'needsRefresh', async (needsRefresh) => {\n        // GIVEN\n        prepareCreds({\n          config: {\n            'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'Ec2InstanceMetadata', $account: '22222' },\n          },\n        });\n        const provider = await providerFromProfile('ecs');\n\n        // WHEN\n        await provider.defaultAccount();\n\n        // THEN\n        expect(needsRefresh).toHaveBeenCalled();\n\n      });\n\n    });\n\n    test('can assume role with env credentials', async () => {\n      return withMocked(AWS.EnvironmentCredentials.prototype, 'needsRefresh', async (needsRefresh) => {\n        // GIVEN\n        prepareCreds({\n          config: {\n            'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'Environment', $account: '22222' },\n          },\n        });\n        const provider = await providerFromProfile('ecs');\n\n        // WHEN\n        await provider.defaultAccount();\n\n        // THEN\n        expect(needsRefresh).toHaveBeenCalled();\n      });\n    });\n\n    test('assume fails with unsupported credential_source', async () => {\n      // GIVEN\n      prepareCreds({\n        config: {\n          'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'unsupported', $account: '22222' },\n        },\n      });\n      const provider = await providerFromProfile('ecs');\n\n      // WHEN\n      const account = await provider.defaultAccount();\n\n      // THEN\n      expect(account?.accountId).toEqual(undefined);\n    });\n  });\n\n  test('defaultAccount returns undefined if STS call fails', async () => {\n    // GIVEN\n    process.env.AWS_ACCESS_KEY_ID = `${uid}akid`;\n    process.env.AWS_SECRET_ACCESS_KEY = 'sekrit';\n\n    // WHEN\n    const provider = await providerFromProfile(undefined);\n\n    // THEN\n    await expect(provider.defaultAccount()).resolves.toBe(undefined);\n  });\n});\n\ntest('even when using a profile to assume another profile, STS calls goes through the proxy', async () => {\n  prepareCreds({\n    credentials: {\n      assumer: { aws_access_key_id: 'assumer' },\n    },\n    config: {\n      'default': { region: 'eu-bla-5' },\n      'profile assumable': { role_arn: 'arn:aws:iam::66666:role/Assumable', source_profile: 'assumer', $account: '66666' },\n      'profile assumer': { region: 'us-east-2' },\n    },\n  });\n\n  // Messy mocking\n  let called = false;\n  jest.mock('proxy-agent', () => {\n    // eslint-disable-next-line @typescript-eslint/no-require-imports\n    class FakeAgent extends require('https').Agent {\n      public addRequest(_: any, __: any) {\n        // FIXME: this error takes 6 seconds to be completely handled. It\n        // might be retries in the SDK somewhere, or something about the Node\n        // event loop. I've spent an hour trying to figure it out and I can't,\n        // and I gave up. We'll just have to live with this until someone gets\n        // inspired.\n        const error = new Error('ABORTED BY TEST');\n        (error as any).code = 'RequestAbortedError';\n        (error as any).retryable = false;\n        called = true;\n        throw error;\n      }\n    }\n    return FakeAgent;\n  });\n\n  // WHEN\n  const provider = await SdkProvider.withAwsCliCompatibleDefaults({\n    ...defaultCredOptions,\n    profile: 'assumable',\n    httpOptions: {\n      proxyAddress: 'http://DOESNTMATTER/',\n    },\n  });\n\n  await provider.defaultAccount();\n\n  // THEN -- the fake proxy agent got called, we don't care about the result\n  expect(called).toEqual(true);\n});\n\n/**\n * Use object hackery to get the credentials out of the SDK object\n */\nfunction sdkConfig(sdk: ISDK): ConfigurationOptions {\n  return (sdk as any).config;\n}\n\n/**\n * Fixture for SDK auth for this test suite\n *\n * Has knowledge of the cache buster, will write proper fake config files and\n * register users and roles in FakeSts at the same time.\n */\nfunction prepareCreds(options: PrepareCredsOptions) {\n  function convertSections(sections?: Record<string, ProfileUser | ProfileRole>) {\n    const ret = [];\n    for (const [profile, user] of Object.entries(sections ?? {})) {\n      ret.push(`[${profile}]`);\n\n      if (isProfileRole(user)) {\n        ret.push(`role_arn=${user.role_arn}`);\n        if ('source_profile' in user) {\n          ret.push(`source_profile=${user.source_profile}`);\n        }\n        if ('credential_source' in user) {\n          ret.push(`credential_source=${user.credential_source}`);\n        }\n        if (user.mfa_serial) {\n          ret.push(`mfa_serial=${user.mfa_serial}`);\n        }\n        options.fakeSts?.registerRole(uniq(user.$account ?? '00000'), user.role_arn, {\n          ...user.$fakeStsOptions,\n          allowedAccounts: user.$fakeStsOptions?.allowedAccounts?.map(uniq),\n        });\n      } else {\n        if (user.aws_access_key_id) {\n          ret.push(`aws_access_key_id=${uniq(user.aws_access_key_id)}`);\n          ret.push('aws_secret_access_key=secret');\n          options.fakeSts?.registerUser(uniq(user.$account ?? '00000'), uniq(user.aws_access_key_id), user.$fakeStsOptions);\n        }\n      }\n\n      if (user.region) {\n        ret.push(`region=${user.region}`);\n      }\n    }\n    return ret.join('\\n');\n  }\n\n  bockfs({\n    '/home/me/.bxt/credentials': convertSections(options.credentials),\n    '/home/me/.bxt/config': convertSections(options.config),\n  });\n\n  // Set environment variables that we want\n  process.env.AWS_CONFIG_FILE = bockfs.path('/home/me/.bxt/config');\n  process.env.AWS_SHARED_CREDENTIALS_FILE = bockfs.path('/home/me/.bxt/credentials');\n}\n\ninterface PrepareCredsOptions {\n  /**\n   * Write the aws/credentials file\n   */\n  readonly credentials?: Record<string, ProfileUser | ProfileRole>;\n\n  /**\n   * Write the aws/config file\n   */\n  readonly config?: Record<string, ProfileUser | ProfileRole>;\n\n  /**\n   * If given, add users to FakeSTS\n   */\n  readonly fakeSts?: FakeSts;\n}\n\ninterface ProfileUser {\n  readonly aws_access_key_id?: string;\n  readonly $account?: string;\n  readonly region?: string;\n  readonly $fakeStsOptions?: RegisterUserOptions;\n}\n\ntype ProfileRole = {\n  readonly role_arn: string;\n  readonly mfa_serial?: string;\n  readonly $account: string;\n  readonly region?: string;\n  readonly $fakeStsOptions?: RegisterRoleOptions;\n} & ({ readonly source_profile: string } | { readonly credential_source: string });\n\nfunction isProfileRole(x: ProfileUser | ProfileRole): x is ProfileRole {\n  return 'role_arn' in x;\n}\n\nfunction providerFromProfile(profile: string | undefined) {\n  return SdkProvider.withAwsCliCompatibleDefaults({ ...defaultCredOptions, profile });\n}\n"]}
|