PHP WebShell
Текущая директория: /opt/BitGoJS/modules/express/dist
Просмотр файла: clientRoutes.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.handleV2GenerateShareTSS = handleV2GenerateShareTSS;
exports.handleV2SignTSSWalletTx = handleV2SignTSSWalletTx;
exports.handleV2Sign = handleV2Sign;
exports.handleV2OFCSignPayloadInExtSigningMode = handleV2OFCSignPayloadInExtSigningMode;
exports.handleV2OFCSignPayload = handleV2OFCSignPayload;
exports.handleV2GenerateWallet = handleV2GenerateWallet;
exports.handleV2CreateAddress = handleV2CreateAddress;
exports.handleV2ConsolidateAccount = handleV2ConsolidateAccount;
exports.handleV2PrebuildAndSignTransaction = handleV2PrebuildAndSignTransaction;
exports.handleV2EnableTokens = handleV2EnableTokens;
exports.handleKeychainChangePassword = handleKeychainChangePassword;
exports.redirectRequest = redirectRequest;
exports.promiseWrapper = promiseWrapper;
exports.createCustomSigningFunction = createCustomSigningFunction;
exports.createCustomPaillierModulusGetter = createCustomPaillierModulusGetter;
exports.createCustomKShareGenerator = createCustomKShareGenerator;
exports.createCustomMuDeltaShareGenerator = createCustomMuDeltaShareGenerator;
exports.createCustomSShareGenerator = createCustomSShareGenerator;
exports.createCustomCommitmentGenerator = createCustomCommitmentGenerator;
exports.createCustomRShareGenerator = createCustomRShareGenerator;
exports.createCustomGShareGenerator = createCustomGShareGenerator;
exports.createCustomMPCv2SigningRound1Generator = createCustomMPCv2SigningRound1Generator;
exports.createCustomMPCv2SigningRound2Generator = createCustomMPCv2SigningRound2Generator;
exports.createCustomMPCv2SigningRound3Generator = createCustomMPCv2SigningRound3Generator;
exports.setupAPIRoutes = setupAPIRoutes;
exports.setupSigningRoutes = setupSigningRoutes;
exports.setupLightningSignerNodeRoutes = setupLightningSignerNodeRoutes;
/**
* @prettier
*/
const sdk_core_1 = require("@bitgo/sdk-core");
const bitgo_1 = require("bitgo");
const bodyParser = require("body-parser");
const debugLib = require("debug");
const _ = require("lodash");
const url = require("url");
const superagent = require("superagent");
// RequestTracer should be extracted into a separate npm package (along with
// the rest of the BitGoJS HTTP request machinery)
const util_1 = require("bitgo/dist/src/v2/internal/util");
const errors_1 = require("./errors");
const fs_1 = require("fs");
const retryPromise_1 = require("./retryPromise");
const lightningSignerRoutes_1 = require("./lightning/lightningSignerRoutes");
const lightningInvoiceRoutes_1 = require("./lightning/lightningInvoiceRoutes");
const lightningWalletRoutes_1 = require("./lightning/lightningWalletRoutes");
const proxy_agent_1 = require("proxy-agent");
const abstract_lightning_1 = require("@bitgo/abstract-lightning");
const lightningWithdrawRoutes_1 = require("./lightning/lightningWithdrawRoutes");
const { version } = require('bitgo/package.json');
const pjson = require('../package.json');
const debug = debugLib('bitgo:express');
const BITGOEXPRESS_USER_AGENT = `BitGoExpress/${pjson.version} BitGoJS/${version}`;
function handlePing(req, res, next) {
return req.bitgo.ping();
}
function handlePingExpress(req) {
return {
status: 'express server is ok!',
};
}
function handleLogin(req) {
const username = req.body.username || req.body.email;
const body = req.body;
body.username = username;
return req.bitgo.authenticate(body);
}
function handleDecrypt(req) {
return {
decrypted: req.bitgo.decrypt(req.body),
};
}
function handleEncrypt(req) {
return {
encrypted: req.bitgo.encrypt(req.body),
};
}
/**
* @deprecated
* @param req
*/
function handleVerifyAddress(req) {
return {
verified: req.bitgo.verifyAddress(req.body),
};
}
/**
* @deprecated
* @param req
*/
function handleCreateLocalKeyChain(req) {
return req.bitgo.keychains().create(req.body);
}
/**
* @deprecated
* @param req
*/
function handleDeriveLocalKeyChain(req) {
return req.bitgo.keychains().deriveLocal(req.body);
}
/**
* @deprecated
* @param req
*/
function handleCreateWalletWithKeychains(req) {
return req.bitgo.wallets().createWalletWithKeychains(req.body);
}
/**
* @deprecated
* @param req
*/
function handleSendCoins(req) {
return req.bitgo
.wallets()
.get({ id: req.params.id })
.then(function (wallet) {
return wallet.sendCoins(req.body);
})
.catch(function (err) {
err.status = 400;
throw err;
})
.then(function (result) {
if (result.status === 'pendingApproval') {
throw apiResponse(202, result);
}
return result;
});
}
/**
* @deprecated
* @param req
*/
function handleSendMany(req) {
return req.bitgo
.wallets()
.get({ id: req.params.id })
.then(function (wallet) {
return wallet.sendMany(req.body);
})
.catch(function (err) {
err.status = 400;
throw err;
})
.then(function (result) {
if (result.status === 'pendingApproval') {
throw apiResponse(202, result);
}
return result;
});
}
/**
* @deprecated
* @param req
*/
function handleCreateTransaction(req) {
return req.bitgo
.wallets()
.get({ id: req.params.id })
.then(function (wallet) {
return wallet.createTransaction(req.body);
})
.catch(function (err) {
err.status = 400;
throw err;
});
}
/**
* @deprecated
* @param req
*/
function handleSignTransaction(req) {
return req.bitgo
.wallets()
.get({ id: req.params.id })
.then(function (wallet) {
return wallet.signTransaction(req.body);
});
}
/**
* @deprecated
* @param req
*/
function handleShareWallet(req) {
return req.bitgo
.wallets()
.get({ id: req.params.id })
.then(function (wallet) {
return wallet.shareWallet(req.body);
});
}
/**
* @deprecated
* @param req
*/
function handleAcceptShare(req) {
const params = req.body || {};
params.walletShareId = req.params.shareId;
return req.bitgo.wallets().acceptShare(params);
}
/**
* @deprecated
* @param req
*/
function handleApproveTransaction(req) {
const params = req.body || {};
return req.bitgo
.pendingApprovals()
.get({ id: req.params.id })
.then(function (pendingApproval) {
if (params.state === 'approved') {
return pendingApproval.approve(params);
}
return pendingApproval.reject(params);
});
}
/**
* @deprecated
* @param req
*/
function handleConstructApprovalTx(req) {
const params = req.body || {};
return req.bitgo
.pendingApprovals()
.get({ id: req.params.id })
.then(function (pendingApproval) {
return pendingApproval.constructApprovalTx(params);
});
}
/**
* @deprecated
* @param req
*/
function handleConsolidateUnspents(req) {
return req.bitgo
.wallets()
.get({ id: req.params.id })
.then(function (wallet) {
return wallet.consolidateUnspents(req.body);
});
}
/**
* @deprecated
* @param req
*/
function handleFanOutUnspents(req) {
return req.bitgo
.wallets()
.get({ id: req.params.id })
.then(function (wallet) {
return wallet.fanOutUnspents(req.body);
});
}
/**
* @deprecated
* @param req
*/
function handleCalculateMinerFeeInfo(req) {
return req.bitgo.calculateMinerFeeInfo({
bitgo: req.bitgo,
feeRate: req.body.feeRate,
nP2shInputs: req.body.nP2shInputs,
nP2pkhInputs: req.body.nP2pkhInputs,
nP2shP2wshInputs: req.body.nP2shP2wshInputs,
nOutputs: req.body.nOutputs,
});
}
/**
* Builds the API's URL string, optionally building the querystring if parameters exist
* @param req
* @return {string}
*/
function createAPIPath(req) {
let apiPath = '/' + req.params[0];
if (!_.isEmpty(req.query)) {
// req.params does not contain the querystring, so we manually add them here
const urlDetails = url.parse(req.url);
if (urlDetails.search) {
// "search" is the properly URL encoded query params, prefixed with "?"
apiPath += urlDetails.search;
}
}
return apiPath;
}
/**
* handle any other V1 API call
* @deprecated
* @param req
* @param res
* @param next
*/
function handleREST(req, res, next) {
const method = req.method;
const bitgo = req.bitgo;
const bitgoURL = bitgo.url(createAPIPath(req));
return redirectRequest(bitgo, method, bitgoURL, req, next);
}
/**
* handle any other V2 API call
* @param req
* @param res
* @param next
*/
function handleV2UserREST(req, res, next) {
const method = req.method;
const bitgo = req.bitgo;
const bitgoURL = bitgo.url('/user' + createAPIPath(req), 2);
return redirectRequest(bitgo, method, bitgoURL, req, next);
}
/**
* handle v2 address validation
* @param req
*/
function handleV2VerifyAddress(req) {
if (!_.isString(req.body.address)) {
throw new Error('Expected address to be a string');
}
if (req.body.supportOldScriptHashVersion !== undefined && !_.isBoolean(req.body.supportOldScriptHashVersion)) {
throw new Error('Expected supportOldScriptHashVersion to be a boolean.');
}
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
if (coin instanceof bitgo_1.Coin.AbstractUtxoCoin) {
return {
isValid: coin.isValidAddress(req.body.address, !!req.body.supportOldScriptHashVersion),
};
}
return {
isValid: coin.isValidAddress(req.body.address),
};
}
/**
* handle address canonicalization
* @param req
*/
function handleCanonicalAddress(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
if (!['ltc', 'bch', 'bsv'].includes(coin.getFamily())) {
throw new Error('only Litecoin/Bitcoin Cash/Bitcoin SV address canonicalization is supported');
}
const address = req.body.address;
const fallbackVersion = req.body.scriptHashVersion; // deprecate
const version = req.body.version;
return coin.canonicalAddress(address, version || fallbackVersion);
}
function getWalletPwFromEnv(walletId) {
const name = `WALLET_${walletId}_PASSPHRASE`;
const walletPw = process.env[name];
if (walletPw === undefined) {
throw new Error(`Could not find wallet passphrase ${name} in environment`);
}
return walletPw;
}
async function getEncryptedPrivKey(path, walletId) {
const privKeyFile = await fs_1.promises.readFile(path, { encoding: 'utf8' });
const encryptedPrivKey = JSON.parse(privKeyFile);
if (encryptedPrivKey[walletId] === undefined) {
throw new Error(`Could not find a field for walletId: ${walletId} in ${path}`);
}
return encryptedPrivKey[walletId];
}
function decryptPrivKey(bg, encryptedPrivKey, walletPw) {
try {
return bg.decrypt({ password: walletPw, input: encryptedPrivKey });
}
catch (e) {
throw new Error(`Error when trying to decrypt private key: ${e}`);
}
}
async function handleV2GenerateShareTSS(req) {
const walletId = req.body.txRequest ? req.body.txRequest.walletId : req.body.tssParams.txRequest.walletId;
if (!walletId) {
throw new Error('Missing required field: walletId');
}
const walletPw = getWalletPwFromEnv(walletId);
const { signerFileSystemPath } = req.config;
if (!signerFileSystemPath) {
throw new Error('Missing required configuration: signerFileSystemPath');
}
const encryptedPrivKey = await getEncryptedPrivKey(signerFileSystemPath, walletId);
const bitgo = req.bitgo;
const privKey = decryptPrivKey(bitgo, encryptedPrivKey, walletPw);
const coin = bitgo.coin(req.params.coin);
req.body.prv = privKey;
req.body.walletPassphrase = walletPw;
try {
if (coin.getMPCAlgorithm() === sdk_core_1.MPCType.EDDSA) {
const eddsaUtils = new sdk_core_1.EddsaUtils(bitgo, coin);
switch (req.params.sharetype) {
case sdk_core_1.ShareType.Commitment:
return await eddsaUtils.createCommitmentShareFromTxRequest(req.body);
case sdk_core_1.ShareType.R:
return await eddsaUtils.createRShareFromTxRequest(req.body);
case sdk_core_1.ShareType.G:
return await eddsaUtils.createGShareFromTxRequest(req.body);
default:
throw new Error(`Share type ${req.params.sharetype} not supported, only commitment, G and R share generation is supported.`);
}
}
else if (coin.getMPCAlgorithm() === sdk_core_1.MPCType.ECDSA) {
const isMPCv2 = [
sdk_core_1.ShareType.MPCv2Round1.toString(),
sdk_core_1.ShareType.MPCv2Round2.toString(),
sdk_core_1.ShareType.MPCv2Round3.toString(),
].includes(req.params.sharetype);
if (isMPCv2) {
const ecdsaMPCv2Utils = new sdk_core_1.EcdsaMPCv2Utils(bitgo, coin);
switch (req.params.sharetype) {
case sdk_core_1.ShareType.MPCv2Round1:
return await ecdsaMPCv2Utils.createOfflineRound1Share(req.body);
case sdk_core_1.ShareType.MPCv2Round2:
return await ecdsaMPCv2Utils.createOfflineRound2Share(req.body);
case sdk_core_1.ShareType.MPCv2Round3:
return await ecdsaMPCv2Utils.createOfflineRound3Share(req.body);
default:
throw new Error(`Share type ${req.params.sharetype} not supported for MPCv2, only MPCv2Round1, MPCv2Round2 and MPCv2Round3 is supported.`);
}
}
else {
const ecdsaUtils = new sdk_core_1.EcdsaUtils(bitgo, coin);
switch (req.params.sharetype) {
case sdk_core_1.ShareType.PaillierModulus:
return ecdsaUtils.getOfflineSignerPaillierModulus(req.body);
case sdk_core_1.ShareType.K:
return await ecdsaUtils.createOfflineKShare(req.body);
case sdk_core_1.ShareType.MuDelta:
return await ecdsaUtils.createOfflineMuDeltaShare(req.body);
case sdk_core_1.ShareType.S:
return await ecdsaUtils.createOfflineSShare(req.body);
default:
throw new Error(`Share type ${req.params.sharetype} not supported, only PaillierModulus, K, MUDelta, and S share generation is supported.`);
}
}
}
else {
throw new Error(`MPC Algorithm ${coin.getMPCAlgorithm()} is not supported.`);
}
}
catch (error) {
console.error('error while signing wallet transaction ', error);
throw error;
}
}
async function handleV2SignTSSWalletTx(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const wallet = await coin.wallets().get({ id: req.params.id });
try {
return await wallet.signTransaction(createTSSSendParams(req, wallet));
}
catch (error) {
console.error('error while signing wallet transaction ', error);
throw error;
}
}
/**
* This route is used to sign while external express signer is enabled
*/
async function handleV2Sign(req) {
const walletId = req.body.txPrebuild?.walletId;
if (!walletId) {
throw new Error('Missing required field: walletId');
}
const walletPw = getWalletPwFromEnv(walletId);
const { signerFileSystemPath } = req.config;
if (!signerFileSystemPath) {
throw new Error('Missing required configuration: signerFileSystemPath');
}
const encryptedPrivKey = await getEncryptedPrivKey(signerFileSystemPath, walletId);
const bitgo = req.bitgo;
let privKey = decryptPrivKey(bitgo, encryptedPrivKey, walletPw);
const coin = bitgo.coin(req.params.coin);
if (req.body.derivationSeed) {
privKey = coin.deriveKeyWithSeed({ key: privKey, seed: req.body.derivationSeed }).key;
}
try {
return await coin.signTransaction({ ...req.body, prv: privKey });
}
catch (error) {
console.log('error while signing wallet transaction ', error);
throw error;
}
}
async function handleV2OFCSignPayloadInExtSigningMode(req) {
const walletId = req.body.walletId;
const payload = req.body.payload;
const bodyWalletPassphrase = req.body.walletPassphrase;
const ofcCoinName = 'ofc';
if (!payload) {
throw new errors_1.ApiResponseError('Missing required field: payload', 400);
}
if (!walletId) {
throw new errors_1.ApiResponseError('Missing required field: walletId', 400);
}
// fetch the password for the given walletId from the body or the env. This is required for decrypting the private key that belongs to that wallet.
const walletPw = bodyWalletPassphrase || getWalletPwFromEnv(walletId);
const { signerFileSystemPath } = req.config;
if (!signerFileSystemPath) {
throw new errors_1.ApiResponseError('Missing required configuration: signerFileSystemPath', 500);
}
// get the encrypted private key from the local JSON file (encryptedPrivKeys.json) (populated using fetchEncryptedPrivateKeys.ts)
const encryptedPrivKey = await getEncryptedPrivKey(signerFileSystemPath, walletId);
const bitgo = req.bitgo;
// decrypt the encrypted private key using the wallet pwd
const privKey = decryptPrivKey(bitgo, encryptedPrivKey, walletPw);
// create a BaseCoin instance for 'ofc'
const coin = bitgo.coin(ofcCoinName);
// stringify the payload if not already a string
const stringifiedPayload = typeof payload === 'string' ? payload : JSON.stringify(payload);
try {
// sign the message using the decrypted private key
const signature = (await coin.signMessage({ prv: privKey }, stringifiedPayload)).toString('hex');
return {
payload: stringifiedPayload,
signature,
};
}
catch (error) {
console.log('Error while signing message.', error);
throw error;
}
}
async function handleV2OFCSignPayload(req) {
const walletId = req.body.walletId;
const payload = req.body.payload;
const bodyWalletPassphrase = req.body.walletPassphrase;
const ofcCoinName = 'ofc';
// If the externalSignerUrl is set, forward the request to the express server hosted on the externalSignerUrl
const externalSignerUrl = req.config?.externalSignerUrl;
if (externalSignerUrl) {
const { body: payloadWithSignature } = await (0, retryPromise_1.retryPromise)(() => superagent
.post(`${externalSignerUrl}/api/v2/ofc/signPayload`)
.type('json')
.send({ walletId: walletId, payload: payload }), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return payloadWithSignature;
}
if (!payload) {
throw new errors_1.ApiResponseError('Missing required field: payload', 400);
}
if (!walletId) {
throw new errors_1.ApiResponseError('Missing required field: walletId', 400);
}
const bitgo = req.bitgo;
// This is to set us up for multiple trading accounts per enterprise
const wallet = await bitgo.coin(ofcCoinName).wallets().get({ id: walletId });
if (wallet === undefined) {
throw new errors_1.ApiResponseError(`Could not find OFC wallet ${walletId}`, 404);
}
const walletPassphrase = bodyWalletPassphrase || getWalletPwFromEnv(wallet.id());
const tradingAccount = wallet.toTradingAccount();
const stringifiedPayload = JSON.stringify(req.body.payload);
const signature = await tradingAccount.signPayload({
payload: stringifiedPayload,
walletPassphrase,
});
return {
payload: stringifiedPayload,
signature,
};
}
/**
* handle new wallet creation
* @param req
*/
async function handleV2GenerateWallet(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const result = await coin.wallets().generateWallet(req.body);
if (req.query.includeKeychains === 'false') {
return result.wallet.toJSON();
}
return { ...result, wallet: result.wallet.toJSON() };
}
/**
* handle new address creation
* @param req
*/
async function handleV2CreateAddress(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const wallet = await coin.wallets().get({ id: req.params.id });
return wallet.createAddress(req.body);
}
/**
* handle v2 approve transaction
* @param req
*/
async function handleV2PendingApproval(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const params = req.body || {};
const pendingApproval = await coin.pendingApprovals().get({ id: req.params.id });
if (params.state === 'approved') {
return pendingApproval.approve(params);
}
return pendingApproval.reject(params);
}
/**
* create a keychain
* @param req
*/
function handleV2CreateLocalKeyChain(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
return coin.keychains().create(req.body);
}
/**
* handle wallet share
* @param req
*/
async function handleV2ShareWallet(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const wallet = await coin.wallets().get({ id: req.params.id });
return wallet.shareWallet(req.body);
}
/**
* handle accept wallet share
* @param req
*/
async function handleV2AcceptWalletShare(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const params = _.extend({}, req.body, { walletShareId: req.params.id });
return coin.wallets().acceptShare(params);
}
/**
* handle wallet sign transaction
*/
async function handleV2SignTxWallet(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const wallet = await coin.wallets().get({ id: req.params.id });
try {
return await wallet.signTransaction(createSendParams(req));
}
catch (error) {
console.log('error while signing wallet transaction ', error);
throw error;
}
}
/**
* handle sign transaction
* @param req
*/
async function handleV2SignTx(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
try {
return await coin.signTransaction(req.body);
}
catch (error) {
console.log('error while signing the transaction ', error);
throw error;
}
}
/**
* handle wallet recover token
* @param req
*/
async function handleV2RecoverToken(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const wallet = await coin.wallets().get({ id: req.params.id });
return wallet.recoverToken(req.body);
}
/**
* handle wallet fanout unspents
* @param req
*/
async function handleV2ConsolidateUnspents(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const wallet = await coin.wallets().get({ id: req.params.id });
return wallet.consolidateUnspents(createSendParams(req));
}
/**
* Handle Wallet Account Consolidation.
*
* @param req
*/
async function handleV2ConsolidateAccount(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
if (req.body.consolidateAddresses && !_.isArray(req.body.consolidateAddresses)) {
throw new Error('consolidate address must be an array of addresses');
}
if (!coin.allowsAccountConsolidations()) {
throw new Error('invalid coin selected');
}
const wallet = await coin.wallets().get({ id: req.params.id });
let result;
try {
if (coin.supportsTss()) {
result = await wallet.sendAccountConsolidations(createTSSSendParams(req, wallet));
}
else {
result = await wallet.sendAccountConsolidations(createSendParams(req));
}
}
catch (err) {
err.status = 400;
throw err;
}
// we had failures to handle
if (result.failure.length && result.failure.length > 0) {
let msg = '';
let status = 202;
if (result.success.length && result.success.length > 0) {
// but we also had successes
msg = `Transactions failed: ${result.failure.length} and succeeded: ${result.success.length}`;
}
else {
// or in this case only failures
status = 400;
msg = `All transactions failed`;
}
throw apiResponse(status, result, msg);
}
return result;
}
/**
* handle wallet fanout unspents
* @param req
*/
async function handleV2FanOutUnspents(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const wallet = await coin.wallets().get({ id: req.params.id });
return wallet.fanoutUnspents(createSendParams(req));
}
/**
* handle wallet sweep
* @param req
*/
async function handleV2Sweep(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const wallet = await coin.wallets().get({ id: req.params.id });
return wallet.sweep(createSendParams(req));
}
/**
* handle CPFP accelerate transaction creation
* @param req
*/
async function handleV2AccelerateTransaction(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const wallet = await coin.wallets().get({ id: req.params.id });
return wallet.accelerateTransaction(createSendParams(req));
}
function createSendParams(req) {
if (req.config?.externalSignerUrl !== undefined) {
return {
...req.body,
customSigningFunction: createCustomSigningFunction(req.config.externalSignerUrl),
};
}
else {
return req.body;
}
}
function createTSSSendParams(req, wallet) {
if (req.config?.externalSignerUrl !== undefined) {
const coin = req.bitgo.coin(req.params.coin);
if (coin.getMPCAlgorithm() === sdk_core_1.MPCType.EDDSA) {
return {
...req.body,
customCommitmentGeneratingFunction: createCustomCommitmentGenerator(req.config.externalSignerUrl, req.params.coin),
customRShareGeneratingFunction: createCustomRShareGenerator(req.config.externalSignerUrl, req.params.coin),
customGShareGeneratingFunction: createCustomGShareGenerator(req.config.externalSignerUrl, req.params.coin),
};
}
else if (coin.getMPCAlgorithm() === sdk_core_1.MPCType.ECDSA) {
if (wallet._wallet.multisigTypeVersion === 'MPCv2') {
return {
...req.body,
customMPCv2SigningRound1GenerationFunction: createCustomMPCv2SigningRound1Generator(req.config.externalSignerUrl, req.params.coin),
customMPCv2SigningRound2GenerationFunction: createCustomMPCv2SigningRound2Generator(req.config.externalSignerUrl, req.params.coin),
customMPCv2SigningRound3GenerationFunction: createCustomMPCv2SigningRound3Generator(req.config.externalSignerUrl, req.params.coin),
};
}
else {
return {
...req.body,
customPaillierModulusGeneratingFunction: createCustomPaillierModulusGetter(req.config.externalSignerUrl, req.params.coin),
customKShareGeneratingFunction: createCustomKShareGenerator(req.config.externalSignerUrl, req.params.coin),
customMuDeltaShareGeneratingFunction: createCustomMuDeltaShareGenerator(req.config.externalSignerUrl, req.params.coin),
customSShareGeneratingFunction: createCustomSShareGenerator(req.config.externalSignerUrl, req.params.coin),
};
}
}
else {
throw new Error(`MPC Algorithm ${coin.getMPCAlgorithm()} is not supported.`);
}
}
else {
return req.body;
}
}
/**
* handle send one
* @param req
*/
async function handleV2SendOne(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const reqId = new util_1.RequestTracer();
const wallet = await coin.wallets().get({ id: req.params.id, reqId });
req.body.reqId = reqId;
let result;
try {
result = await wallet.send(createSendParams(req));
}
catch (err) {
err.status = 400;
throw err;
}
if (result.status === 'pendingApproval') {
throw apiResponse(202, result);
}
return result;
}
/**
* handle send many
* @param req
*/
async function handleV2SendMany(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const reqId = new util_1.RequestTracer();
const wallet = await coin.wallets().get({ id: req.params.id, reqId });
req.body.reqId = reqId;
let result;
try {
if (wallet._wallet.multisigType === 'tss') {
result = await wallet.sendMany(createTSSSendParams(req, wallet));
}
else {
result = await wallet.sendMany(createSendParams(req));
}
}
catch (err) {
err.status = 400;
throw err;
}
if (result.status === 'pendingApproval') {
throw apiResponse(202, result);
}
return result;
}
/**
* payload meant for prebuildAndSignTransaction() in sdk-core which
* validates the payload and makes the appropriate request to WP to
* build, sign, and send a tx.
* - sends request to Platform to build the transaction
* - signs with user key
* - request signature from the second key (BitGo HSM)
* - send/broadcast transaction
* @param req where req.body is {@link PrebuildAndSignTransactionOptions}
*/
async function handleV2PrebuildAndSignTransaction(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const reqId = new util_1.RequestTracer();
const wallet = await coin.wallets().get({ id: req.params.id, reqId });
req.body.reqId = reqId;
let result;
try {
result = await wallet.prebuildAndSignTransaction(createSendParams(req));
}
catch (err) {
err.status = 400;
throw err;
}
return result;
}
/**
* Enables tokens on a wallet
* @param req
*/
async function handleV2EnableTokens(req) {
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
const reqId = new util_1.RequestTracer();
const wallet = await coin.wallets().get({ id: req.params.id, reqId });
req.body.reqId = reqId;
try {
return wallet.sendTokenEnablements(createSendParams(req));
}
catch (err) {
err.status = 400;
throw err;
}
}
/**
* Handle Update Wallet
* @param req
*/
async function handleWalletUpdate(req) {
// If it's a lightning coin, use the lightning-specific handler
if ((0, abstract_lightning_1.isLightningCoinName)(req.params.coin)) {
return (0, lightningWalletRoutes_1.handleUpdateLightningWalletCoinSpecific)(req);
}
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
// For non-lightning coins, directly update the wallet
const wallet = await coin.wallets().get({ id: req.params.id });
return await bitgo.put(wallet.url()).send(req.body).result();
}
/**
* Changes a keychain's passphrase, re-encrypting the key to a new password
* @param req
*/
async function handleKeychainChangePassword(req) {
const { oldPassword, newPassword, otp } = req.body;
if (!oldPassword || !newPassword) {
throw new errors_1.ApiResponseError('Missing 1 or more required fields: [oldPassword, newPassword]', 400);
}
const reqId = new util_1.RequestTracer();
const bitgo = req.bitgo;
const coin = bitgo.coin(req.params.coin);
if (otp) {
await bitgo.unlock({ otp });
}
const keychain = await coin.keychains().get({
id: req.params.id,
reqId,
});
if (!keychain) {
throw new errors_1.ApiResponseError(`Keychain ${req.params.id} not found`, 404);
}
const updatedKeychain = coin.keychains().updateSingleKeychainPassword({
keychain,
oldPassword,
newPassword,
});
return bitgo.put(coin.url(`/key/${updatedKeychain.id}`)).send({
encryptedPrv: updatedKeychain.encryptedPrv,
});
}
/**
* handle any other API call
* @param req
* @param res
* @param next
*/
function handleV2CoinSpecificREST(req, res, next) {
const method = req.method;
const bitgo = req.bitgo;
debug('handling v2 coin specific rest req');
try {
const coin = bitgo.coin(req.params.coin);
const coinURL = coin.url(createAPIPath(req));
return redirectRequest(bitgo, method, coinURL, req, next);
}
catch (e) {
if (e instanceof sdk_core_1.UnsupportedCoinError) {
const queryParams = _.transform(req.query, (acc, value, key) => {
for (const val of _.castArray(value)) {
acc.push(`${key}=${val}`);
}
}, []);
const baseUrl = bitgo.url(req.baseUrl.replace(/^\/api\/v2/, ''), 2);
const url = _.isEmpty(queryParams) ? baseUrl : `${baseUrl}?${queryParams.join('&')}`;
debug(`coin ${req.params.coin} not supported, attempting to handle as a coinless route with url ${url}`);
return redirectRequest(bitgo, method, url, req, next);
}
throw e;
}
}
/**
* Handle additional option to encrypt on the express route for partners requiring value encryption
* @param req.body.encrypt - boolean to determine if the request should handle encryption on behalf of the submission.
*/
async function handleNetworkV1EnterpriseClientConnections(req, res, next) {
debug('handling network v1 partner connection creation');
const bitgo = req.bitgo;
const params = req.params;
const body = req.body;
if (body.encrypt === true) {
if (!body.partnerId) {
throw new errors_1.ApiResponseError('Missing required field: partnerId', 400);
}
const partnersUrl = bitgo.microservicesUrl(`/api/network/v1/enterprises/${params.enterpriseId}/partners`);
const response = await bitgo
.get(partnersUrl)
.set('enterprise-id', params.enterpriseId)
.send({ ids: [params.partnerId] })
.result();
const partners = response.partners;
const partner = partners.find((p) => p.id === body.partnerId);
if (!partner) {
throw new errors_1.ApiResponseError(`Partner not found for partnerId: ${body.partnerId}`, 400);
}
if (!partner.publicKey) {
throw new errors_1.ApiResponseError('Partner does not require encryption', 400);
}
switch (body.connectionKey.schema) {
case 'token':
req.body.connectionKey.connectionToken = await (0, sdk_core_1.encryptRsaWithAesGcm)(partner.publicKey, body.connectionKey.connectionToken);
break;
case 'tokenAndSignature':
req.body.connectionKey.connectionToken = await (0, sdk_core_1.encryptRsaWithAesGcm)(partner.publicKey, body.connectionKey.connectionToken);
req.body.connectionKey.signature = await (0, sdk_core_1.encryptRsaWithAesGcm)(partner.publicKey, body.connectionKey.signature);
break;
case 'apiKeyAndSecret':
case 'clearloop':
req.body.connectionKey.apiKey = await (0, sdk_core_1.encryptRsaWithAesGcm)(partner.publicKey, body.connectionKey.apiKey);
req.body.connectionKey.apiSecret = await (0, sdk_core_1.encryptRsaWithAesGcm)(partner.publicKey, body.connectionKey.apiSecret);
break;
}
}
return handleProxyReq(req, res, next);
}
/**
* Redirect a request using the bitgo request functions.
* @param bitgo
* @param method
* @param url
* @param req
* @param next
*/
function redirectRequest(bitgo, method, url, req, next) {
let request;
switch (method) {
case 'GET':
request = bitgo.get(url);
break;
case 'POST':
request = bitgo.post(url).send(req.body);
break;
case 'PUT':
request = bitgo.put(url).send(req.body);
break;
case 'PATCH':
request = bitgo.patch(url).send(req.body);
break;
case 'OPTIONS':
request = bitgo.options(url).send(req.body);
break;
case 'DELETE':
request = bitgo.del(url).send(req.body);
break;
}
if (request) {
if (req.params.enterpriseId) {
request.set('enterprise-id', req.params.enterpriseId);
}
return request.result().then((result) => {
const status = request.res?.statusCode || 200;
return { status, body: result };
});
}
// something has presumably gone wrong
next();
}
async function handleProxyReq(req, res, next) {
const fullUrl = req.bitgo.microservicesUrl(req.originalUrl);
if (req.url && (/^\/api.*$/.test(req.originalUrl) || /^\/oauth\/token.*$/.test(req.url))) {
req.isProxy = true;
debug('proxying %s request to %s', req.method, fullUrl);
return await redirectRequest(req.bitgo, req.method, fullUrl, req, next);
}
// user tried to access a url which is not an api route, do not proxy
debug('unable to proxy %s request to %s', req.method, fullUrl);
throw new errors_1.ApiResponseError('bitgo-express can only proxy BitGo API requests', 404);
}
/**
*
* @param status
* @param result
* @param message
*/
function apiResponse(status, result, message) {
return new errors_1.ApiResponseError(message, status, result);
}
const expressJSONParser = bodyParser.json({ limit: '20mb' });
/**
* Perform body parsing here only on routes we want
*/
function parseBody(req, res, next) {
// Set the default Content-Type, in case the client doesn't set it. If
// Content-Type isn't specified, Express silently refuses to parse the
// request body.
req.headers['content-type'] = req.headers['content-type'] || 'application/json';
return expressJSONParser(req, res, next);
}
/**
* Create the bitgo object in the request
* @param config
*/
function prepareBitGo(config) {
const { env, customRootUri, customBitcoinNetwork } = config;
return function prepBitGo(req, res, next) {
// Get access token
let accessToken;
if (req.headers.authorization) {
const authSplit = req.headers.authorization.split(' ');
if (authSplit.length === 2 && authSplit[0].toLowerCase() === 'bearer') {
accessToken = authSplit[1];
}
}
const userAgent = req.headers['user-agent']
? BITGOEXPRESS_USER_AGENT + ' ' + req.headers['user-agent']
: BITGOEXPRESS_USER_AGENT;
const useProxyUrl = process.env.BITGO_USE_PROXY;
const bitgoConstructorParams = {
env,
customRootURI: customRootUri,
customBitcoinNetwork,
accessToken,
userAgent,
...(useProxyUrl
? {
customProxyAgent: new proxy_agent_1.ProxyAgent({
getProxyForUrl: () => useProxyUrl,
}),
}
: {}),
};
req.bitgo = new bitgo_1.BitGo(bitgoConstructorParams);
req.config = config;
next();
};
}
function handleRequestHandlerError(res, error) {
let err;
if (error instanceof Error) {
err = error;
}
else if (typeof error === 'string') {
err = new Error('(string_error) ' + error);
}
else {
err = new Error('(object_error) ' + JSON.stringify(error));
}
const message = err.message || 'local error';
// use attached result, or make one
let result = err.result || { error: message };
result = _.extend({}, result, {
message: err.message,
bitgoJsVersion: version,
bitgoExpressVersion: pjson.version,
});
const status = err.status || 500;
if (!(status >= 200 && status < 300)) {
console.log('error %s: %s', status, err.message);
}
if (status >= 500 && status <= 599) {
if (err.response && err.response.request) {
console.log(`failed to make ${err.response.request.method} request to ${err.response.request.url}`);
}
console.log(err.stack);
}
res.status(status).send(result);
}
/**
* Promise handler wrapper to handle sending responses and error cases
* @param promiseRequestHandler
*/
function promiseWrapper(promiseRequestHandler) {
return async function promWrapper(req, res, next) {
debug(`handle: ${req.method} ${req.originalUrl}`);
try {
const result = await promiseRequestHandler(req, res, next);
if (typeof result === 'object' && result !== null && 'body' in result && 'status' in result) {
const { status, body } = result;
res.status(status).send(body);
}
else {
res.status(200).send(result);
}
}
catch (e) {
handleRequestHandlerError(res, e);
}
};
}
function createCustomSigningFunction(externalSignerUrl) {
return async function (params) {
const { body: signedTx } = await (0, retryPromise_1.retryPromise)(() => superagent.post(`${externalSignerUrl}/api/v2/${params.coin.getChain()}/sign`).type('json').send({
txPrebuild: params.txPrebuild,
pubs: params.pubs,
derivationSeed: params.derivationSeed,
signingStep: params.signingStep,
}), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return signedTx;
};
}
function createCustomPaillierModulusGetter(externalSignerUrl, coin) {
return async function (params) {
const { body: result } = await (0, retryPromise_1.retryPromise)(() => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/PaillierModulus`).type('json').send(params), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return result;
};
}
function createCustomKShareGenerator(externalSignerUrl, coin) {
return async function (params) {
const { body: result } = await (0, retryPromise_1.retryPromise)(() => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/K`).type('json').send(params), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return result;
};
}
function createCustomMuDeltaShareGenerator(externalSignerUrl, coin) {
return async function (params) {
const { body: result } = await (0, retryPromise_1.retryPromise)(() => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/MuDelta`).type('json').send(params), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return result;
};
}
function createCustomSShareGenerator(externalSignerUrl, coin) {
return async function (params) {
const { body: result } = await (0, retryPromise_1.retryPromise)(() => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/S`).type('json').send(params), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return result;
};
}
function createCustomCommitmentGenerator(externalSignerUrl, coin) {
return async function (params) {
const { body: result } = await (0, retryPromise_1.retryPromise)(() => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/commitment`).type('json').send(params), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return result;
};
}
function createCustomRShareGenerator(externalSignerUrl, coin) {
return async function (params) {
const { body: rShare } = await (0, retryPromise_1.retryPromise)(() => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/R`).type('json').send(params), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return rShare;
};
}
function createCustomGShareGenerator(externalSignerUrl, coin) {
return async function (params) {
const { body: signedTx } = await (0, retryPromise_1.retryPromise)(() => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/G`).type('json').send(params), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return signedTx;
};
}
function createCustomMPCv2SigningRound1Generator(externalSignerUrl, coin) {
return async function (params) {
const { body: result } = await (0, retryPromise_1.retryPromise)(() => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/MPCv2Round1`).type('json').send(params), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return result;
};
}
function createCustomMPCv2SigningRound2Generator(externalSignerUrl, coin) {
return async function (params) {
const { body: result } = await (0, retryPromise_1.retryPromise)(() => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/MPCv2Round2`).type('json').send(params), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return result;
};
}
function createCustomMPCv2SigningRound3Generator(externalSignerUrl, coin) {
return async function (params) {
const { body: result } = await (0, retryPromise_1.retryPromise)(() => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/MPCv2Round3`).type('json').send(params), (err, tryCount) => {
debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);
});
return result;
};
}
function setupAPIRoutes(app, config) {
// When adding new routes to BitGo Express make sure that you also add the exact same routes to the server. Since
// some customers were confused when calling a BitGo Express route on the BitGo server, we now handle all BitGo
// Express routes on the BitGo server and return an error message that says that one should call BitGo Express
// instead.
// V1 routes should be added to www/config/routes.js
// V2 routes should be added to www/config/routesV2.js
// ping
// /api/v[12]/pingexpress is the only exception to the rule above, as it explicitly checks the health of the
// express server without running into rate limiting with the BitGo server.
app.get('/api/v[12]/ping', prepareBitGo(config), promiseWrapper(handlePing));
app.get('/api/v[12]/pingexpress', promiseWrapper(handlePingExpress));
// auth
app.post('/api/v[12]/user/login', parseBody, prepareBitGo(config), promiseWrapper(handleLogin));
app.post('/api/v[12]/decrypt', parseBody, prepareBitGo(config), promiseWrapper(handleDecrypt));
app.post('/api/v[12]/encrypt', parseBody, prepareBitGo(config), promiseWrapper(handleEncrypt));
app.post('/api/v[12]/verifyaddress', parseBody, prepareBitGo(config), promiseWrapper(handleVerifyAddress));
app.post('/api/v[12]/calculateminerfeeinfo', parseBody, prepareBitGo(config), promiseWrapper(handleCalculateMinerFeeInfo));
app.post('/api/v1/keychain/local', parseBody, prepareBitGo(config), promiseWrapper(handleCreateLocalKeyChain));
app.post('/api/v1/keychain/derive', parseBody, prepareBitGo(config), promiseWrapper(handleDeriveLocalKeyChain));
app.post('/api/v1/wallets/simplecreate', parseBody, prepareBitGo(config), promiseWrapper(handleCreateWalletWithKeychains));
app.post('/api/v1/wallet/:id/sendcoins', parseBody, prepareBitGo(config), promiseWrapper(handleSendCoins));
app.post('/api/v1/wallet/:id/sendmany', parseBody, prepareBitGo(config), promiseWrapper(handleSendMany));
app.post('/api/v1/wallet/:id/createtransaction', parseBody, prepareBitGo(config), promiseWrapper(handleCreateTransaction));
app.post('/api/v1/wallet/:id/signtransaction', parseBody, prepareBitGo(config), promiseWrapper(handleSignTransaction));
app.post('/api/v1/wallet/:id/simpleshare', parseBody, prepareBitGo(config), promiseWrapper(handleShareWallet));
app.post('/api/v1/walletshare/:shareId/acceptShare', parseBody, prepareBitGo(config), promiseWrapper(handleAcceptShare));
app.put('/api/v1/pendingapprovals/:id/express', parseBody, prepareBitGo(config), promiseWrapper(handleApproveTransaction));
app.put('/api/v1/pendingapprovals/:id/constructTx', parseBody, prepareBitGo(config), promiseWrapper(handleConstructApprovalTx));
app.put('/api/v1/wallet/:id/consolidateunspents', parseBody, prepareBitGo(config), promiseWrapper(handleConsolidateUnspents));
app.put('/api/v1/wallet/:id/fanoutunspents', parseBody, prepareBitGo(config), promiseWrapper(handleFanOutUnspents));
// any other API call
app.use('/api/v[1]/*', parseBody, prepareBitGo(config), promiseWrapper(handleREST));
// API v2
// create keychain
app.post('/api/v2/:coin/keychain/local', parseBody, prepareBitGo(config), promiseWrapper(handleV2CreateLocalKeyChain));
// generate wallet
app.post('/api/v2/:coin/wallet/generate', parseBody, prepareBitGo(config), promiseWrapper(handleV2GenerateWallet));
app.put('/express/api/v2/:coin/wallet/:id', parseBody, prepareBitGo(config), promiseWrapper(handleWalletUpdate));
// change wallet passphrase
app.post('/api/v2/:coin/keychain/:id/changepassword', parseBody, prepareBitGo(config), promiseWrapper(handleKeychainChangePassword));
// create address
app.post('/api/v2/:coin/wallet/:id/address', parseBody, prepareBitGo(config), promiseWrapper(handleV2CreateAddress));
// share wallet
app.post('/api/v2/:coin/wallet/:id/share', parseBody, prepareBitGo(config), promiseWrapper(handleV2ShareWallet));
app.post('/api/v2/:coin/walletshare/:id/acceptshare', parseBody, prepareBitGo(config), promiseWrapper(handleV2AcceptWalletShare));
// sign arbitrary payloads w/ trading account key
app.post(`/api/v2/ofc/signPayload`, parseBody, prepareBitGo(config), promiseWrapper(handleV2OFCSignPayload));
// sign transaction
app.post('/api/v2/:coin/signtx', parseBody, prepareBitGo(config), promiseWrapper(handleV2SignTx));
app.post('/api/v2/:coin/wallet/:id/signtx', parseBody, prepareBitGo(config), promiseWrapper(handleV2SignTxWallet));
app.post('/api/v2/:coin/wallet/:id/signtxtss', parseBody, prepareBitGo(config), promiseWrapper(handleV2SignTSSWalletTx));
app.post('/api/v2/:coin/wallet/:id/recovertoken', parseBody, prepareBitGo(config), promiseWrapper(handleV2RecoverToken));
// send transaction
app.post('/api/v2/:coin/wallet/:id/sendcoins', parseBody, prepareBitGo(config), promiseWrapper(handleV2SendOne));
app.post('/api/v2/:coin/wallet/:id/sendmany', parseBody, prepareBitGo(config), promiseWrapper(handleV2SendMany));
app.post('/api/v2/:coin/wallet/:id/prebuildAndSignTransaction', parseBody, prepareBitGo(config), promiseWrapper(handleV2PrebuildAndSignTransaction));
// token enablement
app.post('/api/v2/:coin/wallet/:id/enableTokens', parseBody, prepareBitGo(config), promiseWrapper(handleV2EnableTokens));
// unspent changes
app.post('/api/v2/:coin/wallet/:id/consolidateunspents', parseBody, prepareBitGo(config), promiseWrapper(handleV2ConsolidateUnspents));
app.post('/api/v2/:coin/wallet/:id/fanoutunspents', parseBody, prepareBitGo(config), promiseWrapper(handleV2FanOutUnspents));
app.post('/api/v2/:coin/wallet/:id/sweep', parseBody, prepareBitGo(config), promiseWrapper(handleV2Sweep));
// CPFP
app.post('/api/v2/:coin/wallet/:id/acceleratetx', parseBody, prepareBitGo(config), promiseWrapper(handleV2AccelerateTransaction));
// account-based
app.post('/api/v2/:coin/wallet/:id/consolidateAccount', parseBody, prepareBitGo(config), promiseWrapper(handleV2ConsolidateAccount));
// Miscellaneous
app.post('/api/v2/:coin/canonicaladdress', parseBody, prepareBitGo(config), promiseWrapper(handleCanonicalAddress));
app.post('/api/v2/:coin/verifyaddress', parseBody, prepareBitGo(config), promiseWrapper(handleV2VerifyAddress));
app.put('/api/v2/:coin/pendingapprovals/:id', parseBody, prepareBitGo(config), promiseWrapper(handleV2PendingApproval));
// lightning - pay invoice
app.post('/api/v2/:coin/wallet/:id/lightning/payment', parseBody, prepareBitGo(config), promiseWrapper(lightningInvoiceRoutes_1.handlePayLightningInvoice));
// lightning - onchain withdrawal
app.post('/api/v2/:coin/wallet/:id/lightning/withdraw', parseBody, prepareBitGo(config), promiseWrapper(lightningWithdrawRoutes_1.handleLightningWithdraw));
// any other API v2 call
app.use('/api/v2/user/*', parseBody, prepareBitGo(config), promiseWrapper(handleV2UserREST));
app.use('/api/v2/:coin/*', parseBody, prepareBitGo(config), promiseWrapper(handleV2CoinSpecificREST));
app.post('/api/network/v1/enterprises/:enterpriseId/clients/connections', parseBody, prepareBitGo(config), promiseWrapper(handleNetworkV1EnterpriseClientConnections));
// everything else should use the proxy handler
if (config.disableProxy !== true) {
app.use('/api/:namespace/v[12]/enterprises/:enterpriseId/*', parseBody, prepareBitGo(config), promiseWrapper(handleProxyReq));
app.use(parseBody, prepareBitGo(config), promiseWrapper(handleProxyReq));
}
}
function setupSigningRoutes(app, config) {
app.post('/api/v2/:coin/sign', parseBody, prepareBitGo(config), promiseWrapper(handleV2Sign));
app.post('/api/v2/:coin/tssshare/:sharetype', parseBody, prepareBitGo(config), promiseWrapper(handleV2GenerateShareTSS));
app.post(`/api/v2/ofc/signPayload`, parseBody, prepareBitGo(config), promiseWrapper(handleV2OFCSignPayloadInExtSigningMode));
}
function setupLightningSignerNodeRoutes(app, config) {
app.post('/api/v2/:coin/wallet/:id/initwallet', parseBody, prepareBitGo(config), promiseWrapper(lightningSignerRoutes_1.handleInitLightningWallet));
app.post('/api/v2/:coin/wallet/:id/signermacaroon', parseBody, prepareBitGo(config), promiseWrapper(lightningSignerRoutes_1.handleCreateSignerMacaroon));
app.post('/api/v2/:coin/wallet/:id/unlockwallet', parseBody, prepareBitGo(config), promiseWrapper(lightningSignerRoutes_1.handleUnlockLightningWallet));
app.get('/api/v2/:coin/wallet/:id/state', prepareBitGo(config), promiseWrapper(lightningSignerRoutes_1.handleGetLightningWalletState));
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"clientRoutes.js","sourceRoot":"","sources":["../src/clientRoutes.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport {\n  CommitmentShareRecord,\n  CreateNetworkConnectionParams,\n  CustomCommitmentGeneratingFunction,\n  CustomGShareGeneratingFunction,\n  CustomKShareGeneratingFunction,\n  CustomMPCv2SigningRound1GeneratingFunction,\n  CustomMPCv2SigningRound2GeneratingFunction,\n  CustomMPCv2SigningRound3GeneratingFunction,\n  CustomMuDeltaShareGeneratingFunction,\n  CustomPaillierModulusGetterFunction,\n  CustomRShareGeneratingFunction,\n  CustomSShareGeneratingFunction,\n  EcdsaMPCv2Utils,\n  EcdsaUtils,\n  EddsaUtils,\n  EncryptedSignerShareRecord,\n  encryptRsaWithAesGcm,\n  GetNetworkPartnersResponse,\n  GShare,\n  MPCType,\n  ShareType,\n  SignShare,\n  SShare,\n  TssEcdsaStep1ReturnMessage,\n  TssEcdsaStep2ReturnMessage,\n  UnsupportedCoinError,\n  Wallet,\n} from '@bitgo/sdk-core';\nimport { BitGo, BitGoOptions, Coin, CustomSigningFunction, SignedTransaction, SignedTransactionRequest } from 'bitgo';\nimport * as bodyParser from 'body-parser';\nimport * as debugLib from 'debug';\nimport * as express from 'express';\nimport type { ParamsDictionary } from 'express-serve-static-core';\nimport * as _ from 'lodash';\nimport * as url from 'url';\nimport * as superagent from 'superagent';\n\n// RequestTracer should be extracted into a separate npm package (along with\n// the rest of the BitGoJS HTTP request machinery)\nimport { RequestTracer } from 'bitgo/dist/src/v2/internal/util';\n\nimport { Config } from './config';\nimport { ApiResponseError } from './errors';\nimport { promises as fs } from 'fs';\nimport { retryPromise } from './retryPromise';\nimport {\n  handleCreateSignerMacaroon,\n  handleGetLightningWalletState,\n  handleInitLightningWallet,\n  handleUnlockLightningWallet,\n} from './lightning/lightningSignerRoutes';\nimport { handlePayLightningInvoice } from './lightning/lightningInvoiceRoutes';\nimport { handleUpdateLightningWalletCoinSpecific } from './lightning/lightningWalletRoutes';\nimport { ProxyAgent } from 'proxy-agent';\nimport { isLightningCoinName } from '@bitgo/abstract-lightning';\nimport { handleLightningWithdraw } from './lightning/lightningWithdrawRoutes';\n\nconst { version } = require('bitgo/package.json');\nconst pjson = require('../package.json');\nconst debug = debugLib('bitgo:express');\n\nconst BITGOEXPRESS_USER_AGENT = `BitGoExpress/${pjson.version} BitGoJS/${version}`;\n\nfunction handlePing(req: express.Request, res: express.Response, next: express.NextFunction) {\n  return req.bitgo.ping();\n}\n\nfunction handlePingExpress(req: express.Request) {\n  return {\n    status: 'express server is ok!',\n  };\n}\n\nfunction handleLogin(req: express.Request) {\n  const username = req.body.username || req.body.email;\n  const body = req.body;\n  body.username = username;\n  return req.bitgo.authenticate(body);\n}\n\nfunction handleDecrypt(req: express.Request) {\n  return {\n    decrypted: req.bitgo.decrypt(req.body),\n  };\n}\n\nfunction handleEncrypt(req: express.Request) {\n  return {\n    encrypted: req.bitgo.encrypt(req.body),\n  };\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleVerifyAddress(req: express.Request) {\n  return {\n    verified: req.bitgo.verifyAddress(req.body),\n  };\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleCreateLocalKeyChain(req: express.Request) {\n  return req.bitgo.keychains().create(req.body);\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleDeriveLocalKeyChain(req: express.Request) {\n  return req.bitgo.keychains().deriveLocal(req.body);\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleCreateWalletWithKeychains(req: express.Request) {\n  return req.bitgo.wallets().createWalletWithKeychains(req.body);\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleSendCoins(req: express.Request) {\n  return req.bitgo\n    .wallets()\n    .get({ id: req.params.id })\n    .then(function (wallet) {\n      return wallet.sendCoins(req.body);\n    })\n    .catch(function (err) {\n      err.status = 400;\n      throw err;\n    })\n    .then(function (result) {\n      if (result.status === 'pendingApproval') {\n        throw apiResponse(202, result);\n      }\n      return result;\n    });\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleSendMany(req: express.Request) {\n  return req.bitgo\n    .wallets()\n    .get({ id: req.params.id })\n    .then(function (wallet) {\n      return wallet.sendMany(req.body);\n    })\n    .catch(function (err) {\n      err.status = 400;\n      throw err;\n    })\n    .then(function (result) {\n      if (result.status === 'pendingApproval') {\n        throw apiResponse(202, result);\n      }\n      return result;\n    });\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleCreateTransaction(req: express.Request) {\n  return req.bitgo\n    .wallets()\n    .get({ id: req.params.id })\n    .then(function (wallet) {\n      return wallet.createTransaction(req.body);\n    })\n    .catch(function (err) {\n      err.status = 400;\n      throw err;\n    });\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleSignTransaction(req: express.Request) {\n  return req.bitgo\n    .wallets()\n    .get({ id: req.params.id })\n    .then(function (wallet) {\n      return wallet.signTransaction(req.body);\n    });\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleShareWallet(req: express.Request) {\n  return req.bitgo\n    .wallets()\n    .get({ id: req.params.id })\n    .then(function (wallet) {\n      return wallet.shareWallet(req.body);\n    });\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleAcceptShare(req: express.Request) {\n  const params = req.body || {};\n  params.walletShareId = req.params.shareId;\n  return req.bitgo.wallets().acceptShare(params);\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleApproveTransaction(req: express.Request) {\n  const params = req.body || {};\n  return req.bitgo\n    .pendingApprovals()\n    .get({ id: req.params.id })\n    .then(function (pendingApproval) {\n      if (params.state === 'approved') {\n        return pendingApproval.approve(params);\n      }\n      return pendingApproval.reject(params);\n    });\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleConstructApprovalTx(req: express.Request) {\n  const params = req.body || {};\n  return req.bitgo\n    .pendingApprovals()\n    .get({ id: req.params.id })\n    .then(function (pendingApproval) {\n      return pendingApproval.constructApprovalTx(params);\n    });\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleConsolidateUnspents(req: express.Request) {\n  return req.bitgo\n    .wallets()\n    .get({ id: req.params.id })\n    .then(function (wallet) {\n      return wallet.consolidateUnspents(req.body);\n    });\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleFanOutUnspents(req: express.Request) {\n  return req.bitgo\n    .wallets()\n    .get({ id: req.params.id })\n    .then(function (wallet) {\n      return wallet.fanOutUnspents(req.body);\n    });\n}\n\n/**\n * @deprecated\n * @param req\n */\nfunction handleCalculateMinerFeeInfo(req: express.Request) {\n  return req.bitgo.calculateMinerFeeInfo({\n    bitgo: req.bitgo,\n    feeRate: req.body.feeRate,\n    nP2shInputs: req.body.nP2shInputs,\n    nP2pkhInputs: req.body.nP2pkhInputs,\n    nP2shP2wshInputs: req.body.nP2shP2wshInputs,\n    nOutputs: req.body.nOutputs,\n  });\n}\n\n/**\n * Builds the API's URL string, optionally building the querystring if parameters exist\n * @param req\n * @return {string}\n */\nfunction createAPIPath(req: express.Request) {\n  let apiPath = '/' + req.params[0];\n  if (!_.isEmpty(req.query)) {\n    // req.params does not contain the querystring, so we manually add them here\n    const urlDetails = url.parse(req.url);\n    if (urlDetails.search) {\n      // \"search\" is the properly URL encoded query params, prefixed with \"?\"\n      apiPath += urlDetails.search;\n    }\n  }\n  return apiPath;\n}\n\n/**\n * handle any other V1 API call\n * @deprecated\n * @param req\n * @param res\n * @param next\n */\nfunction handleREST(req: express.Request, res: express.Response, next: express.NextFunction) {\n  const method = req.method;\n  const bitgo = req.bitgo;\n  const bitgoURL = bitgo.url(createAPIPath(req));\n  return redirectRequest(bitgo, method, bitgoURL, req, next);\n}\n\n/**\n * handle any other V2 API call\n * @param req\n * @param res\n * @param next\n */\nfunction handleV2UserREST(req: express.Request, res: express.Response, next: express.NextFunction) {\n  const method = req.method;\n  const bitgo = req.bitgo;\n  const bitgoURL = bitgo.url('/user' + createAPIPath(req), 2);\n  return redirectRequest(bitgo, method, bitgoURL, req, next);\n}\n\n/**\n * handle v2 address validation\n * @param req\n */\nfunction handleV2VerifyAddress(req: express.Request): { isValid: boolean } {\n  if (!_.isString(req.body.address)) {\n    throw new Error('Expected address to be a string');\n  }\n\n  if (req.body.supportOldScriptHashVersion !== undefined && !_.isBoolean(req.body.supportOldScriptHashVersion)) {\n    throw new Error('Expected supportOldScriptHashVersion to be a boolean.');\n  }\n\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n\n  if (coin instanceof Coin.AbstractUtxoCoin) {\n    return {\n      isValid: coin.isValidAddress(req.body.address, !!req.body.supportOldScriptHashVersion),\n    };\n  }\n\n  return {\n    isValid: coin.isValidAddress(req.body.address),\n  };\n}\n\n/**\n * handle address canonicalization\n * @param req\n */\nfunction handleCanonicalAddress(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  if (!['ltc', 'bch', 'bsv'].includes(coin.getFamily())) {\n    throw new Error('only Litecoin/Bitcoin Cash/Bitcoin SV address canonicalization is supported');\n  }\n\n  const address = req.body.address;\n  const fallbackVersion = req.body.scriptHashVersion; // deprecate\n  const version = req.body.version;\n  return (coin as Coin.Bch | Coin.Bsv | Coin.Ltc).canonicalAddress(address, version || fallbackVersion);\n}\n\nfunction getWalletPwFromEnv(walletId: string): string {\n  const name = `WALLET_${walletId}_PASSPHRASE`;\n  const walletPw = process.env[name];\n  if (walletPw === undefined) {\n    throw new Error(`Could not find wallet passphrase ${name} in environment`);\n  }\n  return walletPw;\n}\n\nasync function getEncryptedPrivKey(path: string, walletId: string): Promise<string> {\n  const privKeyFile = await fs.readFile(path, { encoding: 'utf8' });\n  const encryptedPrivKey = JSON.parse(privKeyFile);\n  if (encryptedPrivKey[walletId] === undefined) {\n    throw new Error(`Could not find a field for walletId: ${walletId} in ${path}`);\n  }\n  return encryptedPrivKey[walletId];\n}\n\nfunction decryptPrivKey(bg: BitGo, encryptedPrivKey: string, walletPw: string): string {\n  try {\n    return bg.decrypt({ password: walletPw, input: encryptedPrivKey });\n  } catch (e) {\n    throw new Error(`Error when trying to decrypt private key: ${e}`);\n  }\n}\n\nexport async function handleV2GenerateShareTSS(req: express.Request): Promise<any> {\n  const walletId = req.body.txRequest ? req.body.txRequest.walletId : req.body.tssParams.txRequest.walletId;\n  if (!walletId) {\n    throw new Error('Missing required field: walletId');\n  }\n\n  const walletPw = getWalletPwFromEnv(walletId);\n  const { signerFileSystemPath } = req.config;\n\n  if (!signerFileSystemPath) {\n    throw new Error('Missing required configuration: signerFileSystemPath');\n  }\n\n  const encryptedPrivKey = await getEncryptedPrivKey(signerFileSystemPath, walletId);\n  const bitgo = req.bitgo;\n  const privKey = decryptPrivKey(bitgo, encryptedPrivKey, walletPw);\n  const coin = bitgo.coin(req.params.coin);\n  req.body.prv = privKey;\n  req.body.walletPassphrase = walletPw;\n  try {\n    if (coin.getMPCAlgorithm() === MPCType.EDDSA) {\n      const eddsaUtils = new EddsaUtils(bitgo, coin);\n      switch (req.params.sharetype) {\n        case ShareType.Commitment:\n          return await eddsaUtils.createCommitmentShareFromTxRequest(req.body);\n        case ShareType.R:\n          return await eddsaUtils.createRShareFromTxRequest(req.body);\n        case ShareType.G:\n          return await eddsaUtils.createGShareFromTxRequest(req.body);\n        default:\n          throw new Error(\n            `Share type ${req.params.sharetype} not supported, only commitment, G and R share generation is supported.`\n          );\n      }\n    } else if (coin.getMPCAlgorithm() === MPCType.ECDSA) {\n      const isMPCv2 = [\n        ShareType.MPCv2Round1.toString(),\n        ShareType.MPCv2Round2.toString(),\n        ShareType.MPCv2Round3.toString(),\n      ].includes(req.params.sharetype);\n\n      if (isMPCv2) {\n        const ecdsaMPCv2Utils = new EcdsaMPCv2Utils(bitgo, coin);\n        switch (req.params.sharetype) {\n          case ShareType.MPCv2Round1:\n            return await ecdsaMPCv2Utils.createOfflineRound1Share(req.body);\n          case ShareType.MPCv2Round2:\n            return await ecdsaMPCv2Utils.createOfflineRound2Share(req.body);\n          case ShareType.MPCv2Round3:\n            return await ecdsaMPCv2Utils.createOfflineRound3Share(req.body);\n          default:\n            throw new Error(\n              `Share type ${req.params.sharetype} not supported for MPCv2, only MPCv2Round1, MPCv2Round2 and MPCv2Round3 is supported.`\n            );\n        }\n      } else {\n        const ecdsaUtils = new EcdsaUtils(bitgo, coin);\n        switch (req.params.sharetype) {\n          case ShareType.PaillierModulus:\n            return ecdsaUtils.getOfflineSignerPaillierModulus(req.body);\n          case ShareType.K:\n            return await ecdsaUtils.createOfflineKShare(req.body);\n          case ShareType.MuDelta:\n            return await ecdsaUtils.createOfflineMuDeltaShare(req.body);\n          case ShareType.S:\n            return await ecdsaUtils.createOfflineSShare(req.body);\n          default:\n            throw new Error(\n              `Share type ${req.params.sharetype} not supported, only PaillierModulus, K, MUDelta, and S share generation is supported.`\n            );\n        }\n      }\n    } else {\n      throw new Error(`MPC Algorithm ${coin.getMPCAlgorithm()} is not supported.`);\n    }\n  } catch (error) {\n    console.error('error while signing wallet transaction ', error);\n    throw error;\n  }\n}\n\nexport async function handleV2SignTSSWalletTx(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const wallet = await coin.wallets().get({ id: req.params.id });\n  try {\n    return await wallet.signTransaction(createTSSSendParams(req, wallet));\n  } catch (error) {\n    console.error('error while signing wallet transaction ', error);\n    throw error;\n  }\n}\n\n/**\n * This route is used to sign while external express signer is enabled\n */\nexport async function handleV2Sign(req: express.Request) {\n  const walletId = req.body.txPrebuild?.walletId;\n\n  if (!walletId) {\n    throw new Error('Missing required field: walletId');\n  }\n\n  const walletPw = getWalletPwFromEnv(walletId);\n  const { signerFileSystemPath } = req.config;\n\n  if (!signerFileSystemPath) {\n    throw new Error('Missing required configuration: signerFileSystemPath');\n  }\n\n  const encryptedPrivKey = await getEncryptedPrivKey(signerFileSystemPath, walletId);\n  const bitgo = req.bitgo;\n  let privKey = decryptPrivKey(bitgo, encryptedPrivKey, walletPw);\n  const coin = bitgo.coin(req.params.coin);\n  if (req.body.derivationSeed) {\n    privKey = coin.deriveKeyWithSeed({ key: privKey, seed: req.body.derivationSeed }).key;\n  }\n  try {\n    return await coin.signTransaction({ ...req.body, prv: privKey });\n  } catch (error) {\n    console.log('error while signing wallet transaction ', error);\n    throw error;\n  }\n}\n\nexport async function handleV2OFCSignPayloadInExtSigningMode(\n  req: express.Request\n): Promise<{ payload: string; signature: string }> {\n  const walletId = req.body.walletId;\n  const payload = req.body.payload;\n  const bodyWalletPassphrase = req.body.walletPassphrase;\n  const ofcCoinName = 'ofc';\n\n  if (!payload) {\n    throw new ApiResponseError('Missing required field: payload', 400);\n  }\n\n  if (!walletId) {\n    throw new ApiResponseError('Missing required field: walletId', 400);\n  }\n\n  // fetch the password for the given walletId from the body or the env. This is required for decrypting the private key that belongs to that wallet.\n  const walletPw = bodyWalletPassphrase || getWalletPwFromEnv(walletId);\n\n  const { signerFileSystemPath } = req.config;\n  if (!signerFileSystemPath) {\n    throw new ApiResponseError('Missing required configuration: signerFileSystemPath', 500);\n  }\n  // get the encrypted private key from the local JSON file (encryptedPrivKeys.json) (populated using fetchEncryptedPrivateKeys.ts)\n  const encryptedPrivKey = await getEncryptedPrivKey(signerFileSystemPath, walletId);\n\n  const bitgo = req.bitgo;\n\n  // decrypt the encrypted private key using the wallet pwd\n  const privKey = decryptPrivKey(bitgo, encryptedPrivKey, walletPw);\n\n  // create a BaseCoin instance for 'ofc'\n  const coin = bitgo.coin(ofcCoinName);\n\n  // stringify the payload if not already a string\n  const stringifiedPayload = typeof payload === 'string' ? payload : JSON.stringify(payload);\n\n  try {\n    // sign the message using the decrypted private key\n    const signature = (await coin.signMessage({ prv: privKey }, stringifiedPayload)).toString('hex');\n    return {\n      payload: stringifiedPayload,\n      signature,\n    };\n  } catch (error) {\n    console.log('Error while signing message.', error);\n    throw error;\n  }\n}\n\nexport async function handleV2OFCSignPayload(req: express.Request): Promise<{ payload: string; signature: string }> {\n  const walletId = req.body.walletId;\n  const payload = req.body.payload;\n  const bodyWalletPassphrase = req.body.walletPassphrase;\n  const ofcCoinName = 'ofc';\n\n  // If the externalSignerUrl is set, forward the request to the express server hosted on the externalSignerUrl\n  const externalSignerUrl = req.config?.externalSignerUrl;\n  if (externalSignerUrl) {\n    const { body: payloadWithSignature } = await retryPromise(\n      () =>\n        superagent\n          .post(`${externalSignerUrl}/api/v2/ofc/signPayload`)\n          .type('json')\n          .send({ walletId: walletId, payload: payload }),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return payloadWithSignature;\n  }\n\n  if (!payload) {\n    throw new ApiResponseError('Missing required field: payload', 400);\n  }\n\n  if (!walletId) {\n    throw new ApiResponseError('Missing required field: walletId', 400);\n  }\n\n  const bitgo = req.bitgo;\n\n  // This is to set us up for multiple trading accounts per enterprise\n  const wallet = await bitgo.coin(ofcCoinName).wallets().get({ id: walletId });\n\n  if (wallet === undefined) {\n    throw new ApiResponseError(`Could not find OFC wallet ${walletId}`, 404);\n  }\n\n  const walletPassphrase = bodyWalletPassphrase || getWalletPwFromEnv(wallet.id());\n  const tradingAccount = wallet.toTradingAccount();\n  const stringifiedPayload = JSON.stringify(req.body.payload);\n  const signature = await tradingAccount.signPayload({\n    payload: stringifiedPayload,\n    walletPassphrase,\n  });\n  return {\n    payload: stringifiedPayload,\n    signature,\n  };\n}\n\n/**\n * handle new wallet creation\n * @param req\n */\nexport async function handleV2GenerateWallet(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const result = await coin.wallets().generateWallet(req.body);\n  if (req.query.includeKeychains === 'false') {\n    return result.wallet.toJSON();\n  }\n  return { ...result, wallet: result.wallet.toJSON() };\n}\n\n/**\n * handle new address creation\n * @param req\n */\nexport async function handleV2CreateAddress(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const wallet = await coin.wallets().get({ id: req.params.id });\n  return wallet.createAddress(req.body);\n}\n\n/**\n * handle v2 approve transaction\n * @param req\n */\nasync function handleV2PendingApproval(req: express.Request): Promise<any> {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const params = req.body || {};\n  const pendingApproval = await coin.pendingApprovals().get({ id: req.params.id });\n  if (params.state === 'approved') {\n    return pendingApproval.approve(params);\n  }\n  return pendingApproval.reject(params);\n}\n\n/**\n * create a keychain\n * @param req\n */\nfunction handleV2CreateLocalKeyChain(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  return coin.keychains().create(req.body);\n}\n\n/**\n * handle wallet share\n * @param req\n */\nasync function handleV2ShareWallet(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const wallet = await coin.wallets().get({ id: req.params.id });\n  return wallet.shareWallet(req.body);\n}\n\n/**\n * handle accept wallet share\n * @param req\n */\nasync function handleV2AcceptWalletShare(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const params = _.extend({}, req.body, { walletShareId: req.params.id });\n  return coin.wallets().acceptShare(params);\n}\n\n/**\n * handle wallet sign transaction\n */\nasync function handleV2SignTxWallet(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const wallet = await coin.wallets().get({ id: req.params.id });\n  try {\n    return await wallet.signTransaction(createSendParams(req));\n  } catch (error) {\n    console.log('error while signing wallet transaction ', error);\n    throw error;\n  }\n}\n\n/**\n * handle sign transaction\n * @param req\n */\nasync function handleV2SignTx(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  try {\n    return await coin.signTransaction(req.body);\n  } catch (error) {\n    console.log('error while signing the transaction ', error);\n    throw error;\n  }\n}\n\n/**\n * handle wallet recover token\n * @param req\n */\nasync function handleV2RecoverToken(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n\n  const wallet = await coin.wallets().get({ id: req.params.id });\n  return wallet.recoverToken(req.body);\n}\n\n/**\n * handle wallet fanout unspents\n * @param req\n */\nasync function handleV2ConsolidateUnspents(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const wallet = await coin.wallets().get({ id: req.params.id });\n  return wallet.consolidateUnspents(createSendParams(req));\n}\n\n/**\n * Handle Wallet Account Consolidation.\n *\n * @param req\n */\nexport async function handleV2ConsolidateAccount(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n\n  if (req.body.consolidateAddresses && !_.isArray(req.body.consolidateAddresses)) {\n    throw new Error('consolidate address must be an array of addresses');\n  }\n\n  if (!coin.allowsAccountConsolidations()) {\n    throw new Error('invalid coin selected');\n  }\n\n  const wallet = await coin.wallets().get({ id: req.params.id });\n\n  let result: any;\n  try {\n    if (coin.supportsTss()) {\n      result = await wallet.sendAccountConsolidations(createTSSSendParams(req, wallet));\n    } else {\n      result = await wallet.sendAccountConsolidations(createSendParams(req));\n    }\n  } catch (err) {\n    err.status = 400;\n    throw err;\n  }\n\n  // we had failures to handle\n  if (result.failure.length && result.failure.length > 0) {\n    let msg = '';\n    let status = 202;\n\n    if (result.success.length && result.success.length > 0) {\n      // but we also had successes\n      msg = `Transactions failed: ${result.failure.length} and succeeded: ${result.success.length}`;\n    } else {\n      // or in this case only failures\n      status = 400;\n      msg = `All transactions failed`;\n    }\n\n    throw apiResponse(status, result, msg);\n  }\n\n  return result;\n}\n\n/**\n * handle wallet fanout unspents\n * @param req\n */\nasync function handleV2FanOutUnspents(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const wallet = await coin.wallets().get({ id: req.params.id });\n  return wallet.fanoutUnspents(createSendParams(req));\n}\n\n/**\n * handle wallet sweep\n * @param req\n */\nasync function handleV2Sweep(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const wallet = await coin.wallets().get({ id: req.params.id });\n  return wallet.sweep(createSendParams(req));\n}\n\n/**\n * handle CPFP accelerate transaction creation\n * @param req\n */\nasync function handleV2AccelerateTransaction(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const wallet = await coin.wallets().get({ id: req.params.id });\n  return wallet.accelerateTransaction(createSendParams(req));\n}\n\nfunction createSendParams(req: express.Request) {\n  if (req.config?.externalSignerUrl !== undefined) {\n    return {\n      ...req.body,\n      customSigningFunction: createCustomSigningFunction(req.config.externalSignerUrl),\n    };\n  } else {\n    return req.body;\n  }\n}\n\nfunction createTSSSendParams(req: express.Request, wallet: Wallet) {\n  if (req.config?.externalSignerUrl !== undefined) {\n    const coin = req.bitgo.coin(req.params.coin);\n    if (coin.getMPCAlgorithm() === MPCType.EDDSA) {\n      return {\n        ...req.body,\n        customCommitmentGeneratingFunction: createCustomCommitmentGenerator(\n          req.config.externalSignerUrl,\n          req.params.coin\n        ),\n        customRShareGeneratingFunction: createCustomRShareGenerator(req.config.externalSignerUrl, req.params.coin),\n        customGShareGeneratingFunction: createCustomGShareGenerator(req.config.externalSignerUrl, req.params.coin),\n      };\n    } else if (coin.getMPCAlgorithm() === MPCType.ECDSA) {\n      if (wallet._wallet.multisigTypeVersion === 'MPCv2') {\n        return {\n          ...req.body,\n          customMPCv2SigningRound1GenerationFunction: createCustomMPCv2SigningRound1Generator(\n            req.config.externalSignerUrl,\n            req.params.coin\n          ),\n          customMPCv2SigningRound2GenerationFunction: createCustomMPCv2SigningRound2Generator(\n            req.config.externalSignerUrl,\n            req.params.coin\n          ),\n          customMPCv2SigningRound3GenerationFunction: createCustomMPCv2SigningRound3Generator(\n            req.config.externalSignerUrl,\n            req.params.coin\n          ),\n        };\n      } else {\n        return {\n          ...req.body,\n          customPaillierModulusGeneratingFunction: createCustomPaillierModulusGetter(\n            req.config.externalSignerUrl,\n            req.params.coin\n          ),\n          customKShareGeneratingFunction: createCustomKShareGenerator(req.config.externalSignerUrl, req.params.coin),\n          customMuDeltaShareGeneratingFunction: createCustomMuDeltaShareGenerator(\n            req.config.externalSignerUrl,\n            req.params.coin\n          ),\n          customSShareGeneratingFunction: createCustomSShareGenerator(req.config.externalSignerUrl, req.params.coin),\n        };\n      }\n    } else {\n      throw new Error(`MPC Algorithm ${coin.getMPCAlgorithm()} is not supported.`);\n    }\n  } else {\n    return req.body;\n  }\n}\n\n/**\n * handle send one\n * @param req\n */\nasync function handleV2SendOne(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const reqId = new RequestTracer();\n  const wallet = await coin.wallets().get({ id: req.params.id, reqId });\n  req.body.reqId = reqId;\n\n  let result;\n  try {\n    result = await wallet.send(createSendParams(req));\n  } catch (err) {\n    err.status = 400;\n    throw err;\n  }\n  if (result.status === 'pendingApproval') {\n    throw apiResponse(202, result);\n  }\n  return result;\n}\n\n/**\n * handle send many\n * @param req\n */\nasync function handleV2SendMany(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const reqId = new RequestTracer();\n  const wallet = await coin.wallets().get({ id: req.params.id, reqId });\n  req.body.reqId = reqId;\n  let result;\n  try {\n    if (wallet._wallet.multisigType === 'tss') {\n      result = await wallet.sendMany(createTSSSendParams(req, wallet));\n    } else {\n      result = await wallet.sendMany(createSendParams(req));\n    }\n  } catch (err) {\n    err.status = 400;\n    throw err;\n  }\n  if (result.status === 'pendingApproval') {\n    throw apiResponse(202, result);\n  }\n  return result;\n}\n\n/**\n *  payload meant for prebuildAndSignTransaction() in sdk-core which\n * validates the payload and makes the appropriate request to WP to\n * build, sign, and send a tx.\n * - sends request to Platform to build the transaction\n * - signs with user key\n * - request signature from the second key (BitGo HSM)\n * - send/broadcast transaction\n * @param req where req.body is {@link PrebuildAndSignTransactionOptions}\n */\nexport async function handleV2PrebuildAndSignTransaction(req: express.Request): Promise<SignedTransactionRequest> {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const reqId = new RequestTracer();\n  const wallet = await coin.wallets().get({ id: req.params.id, reqId });\n  req.body.reqId = reqId;\n  let result;\n  try {\n    result = await wallet.prebuildAndSignTransaction(createSendParams(req));\n  } catch (err) {\n    err.status = 400;\n    throw err;\n  }\n  return result;\n}\n\n/**\n * Enables tokens on a wallet\n * @param req\n */\nexport async function handleV2EnableTokens(req: express.Request) {\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  const reqId = new RequestTracer();\n  const wallet = await coin.wallets().get({ id: req.params.id, reqId });\n  req.body.reqId = reqId;\n  try {\n    return wallet.sendTokenEnablements(createSendParams(req));\n  } catch (err) {\n    err.status = 400;\n    throw err;\n  }\n}\n\n/**\n * Handle Update Wallet\n * @param req\n */\nasync function handleWalletUpdate(req: express.Request): Promise<unknown> {\n  // If it's a lightning coin, use the lightning-specific handler\n  if (isLightningCoinName(req.params.coin)) {\n    return handleUpdateLightningWalletCoinSpecific(req);\n  }\n\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n  // For non-lightning coins, directly update the wallet\n  const wallet = await coin.wallets().get({ id: req.params.id });\n  return await bitgo.put(wallet.url()).send(req.body).result();\n}\n\n/**\n * Changes a keychain's passphrase, re-encrypting the key to a new password\n * @param req\n */\nexport async function handleKeychainChangePassword(req: express.Request): Promise<unknown> {\n  const { oldPassword, newPassword, otp } = req.body;\n  if (!oldPassword || !newPassword) {\n    throw new ApiResponseError('Missing 1 or more required fields: [oldPassword, newPassword]', 400);\n  }\n  const reqId = new RequestTracer();\n\n  const bitgo = req.bitgo;\n  const coin = bitgo.coin(req.params.coin);\n\n  if (otp) {\n    await bitgo.unlock({ otp });\n  }\n\n  const keychain = await coin.keychains().get({\n    id: req.params.id,\n    reqId,\n  });\n  if (!keychain) {\n    throw new ApiResponseError(`Keychain ${req.params.id} not found`, 404);\n  }\n\n  const updatedKeychain = coin.keychains().updateSingleKeychainPassword({\n    keychain,\n    oldPassword,\n    newPassword,\n  });\n\n  return bitgo.put(coin.url(`/key/${updatedKeychain.id}`)).send({\n    encryptedPrv: updatedKeychain.encryptedPrv,\n  });\n}\n\n/**\n * handle any other API call\n * @param req\n * @param res\n * @param next\n */\nfunction handleV2CoinSpecificREST(req: express.Request, res: express.Response, next: express.NextFunction) {\n  const method = req.method;\n  const bitgo = req.bitgo;\n\n  debug('handling v2 coin specific rest req');\n\n  try {\n    const coin = bitgo.coin(req.params.coin);\n    const coinURL = coin.url(createAPIPath(req));\n    return redirectRequest(bitgo, method, coinURL, req, next);\n  } catch (e) {\n    if (e instanceof UnsupportedCoinError) {\n      const queryParams = _.transform(\n        req.query,\n        (acc: string[], value, key) => {\n          for (const val of _.castArray(value)) {\n            acc.push(`${key}=${val}`);\n          }\n        },\n        []\n      );\n      const baseUrl = bitgo.url(req.baseUrl.replace(/^\\/api\\/v2/, ''), 2);\n      const url = _.isEmpty(queryParams) ? baseUrl : `${baseUrl}?${queryParams.join('&')}`;\n\n      debug(`coin ${req.params.coin} not supported, attempting to handle as a coinless route with url ${url}`);\n      return redirectRequest(bitgo, method, url, req, next);\n    }\n\n    throw e;\n  }\n}\n\n/**\n * Handle additional option to encrypt on the express route for partners requiring value encryption\n * @param req.body.encrypt - boolean to determine if the request should handle encryption on behalf of the submission.\n */\nasync function handleNetworkV1EnterpriseClientConnections(\n  req: express.Request,\n  res: express.Response,\n  next: express.NextFunction\n) {\n  debug('handling network v1 partner connection creation');\n\n  const bitgo = req.bitgo;\n  const params = req.params;\n  const body = req.body as CreateNetworkConnectionParams & {\n    encrypt?: boolean;\n  };\n\n  if (body.encrypt === true) {\n    if (!body.partnerId) {\n      throw new ApiResponseError('Missing required field: partnerId', 400);\n    }\n\n    const partnersUrl = bitgo.microservicesUrl(`/api/network/v1/enterprises/${params.enterpriseId}/partners`);\n\n    const response: GetNetworkPartnersResponse = await bitgo\n      .get(partnersUrl)\n      .set('enterprise-id', params.enterpriseId)\n      .send({ ids: [params.partnerId] })\n      .result();\n\n    const partners = response.partners;\n    const partner = partners.find((p) => p.id === body.partnerId);\n\n    if (!partner) {\n      throw new ApiResponseError(`Partner not found for partnerId: ${body.partnerId}`, 400);\n    }\n\n    if (!partner.publicKey) {\n      throw new ApiResponseError('Partner does not require encryption', 400);\n    }\n\n    switch (body.connectionKey.schema) {\n      case 'token':\n        req.body.connectionKey.connectionToken = await encryptRsaWithAesGcm(\n          partner.publicKey,\n          body.connectionKey.connectionToken\n        );\n        break;\n      case 'tokenAndSignature':\n        req.body.connectionKey.connectionToken = await encryptRsaWithAesGcm(\n          partner.publicKey,\n          body.connectionKey.connectionToken\n        );\n        req.body.connectionKey.signature = await encryptRsaWithAesGcm(partner.publicKey, body.connectionKey.signature);\n        break;\n      case 'apiKeyAndSecret':\n      case 'clearloop':\n        req.body.connectionKey.apiKey = await encryptRsaWithAesGcm(partner.publicKey, body.connectionKey.apiKey);\n        req.body.connectionKey.apiSecret = await encryptRsaWithAesGcm(partner.publicKey, body.connectionKey.apiSecret);\n        break;\n    }\n  }\n\n  return handleProxyReq(req, res, next);\n}\n\n/**\n * Redirect a request using the bitgo request functions.\n * @param bitgo\n * @param method\n * @param url\n * @param req\n * @param next\n */\nexport function redirectRequest(\n  bitgo: BitGo,\n  method: string,\n  url: string,\n  req: express.Request,\n  next: express.NextFunction\n) {\n  let request;\n\n  switch (method) {\n    case 'GET':\n      request = bitgo.get(url);\n      break;\n    case 'POST':\n      request = bitgo.post(url).send(req.body);\n      break;\n    case 'PUT':\n      request = bitgo.put(url).send(req.body);\n      break;\n    case 'PATCH':\n      request = bitgo.patch(url).send(req.body);\n      break;\n    case 'OPTIONS':\n      request = bitgo.options(url).send(req.body);\n      break;\n    case 'DELETE':\n      request = bitgo.del(url).send(req.body);\n      break;\n  }\n\n  if (request) {\n    if (req.params.enterpriseId) {\n      request.set('enterprise-id', req.params.enterpriseId);\n    }\n\n    return request.result().then((result) => {\n      const status = request.res?.statusCode || 200;\n      return { status, body: result };\n    });\n  }\n\n  // something has presumably gone wrong\n  next();\n}\n\nasync function handleProxyReq(req: express.Request, res: express.Response, next: express.NextFunction) {\n  const fullUrl = req.bitgo.microservicesUrl(req.originalUrl);\n  if (req.url && (/^\\/api.*$/.test(req.originalUrl) || /^\\/oauth\\/token.*$/.test(req.url))) {\n    req.isProxy = true;\n    debug('proxying %s request to %s', req.method, fullUrl);\n    return await redirectRequest(req.bitgo, req.method, fullUrl, req, next);\n  }\n  // user tried to access a url which is not an api route, do not proxy\n  debug('unable to proxy %s request to %s', req.method, fullUrl);\n  throw new ApiResponseError('bitgo-express can only proxy BitGo API requests', 404);\n}\n\n/**\n *\n * @param status\n * @param result\n * @param message\n */\nfunction apiResponse(status: number, result: any, message?: string): ApiResponseError {\n  return new ApiResponseError(message, status, result);\n}\n\nconst expressJSONParser = bodyParser.json({ limit: '20mb' });\n\n/**\n * Perform body parsing here only on routes we want\n */\nfunction parseBody(req: express.Request, res: express.Response, next: express.NextFunction) {\n  // Set the default Content-Type, in case the client doesn't set it.  If\n  // Content-Type isn't specified, Express silently refuses to parse the\n  // request body.\n  req.headers['content-type'] = req.headers['content-type'] || 'application/json';\n  return expressJSONParser(req, res, next);\n}\n\n/**\n * Create the bitgo object in the request\n * @param config\n */\nfunction prepareBitGo(config: Config) {\n  const { env, customRootUri, customBitcoinNetwork } = config;\n\n  return function prepBitGo(req: express.Request, res: express.Response, next: express.NextFunction) {\n    // Get access token\n    let accessToken;\n    if (req.headers.authorization) {\n      const authSplit = req.headers.authorization.split(' ');\n      if (authSplit.length === 2 && authSplit[0].toLowerCase() === 'bearer') {\n        accessToken = authSplit[1];\n      }\n    }\n    const userAgent = req.headers['user-agent']\n      ? BITGOEXPRESS_USER_AGENT + ' ' + req.headers['user-agent']\n      : BITGOEXPRESS_USER_AGENT;\n\n    const useProxyUrl = process.env.BITGO_USE_PROXY;\n    const bitgoConstructorParams: BitGoOptions = {\n      env,\n      customRootURI: customRootUri,\n      customBitcoinNetwork,\n      accessToken,\n      userAgent,\n      ...(useProxyUrl\n        ? {\n            customProxyAgent: new ProxyAgent({\n              getProxyForUrl: () => useProxyUrl,\n            }),\n          }\n        : {}),\n    };\n\n    req.bitgo = new BitGo(bitgoConstructorParams);\n    req.config = config;\n\n    next();\n  };\n}\ntype RequestHandlerResponse = string | unknown | undefined | { status: number; body: unknown };\ninterface RequestHandler extends express.RequestHandler<ParamsDictionary, any, RequestHandlerResponse> {\n  (req: express.Request, res: express.Response, next: express.NextFunction):\n    | RequestHandlerResponse\n    | Promise<RequestHandlerResponse>;\n}\n\nfunction handleRequestHandlerError(res: express.Response, error: unknown) {\n  let err;\n  if (error instanceof Error) {\n    err = error;\n  } else if (typeof error === 'string') {\n    err = new Error('(string_error) ' + error);\n  } else {\n    err = new Error('(object_error) ' + JSON.stringify(error));\n  }\n\n  const message = err.message || 'local error';\n  // use attached result, or make one\n  let result = err.result || { error: message };\n  result = _.extend({}, result, {\n    message: err.message,\n    bitgoJsVersion: version,\n    bitgoExpressVersion: pjson.version,\n  });\n  const status = err.status || 500;\n  if (!(status >= 200 && status < 300)) {\n    console.log('error %s: %s', status, err.message);\n  }\n  if (status >= 500 && status <= 599) {\n    if (err.response && err.response.request) {\n      console.log(`failed to make ${err.response.request.method} request to ${err.response.request.url}`);\n    }\n    console.log(err.stack);\n  }\n  res.status(status).send(result);\n}\n\n/**\n * Promise handler wrapper to handle sending responses and error cases\n * @param promiseRequestHandler\n */\nexport function promiseWrapper(promiseRequestHandler: RequestHandler) {\n  return async function promWrapper(req: express.Request, res: express.Response, next: express.NextFunction) {\n    debug(`handle: ${req.method} ${req.originalUrl}`);\n    try {\n      const result = await promiseRequestHandler(req, res, next);\n      if (typeof result === 'object' && result !== null && 'body' in result && 'status' in result) {\n        const { status, body } = result as { status: number; body: unknown };\n        res.status(status).send(body);\n      } else {\n        res.status(200).send(result);\n      }\n    } catch (e) {\n      handleRequestHandlerError(res, e);\n    }\n  };\n}\n\nexport function createCustomSigningFunction(externalSignerUrl: string): CustomSigningFunction {\n  return async function (params): Promise<SignedTransaction> {\n    const { body: signedTx } = await retryPromise(\n      () =>\n        superagent.post(`${externalSignerUrl}/api/v2/${params.coin.getChain()}/sign`).type('json').send({\n          txPrebuild: params.txPrebuild,\n          pubs: params.pubs,\n          derivationSeed: params.derivationSeed,\n          signingStep: params.signingStep,\n        }),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return signedTx;\n  };\n}\nexport function createCustomPaillierModulusGetter(\n  externalSignerUrl: string,\n  coin: string\n): CustomPaillierModulusGetterFunction {\n  return async function (params): Promise<{\n    userPaillierModulus: string;\n  }> {\n    const { body: result } = await retryPromise(\n      () => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/PaillierModulus`).type('json').send(params),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return result;\n  };\n}\n\nexport function createCustomKShareGenerator(externalSignerUrl: string, coin: string): CustomKShareGeneratingFunction {\n  return async function (params): Promise<TssEcdsaStep1ReturnMessage> {\n    const { body: result } = await retryPromise(\n      () => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/K`).type('json').send(params),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return result;\n  };\n}\n\nexport function createCustomMuDeltaShareGenerator(\n  externalSignerUrl: string,\n  coin: string\n): CustomMuDeltaShareGeneratingFunction {\n  return async function (params): Promise<TssEcdsaStep2ReturnMessage> {\n    const { body: result } = await retryPromise(\n      () => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/MuDelta`).type('json').send(params),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return result;\n  };\n}\n\nexport function createCustomSShareGenerator(externalSignerUrl: string, coin: string): CustomSShareGeneratingFunction {\n  return async function (params): Promise<SShare> {\n    const { body: result } = await retryPromise(\n      () => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/S`).type('json').send(params),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return result;\n  };\n}\n\nexport function createCustomCommitmentGenerator(\n  externalSignerUrl: string,\n  coin: string\n): CustomCommitmentGeneratingFunction {\n  return async function (params): Promise<{\n    userToBitgoCommitment: CommitmentShareRecord;\n    encryptedSignerShare: EncryptedSignerShareRecord;\n    encryptedUserToBitgoRShare: EncryptedSignerShareRecord;\n  }> {\n    const { body: result } = await retryPromise(\n      () => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/commitment`).type('json').send(params),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return result;\n  };\n}\n\nexport function createCustomRShareGenerator(externalSignerUrl: string, coin: string): CustomRShareGeneratingFunction {\n  return async function (params): Promise<{ rShare: SignShare }> {\n    const { body: rShare } = await retryPromise(\n      () => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/R`).type('json').send(params),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return rShare;\n  };\n}\n\nexport function createCustomGShareGenerator(externalSignerUrl: string, coin: string): CustomGShareGeneratingFunction {\n  return async function (params): Promise<GShare> {\n    const { body: signedTx } = await retryPromise(\n      () => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/G`).type('json').send(params),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return signedTx;\n  };\n}\n\nexport function createCustomMPCv2SigningRound1Generator(\n  externalSignerUrl: string,\n  coin: string\n): CustomMPCv2SigningRound1GeneratingFunction {\n  return async function (params) {\n    const { body: result } = await retryPromise(\n      () => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/MPCv2Round1`).type('json').send(params),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return result;\n  };\n}\n\nexport function createCustomMPCv2SigningRound2Generator(\n  externalSignerUrl: string,\n  coin: string\n): CustomMPCv2SigningRound2GeneratingFunction {\n  return async function (params) {\n    const { body: result } = await retryPromise(\n      () => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/MPCv2Round2`).type('json').send(params),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return result;\n  };\n}\n\nexport function createCustomMPCv2SigningRound3Generator(\n  externalSignerUrl: string,\n  coin: string\n): CustomMPCv2SigningRound3GeneratingFunction {\n  return async function (params) {\n    const { body: result } = await retryPromise(\n      () => superagent.post(`${externalSignerUrl}/api/v2/${coin}/tssshare/MPCv2Round3`).type('json').send(params),\n      (err, tryCount) => {\n        debug(`failed to connect to external signer (attempt ${tryCount}, error: ${err.message})`);\n      }\n    );\n    return result;\n  };\n}\n\nexport function setupAPIRoutes(app: express.Application, config: Config): void {\n  // When adding new routes to BitGo Express make sure that you also add the exact same routes to the server. Since\n  // some customers were confused when calling a BitGo Express route on the BitGo server, we now handle all BitGo\n  // Express routes on the BitGo server and return an error message that says that one should call BitGo Express\n  // instead.\n  // V1 routes should be added to www/config/routes.js\n  // V2 routes should be added to www/config/routesV2.js\n\n  // ping\n  // /api/v[12]/pingexpress is the only exception to the rule above, as it explicitly checks the health of the\n  // express server without running into rate limiting with the BitGo server.\n  app.get('/api/v[12]/ping', prepareBitGo(config), promiseWrapper(handlePing));\n  app.get('/api/v[12]/pingexpress', promiseWrapper(handlePingExpress));\n\n  // auth\n  app.post('/api/v[12]/user/login', parseBody, prepareBitGo(config), promiseWrapper(handleLogin));\n\n  app.post('/api/v[12]/decrypt', parseBody, prepareBitGo(config), promiseWrapper(handleDecrypt));\n  app.post('/api/v[12]/encrypt', parseBody, prepareBitGo(config), promiseWrapper(handleEncrypt));\n  app.post('/api/v[12]/verifyaddress', parseBody, prepareBitGo(config), promiseWrapper(handleVerifyAddress));\n  app.post(\n    '/api/v[12]/calculateminerfeeinfo',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleCalculateMinerFeeInfo)\n  );\n\n  app.post('/api/v1/keychain/local', parseBody, prepareBitGo(config), promiseWrapper(handleCreateLocalKeyChain));\n  app.post('/api/v1/keychain/derive', parseBody, prepareBitGo(config), promiseWrapper(handleDeriveLocalKeyChain));\n  app.post(\n    '/api/v1/wallets/simplecreate',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleCreateWalletWithKeychains)\n  );\n\n  app.post('/api/v1/wallet/:id/sendcoins', parseBody, prepareBitGo(config), promiseWrapper(handleSendCoins));\n  app.post('/api/v1/wallet/:id/sendmany', parseBody, prepareBitGo(config), promiseWrapper(handleSendMany));\n  app.post(\n    '/api/v1/wallet/:id/createtransaction',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleCreateTransaction)\n  );\n  app.post(\n    '/api/v1/wallet/:id/signtransaction',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleSignTransaction)\n  );\n\n  app.post('/api/v1/wallet/:id/simpleshare', parseBody, prepareBitGo(config), promiseWrapper(handleShareWallet));\n  app.post(\n    '/api/v1/walletshare/:shareId/acceptShare',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleAcceptShare)\n  );\n\n  app.put(\n    '/api/v1/pendingapprovals/:id/express',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleApproveTransaction)\n  );\n  app.put(\n    '/api/v1/pendingapprovals/:id/constructTx',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleConstructApprovalTx)\n  );\n\n  app.put(\n    '/api/v1/wallet/:id/consolidateunspents',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleConsolidateUnspents)\n  );\n  app.put('/api/v1/wallet/:id/fanoutunspents', parseBody, prepareBitGo(config), promiseWrapper(handleFanOutUnspents));\n\n  // any other API call\n  app.use('/api/v[1]/*', parseBody, prepareBitGo(config), promiseWrapper(handleREST));\n\n  // API v2\n\n  // create keychain\n  app.post(\n    '/api/v2/:coin/keychain/local',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2CreateLocalKeyChain)\n  );\n\n  // generate wallet\n  app.post('/api/v2/:coin/wallet/generate', parseBody, prepareBitGo(config), promiseWrapper(handleV2GenerateWallet));\n\n  app.put('/express/api/v2/:coin/wallet/:id', parseBody, prepareBitGo(config), promiseWrapper(handleWalletUpdate));\n\n  // change wallet passphrase\n  app.post(\n    '/api/v2/:coin/keychain/:id/changepassword',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleKeychainChangePassword)\n  );\n\n  // create address\n  app.post('/api/v2/:coin/wallet/:id/address', parseBody, prepareBitGo(config), promiseWrapper(handleV2CreateAddress));\n\n  // share wallet\n  app.post('/api/v2/:coin/wallet/:id/share', parseBody, prepareBitGo(config), promiseWrapper(handleV2ShareWallet));\n  app.post(\n    '/api/v2/:coin/walletshare/:id/acceptshare',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2AcceptWalletShare)\n  );\n\n  // sign arbitrary payloads w/ trading account key\n  app.post(`/api/v2/ofc/signPayload`, parseBody, prepareBitGo(config), promiseWrapper(handleV2OFCSignPayload));\n\n  // sign transaction\n  app.post('/api/v2/:coin/signtx', parseBody, prepareBitGo(config), promiseWrapper(handleV2SignTx));\n  app.post('/api/v2/:coin/wallet/:id/signtx', parseBody, prepareBitGo(config), promiseWrapper(handleV2SignTxWallet));\n  app.post(\n    '/api/v2/:coin/wallet/:id/signtxtss',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2SignTSSWalletTx)\n  );\n  app.post(\n    '/api/v2/:coin/wallet/:id/recovertoken',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2RecoverToken)\n  );\n\n  // send transaction\n  app.post('/api/v2/:coin/wallet/:id/sendcoins', parseBody, prepareBitGo(config), promiseWrapper(handleV2SendOne));\n  app.post('/api/v2/:coin/wallet/:id/sendmany', parseBody, prepareBitGo(config), promiseWrapper(handleV2SendMany));\n  app.post(\n    '/api/v2/:coin/wallet/:id/prebuildAndSignTransaction',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2PrebuildAndSignTransaction)\n  );\n\n  // token enablement\n  app.post(\n    '/api/v2/:coin/wallet/:id/enableTokens',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2EnableTokens)\n  );\n\n  // unspent changes\n  app.post(\n    '/api/v2/:coin/wallet/:id/consolidateunspents',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2ConsolidateUnspents)\n  );\n  app.post(\n    '/api/v2/:coin/wallet/:id/fanoutunspents',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2FanOutUnspents)\n  );\n\n  app.post('/api/v2/:coin/wallet/:id/sweep', parseBody, prepareBitGo(config), promiseWrapper(handleV2Sweep));\n\n  // CPFP\n  app.post(\n    '/api/v2/:coin/wallet/:id/acceleratetx',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2AccelerateTransaction)\n  );\n\n  // account-based\n  app.post(\n    '/api/v2/:coin/wallet/:id/consolidateAccount',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2ConsolidateAccount)\n  );\n\n  // Miscellaneous\n  app.post('/api/v2/:coin/canonicaladdress', parseBody, prepareBitGo(config), promiseWrapper(handleCanonicalAddress));\n  app.post('/api/v2/:coin/verifyaddress', parseBody, prepareBitGo(config), promiseWrapper(handleV2VerifyAddress));\n  app.put(\n    '/api/v2/:coin/pendingapprovals/:id',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2PendingApproval)\n  );\n\n  // lightning - pay invoice\n  app.post(\n    '/api/v2/:coin/wallet/:id/lightning/payment',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handlePayLightningInvoice)\n  );\n\n  // lightning - onchain withdrawal\n  app.post(\n    '/api/v2/:coin/wallet/:id/lightning/withdraw',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleLightningWithdraw)\n  );\n\n  // any other API v2 call\n  app.use('/api/v2/user/*', parseBody, prepareBitGo(config), promiseWrapper(handleV2UserREST));\n  app.use('/api/v2/:coin/*', parseBody, prepareBitGo(config), promiseWrapper(handleV2CoinSpecificREST));\n\n  app.post(\n    '/api/network/v1/enterprises/:enterpriseId/clients/connections',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleNetworkV1EnterpriseClientConnections)\n  );\n\n  // everything else should use the proxy handler\n  if (config.disableProxy !== true) {\n    app.use(\n      '/api/:namespace/v[12]/enterprises/:enterpriseId/*',\n      parseBody,\n      prepareBitGo(config),\n      promiseWrapper(handleProxyReq)\n    );\n\n    app.use(parseBody, prepareBitGo(config), promiseWrapper(handleProxyReq));\n  }\n}\n\nexport function setupSigningRoutes(app: express.Application, config: Config): void {\n  app.post('/api/v2/:coin/sign', parseBody, prepareBitGo(config), promiseWrapper(handleV2Sign));\n  app.post(\n    '/api/v2/:coin/tssshare/:sharetype',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2GenerateShareTSS)\n  );\n  app.post(\n    `/api/v2/ofc/signPayload`,\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleV2OFCSignPayloadInExtSigningMode)\n  );\n}\n\nexport function setupLightningSignerNodeRoutes(app: express.Application, config: Config): void {\n  app.post(\n    '/api/v2/:coin/wallet/:id/initwallet',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleInitLightningWallet)\n  );\n  app.post(\n    '/api/v2/:coin/wallet/:id/signermacaroon',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleCreateSignerMacaroon)\n  );\n  app.post(\n    '/api/v2/:coin/wallet/:id/unlockwallet',\n    parseBody,\n    prepareBitGo(config),\n    promiseWrapper(handleUnlockLightningWallet)\n  );\n  app.get('/api/v2/:coin/wallet/:id/state', prepareBitGo(config), promiseWrapper(handleGetLightningWalletState));\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!