/ cloudformation-templates / node_modules / aws-cdk / test / api / sdk-provider.test.js
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":"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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}"]}