sdk-provider.test.js
1 "use strict"; 2 Object.defineProperty(exports, "__esModule", { value: true }); 3 const os = require("os"); 4 const cxapi = require("@aws-cdk/cx-api"); 5 const AWS = require("aws-sdk"); 6 const promptly = require("promptly"); 7 const uuid = require("uuid"); 8 const aws_auth_1 = require("../../lib/api/aws-auth"); 9 const logging = require("../../lib/logging"); 10 const plugin_1 = require("../../lib/plugin"); 11 const bockfs = require("../bockfs"); 12 const util_1 = require("../util"); 13 const fake_sts_1 = require("./fake-sts"); 14 jest.mock('promptly', () => ({ 15 prompt: jest.fn().mockResolvedValue('1234'), 16 })); 17 const defaultCredOptions = { 18 ec2creds: false, 19 containerCreds: false, 20 }; 21 let uid; 22 let pluginQueried = false; 23 beforeEach(() => { 24 // Cache busters! 25 // We prefix everything with UUIDs because: 26 // 27 // - We have a cache from account# -> credentials 28 // - We have a cache from access key -> account 29 uid = `(${uuid.v4()})`; 30 logging.setLogLevel(2 /* TRACE */); 31 plugin_1.PluginHost.instance.credentialProviderSources.splice(0); 32 plugin_1.PluginHost.instance.credentialProviderSources.push({ 33 isAvailable() { return Promise.resolve(true); }, 34 canProvideCredentials(account) { return Promise.resolve(account === uniq('99999')); }, 35 getProvider() { 36 pluginQueried = true; 37 return Promise.resolve(new AWS.Credentials({ 38 accessKeyId: `${uid}plugin_key`, 39 secretAccessKey: 'plugin_secret', 40 sessionToken: 'plugin_token', 41 })); 42 }, 43 name: 'test plugin', 44 }); 45 // Make sure these point to nonexistant files to start, if we don't call 46 // prepare() then we don't accidentally want to fall back to system config. 47 process.env.AWS_CONFIG_FILE = '/dev/null'; 48 process.env.AWS_SHARED_CREDENTIALS_FILE = '/dev/null'; 49 }); 50 afterEach(() => { 51 bockfs.restore(); 52 }); 53 function uniq(account) { 54 return `${uid}${account}`; 55 } 56 function env(account) { 57 return cxapi.EnvironmentUtils.make(account, 'def'); 58 } 59 describe('with intercepted network calls', () => { 60 // Most tests will use intercepted network calls, except one test that tests 61 // that the right HTTP `Agent` is used. 62 let fakeSts; 63 beforeEach(() => { 64 fakeSts = new fake_sts_1.FakeSts(); 65 fakeSts.begin(); 66 // Make sure the KeyID returned by the plugin is recognized 67 fakeSts.registerUser(uniq('99999'), uniq('plugin_key')); 68 }); 69 afterEach(() => { 70 fakeSts.restore(); 71 }); 72 // Set of tests where the CDK will not trigger assume-role 73 // (the INI file might still do assume-role) 74 describe('when CDK does not AssumeRole', () => { 75 test('uses default credentials by default', async () => { 76 // WHEN 77 prepareCreds({ 78 fakeSts, 79 credentials: { 80 default: { aws_access_key_id: 'access', $account: '11111', $fakeStsOptions: { partition: 'aws-here' } }, 81 }, 82 config: { 83 default: { region: 'eu-bla-5' }, 84 }, 85 }); 86 const provider = await providerFromProfile(undefined); 87 // THEN 88 expect(provider.defaultRegion).toEqual('eu-bla-5'); 89 await expect(provider.defaultAccount()).resolves.toEqual({ accountId: uniq('11111'), partition: 'aws-here' }); 90 // Ask for a different region 91 const sdk = await provider.forEnvironment({ ...env(uniq('11111')), region: 'rgn' }, aws_auth_1.Mode.ForReading); 92 expect(sdkConfig(sdk).credentials.accessKeyId).toEqual(uniq('access')); 93 expect(sdk.currentRegion).toEqual('rgn'); 94 }); 95 test('throws if profile credentials are not for the right account', async () => { 96 // WHEN 97 prepareCreds({ 98 fakeSts, 99 config: { 100 'profile boo': { aws_access_key_id: 'access', $account: '11111' }, 101 }, 102 }); 103 const provider = await providerFromProfile('boo'); 104 await expect(provider.forEnvironment(env(uniq('some_account_#')), aws_auth_1.Mode.ForReading)).rejects.toThrow('Need to perform AWS calls'); 105 }); 106 test('use profile acct/region if agnostic env requested', async () => { 107 // WHEN 108 prepareCreds({ 109 fakeSts, 110 credentials: { 111 default: { aws_access_key_id: 'access', $account: '11111' }, 112 }, 113 config: { 114 default: { region: 'eu-bla-5' }, 115 }, 116 }); 117 const provider = await providerFromProfile(undefined); 118 // THEN 119 const sdk = await provider.forEnvironment(cxapi.EnvironmentUtils.make(cxapi.UNKNOWN_ACCOUNT, cxapi.UNKNOWN_REGION), aws_auth_1.Mode.ForReading); 120 expect(sdkConfig(sdk).credentials.accessKeyId).toEqual(uniq('access')); 121 expect((await sdk.currentAccount()).accountId).toEqual(uniq('11111')); 122 expect(sdk.currentRegion).toEqual('eu-bla-5'); 123 }); 124 test('passing profile skips EnvironmentCredentials', async () => { 125 // GIVEN 126 prepareCreds({ 127 fakeSts, 128 credentials: { 129 foo: { aws_access_key_id: 'access', $account: '11111' }, 130 }, 131 }); 132 const provider = await providerFromProfile('foo'); 133 const environmentCredentialsPrototype = (new AWS.EnvironmentCredentials('AWS')).constructor.prototype; 134 await util_1.withMocked(environmentCredentialsPrototype, 'refresh', async (refresh) => { 135 var _a; 136 refresh.mockImplementation((callback) => callback(new Error('This function should not have been called'))); 137 // WHEN 138 expect((_a = (await provider.defaultAccount())) === null || _a === void 0 ? void 0 : _a.accountId).toEqual(uniq('11111')); 139 expect(refresh).not.toHaveBeenCalled(); 140 }); 141 }); 142 test('supports profile spread over config_file and credentials_file', async () => { 143 // WHEN 144 prepareCreds({ 145 fakeSts, 146 credentials: { 147 foo: { aws_access_key_id: 'fooccess', $account: '22222' }, 148 }, 149 config: { 150 'default': { region: 'eu-bla-5' }, 151 'profile foo': { region: 'eu-west-1' }, 152 }, 153 }); 154 const provider = await aws_auth_1.SdkProvider.withAwsCliCompatibleDefaults({ ...defaultCredOptions, profile: 'foo' }); 155 // THEN 156 expect(provider.defaultRegion).toEqual('eu-west-1'); 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); 159 expect(sdkConfig(sdk).credentials.accessKeyId).toEqual(uniq('fooccess')); 160 }); 161 test('supports profile only in config_file', async () => { 162 // WHEN 163 prepareCreds({ 164 fakeSts, 165 config: { 166 'default': { region: 'eu-bla-5' }, 167 'profile foo': { aws_access_key_id: 'fooccess', $account: '22222' }, 168 }, 169 }); 170 const provider = await providerFromProfile('foo'); 171 // THEN 172 expect(provider.defaultRegion).toEqual('eu-bla-5'); // Fall back to default config 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); 175 expect(sdkConfig(sdk).credentials.accessKeyId).toEqual(uniq('fooccess')); 176 }); 177 test('can assume-role configured in config', async () => { 178 // GIVEN 179 prepareCreds({ 180 fakeSts, 181 credentials: { 182 assumer: { aws_access_key_id: 'assumer', $account: '11111' }, 183 }, 184 config: { 185 'default': { region: 'eu-bla-5' }, 186 'profile assumer': { region: 'us-east-2' }, 187 'profile assumable': { 188 role_arn: 'arn:aws:iam::66666:role/Assumable', 189 source_profile: 'assumer', 190 $account: '66666', 191 $fakeStsOptions: { allowedAccounts: ['11111'] }, 192 }, 193 }, 194 }); 195 const provider = await providerFromProfile('assumable'); 196 // WHEN 197 const sdk = await provider.forEnvironment(env(uniq('66666')), aws_auth_1.Mode.ForReading); 198 // THEN 199 expect((await sdk.currentAccount()).accountId).toEqual(uniq('66666')); 200 }); 201 test('can assume role even if [default] profile is missing', async () => { 202 var _a; 203 // GIVEN 204 prepareCreds({ 205 fakeSts, 206 credentials: { 207 assumer: { aws_access_key_id: 'assumer', $account: '22222' }, 208 assumable: { role_arn: 'arn:aws:iam::12356789012:role/Assumable', source_profile: 'assumer', $account: '22222' }, 209 }, 210 config: { 211 'profile assumable': { region: 'eu-bla-5' }, 212 }, 213 }); 214 // WHEN 215 const provider = await providerFromProfile('assumable'); 216 // THEN 217 expect((_a = (await provider.defaultAccount())) === null || _a === void 0 ? void 0 : _a.accountId).toEqual(uniq('22222')); 218 }); 219 test('mfa_serial in profile will ask user for token', async () => { 220 // GIVEN 221 prepareCreds({ 222 fakeSts, 223 credentials: { 224 assumer: { aws_access_key_id: 'assumer', $account: '66666' }, 225 }, 226 config: { 227 'default': { region: 'eu-bla-5' }, 228 'profile assumer': { region: 'us-east-2' }, 229 'profile mfa-role': { 230 role_arn: 'arn:aws:iam::66666:role/Assumable', 231 source_profile: 'assumer', 232 mfa_serial: 'arn:aws:iam::account:mfa/user', 233 $account: '66666', 234 }, 235 }, 236 }); 237 const provider = await providerFromProfile('mfa-role'); 238 const promptlyMockCalls = promptly.prompt.mock.calls.length; 239 // THEN 240 const sdk = await provider.forEnvironment(env(uniq('66666')), aws_auth_1.Mode.ForReading); 241 expect((await sdk.currentAccount()).accountId).toEqual(uniq('66666')); 242 expect(fakeSts.assumedRoles[0]).toEqual(expect.objectContaining({ 243 roleArn: 'arn:aws:iam::66666:role/Assumable', 244 serialNumber: 'arn:aws:iam::account:mfa/user', 245 tokenCode: '1234', 246 })); 247 // Mock response was set to fail to make sure we don't call STS 248 // Make sure the MFA mock was called during this test 249 expect(promptly.prompt.mock.calls.length).toBe(promptlyMockCalls + 1); 250 }); 251 }); 252 // For DefaultSynthesis we will do an assume-role after having gotten base credentials 253 describe('when CDK AssumeRoles', () => { 254 beforeEach(() => { 255 // All these tests share that 'arn:aws:role' is a role into account 88888 which can be assumed from 11111 256 fakeSts.registerRole(uniq('88888'), 'arn:aws:role', { allowedAccounts: [uniq('11111')] }); 257 }); 258 test('error we get from assuming a role is useful', async () => { 259 // GIVEN 260 prepareCreds({ 261 fakeSts, 262 config: { 263 default: { aws_access_key_id: 'foo' }, 264 }, 265 }); 266 const provider = await providerFromProfile(undefined); 267 // WHEN 268 const promise = provider.forEnvironment(env(uniq('88888')), aws_auth_1.Mode.ForReading, { 269 assumeRoleArn: 'doesnotexist.role.arn', 270 }); 271 // THEN - error message contains both a helpful hint and the underlying AssumeRole message 272 await expect(promise).rejects.toThrow('(re)-bootstrap the environment'); 273 await expect(promise).rejects.toThrow('doesnotexist.role.arn'); 274 }); 275 test('assuming a role sanitizes the username into the session name', async () => { 276 // GIVEN 277 prepareCreds({ 278 fakeSts, 279 config: { 280 default: { aws_access_key_id: 'foo', $account: '11111' }, 281 }, 282 }); 283 await util_1.withMocked(os, 'userInfo', async (userInfo) => { 284 userInfo.mockReturnValue({ username: 'skål', uid: 1, gid: 1, homedir: '/here', shell: '/bin/sh' }); 285 // WHEN 286 const provider = await providerFromProfile(undefined); 287 const sdk = await provider.forEnvironment(env(uniq('88888')), aws_auth_1.Mode.ForReading, { assumeRoleArn: 'arn:aws:role' }); 288 await sdk.currentAccount(); 289 // THEN 290 expect(fakeSts.assumedRoles[0]).toEqual(expect.objectContaining({ 291 roleSessionName: 'aws-cdk-sk@l', 292 })); 293 }); 294 }); 295 test('even if current credentials are for the wrong account, we will still use them to AssumeRole', async () => { 296 // GIVEN 297 prepareCreds({ 298 fakeSts, 299 config: { 300 default: { aws_access_key_id: 'foo', $account: '11111' }, 301 }, 302 }); 303 const provider = await providerFromProfile(undefined); 304 // WHEN 305 const sdk = await provider.forEnvironment(env(uniq('88888')), aws_auth_1.Mode.ForReading, { assumeRoleArn: 'arn:aws:role' }); 306 // THEN 307 expect((await sdk.currentAccount()).accountId).toEqual(uniq('88888')); 308 }); 309 test('if AssumeRole fails but current credentials are for the right account, we will still use them', async () => { 310 // GIVEN 311 prepareCreds({ 312 fakeSts, 313 config: { 314 default: { aws_access_key_id: 'foo', $account: '88888' }, 315 }, 316 }); 317 const provider = await providerFromProfile(undefined); 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' }); 320 // THEN 321 expect((await sdk.currentAccount()).accountId).toEqual(uniq('88888')); 322 }); 323 }); 324 describe('Plugins', () => { 325 test('does not use plugins if current credentials are for expected account', async () => { 326 prepareCreds({ 327 fakeSts, 328 config: { 329 default: { aws_access_key_id: 'foo', $account: '11111' }, 330 }, 331 }); 332 const provider = await providerFromProfile(undefined); 333 await provider.forEnvironment(env(uniq('11111')), aws_auth_1.Mode.ForReading); 334 expect(pluginQueried).toEqual(false); 335 }); 336 test('uses plugin for account 99999', async () => { 337 const provider = await providerFromProfile(undefined); 338 await provider.forEnvironment(env(uniq('99999')), aws_auth_1.Mode.ForReading); 339 expect(pluginQueried).toEqual(true); 340 }); 341 test('can assume role with credentials from plugin', async () => { 342 fakeSts.registerRole(uniq('99999'), 'arn:aws:iam::99999:role/Assumable'); 343 const provider = await providerFromProfile(undefined); 344 await provider.forEnvironment(env(uniq('99999')), aws_auth_1.Mode.ForReading, { 345 assumeRoleArn: 'arn:aws:iam::99999:role/Assumable', 346 }); 347 expect(fakeSts.assumedRoles[0]).toEqual(expect.objectContaining({ 348 roleArn: 'arn:aws:iam::99999:role/Assumable', 349 })); 350 expect(pluginQueried).toEqual(true); 351 }); 352 test('even if AssumeRole fails but current credentials are from a plugin, we will still use them', async () => { 353 const provider = await providerFromProfile(undefined); 354 const sdk = await provider.forEnvironment(env(uniq('99999')), aws_auth_1.Mode.ForReading, { assumeRoleArn: 'does:not:exist' }); 355 // THEN 356 expect((await sdk.currentAccount()).accountId).toEqual(uniq('99999')); 357 }); 358 test('plugins are still queried even if current credentials are expired (or otherwise invalid)', async () => { 359 // GIVEN 360 process.env.AWS_ACCESS_KEY_ID = `${uid}akid`; 361 process.env.AWS_SECRET_ACCESS_KEY = 'sekrit'; 362 const provider = await providerFromProfile(undefined); 363 // WHEN 364 await provider.forEnvironment(env(uniq('99999')), aws_auth_1.Mode.ForReading); 365 // THEN 366 expect(pluginQueried).toEqual(true); 367 }); 368 }); 369 describe('support for credential_source', () => { 370 test('can assume role with ecs credentials', async () => { 371 return util_1.withMocked(AWS.ECSCredentials.prototype, 'needsRefresh', async (needsRefresh) => { 372 // GIVEN 373 prepareCreds({ 374 config: { 375 'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'EcsContainer', $account: '22222' }, 376 }, 377 }); 378 const provider = await providerFromProfile('ecs'); 379 // WHEN 380 await provider.defaultAccount(); 381 // THEN 382 expect(needsRefresh).toHaveBeenCalled(); 383 }); 384 }); 385 test('can assume role with ec2 credentials', async () => { 386 return util_1.withMocked(AWS.EC2MetadataCredentials.prototype, 'needsRefresh', async (needsRefresh) => { 387 // GIVEN 388 prepareCreds({ 389 config: { 390 'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'Ec2InstanceMetadata', $account: '22222' }, 391 }, 392 }); 393 const provider = await providerFromProfile('ecs'); 394 // WHEN 395 await provider.defaultAccount(); 396 // THEN 397 expect(needsRefresh).toHaveBeenCalled(); 398 }); 399 }); 400 test('can assume role with env credentials', async () => { 401 return util_1.withMocked(AWS.EnvironmentCredentials.prototype, 'needsRefresh', async (needsRefresh) => { 402 // GIVEN 403 prepareCreds({ 404 config: { 405 'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'Environment', $account: '22222' }, 406 }, 407 }); 408 const provider = await providerFromProfile('ecs'); 409 // WHEN 410 await provider.defaultAccount(); 411 // THEN 412 expect(needsRefresh).toHaveBeenCalled(); 413 }); 414 }); 415 test('assume fails with unsupported credential_source', async () => { 416 // GIVEN 417 prepareCreds({ 418 config: { 419 'profile ecs': { role_arn: 'arn:aws:iam::12356789012:role/Assumable', credential_source: 'unsupported', $account: '22222' }, 420 }, 421 }); 422 const provider = await providerFromProfile('ecs'); 423 // WHEN 424 const account = await provider.defaultAccount(); 425 // THEN 426 expect(account === null || account === void 0 ? void 0 : account.accountId).toEqual(undefined); 427 }); 428 }); 429 test('defaultAccount returns undefined if STS call fails', async () => { 430 // GIVEN 431 process.env.AWS_ACCESS_KEY_ID = `${uid}akid`; 432 process.env.AWS_SECRET_ACCESS_KEY = 'sekrit'; 433 // WHEN 434 const provider = await providerFromProfile(undefined); 435 // THEN 436 await expect(provider.defaultAccount()).resolves.toBe(undefined); 437 }); 438 }); 439 test('even when using a profile to assume another profile, STS calls goes through the proxy', async () => { 440 prepareCreds({ 441 credentials: { 442 assumer: { aws_access_key_id: 'assumer' }, 443 }, 444 config: { 445 'default': { region: 'eu-bla-5' }, 446 'profile assumable': { role_arn: 'arn:aws:iam::66666:role/Assumable', source_profile: 'assumer', $account: '66666' }, 447 'profile assumer': { region: 'us-east-2' }, 448 }, 449 }); 450 // Messy mocking 451 let called = false; 452 jest.mock('proxy-agent', () => { 453 // eslint-disable-next-line @typescript-eslint/no-require-imports 454 class FakeAgent extends require('https').Agent { 455 addRequest(_, __) { 456 // FIXME: this error takes 6 seconds to be completely handled. It 457 // might be retries in the SDK somewhere, or something about the Node 458 // event loop. I've spent an hour trying to figure it out and I can't, 459 // and I gave up. We'll just have to live with this until someone gets 460 // inspired. 461 const error = new Error('ABORTED BY TEST'); 462 error.code = 'RequestAbortedError'; 463 error.retryable = false; 464 called = true; 465 throw error; 466 } 467 } 468 return FakeAgent; 469 }); 470 // WHEN 471 const provider = await aws_auth_1.SdkProvider.withAwsCliCompatibleDefaults({ 472 ...defaultCredOptions, 473 profile: 'assumable', 474 httpOptions: { 475 proxyAddress: 'http://DOESNTMATTER/', 476 }, 477 }); 478 await provider.defaultAccount(); 479 // THEN -- the fake proxy agent got called, we don't care about the result 480 expect(called).toEqual(true); 481 }); 482 /** 483 * Use object hackery to get the credentials out of the SDK object 484 */ 485 function sdkConfig(sdk) { 486 return sdk.config; 487 } 488 /** 489 * Fixture for SDK auth for this test suite 490 * 491 * Has knowledge of the cache buster, will write proper fake config files and 492 * register users and roles in FakeSts at the same time. 493 */ 494 function prepareCreds(options) { 495 function convertSections(sections) { 496 var _a, _b, _c, _d, _e, _f; 497 const ret = []; 498 for (const [profile, user] of Object.entries(sections !== null && sections !== void 0 ? sections : {})) { 499 ret.push(`[${profile}]`); 500 if (isProfileRole(user)) { 501 ret.push(`role_arn=${user.role_arn}`); 502 if ('source_profile' in user) { 503 ret.push(`source_profile=${user.source_profile}`); 504 } 505 if ('credential_source' in user) { 506 ret.push(`credential_source=${user.credential_source}`); 507 } 508 if (user.mfa_serial) { 509 ret.push(`mfa_serial=${user.mfa_serial}`); 510 } 511 (_a = options.fakeSts) === null || _a === void 0 ? void 0 : _a.registerRole(uniq((_b = user.$account) !== null && _b !== void 0 ? _b : '00000'), user.role_arn, { 512 ...user.$fakeStsOptions, 513 allowedAccounts: (_d = (_c = user.$fakeStsOptions) === null || _c === void 0 ? void 0 : _c.allowedAccounts) === null || _d === void 0 ? void 0 : _d.map(uniq), 514 }); 515 } 516 else { 517 if (user.aws_access_key_id) { 518 ret.push(`aws_access_key_id=${uniq(user.aws_access_key_id)}`); 519 ret.push('aws_secret_access_key=secret'); 520 (_e = options.fakeSts) === null || _e === void 0 ? void 0 : _e.registerUser(uniq((_f = user.$account) !== null && _f !== void 0 ? _f : '00000'), uniq(user.aws_access_key_id), user.$fakeStsOptions); 521 } 522 } 523 if (user.region) { 524 ret.push(`region=${user.region}`); 525 } 526 } 527 return ret.join('\n'); 528 } 529 bockfs({ 530 '/home/me/.bxt/credentials': convertSections(options.credentials), 531 '/home/me/.bxt/config': convertSections(options.config), 532 }); 533 // Set environment variables that we want 534 process.env.AWS_CONFIG_FILE = bockfs.path('/home/me/.bxt/config'); 535 process.env.AWS_SHARED_CREDENTIALS_FILE = bockfs.path('/home/me/.bxt/credentials'); 536 } 537 function isProfileRole(x) { 538 return 'role_arn' in x; 539 } 540 function providerFromProfile(profile) { 541 return aws_auth_1.SdkProvider.withAwsCliCompatibleDefaults({ ...defaultCredOptions, profile }); 542 } 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,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;YACrG,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,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;YACrI,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,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC;YAC/E,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,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC;YAC/E,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,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC;YAE/E,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,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC;YAC/E,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,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,EAAE,EAAE,aAAa,EAAE,cAAc,EAAE,CAAQ,CAAC;gBACzH,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,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,EAAE,EAAE,aAAa,EAAE,cAAc,EAAE,CAAQ,CAAC;YAEzH,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,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,EAAE,EAAE,aAAa,EAAE,cAAc,EAAE,CAAQ,CAAC;YAEzH,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,MAAM,QAAQ,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,eAAI,CAAC,UAAU,EAAE,EAAE,aAAa,EAAE,gBAAgB,EAAE,CAAC,CAAC;YAEpH,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);\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}"]}