@bitgo-beta/sdk-hmac 1.0.1-beta.97 → 1.0.1-beta.970

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,3 +1,4 @@
1
+ import { type BinaryLike, type KeyObject } from 'crypto';
1
2
  import { CalculateHmacSubjectOptions, CalculateRequestHeadersOptions, CalculateRequestHmacOptions, RequestHeaders, VerifyResponseInfo, VerifyResponseOptions } from './types';
2
3
  /**
3
4
  * Calculate the HMAC for the given key and message
@@ -5,7 +6,7 @@ import { CalculateHmacSubjectOptions, CalculateRequestHeadersOptions, CalculateR
5
6
  * @param message {String} - the actual message to HMAC
6
7
  * @returns {*} - the result of the HMAC operation
7
8
  */
8
- export declare function calculateHMAC(key: string, message: string): string;
9
+ export declare function calculateHMAC(key: string | BinaryLike | KeyObject, message: string | BinaryLike): string;
9
10
  /**
10
11
  * Calculate the subject string that is to be HMAC'ed for a HTTP request or response
11
12
  * @param urlPath request url, including query params
@@ -13,19 +14,21 @@ export declare function calculateHMAC(key: string, message: string): string;
13
14
  * @param timestamp request timestamp from `Date.now()`
14
15
  * @param statusCode Only set for HTTP responses, leave blank for requests
15
16
  * @param method request method
16
- * @returns {string}
17
+ * @param authVersion authentication version (2 or 3)
18
+ * @param useOriginalPath whether to use the original urlPath without parsing (default false)
19
+ * @returns {string | Buffer}
17
20
  */
18
- export declare function calculateHMACSubject({ urlPath, text, timestamp, statusCode, method, authVersion, }: CalculateHmacSubjectOptions): string;
21
+ export declare function calculateHMACSubject<T extends string | Buffer = string>({ urlPath, text, timestamp, statusCode, method, authVersion }: CalculateHmacSubjectOptions<T>, useOriginalPath?: boolean): T;
19
22
  /**
20
23
  * Calculate the HMAC for an HTTP request
21
24
  */
22
- export declare function calculateRequestHMAC({ url: urlPath, text, timestamp, token, method, authVersion, }: CalculateRequestHmacOptions): string;
25
+ export declare function calculateRequestHMAC<T extends string | Buffer = string>({ url: urlPath, text, timestamp, token, method, authVersion }: CalculateRequestHmacOptions<T>, useOriginalPath?: boolean): string;
23
26
  /**
24
27
  * Calculate request headers with HMAC
25
28
  */
26
- export declare function calculateRequestHeaders({ url, text, token, method, authVersion, }: CalculateRequestHeadersOptions): RequestHeaders;
29
+ export declare function calculateRequestHeaders<T extends string | Buffer = string>({ url, text, token, method, authVersion }: CalculateRequestHeadersOptions<T>, useOriginalPath?: boolean): RequestHeaders;
27
30
  /**
28
31
  * Verify the HMAC for an HTTP response
29
32
  */
30
- export declare function verifyResponse({ url: urlPath, statusCode, text, timestamp, token, hmac, method, authVersion, }: VerifyResponseOptions): VerifyResponseInfo;
33
+ export declare function verifyResponse<T extends string | Buffer = string>({ url: urlPath, statusCode, text, timestamp, token, hmac, method, authVersion }: VerifyResponseOptions<T>, useOriginalPath?: boolean): VerifyResponseInfo<T>;
31
34
  //# sourceMappingURL=hmac.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"hmac.d.ts","sourceRoot":"","sources":["../../src/hmac.ts"],"names":[],"mappings":"AAGA,OAAO,EACL,2BAA2B,EAC3B,8BAA8B,EAC9B,2BAA2B,EAC3B,cAAc,EACd,kBAAkB,EAClB,qBAAqB,EACtB,MAAM,SAAS,CAAC;AAEjB;;;;;GAKG;AACH,wBAAgB,aAAa,CAAC,GAAG,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,MAAM,CAElE;AAED;;;;;;;;GAQG;AACH,wBAAgB,oBAAoB,CAAC,EACnC,OAAO,EACP,IAAI,EACJ,SAAS,EACT,UAAU,EACV,MAAM,EACN,WAAW,GACZ,EAAE,2BAA2B,GAAG,MAAM,CAatC;AAED;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,EACnC,GAAG,EAAE,OAAO,EACZ,IAAI,EACJ,SAAS,EACT,KAAK,EACL,MAAM,EACN,WAAW,GACZ,EAAE,2BAA2B,GAAG,MAAM,CAKtC;AAED;;GAEG;AACH,wBAAgB,uBAAuB,CAAC,EACtC,GAAG,EACH,IAAI,EACJ,KAAK,EACL,MAAM,EACN,WAAW,GACZ,EAAE,8BAA8B,GAAG,cAAc,CAYjD;AAED;;GAEG;AACH,wBAAgB,cAAc,CAAC,EAC7B,GAAG,EAAE,OAAO,EACZ,UAAU,EACV,IAAI,EACJ,SAAS,EACT,KAAK,EACL,IAAI,EACJ,MAAM,EACN,WAAW,GACZ,EAAE,qBAAqB,GAAG,kBAAkB,CAyB5C"}
1
+ {"version":3,"file":"hmac.d.ts","sourceRoot":"","sources":["../../src/hmac.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,UAAU,EAAE,KAAK,SAAS,EAAE,MAAM,QAAQ,CAAC;AAGzD,OAAO,EACL,2BAA2B,EAC3B,8BAA8B,EAC9B,2BAA2B,EAC3B,cAAc,EACd,kBAAkB,EAClB,qBAAqB,EACtB,MAAM,SAAS,CAAC;AAGjB;;;;;GAKG;AACH,wBAAgB,aAAa,CAAC,GAAG,EAAE,MAAM,GAAG,UAAU,GAAG,SAAS,EAAE,OAAO,EAAE,MAAM,GAAG,UAAU,GAAG,MAAM,CAExG;AAED;;;;;;;;;;GAUG;AACH,wBAAgB,oBAAoB,CAAC,CAAC,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,EACrE,EAAE,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,EAAE,WAAW,EAAE,EAAE,2BAA2B,CAAC,CAAC,CAAC,EAC7F,eAAe,UAAQ,GACtB,CAAC,CA8BH;AAED;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,CAAC,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,EACrE,EAAE,GAAG,EAAE,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,WAAW,EAAE,EAAE,2BAA2B,CAAC,CAAC,CAAC,EAC7F,eAAe,UAAQ,GACtB,MAAM,CAKR;AAED;;GAEG;AACH,wBAAgB,uBAAuB,CAAC,CAAC,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,EACxE,EAAE,GAAG,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,EAAE,WAAW,EAAE,EAAE,8BAA8B,CAAC,CAAC,CAAC,EAC5E,eAAe,UAAQ,GACtB,cAAc,CAYhB;AAED;;GAEG;AACH,wBAAgB,cAAc,CAAC,CAAC,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,EAC/D,EAAE,GAAG,EAAE,OAAO,EAAE,UAAU,EAAE,IAAI,EAAE,SAAS,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,EAAE,qBAAqB,CAAC,CAAC,CAAC,EACzG,eAAe,UAAQ,GACtB,kBAAkB,CAAC,CAAC,CAAC,CA+BvB"}
package/dist/src/hmac.js CHANGED
@@ -38,9 +38,9 @@ exports.calculateHMACSubject = calculateHMACSubject;
38
38
  exports.calculateRequestHMAC = calculateRequestHMAC;
39
39
  exports.calculateRequestHeaders = calculateRequestHeaders;
40
40
  exports.verifyResponse = verifyResponse;
41
- const crypto_1 = require("crypto");
42
41
  const urlLib = __importStar(require("url"));
43
42
  const sjcl = __importStar(require("@bitgo-beta/sjcl"));
43
+ const util_1 = require("./util");
44
44
  /**
45
45
  * Calculate the HMAC for the given key and message
46
46
  * @param key {String} - the key to use for the HMAC
@@ -48,7 +48,7 @@ const sjcl = __importStar(require("@bitgo-beta/sjcl"));
48
48
  * @returns {*} - the result of the HMAC operation
49
49
  */
50
50
  function calculateHMAC(key, message) {
51
- return (0, crypto_1.createHmac)('sha256', key).update(message).digest('hex');
51
+ return (0, util_1.createHmacWithSha256)(key, message);
52
52
  }
53
53
  /**
54
54
  * Calculate the subject string that is to be HMAC'ed for a HTTP request or response
@@ -57,36 +57,53 @@ function calculateHMAC(key, message) {
57
57
  * @param timestamp request timestamp from `Date.now()`
58
58
  * @param statusCode Only set for HTTP responses, leave blank for requests
59
59
  * @param method request method
60
- * @returns {string}
60
+ * @param authVersion authentication version (2 or 3)
61
+ * @param useOriginalPath whether to use the original urlPath without parsing (default false)
62
+ * @returns {string | Buffer}
61
63
  */
62
- function calculateHMACSubject({ urlPath, text, timestamp, statusCode, method, authVersion, }) {
63
- const urlDetails = urlLib.parse(urlPath);
64
- const queryPath = urlDetails.query && urlDetails.query.length > 0 ? urlDetails.path : urlDetails.pathname;
64
+ function calculateHMACSubject({ urlPath, text, timestamp, statusCode, method, authVersion }, useOriginalPath = false) {
65
+ /* Normalize legacy 'del' to 'delete' for backward compatibility */
66
+ if (method === 'del') {
67
+ method = 'delete';
68
+ }
69
+ let queryPath = urlPath;
70
+ if (!useOriginalPath) {
71
+ const urlDetails = urlLib.parse(urlPath);
72
+ queryPath = urlDetails.query && urlDetails.query.length > 0 ? urlDetails.path : urlDetails.pathname;
73
+ }
74
+ let prefixedText;
65
75
  if (statusCode !== undefined && isFinite(statusCode) && Number.isInteger(statusCode)) {
66
- if (authVersion === 3) {
67
- return [method.toUpperCase(), timestamp, queryPath, statusCode, text].join('|');
68
- }
69
- return [timestamp, queryPath, statusCode, text].join('|');
76
+ prefixedText =
77
+ authVersion === 3
78
+ ? [method.toUpperCase(), timestamp, queryPath, statusCode].join('|')
79
+ : [timestamp, queryPath, statusCode].join('|');
80
+ }
81
+ else {
82
+ prefixedText =
83
+ authVersion === 3
84
+ ? [method.toUpperCase(), timestamp, '3.0', queryPath].join('|')
85
+ : [timestamp, queryPath].join('|');
70
86
  }
71
- if (authVersion === 3) {
72
- return [method.toUpperCase(), timestamp, '3.0', queryPath, text].join('|');
87
+ const isBuffer = Buffer.isBuffer(text);
88
+ if (isBuffer) {
89
+ return Buffer.concat([Buffer.from(prefixedText + '|', 'utf-8'), text]);
73
90
  }
74
- return [timestamp, queryPath, text].join('|');
91
+ return [prefixedText, text].join('|');
75
92
  }
76
93
  /**
77
94
  * Calculate the HMAC for an HTTP request
78
95
  */
79
- function calculateRequestHMAC({ url: urlPath, text, timestamp, token, method, authVersion, }) {
80
- const signatureSubject = calculateHMACSubject({ urlPath, text, timestamp, method, authVersion });
96
+ function calculateRequestHMAC({ url: urlPath, text, timestamp, token, method, authVersion }, useOriginalPath = false) {
97
+ const signatureSubject = calculateHMACSubject({ urlPath, text, timestamp, method, authVersion }, useOriginalPath);
81
98
  // calculate the HMAC
82
99
  return calculateHMAC(token, signatureSubject);
83
100
  }
84
101
  /**
85
102
  * Calculate request headers with HMAC
86
103
  */
87
- function calculateRequestHeaders({ url, text, token, method, authVersion, }) {
104
+ function calculateRequestHeaders({ url, text, token, method, authVersion }, useOriginalPath = false) {
88
105
  const timestamp = Date.now();
89
- const hmac = calculateRequestHMAC({ url, text, timestamp, token, method, authVersion });
106
+ const hmac = calculateRequestHMAC({ url, text, timestamp, token, method, authVersion }, useOriginalPath);
90
107
  // calculate the SHA256 hash of the token
91
108
  const hashDigest = sjcl.hash.sha256.hash(token);
92
109
  const tokenHash = sjcl.codec.hex.fromBits(hashDigest);
@@ -99,7 +116,7 @@ function calculateRequestHeaders({ url, text, token, method, authVersion, }) {
99
116
  /**
100
117
  * Verify the HMAC for an HTTP response
101
118
  */
102
- function verifyResponse({ url: urlPath, statusCode, text, timestamp, token, hmac, method, authVersion, }) {
119
+ function verifyResponse({ url: urlPath, statusCode, text, timestamp, token, hmac, method, authVersion }, useOriginalPath = false) {
103
120
  const signatureSubject = calculateHMACSubject({
104
121
  urlPath,
105
122
  text,
@@ -107,12 +124,14 @@ function verifyResponse({ url: urlPath, statusCode, text, timestamp, token, hmac
107
124
  statusCode,
108
125
  method,
109
126
  authVersion,
110
- });
127
+ }, useOriginalPath);
111
128
  // calculate the HMAC
112
129
  const expectedHmac = calculateHMAC(token, signatureSubject);
113
- // determine if the response is still within the validity window (5 minute window)
130
+ // determine if the response is still within the validity window (5-minute backwards window, 1-minute forward window)
114
131
  const now = Date.now();
115
- const isInResponseValidityWindow = timestamp >= now - 1000 * 60 * 5 && timestamp <= now;
132
+ const backwardValidityWindow = 1000 * 60 * 5;
133
+ const forwardValidityWindow = 1000 * 60;
134
+ const isInResponseValidityWindow = timestamp >= now - backwardValidityWindow && timestamp <= now + forwardValidityWindow;
116
135
  // verify the HMAC and timestamp
117
136
  return {
118
137
  isValid: expectedHmac === hmac,
@@ -122,4 +141,4 @@ function verifyResponse({ url: urlPath, statusCode, text, timestamp, token, hmac
122
141
  verificationTime: now,
123
142
  };
124
143
  }
125
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hmac.js","sourceRoot":"","sources":["../../src/hmac.ts"],"names":[],"mappings":"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","sourcesContent":["import { createHmac } from 'crypto';\nimport * as urlLib from 'url';\nimport * as sjcl from '@bitgo-beta/sjcl';\nimport {\n  CalculateHmacSubjectOptions,\n  CalculateRequestHeadersOptions,\n  CalculateRequestHmacOptions,\n  RequestHeaders,\n  VerifyResponseInfo,\n  VerifyResponseOptions,\n} from './types';\n\n/**\n * Calculate the HMAC for the given key and message\n * @param key {String} - the key to use for the HMAC\n * @param message {String} - the actual message to HMAC\n * @returns {*} - the result of the HMAC operation\n */\nexport function calculateHMAC(key: string, message: string): string {\n  return createHmac('sha256', key).update(message).digest('hex');\n}\n\n/**\n * Calculate the subject string that is to be HMAC'ed for a HTTP request or response\n * @param urlPath request url, including query params\n * @param text request body text\n * @param timestamp request timestamp from `Date.now()`\n * @param statusCode Only set for HTTP responses, leave blank for requests\n * @param method request method\n * @returns {string}\n */\nexport function calculateHMACSubject({\n  urlPath,\n  text,\n  timestamp,\n  statusCode,\n  method,\n  authVersion,\n}: CalculateHmacSubjectOptions): string {\n  const urlDetails = urlLib.parse(urlPath);\n  const queryPath = urlDetails.query && urlDetails.query.length > 0 ? urlDetails.path : urlDetails.pathname;\n  if (statusCode !== undefined && isFinite(statusCode) && Number.isInteger(statusCode)) {\n    if (authVersion === 3) {\n      return [method.toUpperCase(), timestamp, queryPath, statusCode, text].join('|');\n    }\n    return [timestamp, queryPath, statusCode, text].join('|');\n  }\n  if (authVersion === 3) {\n    return [method.toUpperCase(), timestamp, '3.0', queryPath, text].join('|');\n  }\n  return [timestamp, queryPath, text].join('|');\n}\n\n/**\n * Calculate the HMAC for an HTTP request\n */\nexport function calculateRequestHMAC({\n  url: urlPath,\n  text,\n  timestamp,\n  token,\n  method,\n  authVersion,\n}: CalculateRequestHmacOptions): string {\n  const signatureSubject = calculateHMACSubject({ urlPath, text, timestamp, method, authVersion });\n\n  // calculate the HMAC\n  return calculateHMAC(token, signatureSubject);\n}\n\n/**\n * Calculate request headers with HMAC\n */\nexport function calculateRequestHeaders({\n  url,\n  text,\n  token,\n  method,\n  authVersion,\n}: CalculateRequestHeadersOptions): RequestHeaders {\n  const timestamp = Date.now();\n  const hmac = calculateRequestHMAC({ url, text, timestamp, token, method, authVersion });\n\n  // calculate the SHA256 hash of the token\n  const hashDigest = sjcl.hash.sha256.hash(token);\n  const tokenHash = sjcl.codec.hex.fromBits(hashDigest);\n  return {\n    hmac,\n    timestamp,\n    tokenHash,\n  };\n}\n\n/**\n * Verify the HMAC for an HTTP response\n */\nexport function verifyResponse({\n  url: urlPath,\n  statusCode,\n  text,\n  timestamp,\n  token,\n  hmac,\n  method,\n  authVersion,\n}: VerifyResponseOptions): VerifyResponseInfo {\n  const signatureSubject = calculateHMACSubject({\n    urlPath,\n    text,\n    timestamp,\n    statusCode,\n    method,\n    authVersion,\n  });\n\n  // calculate the HMAC\n  const expectedHmac = calculateHMAC(token, signatureSubject);\n\n  // determine if the response is still within the validity window (5 minute window)\n  const now = Date.now();\n  const isInResponseValidityWindow = timestamp >= now - 1000 * 60 * 5 && timestamp <= now;\n\n  // verify the HMAC and timestamp\n  return {\n    isValid: expectedHmac === hmac,\n    expectedHmac,\n    signatureSubject,\n    isInResponseValidityWindow,\n    verificationTime: now,\n  };\n}\n"]}
144
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hmac.js","sourceRoot":"","sources":["../../src/hmac.ts"],"names":[],"mappings":"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","sourcesContent":["import { type BinaryLike, type KeyObject } from 'crypto';\nimport * as urlLib from 'url';\nimport * as sjcl from '@bitgo-beta/sjcl';\nimport {\n  CalculateHmacSubjectOptions,\n  CalculateRequestHeadersOptions,\n  CalculateRequestHmacOptions,\n  RequestHeaders,\n  VerifyResponseInfo,\n  VerifyResponseOptions,\n} from './types';\nimport { createHmacWithSha256 } from './util';\n\n/**\n * Calculate the HMAC for the given key and message\n * @param key {String} - the key to use for the HMAC\n * @param message {String} - the actual message to HMAC\n * @returns {*} - the result of the HMAC operation\n */\nexport function calculateHMAC(key: string | BinaryLike | KeyObject, message: string | BinaryLike): string {\n  return createHmacWithSha256(key, message);\n}\n\n/**\n * Calculate the subject string that is to be HMAC'ed for a HTTP request or response\n * @param urlPath request url, including query params\n * @param text request body text\n * @param timestamp request timestamp from `Date.now()`\n * @param statusCode Only set for HTTP responses, leave blank for requests\n * @param method request method\n * @param authVersion authentication version (2 or 3)\n * @param useOriginalPath whether to use the original urlPath without parsing (default false)\n * @returns {string | Buffer}\n */\nexport function calculateHMACSubject<T extends string | Buffer = string>(\n  { urlPath, text, timestamp, statusCode, method, authVersion }: CalculateHmacSubjectOptions<T>,\n  useOriginalPath = false\n): T {\n  /* Normalize legacy 'del' to 'delete' for backward compatibility */\n  if (method === 'del') {\n    method = 'delete';\n  }\n\n  let queryPath: string | null = urlPath;\n  if (!useOriginalPath) {\n    const urlDetails = urlLib.parse(urlPath);\n    queryPath = urlDetails.query && urlDetails.query.length > 0 ? urlDetails.path : urlDetails.pathname;\n  }\n\n  let prefixedText: string;\n  if (statusCode !== undefined && isFinite(statusCode) && Number.isInteger(statusCode)) {\n    prefixedText =\n      authVersion === 3\n        ? [method.toUpperCase(), timestamp, queryPath, statusCode].join('|')\n        : [timestamp, queryPath, statusCode].join('|');\n  } else {\n    prefixedText =\n      authVersion === 3\n        ? [method.toUpperCase(), timestamp, '3.0', queryPath].join('|')\n        : [timestamp, queryPath].join('|');\n  }\n\n  const isBuffer = Buffer.isBuffer(text);\n  if (isBuffer) {\n    return Buffer.concat([Buffer.from(prefixedText + '|', 'utf-8'), text]) as T;\n  }\n  return [prefixedText, text].join('|') as T;\n}\n\n/**\n * Calculate the HMAC for an HTTP request\n */\nexport function calculateRequestHMAC<T extends string | Buffer = string>(\n  { url: urlPath, text, timestamp, token, method, authVersion }: CalculateRequestHmacOptions<T>,\n  useOriginalPath = false\n): string {\n  const signatureSubject = calculateHMACSubject({ urlPath, text, timestamp, method, authVersion }, useOriginalPath);\n\n  // calculate the HMAC\n  return calculateHMAC(token, signatureSubject);\n}\n\n/**\n * Calculate request headers with HMAC\n */\nexport function calculateRequestHeaders<T extends string | Buffer = string>(\n  { url, text, token, method, authVersion }: CalculateRequestHeadersOptions<T>,\n  useOriginalPath = false\n): RequestHeaders {\n  const timestamp = Date.now();\n  const hmac = calculateRequestHMAC({ url, text, timestamp, token, method, authVersion }, useOriginalPath);\n\n  // calculate the SHA256 hash of the token\n  const hashDigest = sjcl.hash.sha256.hash(token);\n  const tokenHash = sjcl.codec.hex.fromBits(hashDigest);\n  return {\n    hmac,\n    timestamp,\n    tokenHash,\n  };\n}\n\n/**\n * Verify the HMAC for an HTTP response\n */\nexport function verifyResponse<T extends string | Buffer = string>(\n  { url: urlPath, statusCode, text, timestamp, token, hmac, method, authVersion }: VerifyResponseOptions<T>,\n  useOriginalPath = false\n): VerifyResponseInfo<T> {\n  const signatureSubject = calculateHMACSubject(\n    {\n      urlPath,\n      text,\n      timestamp,\n      statusCode,\n      method,\n      authVersion,\n    },\n    useOriginalPath\n  );\n\n  // calculate the HMAC\n  const expectedHmac = calculateHMAC(token, signatureSubject);\n\n  // determine if the response is still within the validity window (5-minute backwards window, 1-minute forward window)\n  const now = Date.now();\n  const backwardValidityWindow = 1000 * 60 * 5;\n  const forwardValidityWindow = 1000 * 60;\n  const isInResponseValidityWindow =\n    timestamp >= now - backwardValidityWindow && timestamp <= now + forwardValidityWindow;\n\n  // verify the HMAC and timestamp\n  return {\n    isValid: expectedHmac === hmac,\n    expectedHmac,\n    signatureSubject,\n    isInResponseValidityWindow,\n    verificationTime: now,\n  };\n}\n"]}
@@ -0,0 +1,185 @@
1
+ /**
2
+ * @prettier
3
+ *
4
+ * V4 HMAC Authentication Module
5
+ *
6
+ * This module implements the v4 authentication scheme which uses a canonical
7
+ * preimage construction with newline-separated fields and body hashing.
8
+ *
9
+ * Key differences from v2/v3:
10
+ * - Separator: newline (\n) instead of pipe (|)
11
+ * - Body: SHA256 hash of raw bytes instead of actual body content
12
+ * - Timestamp: seconds instead of milliseconds
13
+ * - New field: authRequestId for request tracking
14
+ * - Trailing newline in preimage
15
+ * - Support for x-original-* headers (proxy scenarios)
16
+ */
17
+ import { type HashableData } from './util';
18
+ import { CalculateV4PreimageOptions, CalculateV4RequestHmacOptions, CalculateV4RequestHeadersOptions, V4RequestHeaders, VerifyV4ResponseOptions, VerifyV4ResponseInfo } from './types';
19
+ /**
20
+ * Build canonical preimage for v4 authentication.
21
+ *
22
+ * The preimage is constructed as newline-separated values with a trailing newline:
23
+ * ```
24
+ * {timestampSec}
25
+ * {METHOD}
26
+ * {pathWithQuery}
27
+ * {bodyHashHex}
28
+ * {authRequestId}
29
+ * ```
30
+ *
31
+ * This function normalizes the HTTP method to uppercase and handles the
32
+ * legacy 'del' method conversion to 'DELETE'.
33
+ *
34
+ * @param options - The preimage components
35
+ * @returns Newline-separated canonical preimage string with trailing newline
36
+ *
37
+ * @example
38
+ * ```typescript
39
+ * const preimage = calculateV4Preimage({
40
+ * timestampSec: 1761100000,
41
+ * method: 'post',
42
+ * pathWithQuery: '/v2/wallets/transfer?foo=bar',
43
+ * bodyHashHex: '0d5e3b7a8f9c1d2e3f4a5b6c7d8e9f0a1b2c3d4e5f6a7b8c9d0e1f2a3b4c5d6e',
44
+ * authRequestId: '1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e',
45
+ * });
46
+ *
47
+ * // Result:
48
+ * // "1761100000\nPOST\n/v2/wallets/transfer?foo=bar\n0d5e3b...d6e\n1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e\n"
49
+ * ```
50
+ */
51
+ export declare function calculateV4Preimage({ timestampSec, method, pathWithQuery, bodyHashHex, authRequestId, }: CalculateV4PreimageOptions): string;
52
+ /**
53
+ * Calculate SHA256 hash of body and return as lowercase hex string.
54
+ *
55
+ * This is used to compute the bodyHashHex field for v4 authentication.
56
+ * The hash is computed over the raw bytes of the request body, ensuring
57
+ * that the exact bytes sent over the wire are used for signature calculation.
58
+ *
59
+ * Accepts common byte representations for Node.js and browser environments,
60
+ * including Uint8Array and ArrayBuffer for Fetch API compatibility.
61
+ *
62
+ * @param body - Raw request body (string, Buffer, Uint8Array, or ArrayBuffer)
63
+ * @returns Lowercase hex SHA256 hash (64 characters)
64
+ *
65
+ * @example
66
+ * ```typescript
67
+ * // Node.js with Buffer
68
+ * const hash1 = calculateBodyHash(Buffer.from('{"address":"tb1q...","amount":100000}'));
69
+ *
70
+ * // Browser with Uint8Array
71
+ * const hash2 = calculateBodyHash(new TextEncoder().encode('{"address":"tb1q..."}'));
72
+ *
73
+ * // Browser with ArrayBuffer
74
+ * const hash3 = calculateBodyHash(await response.arrayBuffer());
75
+ *
76
+ * // All return: '0d5e3b7a8f...' (64 character hex string)
77
+ * ```
78
+ */
79
+ export declare function calculateBodyHash(body: HashableData): string;
80
+ /**
81
+ * Calculate the HMAC-SHA256 signature for a v4 HTTP request.
82
+ *
83
+ * This function:
84
+ * 1. Builds the canonical preimage from the provided options
85
+ * 2. Computes HMAC-SHA256 of the preimage using the raw access token
86
+ *
87
+ * @param options - Request parameters and raw access token
88
+ * @returns Lowercase hex HMAC-SHA256 signature
89
+ *
90
+ * @example
91
+ * ```typescript
92
+ * const hmac = calculateV4RequestHmac({
93
+ * timestampSec: 1761100000,
94
+ * method: 'POST',
95
+ * pathWithQuery: '/v2/wallets/transfer',
96
+ * bodyHashHex: '0d5e3b...',
97
+ * authRequestId: '1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e',
98
+ * rawToken: 'your-token',
99
+ * });
100
+ * ```
101
+ */
102
+ export declare function calculateV4RequestHmac({ timestampSec, method, pathWithQuery, bodyHashHex, authRequestId, rawToken, }: CalculateV4RequestHmacOptions): string;
103
+ /**
104
+ * Generate all headers required for v4 authenticated requests.
105
+ *
106
+ * This is a convenience function that:
107
+ * 1. Generates the current timestamp (in seconds)
108
+ * 2. Calculates the body hash from raw bytes
109
+ * 3. Computes the HMAC signature
110
+ * 4. Returns all values needed for request headers
111
+ *
112
+ * @param options - Request parameters including raw body and raw token
113
+ * @returns Object containing all v4 authentication header values
114
+ *
115
+ * @example
116
+ * ```typescript
117
+ * const headers = calculateV4RequestHeaders({
118
+ * method: 'POST',
119
+ * pathWithQuery: '/v2/wallets/transfer?foo=bar',
120
+ * rawBody: Buffer.from('{"address":"tb1q..."}'),
121
+ * rawToken: 'your-token',
122
+ * authRequestId: '1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e',
123
+ * });
124
+ *
125
+ * // Use headers to set:
126
+ * // - X-Request-Timestamp: headers.timestampSec
127
+ * // - X-Signature: headers.hmac
128
+ * // - X-Content-SHA256: headers.bodyHashHex
129
+ * // - X-Auth-Request-Id: headers.authRequestId
130
+ * ```
131
+ */
132
+ export declare function calculateV4RequestHeaders({ method, pathWithQuery, rawBody, rawToken, authRequestId, }: CalculateV4RequestHeadersOptions): V4RequestHeaders;
133
+ /**
134
+ * Build canonical preimage for v4 response verification.
135
+ *
136
+ * Response preimage includes the status code and uses the same format:
137
+ * ```
138
+ * {timestampSec}
139
+ * {METHOD}
140
+ * {pathWithQuery}
141
+ * {statusCode}
142
+ * {bodyHashHex}
143
+ * {authRequestId}
144
+ * ```
145
+ *
146
+ * @param options - Response verification parameters
147
+ * @returns Newline-separated canonical preimage string with trailing newline
148
+ */
149
+ export declare function calculateV4ResponsePreimage({ timestampSec, method, pathWithQuery, statusCode, bodyHashHex, authRequestId, }: Omit<VerifyV4ResponseOptions, 'hmac' | 'rawToken'>): string;
150
+ /**
151
+ * Verify the HMAC signature of a v4 HTTP response.
152
+ *
153
+ * This function:
154
+ * 1. Reconstructs the canonical preimage from response data
155
+ * 2. Calculates the expected HMAC
156
+ * 3. Compares with the received HMAC
157
+ * 4. Checks if the timestamp is within the validity window
158
+ *
159
+ * The validity window is:
160
+ * - 5 minutes backwards (to account for clock skew and network latency)
161
+ * - 1 minute forwards (to account for minor clock differences)
162
+ *
163
+ * @param options - Response data and raw token for verification
164
+ * @returns Verification result including validity and diagnostic info
165
+ */
166
+ export declare function verifyV4Response({ hmac, timestampSec, method, pathWithQuery, bodyHashHex, authRequestId, statusCode, rawToken, }: VerifyV4ResponseOptions): VerifyV4ResponseInfo;
167
+ /**
168
+ * Extract path with query from x-original-uri header or request URL.
169
+ * Always canonicalizes to pathname + search to handle absolute URLs.
170
+ *
171
+ * @param xOriginalUri - Value of x-original-uri header (if present)
172
+ * @param requestUrl - The actual request URL
173
+ * @returns The canonical path with query to use for preimage calculation
174
+ *
175
+ */
176
+ export declare function getPathWithQuery(xOriginalUri: string | undefined, requestUrl: string): string;
177
+ /**
178
+ * Get method from x-original-method header or actual request method.
179
+ *
180
+ * @param xOriginalMethod - Value of x-original-method header (if present)
181
+ * @param requestMethod - The actual request method
182
+ * @returns The method to use for preimage calculation
183
+ */
184
+ export declare function getMethod(xOriginalMethod: string | undefined, requestMethod: string): string;
185
+ //# sourceMappingURL=hmacv4.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hmacv4.d.ts","sourceRoot":"","sources":["../../src/hmacv4.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAGH,OAAO,EAML,KAAK,YAAY,EAClB,MAAM,QAAQ,CAAC;AAChB,OAAO,EACL,0BAA0B,EAC1B,6BAA6B,EAC7B,gCAAgC,EAChC,gBAAgB,EAChB,uBAAuB,EACvB,oBAAoB,EACrB,MAAM,SAAS,CAAC;AAEjB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BG;AACH,wBAAgB,mBAAmB,CAAC,EAClC,YAAY,EACZ,MAAM,EACN,aAAa,EACb,WAAW,EACX,aAAa,GACd,EAAE,0BAA0B,GAAG,MAAM,CAOrC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BG;AACH,wBAAgB,iBAAiB,CAAC,IAAI,EAAE,YAAY,GAAG,MAAM,CAE5D;AAED;;;;;;;;;;;;;;;;;;;;;GAqBG;AACH,wBAAgB,sBAAsB,CAAC,EACrC,YAAY,EACZ,MAAM,EACN,aAAa,EACb,WAAW,EACX,aAAa,EACb,QAAQ,GACT,EAAE,6BAA6B,GAAG,MAAM,CAUxC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA4BG;AACH,wBAAgB,yBAAyB,CAAC,EACxC,MAAM,EACN,aAAa,EACb,OAAO,EACP,QAAQ,EACR,aAAa,GACd,EAAE,gCAAgC,GAAG,gBAAgB,CAmBrD;AAED;;;;;;;;;;;;;;;GAeG;AACH,wBAAgB,2BAA2B,CAAC,EAC1C,YAAY,EACZ,MAAM,EACN,aAAa,EACb,UAAU,EACV,WAAW,EACX,aAAa,GACd,EAAE,IAAI,CAAC,uBAAuB,EAAE,MAAM,GAAG,UAAU,CAAC,GAAG,MAAM,CAa7D;AAED;;;;;;;;;;;;;;;GAeG;AACH,wBAAgB,gBAAgB,CAAC,EAC/B,IAAI,EACJ,YAAY,EACZ,MAAM,EACN,aAAa,EACb,WAAW,EACX,aAAa,EACb,UAAU,EACV,QAAQ,GACT,EAAE,uBAAuB,GAAG,oBAAoB,CAkChD;AAED;;;;;;;;GAQG;AACH,wBAAgB,gBAAgB,CAAC,YAAY,EAAE,MAAM,GAAG,SAAS,EAAE,UAAU,EAAE,MAAM,GAAG,MAAM,CAI7F;AAED;;;;;;GAMG;AACH,wBAAgB,SAAS,CAAC,eAAe,EAAE,MAAM,GAAG,SAAS,EAAE,aAAa,EAAE,MAAM,GAAG,MAAM,CAG5F"}
@@ -0,0 +1,274 @@
1
+ "use strict";
2
+ /**
3
+ * @prettier
4
+ *
5
+ * V4 HMAC Authentication Module
6
+ *
7
+ * This module implements the v4 authentication scheme which uses a canonical
8
+ * preimage construction with newline-separated fields and body hashing.
9
+ *
10
+ * Key differences from v2/v3:
11
+ * - Separator: newline (\n) instead of pipe (|)
12
+ * - Body: SHA256 hash of raw bytes instead of actual body content
13
+ * - Timestamp: seconds instead of milliseconds
14
+ * - New field: authRequestId for request tracking
15
+ * - Trailing newline in preimage
16
+ * - Support for x-original-* headers (proxy scenarios)
17
+ */
18
+ Object.defineProperty(exports, "__esModule", { value: true });
19
+ exports.calculateV4Preimage = calculateV4Preimage;
20
+ exports.calculateBodyHash = calculateBodyHash;
21
+ exports.calculateV4RequestHmac = calculateV4RequestHmac;
22
+ exports.calculateV4RequestHeaders = calculateV4RequestHeaders;
23
+ exports.calculateV4ResponsePreimage = calculateV4ResponsePreimage;
24
+ exports.verifyV4Response = verifyV4Response;
25
+ exports.getPathWithQuery = getPathWithQuery;
26
+ exports.getMethod = getMethod;
27
+ const crypto_1 = require("crypto");
28
+ const util_1 = require("./util");
29
+ /**
30
+ * Build canonical preimage for v4 authentication.
31
+ *
32
+ * The preimage is constructed as newline-separated values with a trailing newline:
33
+ * ```
34
+ * {timestampSec}
35
+ * {METHOD}
36
+ * {pathWithQuery}
37
+ * {bodyHashHex}
38
+ * {authRequestId}
39
+ * ```
40
+ *
41
+ * This function normalizes the HTTP method to uppercase and handles the
42
+ * legacy 'del' method conversion to 'DELETE'.
43
+ *
44
+ * @param options - The preimage components
45
+ * @returns Newline-separated canonical preimage string with trailing newline
46
+ *
47
+ * @example
48
+ * ```typescript
49
+ * const preimage = calculateV4Preimage({
50
+ * timestampSec: 1761100000,
51
+ * method: 'post',
52
+ * pathWithQuery: '/v2/wallets/transfer?foo=bar',
53
+ * bodyHashHex: '0d5e3b7a8f9c1d2e3f4a5b6c7d8e9f0a1b2c3d4e5f6a7b8c9d0e1f2a3b4c5d6e',
54
+ * authRequestId: '1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e',
55
+ * });
56
+ *
57
+ * // Result:
58
+ * // "1761100000\nPOST\n/v2/wallets/transfer?foo=bar\n0d5e3b...d6e\n1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e\n"
59
+ * ```
60
+ */
61
+ function calculateV4Preimage({ timestampSec, method, pathWithQuery, bodyHashHex, authRequestId, }) {
62
+ const normalizedMethod = (0, util_1.normalizeMethod)(method);
63
+ // Build newline-separated preimage with trailing newline
64
+ const components = [timestampSec.toString(), normalizedMethod, pathWithQuery, bodyHashHex, authRequestId];
65
+ return components.join('\n') + '\n';
66
+ }
67
+ /**
68
+ * Calculate SHA256 hash of body and return as lowercase hex string.
69
+ *
70
+ * This is used to compute the bodyHashHex field for v4 authentication.
71
+ * The hash is computed over the raw bytes of the request body, ensuring
72
+ * that the exact bytes sent over the wire are used for signature calculation.
73
+ *
74
+ * Accepts common byte representations for Node.js and browser environments,
75
+ * including Uint8Array and ArrayBuffer for Fetch API compatibility.
76
+ *
77
+ * @param body - Raw request body (string, Buffer, Uint8Array, or ArrayBuffer)
78
+ * @returns Lowercase hex SHA256 hash (64 characters)
79
+ *
80
+ * @example
81
+ * ```typescript
82
+ * // Node.js with Buffer
83
+ * const hash1 = calculateBodyHash(Buffer.from('{"address":"tb1q...","amount":100000}'));
84
+ *
85
+ * // Browser with Uint8Array
86
+ * const hash2 = calculateBodyHash(new TextEncoder().encode('{"address":"tb1q..."}'));
87
+ *
88
+ * // Browser with ArrayBuffer
89
+ * const hash3 = calculateBodyHash(await response.arrayBuffer());
90
+ *
91
+ * // All return: '0d5e3b7a8f...' (64 character hex string)
92
+ * ```
93
+ */
94
+ function calculateBodyHash(body) {
95
+ return (0, util_1.sha256Hex)(body);
96
+ }
97
+ /**
98
+ * Calculate the HMAC-SHA256 signature for a v4 HTTP request.
99
+ *
100
+ * This function:
101
+ * 1. Builds the canonical preimage from the provided options
102
+ * 2. Computes HMAC-SHA256 of the preimage using the raw access token
103
+ *
104
+ * @param options - Request parameters and raw access token
105
+ * @returns Lowercase hex HMAC-SHA256 signature
106
+ *
107
+ * @example
108
+ * ```typescript
109
+ * const hmac = calculateV4RequestHmac({
110
+ * timestampSec: 1761100000,
111
+ * method: 'POST',
112
+ * pathWithQuery: '/v2/wallets/transfer',
113
+ * bodyHashHex: '0d5e3b...',
114
+ * authRequestId: '1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e',
115
+ * rawToken: 'your-token',
116
+ * });
117
+ * ```
118
+ */
119
+ function calculateV4RequestHmac({ timestampSec, method, pathWithQuery, bodyHashHex, authRequestId, rawToken, }) {
120
+ const preimage = calculateV4Preimage({
121
+ timestampSec,
122
+ method,
123
+ pathWithQuery,
124
+ bodyHashHex,
125
+ authRequestId,
126
+ });
127
+ return (0, util_1.createHmacWithSha256)(rawToken, preimage);
128
+ }
129
+ /**
130
+ * Generate all headers required for v4 authenticated requests.
131
+ *
132
+ * This is a convenience function that:
133
+ * 1. Generates the current timestamp (in seconds)
134
+ * 2. Calculates the body hash from raw bytes
135
+ * 3. Computes the HMAC signature
136
+ * 4. Returns all values needed for request headers
137
+ *
138
+ * @param options - Request parameters including raw body and raw token
139
+ * @returns Object containing all v4 authentication header values
140
+ *
141
+ * @example
142
+ * ```typescript
143
+ * const headers = calculateV4RequestHeaders({
144
+ * method: 'POST',
145
+ * pathWithQuery: '/v2/wallets/transfer?foo=bar',
146
+ * rawBody: Buffer.from('{"address":"tb1q..."}'),
147
+ * rawToken: 'your-token',
148
+ * authRequestId: '1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e',
149
+ * });
150
+ *
151
+ * // Use headers to set:
152
+ * // - X-Request-Timestamp: headers.timestampSec
153
+ * // - X-Signature: headers.hmac
154
+ * // - X-Content-SHA256: headers.bodyHashHex
155
+ * // - X-Auth-Request-Id: headers.authRequestId
156
+ * ```
157
+ */
158
+ function calculateV4RequestHeaders({ method, pathWithQuery, rawBody, rawToken, authRequestId, }) {
159
+ const timestampSec = (0, util_1.getTimestampSec)();
160
+ const bodyHashHex = calculateBodyHash(rawBody);
161
+ const hmac = calculateV4RequestHmac({
162
+ timestampSec,
163
+ method,
164
+ pathWithQuery,
165
+ bodyHashHex,
166
+ authRequestId,
167
+ rawToken,
168
+ });
169
+ return {
170
+ hmac,
171
+ timestampSec,
172
+ bodyHashHex,
173
+ authRequestId,
174
+ };
175
+ }
176
+ /**
177
+ * Build canonical preimage for v4 response verification.
178
+ *
179
+ * Response preimage includes the status code and uses the same format:
180
+ * ```
181
+ * {timestampSec}
182
+ * {METHOD}
183
+ * {pathWithQuery}
184
+ * {statusCode}
185
+ * {bodyHashHex}
186
+ * {authRequestId}
187
+ * ```
188
+ *
189
+ * @param options - Response verification parameters
190
+ * @returns Newline-separated canonical preimage string with trailing newline
191
+ */
192
+ function calculateV4ResponsePreimage({ timestampSec, method, pathWithQuery, statusCode, bodyHashHex, authRequestId, }) {
193
+ const normalizedMethod = (0, util_1.normalizeMethod)(method);
194
+ const components = [
195
+ timestampSec.toString(),
196
+ normalizedMethod,
197
+ pathWithQuery,
198
+ statusCode.toString(),
199
+ bodyHashHex,
200
+ authRequestId,
201
+ ];
202
+ return components.join('\n') + '\n';
203
+ }
204
+ /**
205
+ * Verify the HMAC signature of a v4 HTTP response.
206
+ *
207
+ * This function:
208
+ * 1. Reconstructs the canonical preimage from response data
209
+ * 2. Calculates the expected HMAC
210
+ * 3. Compares with the received HMAC
211
+ * 4. Checks if the timestamp is within the validity window
212
+ *
213
+ * The validity window is:
214
+ * - 5 minutes backwards (to account for clock skew and network latency)
215
+ * - 1 minute forwards (to account for minor clock differences)
216
+ *
217
+ * @param options - Response data and raw token for verification
218
+ * @returns Verification result including validity and diagnostic info
219
+ */
220
+ function verifyV4Response({ hmac, timestampSec, method, pathWithQuery, bodyHashHex, authRequestId, statusCode, rawToken, }) {
221
+ // Build the response preimage
222
+ const preimage = calculateV4ResponsePreimage({
223
+ timestampSec,
224
+ method,
225
+ pathWithQuery,
226
+ statusCode,
227
+ bodyHashHex,
228
+ authRequestId,
229
+ });
230
+ // Calculate expected HMAC
231
+ const expectedHmac = (0, util_1.createHmacWithSha256)(rawToken, preimage);
232
+ // Use constant-time comparison to prevent timing side-channel attacks
233
+ const hmacBuffer = Buffer.from(hmac, 'hex');
234
+ const expectedHmacBuffer = Buffer.from(expectedHmac, 'hex');
235
+ const isHmacValid = hmacBuffer.length === expectedHmacBuffer.length && (0, crypto_1.timingSafeEqual)(hmacBuffer, expectedHmacBuffer);
236
+ // Check timestamp validity window
237
+ const nowSec = (0, util_1.getTimestampSec)();
238
+ const backwardValidityWindowSec = 5 * 60; // 5 minutes
239
+ const forwardValidityWindowSec = 1 * 60; // 1 minute
240
+ const isInResponseValidityWindow = timestampSec >= nowSec - backwardValidityWindowSec && timestampSec <= nowSec + forwardValidityWindowSec;
241
+ return {
242
+ isValid: isHmacValid,
243
+ expectedHmac,
244
+ preimage,
245
+ isInResponseValidityWindow,
246
+ verificationTime: Date.now(),
247
+ };
248
+ }
249
+ /**
250
+ * Extract path with query from x-original-uri header or request URL.
251
+ * Always canonicalizes to pathname + search to handle absolute URLs.
252
+ *
253
+ * @param xOriginalUri - Value of x-original-uri header (if present)
254
+ * @param requestUrl - The actual request URL
255
+ * @returns The canonical path with query to use for preimage calculation
256
+ *
257
+ */
258
+ function getPathWithQuery(xOriginalUri, requestUrl) {
259
+ // Prefer x-original-uri if available (proxy scenario)
260
+ const rawPath = xOriginalUri ?? requestUrl;
261
+ return (0, util_1.extractPathWithQuery)(rawPath);
262
+ }
263
+ /**
264
+ * Get method from x-original-method header or actual request method.
265
+ *
266
+ * @param xOriginalMethod - Value of x-original-method header (if present)
267
+ * @param requestMethod - The actual request method
268
+ * @returns The method to use for preimage calculation
269
+ */
270
+ function getMethod(xOriginalMethod, requestMethod) {
271
+ // Prefer x-original-method if available (proxy scenario)
272
+ return xOriginalMethod ?? requestMethod;
273
+ }
274
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hmacv4.js","sourceRoot":"","sources":["../../src/hmacv4.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n *\n * V4 HMAC Authentication Module\n *\n * This module implements the v4 authentication scheme which uses a canonical\n * preimage construction with newline-separated fields and body hashing.\n *\n * Key differences from v2/v3:\n * - Separator: newline (\\n) instead of pipe (|)\n * - Body: SHA256 hash of raw bytes instead of actual body content\n * - Timestamp: seconds instead of milliseconds\n * - New field: authRequestId for request tracking\n * - Trailing newline in preimage\n * - Support for x-original-* headers (proxy scenarios)\n */\n\nimport { timingSafeEqual } from 'crypto';\nimport {\n  createHmacWithSha256,\n  sha256Hex,\n  normalizeMethod,\n  getTimestampSec,\n  extractPathWithQuery,\n  type HashableData,\n} from './util';\nimport {\n  CalculateV4PreimageOptions,\n  CalculateV4RequestHmacOptions,\n  CalculateV4RequestHeadersOptions,\n  V4RequestHeaders,\n  VerifyV4ResponseOptions,\n  VerifyV4ResponseInfo,\n} from './types';\n\n/**\n * Build canonical preimage for v4 authentication.\n *\n * The preimage is constructed as newline-separated values with a trailing newline:\n * ```\n * {timestampSec}\n * {METHOD}\n * {pathWithQuery}\n * {bodyHashHex}\n * {authRequestId}\n * ```\n *\n * This function normalizes the HTTP method to uppercase and handles the\n * legacy 'del' method conversion to 'DELETE'.\n *\n * @param options - The preimage components\n * @returns Newline-separated canonical preimage string with trailing newline\n *\n * @example\n * ```typescript\n * const preimage = calculateV4Preimage({\n *   timestampSec: 1761100000,\n *   method: 'post',\n *   pathWithQuery: '/v2/wallets/transfer?foo=bar',\n *   bodyHashHex: '0d5e3b7a8f9c1d2e3f4a5b6c7d8e9f0a1b2c3d4e5f6a7b8c9d0e1f2a3b4c5d6e',\n *   authRequestId: '1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e',\n * });\n *\n * // Result:\n * // \"1761100000\\nPOST\\n/v2/wallets/transfer?foo=bar\\n0d5e3b...d6e\\n1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e\\n\"\n * ```\n */\nexport function calculateV4Preimage({\n  timestampSec,\n  method,\n  pathWithQuery,\n  bodyHashHex,\n  authRequestId,\n}: CalculateV4PreimageOptions): string {\n  const normalizedMethod = normalizeMethod(method);\n\n  // Build newline-separated preimage with trailing newline\n  const components = [timestampSec.toString(), normalizedMethod, pathWithQuery, bodyHashHex, authRequestId];\n\n  return components.join('\\n') + '\\n';\n}\n\n/**\n * Calculate SHA256 hash of body and return as lowercase hex string.\n *\n * This is used to compute the bodyHashHex field for v4 authentication.\n * The hash is computed over the raw bytes of the request body, ensuring\n * that the exact bytes sent over the wire are used for signature calculation.\n *\n * Accepts common byte representations for Node.js and browser environments,\n * including Uint8Array and ArrayBuffer for Fetch API compatibility.\n *\n * @param body - Raw request body (string, Buffer, Uint8Array, or ArrayBuffer)\n * @returns Lowercase hex SHA256 hash (64 characters)\n *\n * @example\n * ```typescript\n * // Node.js with Buffer\n * const hash1 = calculateBodyHash(Buffer.from('{\"address\":\"tb1q...\",\"amount\":100000}'));\n *\n * // Browser with Uint8Array\n * const hash2 = calculateBodyHash(new TextEncoder().encode('{\"address\":\"tb1q...\"}'));\n *\n * // Browser with ArrayBuffer\n * const hash3 = calculateBodyHash(await response.arrayBuffer());\n *\n * // All return: '0d5e3b7a8f...' (64 character hex string)\n * ```\n */\nexport function calculateBodyHash(body: HashableData): string {\n  return sha256Hex(body);\n}\n\n/**\n * Calculate the HMAC-SHA256 signature for a v4 HTTP request.\n *\n * This function:\n * 1. Builds the canonical preimage from the provided options\n * 2. Computes HMAC-SHA256 of the preimage using the raw access token\n *\n * @param options - Request parameters and raw access token\n * @returns Lowercase hex HMAC-SHA256 signature\n *\n * @example\n * ```typescript\n * const hmac = calculateV4RequestHmac({\n *   timestampSec: 1761100000,\n *   method: 'POST',\n *   pathWithQuery: '/v2/wallets/transfer',\n *   bodyHashHex: '0d5e3b...',\n *   authRequestId: '1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e',\n *   rawToken: 'your-token',\n * });\n * ```\n */\nexport function calculateV4RequestHmac({\n  timestampSec,\n  method,\n  pathWithQuery,\n  bodyHashHex,\n  authRequestId,\n  rawToken,\n}: CalculateV4RequestHmacOptions): string {\n  const preimage = calculateV4Preimage({\n    timestampSec,\n    method,\n    pathWithQuery,\n    bodyHashHex,\n    authRequestId,\n  });\n\n  return createHmacWithSha256(rawToken, preimage);\n}\n\n/**\n * Generate all headers required for v4 authenticated requests.\n *\n * This is a convenience function that:\n * 1. Generates the current timestamp (in seconds)\n * 2. Calculates the body hash from raw bytes\n * 3. Computes the HMAC signature\n * 4. Returns all values needed for request headers\n *\n * @param options - Request parameters including raw body and raw token\n * @returns Object containing all v4 authentication header values\n *\n * @example\n * ```typescript\n * const headers = calculateV4RequestHeaders({\n *   method: 'POST',\n *   pathWithQuery: '/v2/wallets/transfer?foo=bar',\n *   rawBody: Buffer.from('{\"address\":\"tb1q...\"}'),\n *   rawToken: 'your-token',\n *   authRequestId: '1b7a1d2b-7a2f-4e4b-a1f8-c2a5a0f84e3e',\n * });\n *\n * // Use headers to set:\n * // - X-Request-Timestamp: headers.timestampSec\n * // - X-Signature: headers.hmac\n * // - X-Content-SHA256: headers.bodyHashHex\n * // - X-Auth-Request-Id: headers.authRequestId\n * ```\n */\nexport function calculateV4RequestHeaders({\n  method,\n  pathWithQuery,\n  rawBody,\n  rawToken,\n  authRequestId,\n}: CalculateV4RequestHeadersOptions): V4RequestHeaders {\n  const timestampSec = getTimestampSec();\n  const bodyHashHex = calculateBodyHash(rawBody);\n\n  const hmac = calculateV4RequestHmac({\n    timestampSec,\n    method,\n    pathWithQuery,\n    bodyHashHex,\n    authRequestId,\n    rawToken,\n  });\n\n  return {\n    hmac,\n    timestampSec,\n    bodyHashHex,\n    authRequestId,\n  };\n}\n\n/**\n * Build canonical preimage for v4 response verification.\n *\n * Response preimage includes the status code and uses the same format:\n * ```\n * {timestampSec}\n * {METHOD}\n * {pathWithQuery}\n * {statusCode}\n * {bodyHashHex}\n * {authRequestId}\n * ```\n *\n * @param options - Response verification parameters\n * @returns Newline-separated canonical preimage string with trailing newline\n */\nexport function calculateV4ResponsePreimage({\n  timestampSec,\n  method,\n  pathWithQuery,\n  statusCode,\n  bodyHashHex,\n  authRequestId,\n}: Omit<VerifyV4ResponseOptions, 'hmac' | 'rawToken'>): string {\n  const normalizedMethod = normalizeMethod(method);\n\n  const components = [\n    timestampSec.toString(),\n    normalizedMethod,\n    pathWithQuery,\n    statusCode.toString(),\n    bodyHashHex,\n    authRequestId,\n  ];\n\n  return components.join('\\n') + '\\n';\n}\n\n/**\n * Verify the HMAC signature of a v4 HTTP response.\n *\n * This function:\n * 1. Reconstructs the canonical preimage from response data\n * 2. Calculates the expected HMAC\n * 3. Compares with the received HMAC\n * 4. Checks if the timestamp is within the validity window\n *\n * The validity window is:\n * - 5 minutes backwards (to account for clock skew and network latency)\n * - 1 minute forwards (to account for minor clock differences)\n *\n * @param options - Response data and raw token for verification\n * @returns Verification result including validity and diagnostic info\n */\nexport function verifyV4Response({\n  hmac,\n  timestampSec,\n  method,\n  pathWithQuery,\n  bodyHashHex,\n  authRequestId,\n  statusCode,\n  rawToken,\n}: VerifyV4ResponseOptions): VerifyV4ResponseInfo {\n  // Build the response preimage\n  const preimage = calculateV4ResponsePreimage({\n    timestampSec,\n    method,\n    pathWithQuery,\n    statusCode,\n    bodyHashHex,\n    authRequestId,\n  });\n\n  // Calculate expected HMAC\n  const expectedHmac = createHmacWithSha256(rawToken, preimage);\n\n  // Use constant-time comparison to prevent timing side-channel attacks\n  const hmacBuffer = Buffer.from(hmac, 'hex');\n  const expectedHmacBuffer = Buffer.from(expectedHmac, 'hex');\n  const isHmacValid =\n    hmacBuffer.length === expectedHmacBuffer.length && timingSafeEqual(hmacBuffer, expectedHmacBuffer);\n\n  // Check timestamp validity window\n  const nowSec = getTimestampSec();\n  const backwardValidityWindowSec = 5 * 60; // 5 minutes\n  const forwardValidityWindowSec = 1 * 60; // 1 minute\n  const isInResponseValidityWindow =\n    timestampSec >= nowSec - backwardValidityWindowSec && timestampSec <= nowSec + forwardValidityWindowSec;\n\n  return {\n    isValid: isHmacValid,\n    expectedHmac,\n    preimage,\n    isInResponseValidityWindow,\n    verificationTime: Date.now(),\n  };\n}\n\n/**\n * Extract path with query from x-original-uri header or request URL.\n * Always canonicalizes to pathname + search to handle absolute URLs.\n *\n * @param xOriginalUri - Value of x-original-uri header (if present)\n * @param requestUrl - The actual request URL\n * @returns The canonical path with query to use for preimage calculation\n *\n */\nexport function getPathWithQuery(xOriginalUri: string | undefined, requestUrl: string): string {\n  // Prefer x-original-uri if available (proxy scenario)\n  const rawPath = xOriginalUri ?? requestUrl;\n  return extractPathWithQuery(rawPath);\n}\n\n/**\n * Get method from x-original-method header or actual request method.\n *\n * @param xOriginalMethod - Value of x-original-method header (if present)\n * @param requestMethod - The actual request method\n * @returns The method to use for preimage calculation\n */\nexport function getMethod(xOriginalMethod: string | undefined, requestMethod: string): string {\n  // Prefer x-original-method if available (proxy scenario)\n  return xOriginalMethod ?? requestMethod;\n}\n"]}
@@ -1,3 +1,5 @@
1
1
  export * from './hmac';
2
+ export * from './hmacv4';
3
+ export * from './util';
2
4
  export * from './types';
3
5
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,QAAQ,CAAC;AACvB,cAAc,SAAS,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,QAAQ,CAAC;AACvB,cAAc,UAAU,CAAC;AACzB,cAAc,QAAQ,CAAC;AACvB,cAAc,SAAS,CAAC"}
package/dist/src/index.js CHANGED
@@ -15,5 +15,7 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
15
15
  };
16
16
  Object.defineProperty(exports, "__esModule", { value: true });
17
17
  __exportStar(require("./hmac"), exports);
18
+ __exportStar(require("./hmacv4"), exports);
19
+ __exportStar(require("./util"), exports);
18
20
  __exportStar(require("./types"), exports);
19
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHlDQUF1QjtBQUN2QiwwQ0FBd0IiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2htYWMnO1xuZXhwb3J0ICogZnJvbSAnLi90eXBlcyc7XG4iXX0=
21
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHlDQUF1QjtBQUN2QiwyQ0FBeUI7QUFDekIseUNBQXVCO0FBQ3ZCLDBDQUF3QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vaG1hYyc7XG5leHBvcnQgKiBmcm9tICcuL2htYWN2NCc7XG5leHBvcnQgKiBmcm9tICcuL3V0aWwnO1xuZXhwb3J0ICogZnJvbSAnLi90eXBlcyc7XG4iXX0=
@@ -1,24 +1,24 @@
1
- export declare const supportedRequestMethods: readonly ["get", "post", "put", "del", "patch", "options"];
1
+ export declare const supportedRequestMethods: readonly ["get", "post", "put", "del", "patch", "options", "delete"];
2
2
  export type AuthVersion = 2 | 3;
3
- export interface CalculateHmacSubjectOptions {
3
+ export interface CalculateHmacSubjectOptions<T> {
4
4
  urlPath: string;
5
- text: string;
5
+ text: T;
6
6
  timestamp: number;
7
7
  method: (typeof supportedRequestMethods)[number];
8
8
  statusCode?: number;
9
9
  authVersion: AuthVersion;
10
10
  }
11
- export interface CalculateRequestHmacOptions {
11
+ export interface CalculateRequestHmacOptions<T extends string | Buffer = string> {
12
12
  url: string;
13
- text: string;
13
+ text: T;
14
14
  timestamp: number;
15
15
  token: string;
16
16
  method: (typeof supportedRequestMethods)[number];
17
17
  authVersion: AuthVersion;
18
18
  }
19
- export interface CalculateRequestHeadersOptions {
19
+ export interface CalculateRequestHeadersOptions<T extends string | Buffer = string> {
20
20
  url: string;
21
- text: string;
21
+ text: T;
22
22
  token: string;
23
23
  method: (typeof supportedRequestMethods)[number];
24
24
  authVersion: AuthVersion;
@@ -28,19 +28,69 @@ export interface RequestHeaders {
28
28
  timestamp: number;
29
29
  tokenHash: string;
30
30
  }
31
- export interface VerifyResponseOptions extends CalculateRequestHeadersOptions {
31
+ export interface VerifyResponseOptions<T extends string | Buffer = string> extends CalculateRequestHeadersOptions<T> {
32
32
  hmac: string;
33
33
  url: string;
34
- text: string;
34
+ text: T;
35
35
  timestamp: number;
36
36
  method: (typeof supportedRequestMethods)[number];
37
37
  statusCode?: number;
38
38
  authVersion: AuthVersion;
39
39
  }
40
- export interface VerifyResponseInfo {
40
+ export interface VerifyResponseInfo<T extends string | Buffer = string> {
41
41
  isValid: boolean;
42
42
  expectedHmac: string;
43
- signatureSubject: string;
43
+ signatureSubject: T;
44
+ isInResponseValidityWindow: boolean;
45
+ verificationTime: number;
46
+ }
47
+ export interface CalculateV4PreimageOptions {
48
+ timestampSec: number;
49
+ method: string;
50
+ pathWithQuery: string;
51
+ bodyHashHex: string;
52
+ authRequestId: string;
53
+ }
54
+ export interface CalculateV4RequestHmacOptions extends CalculateV4PreimageOptions {
55
+ rawToken: string;
56
+ }
57
+ import type { HashableData } from './util';
58
+ export interface CalculateV4RequestHeadersOptions {
59
+ method: string;
60
+ pathWithQuery: string;
61
+ rawBody: HashableData;
62
+ rawToken: string;
63
+ authRequestId: string;
64
+ }
65
+ /**
66
+ * Headers generated for V4 authenticated requests.
67
+ */
68
+ export interface V4RequestHeaders {
69
+ hmac: string;
70
+ timestampSec: number;
71
+ bodyHashHex: string;
72
+ authRequestId: string;
73
+ }
74
+ /**
75
+ * Options for verifying V4 response HMAC.
76
+ */
77
+ export interface VerifyV4ResponseOptions {
78
+ hmac: string;
79
+ timestampSec: number;
80
+ method: string;
81
+ pathWithQuery: string;
82
+ bodyHashHex: string;
83
+ authRequestId: string;
84
+ statusCode: number;
85
+ rawToken: string;
86
+ }
87
+ /**
88
+ * Result of V4 response HMAC verification.
89
+ */
90
+ export interface VerifyV4ResponseInfo {
91
+ isValid: boolean;
92
+ expectedHmac: string;
93
+ preimage: string;
44
94
  isInResponseValidityWindow: boolean;
45
95
  verificationTime: number;
46
96
  }
@@ -1 +1 @@
1
- {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/types.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,uBAAuB,4DAA6D,CAAC;AAElG,MAAM,MAAM,WAAW,GAAG,CAAC,GAAG,CAAC,CAAC;AAEhC,MAAM,WAAW,2BAA2B;IAC1C,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,CAAC,OAAO,uBAAuB,CAAC,CAAC,MAAM,CAAC,CAAC;IACjD,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,WAAW,CAAC;CAC1B;AAED,MAAM,WAAW,2BAA2B;IAC1C,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,CAAC,OAAO,uBAAuB,CAAC,CAAC,MAAM,CAAC,CAAC;IACjD,WAAW,EAAE,WAAW,CAAC;CAC1B;AAED,MAAM,WAAW,8BAA8B;IAC7C,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,CAAC,OAAO,uBAAuB,CAAC,CAAC,MAAM,CAAC,CAAC;IACjD,WAAW,EAAE,WAAW,CAAC;CAC1B;AAED,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,qBAAsB,SAAQ,8BAA8B;IAC3E,IAAI,EAAE,MAAM,CAAC;IACb,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,CAAC,OAAO,uBAAuB,CAAC,CAAC,MAAM,CAAC,CAAC;IACjD,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,WAAW,CAAC;CAC1B;AAED,MAAM,WAAW,kBAAkB;IACjC,OAAO,EAAE,OAAO,CAAC;IACjB,YAAY,EAAE,MAAM,CAAC;IACrB,gBAAgB,EAAE,MAAM,CAAC;IACzB,0BAA0B,EAAE,OAAO,CAAC;IACpC,gBAAgB,EAAE,MAAM,CAAC;CAC1B"}
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/types.ts"],"names":[],"mappings":"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"}
package/dist/src/types.js CHANGED
@@ -1,5 +1,5 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.supportedRequestMethods = void 0;
4
- exports.supportedRequestMethods = ['get', 'post', 'put', 'del', 'patch', 'options'];
5
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQWEsUUFBQSx1QkFBdUIsR0FBRyxDQUFDLEtBQUssRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFFLEtBQUssRUFBRSxPQUFPLEVBQUUsU0FBUyxDQUFVLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3Qgc3VwcG9ydGVkUmVxdWVzdE1ldGhvZHMgPSBbJ2dldCcsICdwb3N0JywgJ3B1dCcsICdkZWwnLCAncGF0Y2gnLCAnb3B0aW9ucyddIGFzIGNvbnN0O1xuXG5leHBvcnQgdHlwZSBBdXRoVmVyc2lvbiA9IDIgfCAzO1xuXG5leHBvcnQgaW50ZXJmYWNlIENhbGN1bGF0ZUhtYWNTdWJqZWN0T3B0aW9ucyB7XG4gIHVybFBhdGg6IHN0cmluZztcbiAgdGV4dDogc3RyaW5nO1xuICB0aW1lc3RhbXA6IG51bWJlcjtcbiAgbWV0aG9kOiAodHlwZW9mIHN1cHBvcnRlZFJlcXVlc3RNZXRob2RzKVtudW1iZXJdO1xuICBzdGF0dXNDb2RlPzogbnVtYmVyO1xuICBhdXRoVmVyc2lvbjogQXV0aFZlcnNpb247XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ2FsY3VsYXRlUmVxdWVzdEhtYWNPcHRpb25zIHtcbiAgdXJsOiBzdHJpbmc7XG4gIHRleHQ6IHN0cmluZztcbiAgdGltZXN0YW1wOiBudW1iZXI7XG4gIHRva2VuOiBzdHJpbmc7XG4gIG1ldGhvZDogKHR5cGVvZiBzdXBwb3J0ZWRSZXF1ZXN0TWV0aG9kcylbbnVtYmVyXTtcbiAgYXV0aFZlcnNpb246IEF1dGhWZXJzaW9uO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENhbGN1bGF0ZVJlcXVlc3RIZWFkZXJzT3B0aW9ucyB7XG4gIHVybDogc3RyaW5nO1xuICB0ZXh0OiBzdHJpbmc7XG4gIHRva2VuOiBzdHJpbmc7XG4gIG1ldGhvZDogKHR5cGVvZiBzdXBwb3J0ZWRSZXF1ZXN0TWV0aG9kcylbbnVtYmVyXTtcbiAgYXV0aFZlcnNpb246IEF1dGhWZXJzaW9uO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFJlcXVlc3RIZWFkZXJzIHtcbiAgaG1hYzogc3RyaW5nO1xuICB0aW1lc3RhbXA6IG51bWJlcjtcbiAgdG9rZW5IYXNoOiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVmVyaWZ5UmVzcG9uc2VPcHRpb25zIGV4dGVuZHMgQ2FsY3VsYXRlUmVxdWVzdEhlYWRlcnNPcHRpb25zIHtcbiAgaG1hYzogc3RyaW5nO1xuICB1cmw6IHN0cmluZztcbiAgdGV4dDogc3RyaW5nO1xuICB0aW1lc3RhbXA6IG51bWJlcjtcbiAgbWV0aG9kOiAodHlwZW9mIHN1cHBvcnRlZFJlcXVlc3RNZXRob2RzKVtudW1iZXJdO1xuICBzdGF0dXNDb2RlPzogbnVtYmVyO1xuICBhdXRoVmVyc2lvbjogQXV0aFZlcnNpb247XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVmVyaWZ5UmVzcG9uc2VJbmZvIHtcbiAgaXNWYWxpZDogYm9vbGVhbjtcbiAgZXhwZWN0ZWRIbWFjOiBzdHJpbmc7XG4gIHNpZ25hdHVyZVN1YmplY3Q6IHN0cmluZztcbiAgaXNJblJlc3BvbnNlVmFsaWRpdHlXaW5kb3c6IGJvb2xlYW47XG4gIHZlcmlmaWNhdGlvblRpbWU6IG51bWJlcjtcbn1cbiJdfQ==
4
+ exports.supportedRequestMethods = ['get', 'post', 'put', 'del', 'patch', 'options', 'delete'];
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQWEsUUFBQSx1QkFBdUIsR0FBRyxDQUFDLEtBQUssRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFFLEtBQUssRUFBRSxPQUFPLEVBQUUsU0FBUyxFQUFFLFFBQVEsQ0FBVSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGNvbnN0IHN1cHBvcnRlZFJlcXVlc3RNZXRob2RzID0gWydnZXQnLCAncG9zdCcsICdwdXQnLCAnZGVsJywgJ3BhdGNoJywgJ29wdGlvbnMnLCAnZGVsZXRlJ10gYXMgY29uc3Q7XG5cbmV4cG9ydCB0eXBlIEF1dGhWZXJzaW9uID0gMiB8IDM7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ2FsY3VsYXRlSG1hY1N1YmplY3RPcHRpb25zPFQ+IHtcbiAgdXJsUGF0aDogc3RyaW5nO1xuICB0ZXh0OiBUO1xuICB0aW1lc3RhbXA6IG51bWJlcjtcbiAgbWV0aG9kOiAodHlwZW9mIHN1cHBvcnRlZFJlcXVlc3RNZXRob2RzKVtudW1iZXJdO1xuICBzdGF0dXNDb2RlPzogbnVtYmVyO1xuICBhdXRoVmVyc2lvbjogQXV0aFZlcnNpb247XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ2FsY3VsYXRlUmVxdWVzdEhtYWNPcHRpb25zPFQgZXh0ZW5kcyBzdHJpbmcgfCBCdWZmZXIgPSBzdHJpbmc+IHtcbiAgdXJsOiBzdHJpbmc7XG4gIHRleHQ6IFQ7XG4gIHRpbWVzdGFtcDogbnVtYmVyO1xuICB0b2tlbjogc3RyaW5nO1xuICBtZXRob2Q6ICh0eXBlb2Ygc3VwcG9ydGVkUmVxdWVzdE1ldGhvZHMpW251bWJlcl07XG4gIGF1dGhWZXJzaW9uOiBBdXRoVmVyc2lvbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBDYWxjdWxhdGVSZXF1ZXN0SGVhZGVyc09wdGlvbnM8VCBleHRlbmRzIHN0cmluZyB8IEJ1ZmZlciA9IHN0cmluZz4ge1xuICB1cmw6IHN0cmluZztcbiAgdGV4dDogVDtcbiAgdG9rZW46IHN0cmluZztcbiAgbWV0aG9kOiAodHlwZW9mIHN1cHBvcnRlZFJlcXVlc3RNZXRob2RzKVtudW1iZXJdO1xuICBhdXRoVmVyc2lvbjogQXV0aFZlcnNpb247XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVxdWVzdEhlYWRlcnMge1xuICBobWFjOiBzdHJpbmc7XG4gIHRpbWVzdGFtcDogbnVtYmVyO1xuICB0b2tlbkhhc2g6IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBWZXJpZnlSZXNwb25zZU9wdGlvbnM8VCBleHRlbmRzIHN0cmluZyB8IEJ1ZmZlciA9IHN0cmluZz4gZXh0ZW5kcyBDYWxjdWxhdGVSZXF1ZXN0SGVhZGVyc09wdGlvbnM8VD4ge1xuICBobWFjOiBzdHJpbmc7XG4gIHVybDogc3RyaW5nO1xuICB0ZXh0OiBUO1xuICB0aW1lc3RhbXA6IG51bWJlcjtcbiAgbWV0aG9kOiAodHlwZW9mIHN1cHBvcnRlZFJlcXVlc3RNZXRob2RzKVtudW1iZXJdO1xuICBzdGF0dXNDb2RlPzogbnVtYmVyO1xuICBhdXRoVmVyc2lvbjogQXV0aFZlcnNpb247XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVmVyaWZ5UmVzcG9uc2VJbmZvPFQgZXh0ZW5kcyBzdHJpbmcgfCBCdWZmZXIgPSBzdHJpbmc+IHtcbiAgaXNWYWxpZDogYm9vbGVhbjtcbiAgZXhwZWN0ZWRIbWFjOiBzdHJpbmc7XG4gIHNpZ25hdHVyZVN1YmplY3Q6IFQ7XG4gIGlzSW5SZXNwb25zZVZhbGlkaXR5V2luZG93OiBib29sZWFuO1xuICB2ZXJpZmljYXRpb25UaW1lOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ2FsY3VsYXRlVjRQcmVpbWFnZU9wdGlvbnMge1xuICB0aW1lc3RhbXBTZWM6IG51bWJlcjtcbiAgbWV0aG9kOiBzdHJpbmc7XG4gIHBhdGhXaXRoUXVlcnk6IHN0cmluZztcbiAgYm9keUhhc2hIZXg6IHN0cmluZztcbiAgYXV0aFJlcXVlc3RJZDogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENhbGN1bGF0ZVY0UmVxdWVzdEhtYWNPcHRpb25zIGV4dGVuZHMgQ2FsY3VsYXRlVjRQcmVpbWFnZU9wdGlvbnMge1xuICByYXdUb2tlbjogc3RyaW5nO1xufVxuXG5pbXBvcnQgdHlwZSB7IEhhc2hhYmxlRGF0YSB9IGZyb20gJy4vdXRpbCc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ2FsY3VsYXRlVjRSZXF1ZXN0SGVhZGVyc09wdGlvbnMge1xuICBtZXRob2Q6IHN0cmluZztcbiAgcGF0aFdpdGhRdWVyeTogc3RyaW5nO1xuICByYXdCb2R5OiBIYXNoYWJsZURhdGE7XG4gIHJhd1Rva2VuOiBzdHJpbmc7XG4gIGF1dGhSZXF1ZXN0SWQ6IHN0cmluZztcbn1cblxuLyoqXG4gKiBIZWFkZXJzIGdlbmVyYXRlZCBmb3IgVjQgYXV0aGVudGljYXRlZCByZXF1ZXN0cy5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBWNFJlcXVlc3RIZWFkZXJzIHtcbiAgaG1hYzogc3RyaW5nO1xuICB0aW1lc3RhbXBTZWM6IG51bWJlcjtcbiAgYm9keUhhc2hIZXg6IHN0cmluZztcbiAgYXV0aFJlcXVlc3RJZDogc3RyaW5nO1xufVxuXG4vKipcbiAqIE9wdGlvbnMgZm9yIHZlcmlmeWluZyBWNCByZXNwb25zZSBITUFDLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFZlcmlmeVY0UmVzcG9uc2VPcHRpb25zIHtcbiAgaG1hYzogc3RyaW5nO1xuICB0aW1lc3RhbXBTZWM6IG51bWJlcjtcbiAgbWV0aG9kOiBzdHJpbmc7XG4gIHBhdGhXaXRoUXVlcnk6IHN0cmluZztcbiAgYm9keUhhc2hIZXg6IHN0cmluZztcbiAgYXV0aFJlcXVlc3RJZDogc3RyaW5nO1xuICBzdGF0dXNDb2RlOiBudW1iZXI7XG4gIHJhd1Rva2VuOiBzdHJpbmc7XG59XG5cbi8qKlxuICogUmVzdWx0IG9mIFY0IHJlc3BvbnNlIEhNQUMgdmVyaWZpY2F0aW9uLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFZlcmlmeVY0UmVzcG9uc2VJbmZvIHtcbiAgaXNWYWxpZDogYm9vbGVhbjtcbiAgZXhwZWN0ZWRIbWFjOiBzdHJpbmc7XG4gIHByZWltYWdlOiBzdHJpbmc7XG4gIGlzSW5SZXNwb25zZVZhbGlkaXR5V2luZG93OiBib29sZWFuO1xuICB2ZXJpZmljYXRpb25UaW1lOiBudW1iZXI7XG59XG4iXX0=
@@ -0,0 +1,49 @@
1
+ import { type BinaryLike, type KeyObject } from 'crypto';
2
+ export type HashableData = string | Buffer | Uint8Array | ArrayBuffer;
3
+ /**
4
+ * Calculate SHA256 hash of data and return as lowercase hex string.
5
+ * Used for body hash calculation in v4 authentication.
6
+ *
7
+ * Accepts string, Buffer, Uint8Array, and ArrayBuffer for compatibility
8
+ * with both Node.js and browser environments.
9
+ *
10
+ * Note: ArrayBuffer is converted to Uint8Array internally since Node.js crypto
11
+ * requires TypedArray or DataView, not plain ArrayBuffer.
12
+ *
13
+ * @param data - The data to hash
14
+ * @returns Lowercase hex string of SHA256 hash
15
+ */
16
+ export declare function sha256Hex(data: HashableData): string;
17
+ /**
18
+ * Calculate HMAC-SHA256 and return as lowercase hex string.
19
+ * This is the core cryptographic primitive shared by v2/v3/v4 authentication.
20
+ *
21
+ * @param key - The secret key for HMAC
22
+ * @param message - The message to authenticate
23
+ * @returns Lowercase hex string of HMAC-SHA256
24
+ */
25
+ export declare function createHmacWithSha256(key: string | BinaryLike | KeyObject, message: string | BinaryLike): string;
26
+ /**
27
+ * Normalize HTTP method to uppercase.
28
+ * Handles legacy 'del' → 'DELETE' conversion for backward compatibility.
29
+ *
30
+ * @param method - HTTP method (case-insensitive)
31
+ * @returns Uppercase HTTP method
32
+ */
33
+ export declare function normalizeMethod(method: string): string;
34
+ /**
35
+ * Extract path with query string from a URL.
36
+ * Handles both absolute URLs and relative paths.
37
+ *
38
+ * @param urlPath - Full URL or relative path
39
+ * @returns Path with query string (e.g., '/api/v2/wallet?foo=bar')
40
+ */
41
+ export declare function extractPathWithQuery(urlPath: string): string;
42
+ /**
43
+ * Get current timestamp in seconds (Unix epoch).
44
+ * Used for v4 authentication which uses seconds instead of milliseconds.
45
+ *
46
+ * @returns Current Unix timestamp in seconds
47
+ */
48
+ export declare function getTimestampSec(): number;
49
+ //# sourceMappingURL=util.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"util.d.ts","sourceRoot":"","sources":["../../src/util.ts"],"names":[],"mappings":"AAAA,OAAO,EAA0B,KAAK,UAAU,EAAE,KAAK,SAAS,EAAE,MAAM,QAAQ,CAAC;AAEjF,MAAM,MAAM,YAAY,GAAG,MAAM,GAAG,MAAM,GAAG,UAAU,GAAG,WAAW,CAAC;AAEtE;;;;;;;;;;;;GAYG;AACH,wBAAgB,SAAS,CAAC,IAAI,EAAE,YAAY,GAAG,MAAM,CAKpD;AAED;;;;;;;GAOG;AACH,wBAAgB,oBAAoB,CAAC,GAAG,EAAE,MAAM,GAAG,UAAU,GAAG,SAAS,EAAE,OAAO,EAAE,MAAM,GAAG,UAAU,GAAG,MAAM,CAE/G;AAED;;;;;;GAMG;AACH,wBAAgB,eAAe,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,CAMtD;AAED;;;;;;GAMG;AACH,wBAAgB,oBAAoB,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,CAa5D;AAED;;;;;GAKG;AACH,wBAAgB,eAAe,IAAI,MAAM,CAExC"}
@@ -0,0 +1,85 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.sha256Hex = sha256Hex;
4
+ exports.createHmacWithSha256 = createHmacWithSha256;
5
+ exports.normalizeMethod = normalizeMethod;
6
+ exports.extractPathWithQuery = extractPathWithQuery;
7
+ exports.getTimestampSec = getTimestampSec;
8
+ const crypto_1 = require("crypto");
9
+ /**
10
+ * Calculate SHA256 hash of data and return as lowercase hex string.
11
+ * Used for body hash calculation in v4 authentication.
12
+ *
13
+ * Accepts string, Buffer, Uint8Array, and ArrayBuffer for compatibility
14
+ * with both Node.js and browser environments.
15
+ *
16
+ * Note: ArrayBuffer is converted to Uint8Array internally since Node.js crypto
17
+ * requires TypedArray or DataView, not plain ArrayBuffer.
18
+ *
19
+ * @param data - The data to hash
20
+ * @returns Lowercase hex string of SHA256 hash
21
+ */
22
+ function sha256Hex(data) {
23
+ const normalizedData = data instanceof ArrayBuffer ? new Uint8Array(data) : data;
24
+ return (0, crypto_1.createHash)('sha256')
25
+ .update(normalizedData)
26
+ .digest('hex');
27
+ }
28
+ /**
29
+ * Calculate HMAC-SHA256 and return as lowercase hex string.
30
+ * This is the core cryptographic primitive shared by v2/v3/v4 authentication.
31
+ *
32
+ * @param key - The secret key for HMAC
33
+ * @param message - The message to authenticate
34
+ * @returns Lowercase hex string of HMAC-SHA256
35
+ */
36
+ function createHmacWithSha256(key, message) {
37
+ return (0, crypto_1.createHmac)('sha256', key).update(message).digest('hex');
38
+ }
39
+ /**
40
+ * Normalize HTTP method to uppercase.
41
+ * Handles legacy 'del' → 'DELETE' conversion for backward compatibility.
42
+ *
43
+ * @param method - HTTP method (case-insensitive)
44
+ * @returns Uppercase HTTP method
45
+ */
46
+ function normalizeMethod(method) {
47
+ const lowerMethod = method.toLowerCase();
48
+ if (lowerMethod === 'del') {
49
+ return 'DELETE';
50
+ }
51
+ return method.toUpperCase();
52
+ }
53
+ /**
54
+ * Extract path with query string from a URL.
55
+ * Handles both absolute URLs and relative paths.
56
+ *
57
+ * @param urlPath - Full URL or relative path
58
+ * @returns Path with query string (e.g., '/api/v2/wallet?foo=bar')
59
+ */
60
+ function extractPathWithQuery(urlPath) {
61
+ try {
62
+ // Try parsing as absolute URL first
63
+ const url = new URL(urlPath);
64
+ return url.pathname + url.search;
65
+ }
66
+ catch {
67
+ try {
68
+ const url = new URL(urlPath, 'http://localhost');
69
+ return url.pathname + url.search;
70
+ }
71
+ catch {
72
+ return urlPath;
73
+ }
74
+ }
75
+ }
76
+ /**
77
+ * Get current timestamp in seconds (Unix epoch).
78
+ * Used for v4 authentication which uses seconds instead of milliseconds.
79
+ *
80
+ * @returns Current Unix timestamp in seconds
81
+ */
82
+ function getTimestampSec() {
83
+ return Math.floor(Date.now() / 1000);
84
+ }
85
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy91dGlsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBaUJBLDhCQUtDO0FBVUQsb0RBRUM7QUFTRCwwQ0FNQztBQVNELG9EQWFDO0FBUUQsMENBRUM7QUFqRkQsbUNBQWlGO0FBSWpGOzs7Ozs7Ozs7Ozs7R0FZRztBQUNILFNBQWdCLFNBQVMsQ0FBQyxJQUFrQjtJQUMxQyxNQUFNLGNBQWMsR0FBRyxJQUFJLFlBQVksV0FBVyxDQUFDLENBQUMsQ0FBQyxJQUFJLFVBQVUsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDO0lBQ2pGLE9BQU8sSUFBQSxtQkFBVSxFQUFDLFFBQVEsQ0FBQztTQUN4QixNQUFNLENBQUMsY0FBNEIsQ0FBQztTQUNwQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUM7QUFDbkIsQ0FBQztBQUVEOzs7Ozs7O0dBT0c7QUFDSCxTQUFnQixvQkFBb0IsQ0FBQyxHQUFvQyxFQUFFLE9BQTRCO0lBQ3JHLE9BQU8sSUFBQSxtQkFBVSxFQUFDLFFBQVEsRUFBRSxHQUFHLENBQUMsQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDO0FBQ2pFLENBQUM7QUFFRDs7Ozs7O0dBTUc7QUFDSCxTQUFnQixlQUFlLENBQUMsTUFBYztJQUM1QyxNQUFNLFdBQVcsR0FBRyxNQUFNLENBQUMsV0FBVyxFQUFFLENBQUM7SUFDekMsSUFBSSxXQUFXLEtBQUssS0FBSyxFQUFFLENBQUM7UUFDMUIsT0FBTyxRQUFRLENBQUM7SUFDbEIsQ0FBQztJQUNELE9BQU8sTUFBTSxDQUFDLFdBQVcsRUFBRSxDQUFDO0FBQzlCLENBQUM7QUFFRDs7Ozs7O0dBTUc7QUFDSCxTQUFnQixvQkFBb0IsQ0FBQyxPQUFlO0lBQ2xELElBQUksQ0FBQztRQUNILG9DQUFvQztRQUNwQyxNQUFNLEdBQUcsR0FBRyxJQUFJLEdBQUcsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUM3QixPQUFPLEdBQUcsQ0FBQyxRQUFRLEdBQUcsR0FBRyxDQUFDLE1BQU0sQ0FBQztJQUNuQyxDQUFDO0lBQUMsTUFBTSxDQUFDO1FBQ1AsSUFBSSxDQUFDO1lBQ0gsTUFBTSxHQUFHLEdBQUcsSUFBSSxHQUFHLENBQUMsT0FBTyxFQUFFLGtCQUFrQixDQUFDLENBQUM7WUFDakQsT0FBTyxHQUFHLENBQUMsUUFBUSxHQUFHLEdBQUcsQ0FBQyxNQUFNLENBQUM7UUFDbkMsQ0FBQztRQUFDLE1BQU0sQ0FBQztZQUNQLE9BQU8sT0FBTyxDQUFDO1FBQ2pCLENBQUM7SUFDSCxDQUFDO0FBQ0gsQ0FBQztBQUVEOzs7OztHQUtHO0FBQ0gsU0FBZ0IsZUFBZTtJQUM3QixPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLEdBQUcsRUFBRSxHQUFHLElBQUksQ0FBQyxDQUFDO0FBQ3ZDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjcmVhdGVIYXNoLCBjcmVhdGVIbWFjLCB0eXBlIEJpbmFyeUxpa2UsIHR5cGUgS2V5T2JqZWN0IH0gZnJvbSAnY3J5cHRvJztcblxuZXhwb3J0IHR5cGUgSGFzaGFibGVEYXRhID0gc3RyaW5nIHwgQnVmZmVyIHwgVWludDhBcnJheSB8IEFycmF5QnVmZmVyO1xuXG4vKipcbiAqIENhbGN1bGF0ZSBTSEEyNTYgaGFzaCBvZiBkYXRhIGFuZCByZXR1cm4gYXMgbG93ZXJjYXNlIGhleCBzdHJpbmcuXG4gKiBVc2VkIGZvciBib2R5IGhhc2ggY2FsY3VsYXRpb24gaW4gdjQgYXV0aGVudGljYXRpb24uXG4gKlxuICogQWNjZXB0cyBzdHJpbmcsIEJ1ZmZlciwgVWludDhBcnJheSwgYW5kIEFycmF5QnVmZmVyIGZvciBjb21wYXRpYmlsaXR5XG4gKiB3aXRoIGJvdGggTm9kZS5qcyBhbmQgYnJvd3NlciBlbnZpcm9ubWVudHMuXG4gKlxuICogTm90ZTogQXJyYXlCdWZmZXIgaXMgY29udmVydGVkIHRvIFVpbnQ4QXJyYXkgaW50ZXJuYWxseSBzaW5jZSBOb2RlLmpzIGNyeXB0b1xuICogcmVxdWlyZXMgVHlwZWRBcnJheSBvciBEYXRhVmlldywgbm90IHBsYWluIEFycmF5QnVmZmVyLlxuICpcbiAqIEBwYXJhbSBkYXRhIC0gVGhlIGRhdGEgdG8gaGFzaFxuICogQHJldHVybnMgTG93ZXJjYXNlIGhleCBzdHJpbmcgb2YgU0hBMjU2IGhhc2hcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHNoYTI1NkhleChkYXRhOiBIYXNoYWJsZURhdGEpOiBzdHJpbmcge1xuICBjb25zdCBub3JtYWxpemVkRGF0YSA9IGRhdGEgaW5zdGFuY2VvZiBBcnJheUJ1ZmZlciA/IG5ldyBVaW50OEFycmF5KGRhdGEpIDogZGF0YTtcbiAgcmV0dXJuIGNyZWF0ZUhhc2goJ3NoYTI1NicpXG4gICAgLnVwZGF0ZShub3JtYWxpemVkRGF0YSBhcyBCaW5hcnlMaWtlKVxuICAgIC5kaWdlc3QoJ2hleCcpO1xufVxuXG4vKipcbiAqIENhbGN1bGF0ZSBITUFDLVNIQTI1NiBhbmQgcmV0dXJuIGFzIGxvd2VyY2FzZSBoZXggc3RyaW5nLlxuICogVGhpcyBpcyB0aGUgY29yZSBjcnlwdG9ncmFwaGljIHByaW1pdGl2ZSBzaGFyZWQgYnkgdjIvdjMvdjQgYXV0aGVudGljYXRpb24uXG4gKlxuICogQHBhcmFtIGtleSAtIFRoZSBzZWNyZXQga2V5IGZvciBITUFDXG4gKiBAcGFyYW0gbWVzc2FnZSAtIFRoZSBtZXNzYWdlIHRvIGF1dGhlbnRpY2F0ZVxuICogQHJldHVybnMgTG93ZXJjYXNlIGhleCBzdHJpbmcgb2YgSE1BQy1TSEEyNTZcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUhtYWNXaXRoU2hhMjU2KGtleTogc3RyaW5nIHwgQmluYXJ5TGlrZSB8IEtleU9iamVjdCwgbWVzc2FnZTogc3RyaW5nIHwgQmluYXJ5TGlrZSk6IHN0cmluZyB7XG4gIHJldHVybiBjcmVhdGVIbWFjKCdzaGEyNTYnLCBrZXkpLnVwZGF0ZShtZXNzYWdlKS5kaWdlc3QoJ2hleCcpO1xufVxuXG4vKipcbiAqIE5vcm1hbGl6ZSBIVFRQIG1ldGhvZCB0byB1cHBlcmNhc2UuXG4gKiBIYW5kbGVzIGxlZ2FjeSAnZGVsJyDihpIgJ0RFTEVURScgY29udmVyc2lvbiBmb3IgYmFja3dhcmQgY29tcGF0aWJpbGl0eS5cbiAqXG4gKiBAcGFyYW0gbWV0aG9kIC0gSFRUUCBtZXRob2QgKGNhc2UtaW5zZW5zaXRpdmUpXG4gKiBAcmV0dXJucyBVcHBlcmNhc2UgSFRUUCBtZXRob2RcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIG5vcm1hbGl6ZU1ldGhvZChtZXRob2Q6IHN0cmluZyk6IHN0cmluZyB7XG4gIGNvbnN0IGxvd2VyTWV0aG9kID0gbWV0aG9kLnRvTG93ZXJDYXNlKCk7XG4gIGlmIChsb3dlck1ldGhvZCA9PT0gJ2RlbCcpIHtcbiAgICByZXR1cm4gJ0RFTEVURSc7XG4gIH1cbiAgcmV0dXJuIG1ldGhvZC50b1VwcGVyQ2FzZSgpO1xufVxuXG4vKipcbiAqIEV4dHJhY3QgcGF0aCB3aXRoIHF1ZXJ5IHN0cmluZyBmcm9tIGEgVVJMLlxuICogSGFuZGxlcyBib3RoIGFic29sdXRlIFVSTHMgYW5kIHJlbGF0aXZlIHBhdGhzLlxuICpcbiAqIEBwYXJhbSB1cmxQYXRoIC0gRnVsbCBVUkwgb3IgcmVsYXRpdmUgcGF0aFxuICogQHJldHVybnMgUGF0aCB3aXRoIHF1ZXJ5IHN0cmluZyAoZS5nLiwgJy9hcGkvdjIvd2FsbGV0P2Zvbz1iYXInKVxuICovXG5leHBvcnQgZnVuY3Rpb24gZXh0cmFjdFBhdGhXaXRoUXVlcnkodXJsUGF0aDogc3RyaW5nKTogc3RyaW5nIHtcbiAgdHJ5IHtcbiAgICAvLyBUcnkgcGFyc2luZyBhcyBhYnNvbHV0ZSBVUkwgZmlyc3RcbiAgICBjb25zdCB1cmwgPSBuZXcgVVJMKHVybFBhdGgpO1xuICAgIHJldHVybiB1cmwucGF0aG5hbWUgKyB1cmwuc2VhcmNoO1xuICB9IGNhdGNoIHtcbiAgICB0cnkge1xuICAgICAgY29uc3QgdXJsID0gbmV3IFVSTCh1cmxQYXRoLCAnaHR0cDovL2xvY2FsaG9zdCcpO1xuICAgICAgcmV0dXJuIHVybC5wYXRobmFtZSArIHVybC5zZWFyY2g7XG4gICAgfSBjYXRjaCB7XG4gICAgICByZXR1cm4gdXJsUGF0aDtcbiAgICB9XG4gIH1cbn1cblxuLyoqXG4gKiBHZXQgY3VycmVudCB0aW1lc3RhbXAgaW4gc2Vjb25kcyAoVW5peCBlcG9jaCkuXG4gKiBVc2VkIGZvciB2NCBhdXRoZW50aWNhdGlvbiB3aGljaCB1c2VzIHNlY29uZHMgaW5zdGVhZCBvZiBtaWxsaXNlY29uZHMuXG4gKlxuICogQHJldHVybnMgQ3VycmVudCBVbml4IHRpbWVzdGFtcCBpbiBzZWNvbmRzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRUaW1lc3RhbXBTZWMoKTogbnVtYmVyIHtcbiAgcmV0dXJuIE1hdGguZmxvb3IoRGF0ZS5ub3coKSAvIDEwMDApO1xufVxuIl19
package/package.json CHANGED
@@ -1,13 +1,13 @@
1
1
  {
2
2
  "name": "@bitgo-beta/sdk-hmac",
3
- "version": "1.0.1-beta.97",
3
+ "version": "1.0.1-beta.970",
4
4
  "description": "HMAC module for the BitGo SDK",
5
5
  "main": "./dist/src/index.js",
6
6
  "types": "./dist/src/index.d.ts",
7
7
  "scripts": {
8
8
  "build": "yarn tsc --build --incremental --verbose .",
9
9
  "fmt": "prettier --write .",
10
- "check-fmt": "prettier --check .",
10
+ "check-fmt": "prettier --check '**/*.{ts,js,json}'",
11
11
  "clean": "rm -r ./dist",
12
12
  "lint": "eslint --quiet .",
13
13
  "test": "npm run coverage",
@@ -16,7 +16,7 @@
16
16
  "prepare": "npm run build"
17
17
  },
18
18
  "dependencies": {
19
- "@bitgo-beta/sjcl": "1.0.2-beta.969"
19
+ "@bitgo-beta/sjcl": "1.0.2-beta.1842"
20
20
  },
21
21
  "devDependencies": {
22
22
  "chai": "^4.3.6",
@@ -46,5 +46,5 @@
46
46
  ".ts"
47
47
  ]
48
48
  },
49
- "gitHead": "c8b2277e85fcd4da3e31b82ea8e3c0ac1e98ed80"
49
+ "gitHead": "025735cf19973e1aeb7e13be2aea92a9a2f32657"
50
50
  }