AssumeRoleProvider.mjs 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. import * as http from "http";
  2. import * as https from "https";
  3. import { URL, URLSearchParams } from "url";
  4. import { CredentialProvider } from "./CredentialProvider.mjs";
  5. import { Credentials } from "./Credentials.mjs";
  6. import { makeDateLong, parseXml, toSha256 } from "./internal/helper.mjs";
  7. import { request } from "./internal/request.mjs";
  8. import { readAsString } from "./internal/response.mjs";
  9. import { signV4ByServiceName } from "./signing.mjs";
  10. /**
  11. * @see https://docs.aws.amazon.com/STS/latest/APIReference/API_AssumeRole.html
  12. */
  13. const defaultExpirySeconds = 900;
  14. export class AssumeRoleProvider extends CredentialProvider {
  15. accessExpiresAt = '';
  16. constructor({
  17. stsEndpoint,
  18. accessKey,
  19. secretKey,
  20. durationSeconds = defaultExpirySeconds,
  21. sessionToken,
  22. policy,
  23. region = '',
  24. roleArn,
  25. roleSessionName,
  26. externalId,
  27. token,
  28. webIdentityToken,
  29. action = 'AssumeRole',
  30. transportAgent = undefined
  31. }) {
  32. super({
  33. accessKey,
  34. secretKey,
  35. sessionToken
  36. });
  37. this.stsEndpoint = new URL(stsEndpoint);
  38. this.accessKey = accessKey;
  39. this.secretKey = secretKey;
  40. this.policy = policy;
  41. this.region = region;
  42. this.roleArn = roleArn;
  43. this.roleSessionName = roleSessionName;
  44. this.externalId = externalId;
  45. this.token = token;
  46. this.webIdentityToken = webIdentityToken;
  47. this.action = action;
  48. this.durationSeconds = parseInt(durationSeconds);
  49. let expirySeconds = this.durationSeconds;
  50. if (this.durationSeconds < defaultExpirySeconds) {
  51. expirySeconds = defaultExpirySeconds;
  52. }
  53. this.expirySeconds = expirySeconds; // for calculating refresh of credentials.
  54. // By default, nodejs uses a global agent if the 'agent' property
  55. // is set to undefined. Otherwise, it's okay to assume the users
  56. // know what they're doing if they specify a custom transport agent.
  57. this.transportAgent = transportAgent;
  58. const isHttp = this.stsEndpoint.protocol === 'http:';
  59. this.transport = isHttp ? http : https;
  60. /**
  61. * Internal Tracking variables
  62. */
  63. this._credentials = null;
  64. }
  65. getRequestConfig() {
  66. const hostValue = this.stsEndpoint.hostname;
  67. const portValue = this.stsEndpoint.port;
  68. const qryParams = new URLSearchParams({
  69. Action: this.action,
  70. Version: '2011-06-15'
  71. });
  72. qryParams.set('DurationSeconds', this.expirySeconds.toString());
  73. if (this.policy) {
  74. qryParams.set('Policy', this.policy);
  75. }
  76. if (this.roleArn) {
  77. qryParams.set('RoleArn', this.roleArn);
  78. }
  79. if (this.roleSessionName != null) {
  80. qryParams.set('RoleSessionName', this.roleSessionName);
  81. }
  82. if (this.token != null) {
  83. qryParams.set('Token', this.token);
  84. }
  85. if (this.webIdentityToken) {
  86. qryParams.set('WebIdentityToken', this.webIdentityToken);
  87. }
  88. if (this.externalId) {
  89. qryParams.set('ExternalId', this.externalId);
  90. }
  91. const urlParams = qryParams.toString();
  92. const contentSha256 = toSha256(urlParams);
  93. const date = new Date();
  94. const requestOptions = {
  95. hostname: hostValue,
  96. port: portValue,
  97. path: '/',
  98. protocol: this.stsEndpoint.protocol,
  99. method: 'POST',
  100. headers: {
  101. 'Content-Type': 'application/x-www-form-urlencoded',
  102. 'content-length': urlParams.length.toString(),
  103. host: hostValue,
  104. 'x-amz-date': makeDateLong(date),
  105. 'x-amz-content-sha256': contentSha256
  106. },
  107. agent: this.transportAgent
  108. };
  109. requestOptions.headers.authorization = signV4ByServiceName(requestOptions, this.accessKey, this.secretKey, this.region, date, contentSha256, 'sts');
  110. return {
  111. requestOptions,
  112. requestData: urlParams
  113. };
  114. }
  115. async performRequest() {
  116. const {
  117. requestOptions,
  118. requestData
  119. } = this.getRequestConfig();
  120. const res = await request(this.transport, requestOptions, requestData);
  121. const body = await readAsString(res);
  122. return parseXml(body);
  123. }
  124. parseCredentials(respObj) {
  125. if (respObj.ErrorResponse) {
  126. var _respObj$ErrorRespons, _respObj$ErrorRespons2, _respObj$ErrorRespons3, _respObj$ErrorRespons4;
  127. throw new Error(`Unable to obtain credentials: ${(_respObj$ErrorRespons = respObj.ErrorResponse) === null || _respObj$ErrorRespons === void 0 ? void 0 : (_respObj$ErrorRespons2 = _respObj$ErrorRespons.Error) === null || _respObj$ErrorRespons2 === void 0 ? void 0 : _respObj$ErrorRespons2.Code} ${(_respObj$ErrorRespons3 = respObj.ErrorResponse) === null || _respObj$ErrorRespons3 === void 0 ? void 0 : (_respObj$ErrorRespons4 = _respObj$ErrorRespons3.Error) === null || _respObj$ErrorRespons4 === void 0 ? void 0 : _respObj$ErrorRespons4.Message}`, {
  128. cause: respObj
  129. });
  130. }
  131. const {
  132. AssumeRoleResponse: {
  133. AssumeRoleResult: {
  134. Credentials: {
  135. AccessKeyId: accessKey,
  136. SecretAccessKey: secretKey,
  137. SessionToken: sessionToken,
  138. Expiration: expiresAt
  139. }
  140. }
  141. }
  142. } = respObj;
  143. this.accessExpiresAt = expiresAt;
  144. return new Credentials({
  145. accessKey,
  146. secretKey,
  147. sessionToken
  148. });
  149. }
  150. async refreshCredentials() {
  151. try {
  152. const assumeRoleCredentials = await this.performRequest();
  153. this._credentials = this.parseCredentials(assumeRoleCredentials);
  154. } catch (err) {
  155. throw new Error(`Failed to get Credentials: ${err}`, {
  156. cause: err
  157. });
  158. }
  159. return this._credentials;
  160. }
  161. async getCredentials() {
  162. if (this._credentials && !this.isAboutToExpire()) {
  163. return this._credentials;
  164. }
  165. this._credentials = await this.refreshCredentials();
  166. return this._credentials;
  167. }
  168. isAboutToExpire() {
  169. const expiresAt = new Date(this.accessExpiresAt);
  170. const provisionalExpiry = new Date(Date.now() + 1000 * 10); // check before 10 seconds.
  171. return provisionalExpiry > expiresAt;
  172. }
  173. }
  174. // deprecated default export, please use named exports.
  175. // keep for backward compatibility.
  176. // eslint-disable-next-line import/no-default-export
  177. export default AssumeRoleProvider;
  178. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["http","https","URL","URLSearchParams","CredentialProvider","Credentials","makeDateLong","parseXml","toSha256","request","readAsString","signV4ByServiceName","defaultExpirySeconds","AssumeRoleProvider","accessExpiresAt","constructor","stsEndpoint","accessKey","secretKey","durationSeconds","sessionToken","policy","region","roleArn","roleSessionName","externalId","token","webIdentityToken","action","transportAgent","undefined","parseInt","expirySeconds","isHttp","protocol","transport","_credentials","getRequestConfig","hostValue","hostname","portValue","port","qryParams","Action","Version","set","toString","urlParams","contentSha256","date","Date","requestOptions","path","method","headers","length","host","agent","authorization","requestData","performRequest","res","body","parseCredentials","respObj","ErrorResponse","_respObj$ErrorRespons","_respObj$ErrorRespons2","_respObj$ErrorRespons3","_respObj$ErrorRespons4","Error","Code","Message","cause","AssumeRoleResponse","AssumeRoleResult","AccessKeyId","SecretAccessKey","SessionToken","Expiration","expiresAt","refreshCredentials","assumeRoleCredentials","err","getCredentials","isAboutToExpire","provisionalExpiry","now"],"sources":["AssumeRoleProvider.ts"],"sourcesContent":["import * as http from 'node:http'\nimport * as https from 'node:https'\nimport { URL, URLSearchParams } from 'node:url'\n\nimport { CredentialProvider } from './CredentialProvider.ts'\nimport { Credentials } from './Credentials.ts'\nimport { makeDateLong, parseXml, toSha256 } from './internal/helper.ts'\nimport { request } from './internal/request.ts'\nimport { readAsString } from './internal/response.ts'\nimport type { Transport } from './internal/type.ts'\nimport { signV4ByServiceName } from './signing.ts'\n\n/**\n * @see https://docs.aws.amazon.com/STS/latest/APIReference/API_AssumeRole.html\n */\ntype CredentialResponse = {\n  ErrorResponse?: {\n    Error?: {\n      Code?: string\n      Message?: string\n    }\n  }\n\n  AssumeRoleResponse: {\n    AssumeRoleResult: {\n      Credentials: {\n        AccessKeyId: string\n        SecretAccessKey: string\n        SessionToken: string\n        Expiration: string\n      }\n    }\n  }\n}\n\nexport interface AssumeRoleProviderOptions {\n  stsEndpoint: string\n  accessKey: string\n  secretKey: string\n  durationSeconds?: number\n  sessionToken?: string\n  policy?: string\n  region?: string\n  roleArn?: string\n  roleSessionName?: string\n  externalId?: string\n  token?: string\n  webIdentityToken?: string\n  action?: string\n  transportAgent?: http.Agent\n}\n\nconst defaultExpirySeconds = 900\n\nexport class AssumeRoleProvider extends CredentialProvider {\n  private readonly stsEndpoint: URL\n  private readonly accessKey: string\n  private readonly secretKey: string\n  private readonly durationSeconds: number\n  private readonly policy?: string\n  private readonly region: string\n  private readonly roleArn?: string\n  private readonly roleSessionName?: string\n  private readonly externalId?: string\n  private readonly token?: string\n  private readonly webIdentityToken?: string\n  private readonly action: string\n\n  private _credentials: Credentials | null\n  private readonly expirySeconds: number\n  private accessExpiresAt = ''\n  private readonly transportAgent?: http.Agent\n\n  private readonly transport: Transport\n\n  constructor({\n    stsEndpoint,\n    accessKey,\n    secretKey,\n    durationSeconds = defaultExpirySeconds,\n    sessionToken,\n    policy,\n    region = '',\n    roleArn,\n    roleSessionName,\n    externalId,\n    token,\n    webIdentityToken,\n    action = 'AssumeRole',\n    transportAgent = undefined,\n  }: AssumeRoleProviderOptions) {\n    super({ accessKey, secretKey, sessionToken })\n\n    this.stsEndpoint = new URL(stsEndpoint)\n    this.accessKey = accessKey\n    this.secretKey = secretKey\n    this.policy = policy\n    this.region = region\n    this.roleArn = roleArn\n    this.roleSessionName = roleSessionName\n    this.externalId = externalId\n    this.token = token\n    this.webIdentityToken = webIdentityToken\n    this.action = action\n\n    this.durationSeconds = parseInt(durationSeconds as unknown as string)\n\n    let expirySeconds = this.durationSeconds\n    if (this.durationSeconds < defaultExpirySeconds) {\n      expirySeconds = defaultExpirySeconds\n    }\n    this.expirySeconds = expirySeconds // for calculating refresh of credentials.\n\n    // By default, nodejs uses a global agent if the 'agent' property\n    // is set to undefined. Otherwise, it's okay to assume the users\n    // know what they're doing if they specify a custom transport agent.\n    this.transportAgent = transportAgent\n    const isHttp: boolean = this.stsEndpoint.protocol === 'http:'\n    this.transport = isHttp ? http : https\n\n    /**\n     * Internal Tracking variables\n     */\n    this._credentials = null\n  }\n\n  getRequestConfig(): {\n    requestOptions: http.RequestOptions\n    requestData: string\n  } {\n    const hostValue = this.stsEndpoint.hostname\n    const portValue = this.stsEndpoint.port\n    const qryParams = new URLSearchParams({ Action: this.action, Version: '2011-06-15' })\n\n    qryParams.set('DurationSeconds', this.expirySeconds.toString())\n\n    if (this.policy) {\n      qryParams.set('Policy', this.policy)\n    }\n    if (this.roleArn) {\n      qryParams.set('RoleArn', this.roleArn)\n    }\n\n    if (this.roleSessionName != null) {\n      qryParams.set('RoleSessionName', this.roleSessionName)\n    }\n    if (this.token != null) {\n      qryParams.set('Token', this.token)\n    }\n\n    if (this.webIdentityToken) {\n      qryParams.set('WebIdentityToken', this.webIdentityToken)\n    }\n\n    if (this.externalId) {\n      qryParams.set('ExternalId', this.externalId)\n    }\n\n    const urlParams = qryParams.toString()\n    const contentSha256 = toSha256(urlParams)\n\n    const date = new Date()\n\n    const requestOptions = {\n      hostname: hostValue,\n      port: portValue,\n      path: '/',\n      protocol: this.stsEndpoint.protocol,\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/x-www-form-urlencoded',\n        'content-length': urlParams.length.toString(),\n        host: hostValue,\n        'x-amz-date': makeDateLong(date),\n        'x-amz-content-sha256': contentSha256,\n      } as Record<string, string>,\n      agent: this.transportAgent,\n    } satisfies http.RequestOptions\n\n    requestOptions.headers.authorization = signV4ByServiceName(\n      requestOptions,\n      this.accessKey,\n      this.secretKey,\n      this.region,\n      date,\n      contentSha256,\n      'sts',\n    )\n\n    return {\n      requestOptions,\n      requestData: urlParams,\n    }\n  }\n\n  async performRequest(): Promise<CredentialResponse> {\n    const { requestOptions, requestData } = this.getRequestConfig()\n\n    const res = await request(this.transport, requestOptions, requestData)\n\n    const body = await readAsString(res)\n\n    return parseXml(body)\n  }\n\n  parseCredentials(respObj: CredentialResponse): Credentials {\n    if (respObj.ErrorResponse) {\n      throw new Error(\n        `Unable to obtain credentials: ${respObj.ErrorResponse?.Error?.Code} ${respObj.ErrorResponse?.Error?.Message}`,\n        { cause: respObj },\n      )\n    }\n\n    const {\n      AssumeRoleResponse: {\n        AssumeRoleResult: {\n          Credentials: {\n            AccessKeyId: accessKey,\n            SecretAccessKey: secretKey,\n            SessionToken: sessionToken,\n            Expiration: expiresAt,\n          },\n        },\n      },\n    } = respObj\n\n    this.accessExpiresAt = expiresAt\n\n    return new Credentials({ accessKey, secretKey, sessionToken })\n  }\n\n  async refreshCredentials(): Promise<Credentials> {\n    try {\n      const assumeRoleCredentials = await this.performRequest()\n      this._credentials = this.parseCredentials(assumeRoleCredentials)\n    } catch (err) {\n      throw new Error(`Failed to get Credentials: ${err}`, { cause: err })\n    }\n\n    return this._credentials\n  }\n\n  async getCredentials(): Promise<Credentials> {\n    if (this._credentials && !this.isAboutToExpire()) {\n      return this._credentials\n    }\n\n    this._credentials = await this.refreshCredentials()\n    return this._credentials\n  }\n\n  isAboutToExpire() {\n    const expiresAt = new Date(this.accessExpiresAt)\n    const provisionalExpiry = new Date(Date.now() + 1000 * 10) // check before 10 seconds.\n    return provisionalExpiry > expiresAt\n  }\n}\n\n// deprecated default export, please use named exports.\n// keep for backward compatibility.\n// eslint-disable-next-line import/no-default-export\nexport default AssumeRoleProvider\n"],"mappings":"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"}