signing.mjs 40 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. /*
  2. * MinIO Javascript Library for Amazon S3 Compatible Cloud Storage, (C) 2016 MinIO, Inc.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. import * as crypto from "crypto";
  17. import * as errors from "./errors.mjs";
  18. import { PRESIGN_EXPIRY_DAYS_MAX } from "./helpers.mjs";
  19. import { getScope, isNumber, isObject, isString, makeDateLong, makeDateShort, uriEscape } from "./internal/helper.mjs";
  20. const signV4Algorithm = 'AWS4-HMAC-SHA256';
  21. // getCanonicalRequest generate a canonical request of style.
  22. //
  23. // canonicalRequest =
  24. // <HTTPMethod>\n
  25. // <CanonicalURI>\n
  26. // <CanonicalQueryString>\n
  27. // <CanonicalHeaders>\n
  28. // <SignedHeaders>\n
  29. // <HashedPayload>
  30. //
  31. function getCanonicalRequest(method, path, headers, signedHeaders, hashedPayload) {
  32. if (!isString(method)) {
  33. throw new TypeError('method should be of type "string"');
  34. }
  35. if (!isString(path)) {
  36. throw new TypeError('path should be of type "string"');
  37. }
  38. if (!isObject(headers)) {
  39. throw new TypeError('headers should be of type "object"');
  40. }
  41. if (!Array.isArray(signedHeaders)) {
  42. throw new TypeError('signedHeaders should be of type "array"');
  43. }
  44. if (!isString(hashedPayload)) {
  45. throw new TypeError('hashedPayload should be of type "string"');
  46. }
  47. const headersArray = signedHeaders.reduce((acc, i) => {
  48. // Trim spaces from the value (required by V4 spec)
  49. const val = `${headers[i]}`.replace(/ +/g, ' ');
  50. acc.push(`${i.toLowerCase()}:${val}`);
  51. return acc;
  52. }, []);
  53. const requestResource = path.split('?')[0];
  54. let requestQuery = path.split('?')[1];
  55. if (!requestQuery) {
  56. requestQuery = '';
  57. }
  58. if (requestQuery) {
  59. requestQuery = requestQuery.split('&').sort().map(element => !element.includes('=') ? element + '=' : element).join('&');
  60. }
  61. return [method.toUpperCase(), requestResource, requestQuery, headersArray.join('\n') + '\n', signedHeaders.join(';').toLowerCase(), hashedPayload].join('\n');
  62. }
  63. // generate a credential string
  64. function getCredential(accessKey, region, requestDate, serviceName = 's3') {
  65. if (!isString(accessKey)) {
  66. throw new TypeError('accessKey should be of type "string"');
  67. }
  68. if (!isString(region)) {
  69. throw new TypeError('region should be of type "string"');
  70. }
  71. if (!isObject(requestDate)) {
  72. throw new TypeError('requestDate should be of type "object"');
  73. }
  74. return `${accessKey}/${getScope(region, requestDate, serviceName)}`;
  75. }
  76. // Returns signed headers array - alphabetically sorted
  77. function getSignedHeaders(headers) {
  78. if (!isObject(headers)) {
  79. throw new TypeError('request should be of type "object"');
  80. }
  81. // Excerpts from @lsegal - https://github.com/aws/aws-sdk-js/issues/659#issuecomment-120477258
  82. //
  83. // User-Agent:
  84. //
  85. // This is ignored from signing because signing this causes problems with generating pre-signed URLs
  86. // (that are executed by other agents) or when customers pass requests through proxies, which may
  87. // modify the user-agent.
  88. //
  89. // Content-Length:
  90. //
  91. // This is ignored from signing because generating a pre-signed URL should not provide a content-length
  92. // constraint, specifically when vending a S3 pre-signed PUT URL. The corollary to this is that when
  93. // sending regular requests (non-pre-signed), the signature contains a checksum of the body, which
  94. // implicitly validates the payload length (since changing the number of bytes would change the checksum)
  95. // and therefore this header is not valuable in the signature.
  96. //
  97. // Content-Type:
  98. //
  99. // Signing this header causes quite a number of problems in browser environments, where browsers
  100. // like to modify and normalize the content-type header in different ways. There is more information
  101. // on this in https://github.com/aws/aws-sdk-js/issues/244. Avoiding this field simplifies logic
  102. // and reduces the possibility of future bugs
  103. //
  104. // Authorization:
  105. //
  106. // Is skipped for obvious reasons
  107. const ignoredHeaders = ['authorization', 'content-length', 'content-type', 'user-agent'];
  108. return Object.keys(headers).filter(header => !ignoredHeaders.includes(header)).sort();
  109. }
  110. // returns the key used for calculating signature
  111. function getSigningKey(date, region, secretKey, serviceName = 's3') {
  112. if (!isObject(date)) {
  113. throw new TypeError('date should be of type "object"');
  114. }
  115. if (!isString(region)) {
  116. throw new TypeError('region should be of type "string"');
  117. }
  118. if (!isString(secretKey)) {
  119. throw new TypeError('secretKey should be of type "string"');
  120. }
  121. const dateLine = makeDateShort(date);
  122. const hmac1 = crypto.createHmac('sha256', 'AWS4' + secretKey).update(dateLine).digest(),
  123. hmac2 = crypto.createHmac('sha256', hmac1).update(region).digest(),
  124. hmac3 = crypto.createHmac('sha256', hmac2).update(serviceName).digest();
  125. return crypto.createHmac('sha256', hmac3).update('aws4_request').digest();
  126. }
  127. // returns the string that needs to be signed
  128. function getStringToSign(canonicalRequest, requestDate, region, serviceName = 's3') {
  129. if (!isString(canonicalRequest)) {
  130. throw new TypeError('canonicalRequest should be of type "string"');
  131. }
  132. if (!isObject(requestDate)) {
  133. throw new TypeError('requestDate should be of type "object"');
  134. }
  135. if (!isString(region)) {
  136. throw new TypeError('region should be of type "string"');
  137. }
  138. const hash = crypto.createHash('sha256').update(canonicalRequest).digest('hex');
  139. const scope = getScope(region, requestDate, serviceName);
  140. const stringToSign = [signV4Algorithm, makeDateLong(requestDate), scope, hash];
  141. return stringToSign.join('\n');
  142. }
  143. // calculate the signature of the POST policy
  144. export function postPresignSignatureV4(region, date, secretKey, policyBase64) {
  145. if (!isString(region)) {
  146. throw new TypeError('region should be of type "string"');
  147. }
  148. if (!isObject(date)) {
  149. throw new TypeError('date should be of type "object"');
  150. }
  151. if (!isString(secretKey)) {
  152. throw new TypeError('secretKey should be of type "string"');
  153. }
  154. if (!isString(policyBase64)) {
  155. throw new TypeError('policyBase64 should be of type "string"');
  156. }
  157. const signingKey = getSigningKey(date, region, secretKey);
  158. return crypto.createHmac('sha256', signingKey).update(policyBase64).digest('hex').toLowerCase();
  159. }
  160. // Returns the authorization header
  161. export function signV4(request, accessKey, secretKey, region, requestDate, sha256sum, serviceName = 's3') {
  162. if (!isObject(request)) {
  163. throw new TypeError('request should be of type "object"');
  164. }
  165. if (!isString(accessKey)) {
  166. throw new TypeError('accessKey should be of type "string"');
  167. }
  168. if (!isString(secretKey)) {
  169. throw new TypeError('secretKey should be of type "string"');
  170. }
  171. if (!isString(region)) {
  172. throw new TypeError('region should be of type "string"');
  173. }
  174. if (!accessKey) {
  175. throw new errors.AccessKeyRequiredError('accessKey is required for signing');
  176. }
  177. if (!secretKey) {
  178. throw new errors.SecretKeyRequiredError('secretKey is required for signing');
  179. }
  180. const signedHeaders = getSignedHeaders(request.headers);
  181. const canonicalRequest = getCanonicalRequest(request.method, request.path, request.headers, signedHeaders, sha256sum);
  182. const serviceIdentifier = serviceName || 's3';
  183. const stringToSign = getStringToSign(canonicalRequest, requestDate, region, serviceIdentifier);
  184. const signingKey = getSigningKey(requestDate, region, secretKey, serviceIdentifier);
  185. const credential = getCredential(accessKey, region, requestDate, serviceIdentifier);
  186. const signature = crypto.createHmac('sha256', signingKey).update(stringToSign).digest('hex').toLowerCase();
  187. return `${signV4Algorithm} Credential=${credential}, SignedHeaders=${signedHeaders.join(';').toLowerCase()}, Signature=${signature}`;
  188. }
  189. export function signV4ByServiceName(request, accessKey, secretKey, region, requestDate, contentSha256, serviceName = 's3') {
  190. return signV4(request, accessKey, secretKey, region, requestDate, contentSha256, serviceName);
  191. }
  192. // returns a presigned URL string
  193. export function presignSignatureV4(request, accessKey, secretKey, sessionToken, region, requestDate, expires) {
  194. if (!isObject(request)) {
  195. throw new TypeError('request should be of type "object"');
  196. }
  197. if (!isString(accessKey)) {
  198. throw new TypeError('accessKey should be of type "string"');
  199. }
  200. if (!isString(secretKey)) {
  201. throw new TypeError('secretKey should be of type "string"');
  202. }
  203. if (!isString(region)) {
  204. throw new TypeError('region should be of type "string"');
  205. }
  206. if (!accessKey) {
  207. throw new errors.AccessKeyRequiredError('accessKey is required for presigning');
  208. }
  209. if (!secretKey) {
  210. throw new errors.SecretKeyRequiredError('secretKey is required for presigning');
  211. }
  212. if (expires && !isNumber(expires)) {
  213. throw new TypeError('expires should be of type "number"');
  214. }
  215. if (expires && expires < 1) {
  216. throw new errors.ExpiresParamError('expires param cannot be less than 1 seconds');
  217. }
  218. if (expires && expires > PRESIGN_EXPIRY_DAYS_MAX) {
  219. throw new errors.ExpiresParamError('expires param cannot be greater than 7 days');
  220. }
  221. const iso8601Date = makeDateLong(requestDate);
  222. const signedHeaders = getSignedHeaders(request.headers);
  223. const credential = getCredential(accessKey, region, requestDate);
  224. const hashedPayload = 'UNSIGNED-PAYLOAD';
  225. const requestQuery = [];
  226. requestQuery.push(`X-Amz-Algorithm=${signV4Algorithm}`);
  227. requestQuery.push(`X-Amz-Credential=${uriEscape(credential)}`);
  228. requestQuery.push(`X-Amz-Date=${iso8601Date}`);
  229. requestQuery.push(`X-Amz-Expires=${expires}`);
  230. requestQuery.push(`X-Amz-SignedHeaders=${uriEscape(signedHeaders.join(';').toLowerCase())}`);
  231. if (sessionToken) {
  232. requestQuery.push(`X-Amz-Security-Token=${uriEscape(sessionToken)}`);
  233. }
  234. const resource = request.path.split('?')[0];
  235. let query = request.path.split('?')[1];
  236. if (query) {
  237. query = query + '&' + requestQuery.join('&');
  238. } else {
  239. query = requestQuery.join('&');
  240. }
  241. const path = resource + '?' + query;
  242. const canonicalRequest = getCanonicalRequest(request.method, path, request.headers, signedHeaders, hashedPayload);
  243. const stringToSign = getStringToSign(canonicalRequest, requestDate, region);
  244. const signingKey = getSigningKey(requestDate, region, secretKey);
  245. const signature = crypto.createHmac('sha256', signingKey).update(stringToSign).digest('hex').toLowerCase();
  246. return request.protocol + '//' + request.headers.host + path + `&X-Amz-Signature=${signature}`;
  247. }
  248. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["crypto","errors","PRESIGN_EXPIRY_DAYS_MAX","getScope","isNumber","isObject","isString","makeDateLong","makeDateShort","uriEscape","signV4Algorithm","getCanonicalRequest","method","path","headers","signedHeaders","hashedPayload","TypeError","Array","isArray","headersArray","reduce","acc","i","val","replace","push","toLowerCase","requestResource","split","requestQuery","sort","map","element","includes","join","toUpperCase","getCredential","accessKey","region","requestDate","serviceName","getSignedHeaders","ignoredHeaders","Object","keys","filter","header","getSigningKey","date","secretKey","dateLine","hmac1","createHmac","update","digest","hmac2","hmac3","getStringToSign","canonicalRequest","hash","createHash","scope","stringToSign","postPresignSignatureV4","policyBase64","signingKey","signV4","request","sha256sum","AccessKeyRequiredError","SecretKeyRequiredError","serviceIdentifier","credential","signature","signV4ByServiceName","contentSha256","presignSignatureV4","sessionToken","expires","ExpiresParamError","iso8601Date","resource","query","protocol","host"],"sources":["signing.ts"],"sourcesContent":["/*\n * MinIO Javascript Library for Amazon S3 Compatible Cloud Storage, (C) 2016 MinIO, Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n *     http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as crypto from 'node:crypto'\n\nimport * as errors from './errors.ts'\nimport { PRESIGN_EXPIRY_DAYS_MAX } from './helpers.ts'\nimport { getScope, isNumber, isObject, isString, makeDateLong, makeDateShort, uriEscape } from './internal/helper.ts'\nimport type { ICanonicalRequest, IRequest, RequestHeaders } from './internal/type.ts'\n\nconst signV4Algorithm = 'AWS4-HMAC-SHA256'\n\n// getCanonicalRequest generate a canonical request of style.\n//\n// canonicalRequest =\n//  <HTTPMethod>\\n\n//  <CanonicalURI>\\n\n//  <CanonicalQueryString>\\n\n//  <CanonicalHeaders>\\n\n//  <SignedHeaders>\\n\n//  <HashedPayload>\n//\nfunction getCanonicalRequest(\n  method: string,\n  path: string,\n  headers: RequestHeaders,\n  signedHeaders: string[],\n  hashedPayload: string,\n): ICanonicalRequest {\n  if (!isString(method)) {\n    throw new TypeError('method should be of type \"string\"')\n  }\n  if (!isString(path)) {\n    throw new TypeError('path should be of type \"string\"')\n  }\n  if (!isObject(headers)) {\n    throw new TypeError('headers should be of type \"object\"')\n  }\n  if (!Array.isArray(signedHeaders)) {\n    throw new TypeError('signedHeaders should be of type \"array\"')\n  }\n  if (!isString(hashedPayload)) {\n    throw new TypeError('hashedPayload should be of type \"string\"')\n  }\n\n  const headersArray = signedHeaders.reduce((acc, i) => {\n    // Trim spaces from the value (required by V4 spec)\n    const val = `${headers[i]}`.replace(/ +/g, ' ')\n    acc.push(`${i.toLowerCase()}:${val}`)\n    return acc\n  }, [] as string[])\n\n  const requestResource = path.split('?')[0]\n  let requestQuery = path.split('?')[1]\n  if (!requestQuery) {\n    requestQuery = ''\n  }\n\n  if (requestQuery) {\n    requestQuery = requestQuery\n      .split('&')\n      .sort()\n      .map((element) => (!element.includes('=') ? element + '=' : element))\n      .join('&')\n  }\n\n  return [\n    method.toUpperCase(),\n    requestResource,\n    requestQuery,\n    headersArray.join('\\n') + '\\n',\n    signedHeaders.join(';').toLowerCase(),\n    hashedPayload,\n  ].join('\\n')\n}\n\n// generate a credential string\nfunction getCredential(accessKey: string, region: string, requestDate?: Date, serviceName = 's3') {\n  if (!isString(accessKey)) {\n    throw new TypeError('accessKey should be of type \"string\"')\n  }\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n  if (!isObject(requestDate)) {\n    throw new TypeError('requestDate should be of type \"object\"')\n  }\n  return `${accessKey}/${getScope(region, requestDate, serviceName)}`\n}\n\n// Returns signed headers array - alphabetically sorted\nfunction getSignedHeaders(headers: RequestHeaders): string[] {\n  if (!isObject(headers)) {\n    throw new TypeError('request should be of type \"object\"')\n  }\n  // Excerpts from @lsegal - https://github.com/aws/aws-sdk-js/issues/659#issuecomment-120477258\n  //\n  //  User-Agent:\n  //\n  //      This is ignored from signing because signing this causes problems with generating pre-signed URLs\n  //      (that are executed by other agents) or when customers pass requests through proxies, which may\n  //      modify the user-agent.\n  //\n  //  Content-Length:\n  //\n  //      This is ignored from signing because generating a pre-signed URL should not provide a content-length\n  //      constraint, specifically when vending a S3 pre-signed PUT URL. The corollary to this is that when\n  //      sending regular requests (non-pre-signed), the signature contains a checksum of the body, which\n  //      implicitly validates the payload length (since changing the number of bytes would change the checksum)\n  //      and therefore this header is not valuable in the signature.\n  //\n  //  Content-Type:\n  //\n  //      Signing this header causes quite a number of problems in browser environments, where browsers\n  //      like to modify and normalize the content-type header in different ways. There is more information\n  //      on this in https://github.com/aws/aws-sdk-js/issues/244. Avoiding this field simplifies logic\n  //      and reduces the possibility of future bugs\n  //\n  //  Authorization:\n  //\n  //      Is skipped for obvious reasons\n\n  const ignoredHeaders = ['authorization', 'content-length', 'content-type', 'user-agent']\n  return Object.keys(headers)\n    .filter((header) => !ignoredHeaders.includes(header))\n    .sort()\n}\n\n// returns the key used for calculating signature\nfunction getSigningKey(date: Date, region: string, secretKey: string, serviceName = 's3') {\n  if (!isObject(date)) {\n    throw new TypeError('date should be of type \"object\"')\n  }\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n  if (!isString(secretKey)) {\n    throw new TypeError('secretKey should be of type \"string\"')\n  }\n  const dateLine = makeDateShort(date)\n  const hmac1 = crypto\n      .createHmac('sha256', 'AWS4' + secretKey)\n      .update(dateLine)\n      .digest(),\n    hmac2 = crypto.createHmac('sha256', hmac1).update(region).digest(),\n    hmac3 = crypto.createHmac('sha256', hmac2).update(serviceName).digest()\n  return crypto.createHmac('sha256', hmac3).update('aws4_request').digest()\n}\n\n// returns the string that needs to be signed\nfunction getStringToSign(canonicalRequest: ICanonicalRequest, requestDate: Date, region: string, serviceName = 's3') {\n  if (!isString(canonicalRequest)) {\n    throw new TypeError('canonicalRequest should be of type \"string\"')\n  }\n  if (!isObject(requestDate)) {\n    throw new TypeError('requestDate should be of type \"object\"')\n  }\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n  const hash = crypto.createHash('sha256').update(canonicalRequest).digest('hex')\n  const scope = getScope(region, requestDate, serviceName)\n  const stringToSign = [signV4Algorithm, makeDateLong(requestDate), scope, hash]\n\n  return stringToSign.join('\\n')\n}\n\n// calculate the signature of the POST policy\nexport function postPresignSignatureV4(region: string, date: Date, secretKey: string, policyBase64: string): string {\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n  if (!isObject(date)) {\n    throw new TypeError('date should be of type \"object\"')\n  }\n  if (!isString(secretKey)) {\n    throw new TypeError('secretKey should be of type \"string\"')\n  }\n  if (!isString(policyBase64)) {\n    throw new TypeError('policyBase64 should be of type \"string\"')\n  }\n  const signingKey = getSigningKey(date, region, secretKey)\n  return crypto.createHmac('sha256', signingKey).update(policyBase64).digest('hex').toLowerCase()\n}\n\n// Returns the authorization header\nexport function signV4(\n  request: IRequest,\n  accessKey: string,\n  secretKey: string,\n  region: string,\n  requestDate: Date,\n  sha256sum: string,\n  serviceName = 's3',\n) {\n  if (!isObject(request)) {\n    throw new TypeError('request should be of type \"object\"')\n  }\n  if (!isString(accessKey)) {\n    throw new TypeError('accessKey should be of type \"string\"')\n  }\n  if (!isString(secretKey)) {\n    throw new TypeError('secretKey should be of type \"string\"')\n  }\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n\n  if (!accessKey) {\n    throw new errors.AccessKeyRequiredError('accessKey is required for signing')\n  }\n  if (!secretKey) {\n    throw new errors.SecretKeyRequiredError('secretKey is required for signing')\n  }\n\n  const signedHeaders = getSignedHeaders(request.headers)\n  const canonicalRequest = getCanonicalRequest(request.method, request.path, request.headers, signedHeaders, sha256sum)\n  const serviceIdentifier = serviceName || 's3'\n  const stringToSign = getStringToSign(canonicalRequest, requestDate, region, serviceIdentifier)\n  const signingKey = getSigningKey(requestDate, region, secretKey, serviceIdentifier)\n  const credential = getCredential(accessKey, region, requestDate, serviceIdentifier)\n  const signature = crypto.createHmac('sha256', signingKey).update(stringToSign).digest('hex').toLowerCase()\n\n  return `${signV4Algorithm} Credential=${credential}, SignedHeaders=${signedHeaders\n    .join(';')\n    .toLowerCase()}, Signature=${signature}`\n}\n\nexport function signV4ByServiceName(\n  request: IRequest,\n  accessKey: string,\n  secretKey: string,\n  region: string,\n  requestDate: Date,\n  contentSha256: string,\n  serviceName = 's3',\n): string {\n  return signV4(request, accessKey, secretKey, region, requestDate, contentSha256, serviceName)\n}\n\n// returns a presigned URL string\nexport function presignSignatureV4(\n  request: IRequest,\n  accessKey: string,\n  secretKey: string,\n  sessionToken: string | undefined,\n  region: string,\n  requestDate: Date,\n  expires: number | undefined,\n) {\n  if (!isObject(request)) {\n    throw new TypeError('request should be of type \"object\"')\n  }\n  if (!isString(accessKey)) {\n    throw new TypeError('accessKey should be of type \"string\"')\n  }\n  if (!isString(secretKey)) {\n    throw new TypeError('secretKey should be of type \"string\"')\n  }\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n\n  if (!accessKey) {\n    throw new errors.AccessKeyRequiredError('accessKey is required for presigning')\n  }\n  if (!secretKey) {\n    throw new errors.SecretKeyRequiredError('secretKey is required for presigning')\n  }\n\n  if (expires && !isNumber(expires)) {\n    throw new TypeError('expires should be of type \"number\"')\n  }\n  if (expires && expires < 1) {\n    throw new errors.ExpiresParamError('expires param cannot be less than 1 seconds')\n  }\n  if (expires && expires > PRESIGN_EXPIRY_DAYS_MAX) {\n    throw new errors.ExpiresParamError('expires param cannot be greater than 7 days')\n  }\n\n  const iso8601Date = makeDateLong(requestDate)\n  const signedHeaders = getSignedHeaders(request.headers)\n  const credential = getCredential(accessKey, region, requestDate)\n  const hashedPayload = 'UNSIGNED-PAYLOAD'\n\n  const requestQuery: string[] = []\n  requestQuery.push(`X-Amz-Algorithm=${signV4Algorithm}`)\n  requestQuery.push(`X-Amz-Credential=${uriEscape(credential)}`)\n  requestQuery.push(`X-Amz-Date=${iso8601Date}`)\n  requestQuery.push(`X-Amz-Expires=${expires}`)\n  requestQuery.push(`X-Amz-SignedHeaders=${uriEscape(signedHeaders.join(';').toLowerCase())}`)\n  if (sessionToken) {\n    requestQuery.push(`X-Amz-Security-Token=${uriEscape(sessionToken)}`)\n  }\n\n  const resource = request.path.split('?')[0]\n  let query = request.path.split('?')[1]\n  if (query) {\n    query = query + '&' + requestQuery.join('&')\n  } else {\n    query = requestQuery.join('&')\n  }\n\n  const path = resource + '?' + query\n\n  const canonicalRequest = getCanonicalRequest(request.method, path, request.headers, signedHeaders, hashedPayload)\n\n  const stringToSign = getStringToSign(canonicalRequest, requestDate, region)\n  const signingKey = getSigningKey(requestDate, region, secretKey)\n  const signature = crypto.createHmac('sha256', signingKey).update(stringToSign).digest('hex').toLowerCase()\n  return request.protocol + '//' + request.headers.host + path + `&X-Amz-Signature=${signature}`\n}\n"],"mappings":"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"}