@bitgo-beta/sdk-api 1.10.1-beta.503 → 1.10.1-beta.505
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/CHANGELOG.md +14 -0
- package/dist/package.json +6 -6
- package/dist/src/api.d.ts +1 -1
- package/dist/src/api.d.ts.map +1 -1
- package/dist/src/api.js +22 -19
- package/dist/src/bitgoAPI.js +35 -9
- package/dist/src/encrypt.js +25 -2
- package/dist/src/util.js +25 -2
- package/dist/src/v1/blockchain.js +14 -11
- package/dist/src/v1/keychains.js +12 -9
- package/dist/src/v1/markets.js +8 -5
- package/dist/src/v1/pendingapproval.js +43 -17
- package/dist/src/v1/pendingapprovals.js +31 -5
- package/dist/src/v1/signPsbt.js +30 -4
- package/dist/src/v1/transactionBuilder.js +103 -77
- package/dist/src/v1/travelRule.js +42 -16
- package/dist/src/v1/verifyAddress.js +25 -2
- package/dist/src/v1/wallet.js +206 -180
- package/dist/src/v1/wallets.js +48 -22
- package/package.json +6 -6
package/CHANGELOG.md
CHANGED
|
@@ -3,6 +3,20 @@
|
|
|
3
3
|
All notable changes to this project will be documented in this file.
|
|
4
4
|
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
|
|
5
5
|
|
|
6
|
+
# [1.55.0](https://github.com/BitGo/BitGoJS/compare/@bitgo/sdk-api@1.54.3...@bitgo/sdk-api@1.55.0) (2024-10-04)
|
|
7
|
+
|
|
8
|
+
### Bug Fixes
|
|
9
|
+
|
|
10
|
+
- move spec to unit-test scripto ([3a7f032](https://github.com/BitGo/BitGoJS/commit/3a7f0326d105a956597f66d2742c7c8a670ce9a7))
|
|
11
|
+
- **sdk-api:** use createPsbtDecode instead of createPsbtFromBuffer ([06d0b30](https://github.com/BitGo/BitGoJS/commit/06d0b30e6987678c8147f98781f17cdc4e9cb37b))
|
|
12
|
+
|
|
13
|
+
### Features
|
|
14
|
+
|
|
15
|
+
- add psbt signing for v1 wallets ([2c01a77](https://github.com/BitGo/BitGoJS/commit/2c01a77b42904bad9ab63cc31af4c9fdeea58f5e))
|
|
16
|
+
- **sdk-api:** assume unencrypted access token ([6b8dec1](https://github.com/BitGo/BitGoJS/commit/6b8dec1975ce99d6b144829f5a3d77a293e51d57))
|
|
17
|
+
- **sdk-api:** authenticateWithPasskey ([0dce834](https://github.com/BitGo/BitGoJS/commit/0dce834c5415d469e2332723998ac02f9ce1f98c))
|
|
18
|
+
- **sdk-api:** authenticateWithPasskey method ([4320e2b](https://github.com/BitGo/BitGoJS/commit/4320e2b57d52cd903e9d7e9721f5bbfe46eddb4f))
|
|
19
|
+
|
|
6
20
|
## [1.54.3](https://github.com/BitGo/BitGoJS/compare/@bitgo/sdk-api@1.54.2...@bitgo/sdk-api@1.54.3) (2024-09-24)
|
|
7
21
|
|
|
8
22
|
**Note:** Version bump only for package @bitgo/sdk-api
|
package/dist/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@bitgo-beta/sdk-api",
|
|
3
|
-
"version": "1.10.1-beta.
|
|
3
|
+
"version": "1.10.1-beta.505",
|
|
4
4
|
"description": "REST wrapper for BitGoJS",
|
|
5
5
|
"main": "./dist/src/index.js",
|
|
6
6
|
"types": "./dist/src/index.d.ts",
|
|
@@ -40,10 +40,10 @@
|
|
|
40
40
|
]
|
|
41
41
|
},
|
|
42
42
|
"dependencies": {
|
|
43
|
-
"@bitgo-beta/sdk-core": "8.2.1-beta.
|
|
44
|
-
"@bitgo-beta/sjcl": "1.0.2-beta.
|
|
45
|
-
"@bitgo-beta/unspents": "0.13.2-beta.
|
|
46
|
-
"@bitgo-beta/utxo-lib": "8.0.3-beta.
|
|
43
|
+
"@bitgo-beta/sdk-core": "8.2.1-beta.506",
|
|
44
|
+
"@bitgo-beta/sjcl": "1.0.2-beta.744",
|
|
45
|
+
"@bitgo-beta/unspents": "0.13.2-beta.506",
|
|
46
|
+
"@bitgo-beta/utxo-lib": "8.0.3-beta.507",
|
|
47
47
|
"@types/superagent": "4.1.15",
|
|
48
48
|
"bitcoinjs-message": "npm:@bitgo-forks/bitcoinjs-message@1.0.0-master.2",
|
|
49
49
|
"bluebird": "^3.5.3",
|
|
@@ -63,5 +63,5 @@
|
|
|
63
63
|
"resolutions": {
|
|
64
64
|
"degenerator": "5.0.0"
|
|
65
65
|
},
|
|
66
|
-
"gitHead": "
|
|
66
|
+
"gitHead": "3862d6687dab985ca7d8ce9d5f76b454e363e43b"
|
|
67
67
|
}
|
package/dist/src/api.d.ts
CHANGED
package/dist/src/api.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"api.d.ts","sourceRoot":"","sources":["../../src/api.ts"],"names":[],"mappings":"AAOA,OAAO,
|
|
1
|
+
{"version":3,"file":"api.d.ts","sourceRoot":"","sources":["../../src/api.ts"],"names":[],"mappings":"AAOA,OAAO,UAAU,MAAM,YAAY,CAAC;AAIpC,OAAO,EAAoB,YAAY,EAAE,MAAM,sBAAsB,CAAC;AAEtE,OAAO,EAAE,qBAAqB,EAAE,MAAM,SAAS,CAAC;AAChD,OAAO,EAAE,QAAQ,EAAE,MAAM,YAAY,CAAC;AAItC;;;;;;;;;;GAUG;AACH,wBAAgB,cAAc,CAAC,kBAAkB,GAAG,GAAG,EACrD,GAAG,EAAE,UAAU,CAAC,iBAAiB,GAChC,YAAY,CAAC,kBAAkB,CAAC,CASlC;AAED;;;;GAIG;AACH,wBAAgB,oBAAoB,CAAC,kBAAkB,EACrD,aAAa,CAAC,EAAE,MAAM,GACrB,CAAC,GAAG,EAAE,UAAU,CAAC,QAAQ,KAAK,kBAAkB,CAOlD;AAkBD;;;;GAIG;AACH,wBAAgB,mBAAmB,CAAC,CAAC,EAAE,KAAK,GAAG;IAAE,QAAQ,CAAC,EAAE,UAAU,CAAC,QAAQ,CAAA;CAAE,GAAG,KAAK,CAKxF;AAgCD;;;;GAIG;AACH,wBAAgB,oBAAoB,CAAC,GAAG,EAAE,UAAU,CAAC,OAAO,GAAG,MAAM,GAAG,SAAS,CAkBhF;AAED;;;GAGG;AACH,wBAAgB,qBAAqB,CAAC,GAAG,EAAE,UAAU,CAAC,iBAAiB,GAAG,IAAI,CAwB7E;AAED;;;GAGG;AACH,wBAAgB,cAAc,CAC5B,KAAK,EAAE,QAAQ,EACf,KAAK,EAAE,MAAM,GAAG,SAAS,EACzB,MAAM,EAAE,qBAAqB,CAAC,QAAQ,CAAC,EACvC,GAAG,EAAE,UAAU,CAAC,iBAAiB,EACjC,QAAQ,EAAE,UAAU,CAAC,QAAQ,GAC5B,UAAU,CAAC,QAAQ,CA+CrB"}
|
package/dist/src/api.js
CHANGED
|
@@ -1,16 +1,19 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
2
5
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
6
|
exports.verifyResponse = exports.setRequestQueryString = exports.serializeRequestData = exports.handleResponseError = exports.handleResponseResult = exports.toBitgoRequest = void 0;
|
|
4
7
|
/**
|
|
5
8
|
* @prettier
|
|
6
9
|
*/
|
|
7
|
-
const debug_1 = require("debug");
|
|
8
|
-
const
|
|
9
|
-
const
|
|
10
|
-
const
|
|
11
|
-
const
|
|
12
|
-
const
|
|
13
|
-
const
|
|
10
|
+
const debug_1 = __importDefault(require("debug"));
|
|
11
|
+
const eol_1 = __importDefault(require("eol"));
|
|
12
|
+
const lodash_1 = __importDefault(require("lodash"));
|
|
13
|
+
const sanitize_html_1 = __importDefault(require("sanitize-html"));
|
|
14
|
+
const superagent_1 = __importDefault(require("superagent"));
|
|
15
|
+
const url_1 = __importDefault(require("url"));
|
|
16
|
+
const querystring_1 = __importDefault(require("querystring"));
|
|
14
17
|
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
15
18
|
const debug = (0, debug_1.default)('bitgo:api');
|
|
16
19
|
/**
|
|
@@ -39,7 +42,7 @@ exports.toBitgoRequest = toBitgoRequest;
|
|
|
39
42
|
*/
|
|
40
43
|
function handleResponseResult(optionalField) {
|
|
41
44
|
return function (res) {
|
|
42
|
-
if (
|
|
45
|
+
if (lodash_1.default.isNumber(res.status) && res.status >= 200 && res.status < 300) {
|
|
43
46
|
return optionalField ? res.body[optionalField] : res.body;
|
|
44
47
|
}
|
|
45
48
|
throw errFromResponse(res);
|
|
@@ -58,7 +61,7 @@ function errFromResponse(res) {
|
|
|
58
61
|
const message = createResponseErrorString(res);
|
|
59
62
|
const status = res.status;
|
|
60
63
|
const result = res.body;
|
|
61
|
-
const invalidToken =
|
|
64
|
+
const invalidToken = lodash_1.default.has(res.header, 'x-auth-required') && res.header['x-auth-required'] === 'true';
|
|
62
65
|
const needsOtp = ((_a = res.body) === null || _a === void 0 ? void 0 : _a.needsOTP) !== undefined;
|
|
63
66
|
return new sdk_core_1.ApiResponseError(message, status, result, invalidToken, needsOtp);
|
|
64
67
|
}
|
|
@@ -90,11 +93,11 @@ function createResponseErrorString(res) {
|
|
|
90
93
|
// if the response came back as text, we try to parse it as HTML and remove all tags, leaving us
|
|
91
94
|
// just the bare text, which we then trim of excessive newlines and limit to a certain length
|
|
92
95
|
try {
|
|
93
|
-
let sanitizedText =
|
|
96
|
+
let sanitizedText = (0, sanitize_html_1.default)(res.text, { allowedTags: [] });
|
|
94
97
|
sanitizedText = sanitizedText.trim();
|
|
95
|
-
sanitizedText =
|
|
96
|
-
sanitizedText =
|
|
97
|
-
sanitizedText =
|
|
98
|
+
sanitizedText = eol_1.default.lf(sanitizedText); // use '\n' for all newlines
|
|
99
|
+
sanitizedText = lodash_1.default.replace(sanitizedText, /\n[ |\t]{1,}\n/g, '\n\n'); // remove the spaces/tabs between newlines
|
|
100
|
+
sanitizedText = lodash_1.default.replace(sanitizedText, /[\n]{3,}/g, '\n\n'); // have at most 2 consecutive newlines
|
|
98
101
|
sanitizedText = sanitizedText.substring(0, 5000); // prevent message from getting too large
|
|
99
102
|
errString = errString + '\n' + sanitizedText; // add it to our existing errString (at this point the more info the better!)
|
|
100
103
|
}
|
|
@@ -118,9 +121,9 @@ function serializeRequestData(req) {
|
|
|
118
121
|
if (contentType) {
|
|
119
122
|
contentType = contentType.split(';')[0];
|
|
120
123
|
}
|
|
121
|
-
let serialize =
|
|
124
|
+
let serialize = superagent_1.default.serialize[contentType];
|
|
122
125
|
if (!serialize && /[\/+]json\b/.test(contentType)) {
|
|
123
|
-
serialize =
|
|
126
|
+
serialize = superagent_1.default.serialize['application/json'];
|
|
124
127
|
}
|
|
125
128
|
if (serialize) {
|
|
126
129
|
data = serialize(data);
|
|
@@ -135,7 +138,7 @@ exports.serializeRequestData = serializeRequestData;
|
|
|
135
138
|
* @param req
|
|
136
139
|
*/
|
|
137
140
|
function setRequestQueryString(req) {
|
|
138
|
-
const urlDetails =
|
|
141
|
+
const urlDetails = url_1.default.parse(req.url);
|
|
139
142
|
let queryString;
|
|
140
143
|
const query = req._query;
|
|
141
144
|
const qs = req.qs;
|
|
@@ -146,7 +149,7 @@ function setRequestQueryString(req) {
|
|
|
146
149
|
}
|
|
147
150
|
else if (qs) {
|
|
148
151
|
// node version
|
|
149
|
-
queryString =
|
|
152
|
+
queryString = querystring_1.default.stringify(qs);
|
|
150
153
|
req.qs = null;
|
|
151
154
|
}
|
|
152
155
|
if (queryString) {
|
|
@@ -156,7 +159,7 @@ function setRequestQueryString(req) {
|
|
|
156
159
|
else {
|
|
157
160
|
urlDetails.search = '?' + queryString;
|
|
158
161
|
}
|
|
159
|
-
req.url =
|
|
162
|
+
req.url = url_1.default.format(urlDetails);
|
|
160
163
|
}
|
|
161
164
|
}
|
|
162
165
|
exports.setRequestQueryString = setRequestQueryString;
|
|
@@ -206,4 +209,4 @@ function verifyResponse(bitgo, token, method, req, response) {
|
|
|
206
209
|
return response;
|
|
207
210
|
}
|
|
208
211
|
exports.verifyResponse = verifyResponse;
|
|
209
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api.js","sourceRoot":"","sources":["../../src/api.ts"],"names":[],"mappings":";;;AAAA;;GAEG;AACH,iCAA0B;AAC1B,2BAA2B;AAC3B,4BAA4B;AAC5B,8CAA8C;AAC9C,yCAAyC;AACzC,8BAA8B;AAC9B,2CAA2C;AAE3C,mDAAsE;AAKtE,MAAM,KAAK,GAAG,IAAA,eAAK,EAAC,WAAW,CAAC,CAAC;AAEjC;;;;;;;;;;GAUG;AACH,SAAgB,cAAc,CAC5B,GAAiC;IAEjC,OAAO,MAAM,CAAC,MAAM,CAAC,GAAG,EAAE;QACxB,MAAM,CAAC,aAAsB;YAC3B,OAAO,GAAG,CAAC,IAAI,CACb,CAAC,QAAQ,EAAE,EAAE,CAAC,oBAAoB,CAAqB,aAAa,CAAC,CAAC,QAAQ,CAAC,EAC/E,CAAC,KAAK,EAAE,EAAE,CAAC,mBAAmB,CAAC,KAAK,CAAC,CACtC,CAAC;QACJ,CAAC;KACF,CAAC,CAAC;AACL,CAAC;AAXD,wCAWC;AAED;;;;GAIG;AACH,SAAgB,oBAAoB,CAClC,aAAsB;IAEtB,OAAO,UAAU,GAAwB;QACvC,IAAI,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,GAAG,CAAC,MAAM,IAAI,GAAG,IAAI,GAAG,CAAC,MAAM,GAAG,GAAG,EAAE;YACnE,OAAO,aAAa,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC;SAC3D;QACD,MAAM,eAAe,CAAC,GAAG,CAAC,CAAC;IAC7B,CAAC,CAAC;AACJ,CAAC;AATD,oDASC;AAED;;;;;;GAMG;AACH,SAAS,eAAe,CAAmB,GAAwB;;IACjE,MAAM,OAAO,GAAG,yBAAyB,CAAC,GAAG,CAAC,CAAC;IAC/C,MAAM,MAAM,GAAG,GAAG,CAAC,MAAM,CAAC;IAC1B,MAAM,MAAM,GAAG,GAAG,CAAC,IAAwB,CAAC;IAC5C,MAAM,YAAY,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,EAAE,iBAAiB,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,iBAAiB,CAAC,KAAK,MAAM,CAAC;IACtG,MAAM,QAAQ,GAAG,CAAA,MAAA,GAAG,CAAC,IAAI,0CAAE,QAAQ,MAAK,SAAS,CAAC;IAClD,OAAO,IAAI,2BAAgB,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,YAAY,EAAE,QAAQ,CAAC,CAAC;AAC/E,CAAC;AAED;;;;GAIG;AACH,SAAgB,mBAAmB,CAAC,CAA6C;IAC/E,IAAI,CAAC,CAAC,QAAQ,EAAE;QACd,MAAM,eAAe,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;KACnC;IACD,MAAM,CAAC,CAAC;AACV,CAAC;AALD,kDAKC;AAED;;;;GAIG;AACH,SAAS,yBAAyB,CAAC,GAAwB;;IACzD,IAAI,SAAS,GAAG,GAAG,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC,+CAA+C;IACtF,IAAI,MAAA,GAAG,CAAC,IAAI,0CAAE,KAAK,EAAE;QACnB,0FAA0F;QAC1F,SAAS,GAAG,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC;KAC5B;SAAM,IAAI,GAAG,CAAC,IAAI,EAAE;QACnB,gGAAgG;QAChG,6FAA6F;QAC7F,IAAI;YACF,IAAI,aAAa,GAAG,YAAY,CAAC,GAAG,CAAC,IAAI,EAAE,EAAE,WAAW,EAAE,EAAE,EAAE,CAAC,CAAC;YAChE,aAAa,GAAG,aAAa,CAAC,IAAI,EAAE,CAAC;YACrC,aAAa,GAAG,GAAG,CAAC,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC,4BAA4B;YACnE,aAAa,GAAG,CAAC,CAAC,OAAO,CAAC,aAAa,EAAE,iBAAiB,EAAE,MAAM,CAAC,CAAC,CAAC,0CAA0C;YAC/G,aAAa,GAAG,CAAC,CAAC,OAAO,CAAC,aAAa,EAAE,WAAW,EAAE,MAAM,CAAC,CAAC,CAAC,sCAAsC;YACrG,aAAa,GAAG,aAAa,CAAC,SAAS,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,yCAAyC;YAC3F,SAAS,GAAG,SAAS,GAAG,IAAI,GAAG,aAAa,CAAC,CAAC,6EAA6E;SAC5H;QAAC,OAAO,CAAC,EAAE;YACV,qEAAqE;YACrE,KAAK,CAAC,oFAAoF,EAAE,CAAC,CAAC,OAAO,EAAE,GAAG,CAAC,IAAI,CAAC,CAAC;SAClH;KACF;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;;;GAIG;AACH,SAAgB,oBAAoB,CAAC,GAAuB;IAC1D,IAAI,IAAI,GAAsC,GAAW,CAAC,KAAK,CAAC;IAChE,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE;QAC5B,IAAI,WAAW,GAAG,GAAG,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;QAC1C,uEAAuE;QACvE,IAAI,WAAW,EAAE;YACf,WAAW,GAAG,WAAW,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;SACzC;QACD,IAAI,SAAS,GAAG,UAAU,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC;QAClD,IAAI,CAAC,SAAS,IAAI,aAAa,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE;YACjD,SAAS,GAAG,UAAU,CAAC,SAAS,CAAC,kBAAkB,CAAC,CAAC;SACtD;QACD,IAAI,SAAS,EAAE;YACb,IAAI,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;YACtB,GAAW,CAAC,KAAK,GAAG,IAAI,CAAC;YAC1B,OAAO,IAAI,CAAC;SACb;KACF;AACH,CAAC;AAlBD,oDAkBC;AAED;;;GAGG;AACH,SAAgB,qBAAqB,CAAC,GAAiC;IACrE,MAAM,UAAU,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IAEzC,IAAI,WAA+B,CAAC;IACpC,MAAM,KAAK,GAAc,GAAW,CAAC,MAAM,CAAC;IAC5C,MAAM,EAAE,GAA+B,GAAW,CAAC,EAAE,CAAC;IACtD,IAAI,KAAK,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE;QAC7B,kBAAkB;QAClB,WAAW,GAAG,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC7B,GAAW,CAAC,MAAM,GAAG,EAAE,CAAC;KAC1B;SAAM,IAAI,EAAE,EAAE;QACb,eAAe;QACf,WAAW,GAAG,WAAW,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;QACvC,GAAW,CAAC,EAAE,GAAG,IAAI,CAAC;KACxB;IAED,IAAI,WAAW,EAAE;QACf,IAAI,UAAU,CAAC,MAAM,EAAE;YACrB,UAAU,CAAC,MAAM,IAAI,GAAG,GAAG,WAAW,CAAC;SACxC;aAAM;YACL,UAAU,CAAC,MAAM,GAAG,GAAG,GAAG,WAAW,CAAC;SACvC;QACD,GAAG,CAAC,GAAG,GAAG,MAAM,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;KACrC;AACH,CAAC;AAxBD,sDAwBC;AAED;;;GAGG;AACH,SAAgB,cAAc,CAC5B,KAAe,EACf,KAAyB,EACzB,MAAuC,EACvC,GAAiC,EACjC,QAA6B;IAE7B,0DAA0D;IAC1D,IAAI,CAAC,GAAG,CAAC,iBAAiB,IAAI,CAAC,GAAG,CAAC,mBAAmB,EAAE;QACtD,OAAO,QAAQ,CAAC;KACjB;IAED,MAAM,oBAAoB,GAAG,KAAK,CAAC,cAAc,CAAC;QAChD,GAAG,EAAE,GAAG,CAAC,GAAG;QACZ,IAAI,EAAE,QAAQ,CAAC,MAAM,CAAC,IAAI;QAC1B,UAAU,EAAE,QAAQ,CAAC,MAAM;QAC3B,IAAI,EAAE,QAAQ,CAAC,IAAI;QACnB,SAAS,EAAE,QAAQ,CAAC,MAAM,CAAC,SAAS;QACpC,KAAK,EAAE,GAAG,CAAC,mBAAmB;QAC9B,MAAM;KACP,CAAC,CAAC;IAEH,IAAI,CAAC,oBAAoB,CAAC,OAAO,EAAE;QACjC,qBAAqB;QACrB,MAAM,YAAY,GAAG,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC;QAC1C,MAAM,YAAY,GAAG,oBAAoB,CAAC,YAAY,CAAC;QACvD,MAAM,gBAAgB,GAAG,oBAAoB,CAAC,gBAAgB,CAAC;QAC/D,uFAAuF;QACvF,MAAM,iBAAiB,GAAG,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QAC9D,MAAM,YAAY,GAAG;YACnB,YAAY;YACZ,YAAY;YACZ,SAAS,EAAE,gBAAgB;YAC3B,YAAY,EAAE,GAAG,CAAC,mBAAmB;YACrC,UAAU,EAAE,iBAAiB;SAC9B,CAAC;QACF,KAAK,CAAC,2BAA2B,EAAE,YAAY,CAAC,CAAC;QACjD,MAAM,IAAI,2BAAgB,CAAC,0DAA0D,EAAE,GAAG,EAAE,YAAY,CAAC,CAAC;KAC3G;IAED,IAAI,KAAK,CAAC,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,oBAAoB,CAAC,0BAA0B,EAAE;QACpF,MAAM,YAAY,GAAG;YACnB,SAAS,EAAE,QAAQ,CAAC,MAAM,CAAC,SAAS;YACpC,gBAAgB,EAAE,oBAAoB,CAAC,gBAAgB;SACxD,CAAC;QACF,KAAK,CAAC,2DAA2D,EAAE,YAAY,CAAC,CAAC;QACjF,MAAM,IAAI,2BAAgB,CACxB,0FAA0F,EAC1F,GAAG,EACH,YAAY,CACb,CAAC;KACH;IACD,OAAO,QAAQ,CAAC;AAClB,CAAC;AArDD,wCAqDC","sourcesContent":["/**\n * @prettier\n */\nimport Debug from 'debug';\nimport * as eol from 'eol';\nimport * as _ from 'lodash';\nimport * as sanitizeHtml from 'sanitize-html';\nimport * as superagent from 'superagent';\nimport * as urlLib from 'url';\nimport * as querystring from 'querystring';\n\nimport { ApiResponseError, BitGoRequest } from '@bitgo-beta/sdk-core';\n\nimport { VerifyResponseOptions } from './types';\nimport { BitGoAPI } from './bitgoAPI';\n\nconst debug = Debug('bitgo:api');\n\n/**\n * Add the bitgo-specific result() function on a superagent request.\n *\n * If the server response is successful, the `result()` function will return either the entire response body,\n * or the field from the response body specified by the `optionalField` parameter if it is provided.\n *\n * If the server response with an error, `result()` will handle HTTP errors appropriately by\n * rethrowing them as an `ApiResponseError` if possible, and otherwise rethrowing the underlying response error.\n *\n * @param req\n */\nexport function toBitgoRequest<ResponseResultType = any>(\n  req: superagent.SuperAgentRequest\n): BitGoRequest<ResponseResultType> {\n  return Object.assign(req, {\n    result(optionalField?: string) {\n      return req.then(\n        (response) => handleResponseResult<ResponseResultType>(optionalField)(response),\n        (error) => handleResponseError(error)\n      );\n    },\n  });\n}\n\n/**\n * Return a function which extracts the specified response body property from the response if successful,\n * otherwise throw an `ApiErrorResponse` parsed from the response body.\n * @param optionalField\n */\nexport function handleResponseResult<ResponseResultType>(\n  optionalField?: string\n): (res: superagent.Response) => ResponseResultType {\n  return function (res: superagent.Response): ResponseResultType {\n    if (_.isNumber(res.status) && res.status >= 200 && res.status < 300) {\n      return optionalField ? res.body[optionalField] : res.body;\n    }\n    throw errFromResponse(res);\n  };\n}\n\n/**\n * Extract relevant information from a successful response (that is, a response with an HTTP status code\n * between 200 and 299), but which resulted in an application specific error and use it to construct and\n * throw an `ApiErrorResponse`.\n *\n * @param res\n */\nfunction errFromResponse<ResponseBodyType>(res: superagent.Response): ApiResponseError {\n  const message = createResponseErrorString(res);\n  const status = res.status;\n  const result = res.body as ResponseBodyType;\n  const invalidToken = _.has(res.header, 'x-auth-required') && res.header['x-auth-required'] === 'true';\n  const needsOtp = res.body?.needsOTP !== undefined;\n  return new ApiResponseError(message, status, result, invalidToken, needsOtp);\n}\n\n/**\n * Handle an error or an error containing an HTTP response and use it to throw a well-formed error object.\n *\n * @param e\n */\nexport function handleResponseError(e: Error & { response?: superagent.Response }): never {\n  if (e.response) {\n    throw errFromResponse(e.response);\n  }\n  throw e;\n}\n\n/**\n * There are many ways a request can fail, and may ways information on that failure can be\n * communicated to the client. This function tries to handle those cases and create a sane error string\n * @param res Response from an HTTP request\n */\nfunction createResponseErrorString(res: superagent.Response): string {\n  let errString = res.status.toString(); // at the very least we'll have the status code\n  if (res.body?.error) {\n    // this is the case we hope for, where the server gives us a nice error from the JSON body\n    errString = res.body.error;\n  } else if (res.text) {\n    // if the response came back as text, we try to parse it as HTML and remove all tags, leaving us\n    // just the bare text, which we then trim of excessive newlines and limit to a certain length\n    try {\n      let sanitizedText = sanitizeHtml(res.text, { allowedTags: [] });\n      sanitizedText = sanitizedText.trim();\n      sanitizedText = eol.lf(sanitizedText); // use '\\n' for all newlines\n      sanitizedText = _.replace(sanitizedText, /\\n[ |\\t]{1,}\\n/g, '\\n\\n'); // remove the spaces/tabs between newlines\n      sanitizedText = _.replace(sanitizedText, /[\\n]{3,}/g, '\\n\\n'); // have at most 2 consecutive newlines\n      sanitizedText = sanitizedText.substring(0, 5000); // prevent message from getting too large\n      errString = errString + '\\n' + sanitizedText; // add it to our existing errString (at this point the more info the better!)\n    } catch (e) {\n      // do nothing, the response's HTML was too wacky to be parsed cleanly\n      debug('got error with message \"%s\" while creating response error string from response: %s', e.message, res.text);\n    }\n  }\n\n  return errString;\n}\n\n/**\n * Serialize request data based on the request content type\n * Note: Not sure this is still needed or even useful. Consider removing.\n * @param req\n */\nexport function serializeRequestData(req: superagent.Request): string | undefined {\n  let data: string | Record<string, unknown> = (req as any)._data;\n  if (typeof data !== 'string') {\n    let contentType = req.get('Content-Type');\n    // Parse out just the content type from the header (ignore the charset)\n    if (contentType) {\n      contentType = contentType.split(';')[0];\n    }\n    let serialize = superagent.serialize[contentType];\n    if (!serialize && /[\\/+]json\\b/.test(contentType)) {\n      serialize = superagent.serialize['application/json'];\n    }\n    if (serialize) {\n      data = serialize(data);\n      (req as any)._data = data;\n      return data;\n    }\n  }\n}\n\n/**\n * Set the superagent query string correctly for browsers or node.\n * @param req\n */\nexport function setRequestQueryString(req: superagent.SuperAgentRequest): void {\n  const urlDetails = urlLib.parse(req.url);\n\n  let queryString: string | undefined;\n  const query: string[] = (req as any)._query;\n  const qs: { [key: string]: string } = (req as any).qs;\n  if (query && query.length > 0) {\n    // browser version\n    queryString = query.join('&');\n    (req as any)._query = [];\n  } else if (qs) {\n    // node version\n    queryString = querystring.stringify(qs);\n    (req as any).qs = null;\n  }\n\n  if (queryString) {\n    if (urlDetails.search) {\n      urlDetails.search += '&' + queryString;\n    } else {\n      urlDetails.search = '?' + queryString;\n    }\n    req.url = urlLib.format(urlDetails);\n  }\n}\n\n/**\n * Verify that the response received from the server is signed correctly.\n * Right now, it is very permissive with the timestamp variance.\n */\nexport function verifyResponse(\n  bitgo: BitGoAPI,\n  token: string | undefined,\n  method: VerifyResponseOptions['method'],\n  req: superagent.SuperAgentRequest,\n  response: superagent.Response\n): superagent.Response {\n  // we can't verify the response if we're not authenticated\n  if (!req.isV2Authenticated || !req.authenticationToken) {\n    return response;\n  }\n\n  const verificationResponse = bitgo.verifyResponse({\n    url: req.url,\n    hmac: response.header.hmac,\n    statusCode: response.status,\n    text: response.text,\n    timestamp: response.header.timestamp,\n    token: req.authenticationToken,\n    method,\n  });\n\n  if (!verificationResponse.isValid) {\n    // calculate the HMAC\n    const receivedHmac = response.header.hmac;\n    const expectedHmac = verificationResponse.expectedHmac;\n    const signatureSubject = verificationResponse.signatureSubject;\n    // Log only the first 10 characters of the token to ensure the full token isn't logged.\n    const partialBitgoToken = token ? token.substring(0, 10) : '';\n    const errorDetails = {\n      expectedHmac,\n      receivedHmac,\n      hmacInput: signatureSubject,\n      requestToken: req.authenticationToken,\n      bitgoToken: partialBitgoToken,\n    };\n    debug('Invalid response HMAC: %O', errorDetails);\n    throw new ApiResponseError('invalid response HMAC, possible man-in-the-middle-attack', 511, errorDetails);\n  }\n\n  if (bitgo.getAuthVersion() === 3 && !verificationResponse.isInResponseValidityWindow) {\n    const errorDetails = {\n      timestamp: response.header.timestamp,\n      verificationTime: verificationResponse.verificationTime,\n    };\n    debug('Server response outside response validity time window: %O', errorDetails);\n    throw new ApiResponseError(\n      'server response outside response validity time window, possible man-in-the-middle-attack',\n      511,\n      errorDetails\n    );\n  }\n  return response;\n}\n"]}
|
|
212
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api.js","sourceRoot":"","sources":["../../src/api.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport Debug from 'debug';\nimport eol from 'eol';\nimport _ from 'lodash';\nimport sanitizeHtml from 'sanitize-html';\nimport superagent from 'superagent';\nimport urlLib from 'url';\nimport querystring from 'querystring';\n\nimport { ApiResponseError, BitGoRequest } from '@bitgo-beta/sdk-core';\n\nimport { VerifyResponseOptions } from './types';\nimport { BitGoAPI } from './bitgoAPI';\n\nconst debug = Debug('bitgo:api');\n\n/**\n * Add the bitgo-specific result() function on a superagent request.\n *\n * If the server response is successful, the `result()` function will return either the entire response body,\n * or the field from the response body specified by the `optionalField` parameter if it is provided.\n *\n * If the server response with an error, `result()` will handle HTTP errors appropriately by\n * rethrowing them as an `ApiResponseError` if possible, and otherwise rethrowing the underlying response error.\n *\n * @param req\n */\nexport function toBitgoRequest<ResponseResultType = any>(\n  req: superagent.SuperAgentRequest\n): BitGoRequest<ResponseResultType> {\n  return Object.assign(req, {\n    result(optionalField?: string) {\n      return req.then(\n        (response) => handleResponseResult<ResponseResultType>(optionalField)(response),\n        (error) => handleResponseError(error)\n      );\n    },\n  });\n}\n\n/**\n * Return a function which extracts the specified response body property from the response if successful,\n * otherwise throw an `ApiErrorResponse` parsed from the response body.\n * @param optionalField\n */\nexport function handleResponseResult<ResponseResultType>(\n  optionalField?: string\n): (res: superagent.Response) => ResponseResultType {\n  return function (res: superagent.Response): ResponseResultType {\n    if (_.isNumber(res.status) && res.status >= 200 && res.status < 300) {\n      return optionalField ? res.body[optionalField] : res.body;\n    }\n    throw errFromResponse(res);\n  };\n}\n\n/**\n * Extract relevant information from a successful response (that is, a response with an HTTP status code\n * between 200 and 299), but which resulted in an application specific error and use it to construct and\n * throw an `ApiErrorResponse`.\n *\n * @param res\n */\nfunction errFromResponse<ResponseBodyType>(res: superagent.Response): ApiResponseError {\n  const message = createResponseErrorString(res);\n  const status = res.status;\n  const result = res.body as ResponseBodyType;\n  const invalidToken = _.has(res.header, 'x-auth-required') && res.header['x-auth-required'] === 'true';\n  const needsOtp = res.body?.needsOTP !== undefined;\n  return new ApiResponseError(message, status, result, invalidToken, needsOtp);\n}\n\n/**\n * Handle an error or an error containing an HTTP response and use it to throw a well-formed error object.\n *\n * @param e\n */\nexport function handleResponseError(e: Error & { response?: superagent.Response }): never {\n  if (e.response) {\n    throw errFromResponse(e.response);\n  }\n  throw e;\n}\n\n/**\n * There are many ways a request can fail, and may ways information on that failure can be\n * communicated to the client. This function tries to handle those cases and create a sane error string\n * @param res Response from an HTTP request\n */\nfunction createResponseErrorString(res: superagent.Response): string {\n  let errString = res.status.toString(); // at the very least we'll have the status code\n  if (res.body?.error) {\n    // this is the case we hope for, where the server gives us a nice error from the JSON body\n    errString = res.body.error;\n  } else if (res.text) {\n    // if the response came back as text, we try to parse it as HTML and remove all tags, leaving us\n    // just the bare text, which we then trim of excessive newlines and limit to a certain length\n    try {\n      let sanitizedText = sanitizeHtml(res.text, { allowedTags: [] });\n      sanitizedText = sanitizedText.trim();\n      sanitizedText = eol.lf(sanitizedText); // use '\\n' for all newlines\n      sanitizedText = _.replace(sanitizedText, /\\n[ |\\t]{1,}\\n/g, '\\n\\n'); // remove the spaces/tabs between newlines\n      sanitizedText = _.replace(sanitizedText, /[\\n]{3,}/g, '\\n\\n'); // have at most 2 consecutive newlines\n      sanitizedText = sanitizedText.substring(0, 5000); // prevent message from getting too large\n      errString = errString + '\\n' + sanitizedText; // add it to our existing errString (at this point the more info the better!)\n    } catch (e) {\n      // do nothing, the response's HTML was too wacky to be parsed cleanly\n      debug('got error with message \"%s\" while creating response error string from response: %s', e.message, res.text);\n    }\n  }\n\n  return errString;\n}\n\n/**\n * Serialize request data based on the request content type\n * Note: Not sure this is still needed or even useful. Consider removing.\n * @param req\n */\nexport function serializeRequestData(req: superagent.Request): string | undefined {\n  let data: string | Record<string, unknown> = (req as any)._data;\n  if (typeof data !== 'string') {\n    let contentType = req.get('Content-Type');\n    // Parse out just the content type from the header (ignore the charset)\n    if (contentType) {\n      contentType = contentType.split(';')[0];\n    }\n    let serialize = superagent.serialize[contentType];\n    if (!serialize && /[\\/+]json\\b/.test(contentType)) {\n      serialize = superagent.serialize['application/json'];\n    }\n    if (serialize) {\n      data = serialize(data);\n      (req as any)._data = data;\n      return data;\n    }\n  }\n}\n\n/**\n * Set the superagent query string correctly for browsers or node.\n * @param req\n */\nexport function setRequestQueryString(req: superagent.SuperAgentRequest): void {\n  const urlDetails = urlLib.parse(req.url);\n\n  let queryString: string | undefined;\n  const query: string[] = (req as any)._query;\n  const qs: { [key: string]: string } = (req as any).qs;\n  if (query && query.length > 0) {\n    // browser version\n    queryString = query.join('&');\n    (req as any)._query = [];\n  } else if (qs) {\n    // node version\n    queryString = querystring.stringify(qs);\n    (req as any).qs = null;\n  }\n\n  if (queryString) {\n    if (urlDetails.search) {\n      urlDetails.search += '&' + queryString;\n    } else {\n      urlDetails.search = '?' + queryString;\n    }\n    req.url = urlLib.format(urlDetails);\n  }\n}\n\n/**\n * Verify that the response received from the server is signed correctly.\n * Right now, it is very permissive with the timestamp variance.\n */\nexport function verifyResponse(\n  bitgo: BitGoAPI,\n  token: string | undefined,\n  method: VerifyResponseOptions['method'],\n  req: superagent.SuperAgentRequest,\n  response: superagent.Response\n): superagent.Response {\n  // we can't verify the response if we're not authenticated\n  if (!req.isV2Authenticated || !req.authenticationToken) {\n    return response;\n  }\n\n  const verificationResponse = bitgo.verifyResponse({\n    url: req.url,\n    hmac: response.header.hmac,\n    statusCode: response.status,\n    text: response.text,\n    timestamp: response.header.timestamp,\n    token: req.authenticationToken,\n    method,\n  });\n\n  if (!verificationResponse.isValid) {\n    // calculate the HMAC\n    const receivedHmac = response.header.hmac;\n    const expectedHmac = verificationResponse.expectedHmac;\n    const signatureSubject = verificationResponse.signatureSubject;\n    // Log only the first 10 characters of the token to ensure the full token isn't logged.\n    const partialBitgoToken = token ? token.substring(0, 10) : '';\n    const errorDetails = {\n      expectedHmac,\n      receivedHmac,\n      hmacInput: signatureSubject,\n      requestToken: req.authenticationToken,\n      bitgoToken: partialBitgoToken,\n    };\n    debug('Invalid response HMAC: %O', errorDetails);\n    throw new ApiResponseError('invalid response HMAC, possible man-in-the-middle-attack', 511, errorDetails);\n  }\n\n  if (bitgo.getAuthVersion() === 3 && !verificationResponse.isInResponseValidityWindow) {\n    const errorDetails = {\n      timestamp: response.header.timestamp,\n      verificationTime: verificationResponse.verificationTime,\n    };\n    debug('Server response outside response validity time window: %O', errorDetails);\n    throw new ApiResponseError(\n      'server response outside response validity time window, possible man-in-the-middle-attack',\n      511,\n      errorDetails\n    );\n  }\n  return response;\n}\n"]}
|