@bitgo-beta/sdk-hmac 1.0.1-beta.75 → 1.0.1-beta.751
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 +29 -12
- package/dist/src/types.d.ts +11 -11
- package/dist/src/types.d.ts.map +1 -1
- package/dist/src/types.js +2 -2
- 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, }: CalculateRequestHmacOptions): string;
|
|
25
|
+
export declare function calculateRequestHMAC<T extends string | Buffer = string>({ url: urlPath, text, timestamp, token, method, authVersion, }: CalculateRequestHmacOptions<T>): 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>): 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>): 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,EAAc,KAAK,SAAS,EAAE,MAAM,QAAQ,CAAC;AAGrE,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,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,EAAE,EACvE,GAAG,EAAE,OAAO,EACZ,IAAI,EACJ,SAAS,EACT,KAAK,EACL,MAAM,EACN,WAAW,GACZ,EAAE,2BAA2B,CAAC,CAAC,CAAC,GAAG,MAAM,CAKzC;AAED;;GAEG;AACH,wBAAgB,uBAAuB,CAAC,CAAC,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,EAAE,EAC1E,GAAG,EACH,IAAI,EACJ,KAAK,EACL,MAAM,EACN,WAAW,GACZ,EAAE,8BAA8B,CAAC,CAAC,CAAC,GAAG,cAAc,CAYpD;AAED;;GAEG;AACH,wBAAgB,cAAc,CAAC,CAAC,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,EAAE,EACjE,GAAG,EAAE,OAAO,EACZ,UAAU,EACV,IAAI,EACJ,SAAS,EACT,KAAK,EACL,IAAI,EACJ,MAAM,EACN,WAAW,GACZ,EAAE,qBAAqB,CAAC,CAAC,CAAC,GAAG,kBAAkB,CAAC,CAAC,CAAC,CAyBlD"}
|
package/dist/src/hmac.js
CHANGED
|
@@ -57,21 +57,38 @@ 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
|
|
@@ -122,4 +139,4 @@ function verifyResponse({ url: urlPath, statusCode, text, timestamp, token, hmac
|
|
|
122
139
|
verificationTime: now,
|
|
123
140
|
};
|
|
124
141
|
}
|
|
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"]}
|
|
142
|
+
//# 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, createHmac, 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';\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 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 * @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,\n  text,\n  timestamp,\n  token,\n  method,\n  authVersion,\n}: CalculateRequestHmacOptions<T>): 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<T extends string | Buffer = string>({\n  url,\n  text,\n  token,\n  method,\n  authVersion,\n}: CalculateRequestHeadersOptions<T>): 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<T extends string | Buffer = string>({\n  url: urlPath,\n  statusCode,\n  text,\n  timestamp,\n  token,\n  hmac,\n  method,\n  authVersion,\n}: VerifyResponseOptions<T>): VerifyResponseInfo<T> {\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"]}
|
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,19 @@ 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
44
|
isInResponseValidityWindow: boolean;
|
|
45
45
|
verificationTime: number;
|
|
46
46
|
}
|
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":"AAAA,eAAO,MAAM,uBAAuB,sEAAuE,CAAC;AAE5G,MAAM,MAAM,WAAW,GAAG,CAAC,GAAG,CAAC,CAAC;AAEhC,MAAM,WAAW,2BAA2B,CAAC,CAAC;IAC5C,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,CAAC,CAAC;IACR,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,CAAC,CAAC,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM;IAC7E,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,CAAC,CAAC;IACR,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,CAAC,CAAC,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM;IAChF,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,CAAC,CAAC;IACR,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,qBAAqB,CAAC,CAAC,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,CAAE,SAAQ,8BAA8B,CAAC,CAAC,CAAC;IAClH,IAAI,EAAE,MAAM,CAAC;IACb,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,CAAC,CAAC;IACR,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,CAAC,CAAC,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM;IACpE,OAAO,EAAE,OAAO,CAAC;IACjB,YAAY,EAAE,MAAM,CAAC;IACrB,gBAAgB,EAAE,CAAC,CAAC;IACpB,0BAA0B,EAAE,OAAO,CAAC;IACpC,gBAAgB,EAAE,MAAM,CAAC;CAC1B"}
|
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+IHtcbiAgaXNWYWxpZDogYm9vbGVhbjtcbiAgZXhwZWN0ZWRIbWFjOiBzdHJpbmc7XG4gIHNpZ25hdHVyZVN1YmplY3Q6IFQ7XG4gIGlzSW5SZXNwb25zZVZhbGlkaXR5V2luZG93OiBib29sZWFuO1xuICB2ZXJpZmljYXRpb25UaW1lOiBudW1iZXI7XG59XG4iXX0=
|
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.751",
|
|
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.1623"
|
|
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": "ab17345250c11390525a8896efaca53bcf22b118"
|
|
50
50
|
}
|