@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 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.503",
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.504",
44
- "@bitgo-beta/sjcl": "1.0.2-beta.742",
45
- "@bitgo-beta/unspents": "0.13.2-beta.504",
46
- "@bitgo-beta/utxo-lib": "8.0.3-beta.505",
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": "15f108c6c5dacf98c06663770d5c5ec5989f3349"
66
+ "gitHead": "3862d6687dab985ca7d8ce9d5f76b454e363e43b"
67
67
  }
package/dist/src/api.d.ts CHANGED
@@ -1,4 +1,4 @@
1
- import * as superagent from 'superagent';
1
+ import superagent from 'superagent';
2
2
  import { BitGoRequest } from '@bitgo-beta/sdk-core';
3
3
  import { VerifyResponseOptions } from './types';
4
4
  import { BitGoAPI } from './bitgoAPI';
@@ -1 +1 @@
1
- {"version":3,"file":"api.d.ts","sourceRoot":"","sources":["../../src/api.ts"],"names":[],"mappings":"AAOA,OAAO,KAAK,UAAU,MAAM,YAAY,CAAC;AAIzC,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"}
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 eol = require("eol");
9
- const _ = require("lodash");
10
- const sanitizeHtml = require("sanitize-html");
11
- const superagent = require("superagent");
12
- const urlLib = require("url");
13
- const querystring = require("querystring");
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 (_.isNumber(res.status) && res.status >= 200 && res.status < 300) {
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 = _.has(res.header, 'x-auth-required') && res.header['x-auth-required'] === 'true';
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 = sanitizeHtml(res.text, { allowedTags: [] });
96
+ let sanitizedText = (0, sanitize_html_1.default)(res.text, { allowedTags: [] });
94
97
  sanitizedText = sanitizedText.trim();
95
- sanitizedText = eol.lf(sanitizedText); // use '\n' for all newlines
96
- sanitizedText = _.replace(sanitizedText, /\n[ |\t]{1,}\n/g, '\n\n'); // remove the spaces/tabs between newlines
97
- sanitizedText = _.replace(sanitizedText, /[\n]{3,}/g, '\n\n'); // have at most 2 consecutive newlines
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 = superagent.serialize[contentType];
124
+ let serialize = superagent_1.default.serialize[contentType];
122
125
  if (!serialize && /[\/+]json\b/.test(contentType)) {
123
- serialize = superagent.serialize['application/json'];
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 = urlLib.parse(req.url);
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 = querystring.stringify(qs);
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 = urlLib.format(urlDetails);
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":"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","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"]}