@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.
- package/dist/src/hmac.d.ts +9 -6
- package/dist/src/hmac.d.ts.map +1 -1
- package/dist/src/hmac.js +41 -22
- package/dist/src/hmacv4.d.ts +185 -0
- package/dist/src/hmacv4.d.ts.map +1 -0
- package/dist/src/hmacv4.js +274 -0
- package/dist/src/index.d.ts +2 -0
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +3 -1
- package/dist/src/types.d.ts +61 -11
- package/dist/src/types.d.ts.map +1 -1
- package/dist/src/types.js +2 -2
- package/dist/src/util.d.ts +49 -0
- package/dist/src/util.d.ts.map +1 -0
- package/dist/src/util.js +85 -0
- package/package.json +4 -4
package/dist/src/hmac.d.ts
CHANGED
|
@@ -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
|
-
* @
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
package/dist/src/hmac.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"hmac.d.ts","sourceRoot":"","sources":["../../src/hmac.ts"],"names":[],"mappings":"
|
|
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,
|
|
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
|
-
* @
|
|
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
|
-
|
|
64
|
-
|
|
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
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
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
|
-
|
|
72
|
-
|
|
87
|
+
const isBuffer = Buffer.isBuffer(text);
|
|
88
|
+
if (isBuffer) {
|
|
89
|
+
return Buffer.concat([Buffer.from(prefixedText + '|', 'utf-8'), text]);
|
|
73
90
|
}
|
|
74
|
-
return [
|
|
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
|
|
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":";AAAA;;;;;;;;;;;;;;;GAeG;;AAoDH,kDAaC;AA6BD,8CAEC;AAwBD,wDAiBC;AA+BD,8DAyBC;AAkBD,kEAoBC;AAkBD,4CA2CC;AAWD,4CAIC;AASD,8BAGC;AA7TD,mCAAyC;AACzC,iCAOgB;AAUhB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BG;AACH,SAAgB,mBAAmB,CAAC,EAClC,YAAY,EACZ,MAAM,EACN,aAAa,EACb,WAAW,EACX,aAAa,GACc;IAC3B,MAAM,gBAAgB,GAAG,IAAA,sBAAe,EAAC,MAAM,CAAC,CAAC;IAEjD,yDAAyD;IACzD,MAAM,UAAU,GAAG,CAAC,YAAY,CAAC,QAAQ,EAAE,EAAE,gBAAgB,EAAE,aAAa,EAAE,WAAW,EAAE,aAAa,CAAC,CAAC;IAE1G,OAAO,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC;AACtC,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BG;AACH,SAAgB,iBAAiB,CAAC,IAAkB;IAClD,OAAO,IAAA,gBAAS,EAAC,IAAI,CAAC,CAAC;AACzB,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;GAqBG;AACH,SAAgB,sBAAsB,CAAC,EACrC,YAAY,EACZ,MAAM,EACN,aAAa,EACb,WAAW,EACX,aAAa,EACb,QAAQ,GACsB;IAC9B,MAAM,QAAQ,GAAG,mBAAmB,CAAC;QACnC,YAAY;QACZ,MAAM;QACN,aAAa;QACb,WAAW;QACX,aAAa;KACd,CAAC,CAAC;IAEH,OAAO,IAAA,2BAAoB,EAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;AAClD,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA4BG;AACH,SAAgB,yBAAyB,CAAC,EACxC,MAAM,EACN,aAAa,EACb,OAAO,EACP,QAAQ,EACR,aAAa,GACoB;IACjC,MAAM,YAAY,GAAG,IAAA,sBAAe,GAAE,CAAC;IACvC,MAAM,WAAW,GAAG,iBAAiB,CAAC,OAAO,CAAC,CAAC;IAE/C,MAAM,IAAI,GAAG,sBAAsB,CAAC;QAClC,YAAY;QACZ,MAAM;QACN,aAAa;QACb,WAAW;QACX,aAAa;QACb,QAAQ;KACT,CAAC,CAAC;IAEH,OAAO;QACL,IAAI;QACJ,YAAY;QACZ,WAAW;QACX,aAAa;KACd,CAAC;AACJ,CAAC;AAED;;;;;;;;;;;;;;;GAeG;AACH,SAAgB,2BAA2B,CAAC,EAC1C,YAAY,EACZ,MAAM,EACN,aAAa,EACb,UAAU,EACV,WAAW,EACX,aAAa,GACsC;IACnD,MAAM,gBAAgB,GAAG,IAAA,sBAAe,EAAC,MAAM,CAAC,CAAC;IAEjD,MAAM,UAAU,GAAG;QACjB,YAAY,CAAC,QAAQ,EAAE;QACvB,gBAAgB;QAChB,aAAa;QACb,UAAU,CAAC,QAAQ,EAAE;QACrB,WAAW;QACX,aAAa;KACd,CAAC;IAEF,OAAO,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC;AACtC,CAAC;AAED;;;;;;;;;;;;;;;GAeG;AACH,SAAgB,gBAAgB,CAAC,EAC/B,IAAI,EACJ,YAAY,EACZ,MAAM,EACN,aAAa,EACb,WAAW,EACX,aAAa,EACb,UAAU,EACV,QAAQ,GACgB;IACxB,8BAA8B;IAC9B,MAAM,QAAQ,GAAG,2BAA2B,CAAC;QAC3C,YAAY;QACZ,MAAM;QACN,aAAa;QACb,UAAU;QACV,WAAW;QACX,aAAa;KACd,CAAC,CAAC;IAEH,0BAA0B;IAC1B,MAAM,YAAY,GAAG,IAAA,2BAAoB,EAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;IAE9D,sEAAsE;IACtE,MAAM,UAAU,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;IAC5C,MAAM,kBAAkB,GAAG,MAAM,CAAC,IAAI,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC;IAC5D,MAAM,WAAW,GACf,UAAU,CAAC,MAAM,KAAK,kBAAkB,CAAC,MAAM,IAAI,IAAA,wBAAe,EAAC,UAAU,EAAE,kBAAkB,CAAC,CAAC;IAErG,kCAAkC;IAClC,MAAM,MAAM,GAAG,IAAA,sBAAe,GAAE,CAAC;IACjC,MAAM,yBAAyB,GAAG,CAAC,GAAG,EAAE,CAAC,CAAC,YAAY;IACtD,MAAM,wBAAwB,GAAG,CAAC,GAAG,EAAE,CAAC,CAAC,WAAW;IACpD,MAAM,0BAA0B,GAC9B,YAAY,IAAI,MAAM,GAAG,yBAAyB,IAAI,YAAY,IAAI,MAAM,GAAG,wBAAwB,CAAC;IAE1G,OAAO;QACL,OAAO,EAAE,WAAW;QACpB,YAAY;QACZ,QAAQ;QACR,0BAA0B;QAC1B,gBAAgB,EAAE,IAAI,CAAC,GAAG,EAAE;KAC7B,CAAC;AACJ,CAAC;AAED;;;;;;;;GAQG;AACH,SAAgB,gBAAgB,CAAC,YAAgC,EAAE,UAAkB;IACnF,sDAAsD;IACtD,MAAM,OAAO,GAAG,YAAY,IAAI,UAAU,CAAC;IAC3C,OAAO,IAAA,2BAAoB,EAAC,OAAO,CAAC,CAAC;AACvC,CAAC;AAED;;;;;;GAMG;AACH,SAAgB,SAAS,CAAC,eAAmC,EAAE,aAAqB;IAClF,yDAAyD;IACzD,OAAO,eAAe,IAAI,aAAa,CAAC;AAC1C,CAAC","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"]}
|
package/dist/src/index.d.ts
CHANGED
package/dist/src/index.d.ts.map
CHANGED
|
@@ -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,
|
|
21
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHlDQUF1QjtBQUN2QiwyQ0FBeUI7QUFDekIseUNBQXVCO0FBQ3ZCLDBDQUF3QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vaG1hYyc7XG5leHBvcnQgKiBmcm9tICcuL2htYWN2NCc7XG5leHBvcnQgKiBmcm9tICcuL3V0aWwnO1xuZXhwb3J0ICogZnJvbSAnLi90eXBlcyc7XG4iXX0=
|
package/dist/src/types.d.ts
CHANGED
|
@@ -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:
|
|
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:
|
|
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:
|
|
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:
|
|
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:
|
|
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
|
}
|
package/dist/src/types.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/types.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,uBAAuB,
|
|
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,
|
|
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"}
|
package/dist/src/util.js
ADDED
|
@@ -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.
|
|
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.
|
|
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": "
|
|
49
|
+
"gitHead": "025735cf19973e1aeb7e13be2aea92a9a2f32657"
|
|
50
50
|
}
|