PHP WebShell

Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/sdk-core/dist/src/bitgo/keychain

Просмотр файла: keychains.js

"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    var desc = Object.getOwnPropertyDescriptor(m, k);
    if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
      desc = { enumerable: true, get: function() { return m[k]; } };
    }
    Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
    Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
    o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
    var ownKeys = function(o) {
        ownKeys = Object.getOwnPropertyNames || function (o) {
            var ar = [];
            for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
            return ar;
        };
        return ownKeys(o);
    };
    return function (mod) {
        if (mod && mod.__esModule) return mod;
        var result = {};
        if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
        __setModuleDefault(result, mod);
        return result;
    };
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.Keychains = void 0;
const assert_1 = __importDefault(require("assert"));
const _ = __importStar(require("lodash"));
const common = __importStar(require("../../common"));
const utils_1 = require("../utils");
const ovcJsonCodec_1 = require("./ovcJsonCodec");
class Keychains {
    constructor(bitgo, baseCoin) {
        this.bitgo = bitgo;
        this.baseCoin = baseCoin;
    }
    /**
     * Get a keychain by ID
     * @param params
     * @param params.id
     * @param params.xpub (optional)
     * @param params.ethAddress (optional)
     * @param params.reqId (optional)
     */
    async get(params) {
        common.validateParams(params, [], ['xpub', 'ethAddress']);
        if (_.isUndefined(params.id)) {
            throw new Error('id must be defined');
        }
        const id = params.id;
        if (params.reqId) {
            this.bitgo.setRequestTracer(params.reqId);
        }
        return await this.bitgo.get(this.baseCoin.url('/key/' + encodeURIComponent(id))).result();
    }
    /**
     * list the users keychains
     * @param params
     * @param params.limit - Max number of results in a single call.
     * @param params.prevId - Continue iterating (provided by nextBatchPrevId in the previous list)
     * @returns {*}
     */
    async list(params = {}) {
        const queryObject = {};
        if (!_.isUndefined(params.limit)) {
            if (!_.isNumber(params.limit)) {
                throw new Error('invalid limit argument, expecting number');
            }
            queryObject.limit = params.limit;
        }
        if (!_.isUndefined(params.prevId)) {
            if (!_.isString(params.prevId)) {
                throw new Error('invalid prevId argument, expecting string');
            }
            queryObject.prevId = params.prevId;
        }
        return this.bitgo.get(this.baseCoin.url('/key')).query(queryObject).result();
    }
    /**
     * Change the decryption password for all possible keychains associated with a user.
     *
     * This function iterates through all keys associated with the user, decrypts
     * them with the old password and re-encrypts them with the new password.
     *
     * This should be called when a user changes their login password, and are expecting
     * that their wallet passwords are changed to match the new login password.
     *
     * @param params
     * @param params.oldPassword - The old password used for encrypting the key
     * @param params.newPassword - The new password to be used for encrypting the key
     * @returns changedKeys Object - e.g.:
     *  {
     *    xpub1: encryptedPrv,
     *    ...
     *  }
     */
    async updatePassword(params) {
        common.validateParams(params, ['oldPassword', 'newPassword'], []);
        const changedKeys = {};
        let prevId;
        let keysLeft = true;
        while (keysLeft) {
            const result = await this.list({ limit: 500, prevId });
            for (const key of result.keys) {
                const oldEncryptedPrv = key.encryptedPrv;
                if (_.isUndefined(oldEncryptedPrv)) {
                    continue;
                }
                try {
                    const updatedKeychain = this.updateSingleKeychainPassword({
                        keychain: key,
                        oldPassword: params.oldPassword,
                        newPassword: params.newPassword,
                    });
                    if (updatedKeychain.encryptedPrv) {
                        const changedKeyIdentifier = updatedKeychain.type === 'tss' ? updatedKeychain.id : updatedKeychain.pub;
                        if (changedKeyIdentifier) {
                            changedKeys[changedKeyIdentifier] = updatedKeychain.encryptedPrv;
                        }
                    }
                }
                catch (e) {
                    // if the password was incorrect, silence the error, throw otherwise
                    if (!e.message.includes('private key is incorrect')) {
                        throw e;
                    }
                }
            }
            if (result.nextBatchPrevId) {
                prevId = result.nextBatchPrevId;
            }
            else {
                keysLeft = false;
            }
        }
        return changedKeys;
    }
    /**
     * Update the password used to decrypt a single keychain
     * @param params
     * @param params.keychain - The keychain whose password should be updated
     * @param params.oldPassword - The old password used for encrypting the key
     * @param params.newPassword - The new password to be used for encrypting the key
     * @returns {object}
     */
    updateSingleKeychainPassword(params = {}) {
        if (!_.isString(params.oldPassword)) {
            throw new Error('expected old password to be a string');
        }
        if (!_.isString(params.newPassword)) {
            throw new Error('expected new password to be a string');
        }
        if (!_.isObject(params.keychain) || !_.isString(params.keychain.encryptedPrv)) {
            throw new Error('expected keychain to be an object with an encryptedPrv property');
        }
        const oldEncryptedPrv = params.keychain.encryptedPrv;
        try {
            const decryptedPrv = this.bitgo.decrypt({ input: oldEncryptedPrv, password: params.oldPassword });
            const newEncryptedPrv = this.bitgo.encrypt({ input: decryptedPrv, password: params.newPassword });
            return _.assign({}, params.keychain, { encryptedPrv: newEncryptedPrv });
        }
        catch (e) {
            // catching an error here means that the password was incorrect or, less likely, the input to decrypt is corrupted
            throw new Error('password used to decrypt keychain private key is incorrect');
        }
    }
    /**
     * Create a public/private key pair
     * @param params - optional params
     * @param params.seed optional - seed to use for keypair generation
     * @param params.isRootKey optional - whether the resulting keypair should be a root key
     * @returns {KeyPair} - the generated keypair
     */
    create(params = {}) {
        if (params?.isRootKey) {
            return this.baseCoin.generateRootKeyPair(params.seed);
        }
        return this.baseCoin.generateKeyPair(params.seed);
    }
    /**
     * Add a keychain to BitGo's records
     * @param params
     */
    async add(params = {}) {
        params = params || {};
        common.validateParams(params, [], [
            'pub',
            'encryptedPrv',
            'keyType',
            'type',
            'source',
            'originalPasscodeEncryptionCode',
            'enterprise',
            'derivedFromParentWithSeed',
        ]);
        if (!_.isUndefined(params.disableKRSEmail)) {
            if (!_.isBoolean(params.disableKRSEmail)) {
                throw new Error('invalid disableKRSEmail argument, expecting boolean');
            }
        }
        if (params.reqId) {
            this.bitgo.setRequestTracer(params.reqId);
        }
        return await this.bitgo
            .post(this.baseCoin.url('/key'))
            .send({
            pub: params.pub,
            commonPub: params.commonPub,
            commonKeychain: params.commonKeychain,
            encryptedPrv: params.encryptedPrv,
            type: params.type,
            keyType: params.keyType,
            source: params.source,
            provider: params.provider,
            originalPasscodeEncryptionCode: params.originalPasscodeEncryptionCode,
            enterprise: params.enterprise,
            derivedFromParentWithSeed: params.derivedFromParentWithSeed,
            disableKRSEmail: params.disableKRSEmail,
            krsSpecific: params.krsSpecific,
            keyShares: params.keyShares,
            userGPGPublicKey: params.userGPGPublicKey,
            backupGPGPublicKey: params.backupGPGPublicKey,
            algoUsed: params.algoUsed,
            isDistributedCustody: params.isDistributedCustody,
            isMPCv2: params.isMPCv2,
            coinSpecific: params.coinSpecific,
        })
            .result();
    }
    /**
     * Create a BitGo key
     * @param params (empty)
     */
    async createBitGo(params = {}) {
        params.source = 'bitgo';
        this.baseCoin.preCreateBitGo(params);
        return await this.add(params);
    }
    /**
     * Create a backup key
     * @param params
     * @param params.provider (optional)
     */
    async createBackup(params = {}) {
        params.source = 'backup';
        const isTssBackupKey = params.prv && (params.commonKeychain || params.commonPub);
        if (_.isUndefined(params.provider) && !isTssBackupKey) {
            // if the provider is undefined, we generate a local key and add the source details
            const key = this.create();
            _.extend(params, key);
            if (params.passphrase !== undefined) {
                _.extend(params, { encryptedPrv: this.bitgo.encrypt({ input: key.prv, password: params.passphrase }) });
            }
        }
        const serverResponse = await this.add(params);
        return _.extend({}, serverResponse, _.pick(params, ['prv', 'encryptedPrv', 'provider', 'source']));
    }
    /**
     * Gets keys for signing from a wallet
     * @param params
     * @returns {Promise<Keychain[]>}
     */
    async getKeysForSigning(params = {}) {
        if (!_.isObject(params.wallet)) {
            throw new Error('missing required param wallet');
        }
        const wallet = params.wallet;
        const reqId = params.reqId || new utils_1.RequestTracer();
        const ids = wallet.baseCoin.keyIdsForSigning();
        const keychainQueriesBluebirds = ids.map((id) => this.get({ id: wallet.keyIds()[id], reqId }));
        return Promise.all(keychainQueriesBluebirds);
    }
    /**
     * Convenience function to create and store MPC keychains with BitGo.
     * @param params passphrase used to encrypt secret materials
     * @return {Promise<KeychainsTriplet>} newly created User, Backup, and BitGo keys
     */
    async createMpc(params) {
        let MpcUtils;
        let multisigTypeVersion = undefined;
        if (params.multisigType === 'tss' && this.baseCoin.getMPCAlgorithm() === 'ecdsa') {
            const tssSettings = await this.bitgo
                .get(this.bitgo.microservicesUrl('/api/v2/tss/settings'))
                .result();
            multisigTypeVersion =
                tssSettings.coinSettings[this.baseCoin.getFamily()]?.walletCreationSettings?.multiSigTypeVersion;
        }
        switch (params.multisigType) {
            case 'tss':
                MpcUtils =
                    this.baseCoin.getMPCAlgorithm() === 'eddsa'
                        ? utils_1.EDDSAUtils.default
                        : multisigTypeVersion === 'MPCv2'
                            ? utils_1.ECDSAUtils.EcdsaMPCv2Utils
                            : utils_1.ECDSAUtils.EcdsaUtils;
                break;
            default:
                throw new Error('Unsupported multi-sig type');
        }
        const mpcUtils = new MpcUtils(this.bitgo, this.baseCoin);
        return await mpcUtils.createKeychains({
            passphrase: params.passphrase,
            enterprise: params.enterprise,
            originalPasscodeEncryptionCode: params.originalPasscodeEncryptionCode,
            retrofit: params.retrofit,
        });
    }
    async recreateMpc(params) {
        (0, assert_1.default)(params.coin, new Error('missing required param coin'));
        (0, assert_1.default)(params.walletId, new Error('missing required param walletId'));
        (0, assert_1.default)(params.otp, new Error('missing required param otp'));
        (0, assert_1.default)(params.passphrase, new Error('missing required param passphrase'));
        (0, assert_1.default)(params.encryptedMaterial.encryptedWalletPassphrase, new Error('missing required param encryptedWalletPassphrase'));
        (0, assert_1.default)(params.encryptedMaterial.encryptedUserKey, new Error('missing required param encryptedUserKey'));
        (0, assert_1.default)(params.encryptedMaterial.encryptedBackupKey, new Error('missing required param encryptedBackupKey'));
        await this.bitgo.post(this.bitgo.microservicesUrl('/api/v1/user/unlock')).send({ otp: params.otp }).result();
        const { recoveryInfo } = await this.bitgo
            .post(this.bitgo.microservicesUrl(`/api/v2/${params.coin}/wallet/${params.walletId}/passcoderecovery`))
            .result();
        if (!recoveryInfo || !('passcodeEncryptionCode' in recoveryInfo)) {
            throw new Error('failed to get recovery info');
        }
        const decryptedWalletPassphrase = this.bitgo.decrypt({
            input: params.encryptedMaterial.encryptedWalletPassphrase,
            password: recoveryInfo.passcodeEncryptionCode,
        });
        const decryptedUserKey = this.bitgo.decrypt({
            input: params.encryptedMaterial.encryptedUserKey,
            password: decryptedWalletPassphrase,
        });
        const decryptedBackupKey = this.bitgo.decrypt({
            input: params.encryptedMaterial.encryptedBackupKey,
            password: decryptedWalletPassphrase,
        });
        return this.createMpc({
            ...params,
            multisigType: 'tss',
            retrofit: {
                decryptedUserKey,
                decryptedBackupKey,
                walletId: params.walletId,
            },
        });
    }
    /**
     * It parses the JSON downloaded from the OVC for platform (BitGo),
     * and creates a corresponding TSS BitGo key. It also returns the JSON that needs
     * to be uploaded back to the OVCs containing the BitGo -> OVC shares.
     * @param ovcOutputJson JSON format of the file downloaded from the OVC for platform
     * @param enterprise Optional enterprise ID associated with the key
     * @returns {BitGoKeyFromOvcShares}
     */
    async createTssBitGoKeyFromOvcShares(ovcOutputJson, enterprise) {
        const decodedOvcOutput = (0, utils_1.decodeOrElse)(ovcJsonCodec_1.OvcToBitGoJSON.name, ovcJsonCodec_1.OvcToBitGoJSON, ovcOutputJson, (errors) => {
            throw new Error(`Error(s) parsing OVC JSON: ${errors}`);
        });
        if (decodedOvcOutput.state !== 1) {
            throw new Error('State expected to be "1". Please complete the first two OVC operations');
        }
        // OVC-1 is responsible for the User key
        const ovc1 = decodedOvcOutput.ovc[1];
        // OVC-2 is responsible for the Backup key
        const ovc2 = decodedOvcOutput.ovc[2];
        const keyShares = [
            {
                from: 'user',
                to: 'bitgo',
                publicShare: ovc1.ovcToBitgoShare.publicShare,
                privateShare: ovc1.ovcToBitgoShare.privateShare,
                privateShareProof: ovc1.ovcToBitgoShare.uSig.toString() ?? '',
                vssProof: ovc1.ovcToBitgoShare.vssProof ?? '',
            },
            {
                from: 'backup',
                to: 'bitgo',
                publicShare: ovc2.ovcToBitgoShare.publicShare,
                privateShare: ovc2.ovcToBitgoShare.privateShare,
                privateShareProof: ovc2.ovcToBitgoShare.uSig.toString() ?? '',
                vssProof: ovc2.ovcToBitgoShare.vssProof ?? '',
            },
        ];
        const key = await this.baseCoin.keychains().add({
            source: 'bitgo',
            keyShares,
            keyType: 'tss',
            userGPGPublicKey: ovc1.gpgPubKey,
            backupGPGPublicKey: ovc2.gpgPubKey,
            enterprise,
        });
        (0, assert_1.default)(key.keyShares);
        (0, assert_1.default)(key.commonKeychain);
        (0, assert_1.default)(key.walletHSMGPGPublicKeySigs);
        const bitgoToUserShare = key.keyShares.find((value) => value.from === 'bitgo' && value.to === 'user');
        (0, assert_1.default)(bitgoToUserShare);
        (0, assert_1.default)(bitgoToUserShare.vssProof);
        (0, assert_1.default)(bitgoToUserShare.paillierPublicKey);
        const bitgoToBackupShare = key.keyShares.find((value) => value.from === 'bitgo' && value.to === 'backup');
        (0, assert_1.default)(bitgoToBackupShare);
        (0, assert_1.default)(bitgoToBackupShare.vssProof);
        (0, assert_1.default)(bitgoToBackupShare.paillierPublicKey);
        // Create JSON data with platform shares for OVC-1 and OVC-2
        const bitgoToOvcOutput = {
            wallet: {
                ...decodedOvcOutput,
                platform: {
                    commonKeychain: key.commonKeychain,
                    walletGpgPubKeySigs: key.walletHSMGPGPublicKeySigs,
                    ovc: {
                        // BitGo to User (OVC-1)
                        1: {
                            bitgoToOvcShare: {
                                i: 1,
                                j: 3,
                                publicShare: bitgoToUserShare.publicShare,
                                privateShare: bitgoToUserShare.privateShare,
                                paillierPublicKey: bitgoToUserShare.paillierPublicKey,
                                vssProof: bitgoToUserShare.vssProof,
                            },
                        },
                        // BitGo to Backup (OVC-2)
                        2: {
                            bitgoToOvcShare: {
                                i: 2,
                                j: 3,
                                publicShare: bitgoToBackupShare.publicShare,
                                privateShare: bitgoToBackupShare.privateShare,
                                paillierPublicKey: bitgoToBackupShare.paillierPublicKey,
                                vssProof: bitgoToBackupShare.vssProof,
                            },
                        },
                    },
                },
            },
        };
        // Mark it ready for next operation, should be 2
        bitgoToOvcOutput.wallet.state += 1;
        const output = {
            bitGoKeyId: key.id,
            bitGoOutputJsonForOvc: bitgoToOvcOutput,
        };
        return (0, utils_1.decodeOrElse)(ovcJsonCodec_1.BitGoKeyFromOvcShares.name, ovcJsonCodec_1.BitGoKeyFromOvcShares, output, (errors) => {
            throw new Error(`Error producing the output: ${errors}`);
        });
    }
    /**
     * Create user keychain, encrypt the private key with the wallet passphrase and store it in BitGo.
     * @param walletPassphrase
     * @returns Keychain including the decrypted private key
     */
    async createUserKeychain(walletPassphrase) {
        const keychains = this.baseCoin.keychains();
        const newKeychain = keychains.create();
        const originalPasscodeEncryptionCode = (0, utils_1.generateRandomPassword)(5);
        const encryptedPrv = this.bitgo.encrypt({
            password: walletPassphrase,
            input: newKeychain.prv,
        });
        return {
            ...(await keychains.add({
                encryptedPrv,
                originalPasscodeEncryptionCode,
                pub: newKeychain.pub,
                source: 'user',
            })),
            prv: newKeychain.prv,
        };
    }
}
exports.Keychains = Keychains;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"keychains.js","sourceRoot":"","sources":["../../../../src/bitgo/keychain/keychains.ts"],"names":[],"mappings":"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","sourcesContent":["import { TssSettings } from '@bitgo/public-types';\nimport assert from 'assert';\nimport * as _ from 'lodash';\nimport * as common from '../../common';\nimport { IBaseCoin, KeychainsTriplet, KeyPair } from '../baseCoin';\nimport { BitGoBase } from '../bitgoBase';\nimport { decodeOrElse, ECDSAUtils, EDDSAUtils, generateRandomPassword, RequestTracer } from '../utils';\nimport {\n  AddKeychainOptions,\n  ApiKeyShare,\n  ChangedKeychains,\n  CreateBackupOptions,\n  CreateBitGoOptions,\n  CreateMpcOptions,\n  GetKeychainOptions,\n  GetKeysForSigningOptions,\n  IKeychains,\n  Keychain,\n  ListKeychainOptions,\n  ListKeychainsResult,\n  RecreateMpcOptions,\n  UpdatePasswordOptions,\n  UpdateSingleKeychainPasswordOptions,\n} from './iKeychains';\nimport { BitGoKeyFromOvcShares, BitGoToOvcJSON, OvcToBitGoJSON } from './ovcJsonCodec';\n\nexport class Keychains implements IKeychains {\n  private readonly bitgo: BitGoBase;\n  private readonly baseCoin: IBaseCoin;\n\n  constructor(bitgo: BitGoBase, baseCoin: IBaseCoin) {\n    this.bitgo = bitgo;\n    this.baseCoin = baseCoin;\n  }\n\n  /**\n   * Get a keychain by ID\n   * @param params\n   * @param params.id\n   * @param params.xpub (optional)\n   * @param params.ethAddress (optional)\n   * @param params.reqId (optional)\n   */\n  async get(params: GetKeychainOptions): Promise<Keychain> {\n    common.validateParams(params, [], ['xpub', 'ethAddress']);\n\n    if (_.isUndefined(params.id)) {\n      throw new Error('id must be defined');\n    }\n\n    const id = params.id;\n    if (params.reqId) {\n      this.bitgo.setRequestTracer(params.reqId);\n    }\n    return await this.bitgo.get(this.baseCoin.url('/key/' + encodeURIComponent(id))).result();\n  }\n\n  /**\n   * list the users keychains\n   * @param params\n   * @param params.limit - Max number of results in a single call.\n   * @param params.prevId - Continue iterating (provided by nextBatchPrevId in the previous list)\n   * @returns {*}\n   */\n  async list(params: ListKeychainOptions = {}): Promise<ListKeychainsResult> {\n    const queryObject: any = {};\n\n    if (!_.isUndefined(params.limit)) {\n      if (!_.isNumber(params.limit)) {\n        throw new Error('invalid limit argument, expecting number');\n      }\n      queryObject.limit = params.limit;\n    }\n    if (!_.isUndefined(params.prevId)) {\n      if (!_.isString(params.prevId)) {\n        throw new Error('invalid prevId argument, expecting string');\n      }\n      queryObject.prevId = params.prevId;\n    }\n\n    return this.bitgo.get(this.baseCoin.url('/key')).query(queryObject).result();\n  }\n\n  /**\n   * Change the decryption password for all possible keychains associated with a user.\n   *\n   * This function iterates through all keys associated with the user, decrypts\n   * them with the old password and re-encrypts them with the new password.\n   *\n   * This should be called when a user changes their login password, and are expecting\n   * that their wallet passwords are changed to match the new login password.\n   *\n   * @param params\n   * @param params.oldPassword - The old password used for encrypting the key\n   * @param params.newPassword - The new password to be used for encrypting the key\n   * @returns changedKeys Object - e.g.:\n   *  {\n   *    xpub1: encryptedPrv,\n   *    ...\n   *  }\n   */\n  async updatePassword(params: UpdatePasswordOptions): Promise<ChangedKeychains> {\n    common.validateParams(params, ['oldPassword', 'newPassword'], []);\n    const changedKeys: ChangedKeychains = {};\n    let prevId;\n    let keysLeft = true;\n    while (keysLeft) {\n      const result: ListKeychainsResult = await this.list({ limit: 500, prevId });\n      for (const key of result.keys) {\n        const oldEncryptedPrv = key.encryptedPrv;\n        if (_.isUndefined(oldEncryptedPrv)) {\n          continue;\n        }\n        try {\n          const updatedKeychain = this.updateSingleKeychainPassword({\n            keychain: key,\n            oldPassword: params.oldPassword,\n            newPassword: params.newPassword,\n          });\n          if (updatedKeychain.encryptedPrv) {\n            const changedKeyIdentifier = updatedKeychain.type === 'tss' ? updatedKeychain.id : updatedKeychain.pub;\n            if (changedKeyIdentifier) {\n              changedKeys[changedKeyIdentifier] = updatedKeychain.encryptedPrv;\n            }\n          }\n        } catch (e) {\n          // if the password was incorrect, silence the error, throw otherwise\n          if (!e.message.includes('private key is incorrect')) {\n            throw e;\n          }\n        }\n      }\n      if (result.nextBatchPrevId) {\n        prevId = result.nextBatchPrevId;\n      } else {\n        keysLeft = false;\n      }\n    }\n    return changedKeys;\n  }\n\n  /**\n   * Update the password used to decrypt a single keychain\n   * @param params\n   * @param params.keychain - The keychain whose password should be updated\n   * @param params.oldPassword - The old password used for encrypting the key\n   * @param params.newPassword - The new password to be used for encrypting the key\n   * @returns {object}\n   */\n  updateSingleKeychainPassword(params: UpdateSingleKeychainPasswordOptions = {}): Keychain {\n    if (!_.isString(params.oldPassword)) {\n      throw new Error('expected old password to be a string');\n    }\n\n    if (!_.isString(params.newPassword)) {\n      throw new Error('expected new password to be a string');\n    }\n\n    if (!_.isObject(params.keychain) || !_.isString(params.keychain.encryptedPrv)) {\n      throw new Error('expected keychain to be an object with an encryptedPrv property');\n    }\n\n    const oldEncryptedPrv = params.keychain.encryptedPrv;\n    try {\n      const decryptedPrv = this.bitgo.decrypt({ input: oldEncryptedPrv, password: params.oldPassword });\n      const newEncryptedPrv = this.bitgo.encrypt({ input: decryptedPrv, password: params.newPassword });\n      return _.assign({}, params.keychain, { encryptedPrv: newEncryptedPrv });\n    } catch (e) {\n      // catching an error here means that the password was incorrect or, less likely, the input to decrypt is corrupted\n      throw new Error('password used to decrypt keychain private key is incorrect');\n    }\n  }\n\n  /**\n   * Create a public/private key pair\n   * @param params - optional params\n   * @param params.seed optional - seed to use for keypair generation\n   * @param params.isRootKey optional - whether the resulting keypair should be a root key\n   * @returns {KeyPair} - the generated keypair\n   */\n  create(params: { seed?: Buffer; isRootKey?: boolean } = {}): KeyPair {\n    if (params?.isRootKey) {\n      return this.baseCoin.generateRootKeyPair(params.seed);\n    }\n    return this.baseCoin.generateKeyPair(params.seed);\n  }\n\n  /**\n   * Add a keychain to BitGo's records\n   * @param params\n   */\n  async add(params: AddKeychainOptions = {}): Promise<Keychain> {\n    params = params || {};\n    common.validateParams(\n      params,\n      [],\n      [\n        'pub',\n        'encryptedPrv',\n        'keyType',\n        'type',\n        'source',\n        'originalPasscodeEncryptionCode',\n        'enterprise',\n        'derivedFromParentWithSeed',\n      ]\n    );\n\n    if (!_.isUndefined(params.disableKRSEmail)) {\n      if (!_.isBoolean(params.disableKRSEmail)) {\n        throw new Error('invalid disableKRSEmail argument, expecting boolean');\n      }\n    }\n\n    if (params.reqId) {\n      this.bitgo.setRequestTracer(params.reqId);\n    }\n\n    return await this.bitgo\n      .post(this.baseCoin.url('/key'))\n      .send({\n        pub: params.pub,\n        commonPub: params.commonPub,\n        commonKeychain: params.commonKeychain,\n        encryptedPrv: params.encryptedPrv,\n        type: params.type,\n        keyType: params.keyType,\n        source: params.source,\n        provider: params.provider,\n        originalPasscodeEncryptionCode: params.originalPasscodeEncryptionCode,\n        enterprise: params.enterprise,\n        derivedFromParentWithSeed: params.derivedFromParentWithSeed,\n        disableKRSEmail: params.disableKRSEmail,\n        krsSpecific: params.krsSpecific,\n        keyShares: params.keyShares,\n        userGPGPublicKey: params.userGPGPublicKey,\n        backupGPGPublicKey: params.backupGPGPublicKey,\n        algoUsed: params.algoUsed,\n        isDistributedCustody: params.isDistributedCustody,\n        isMPCv2: params.isMPCv2,\n        coinSpecific: params.coinSpecific,\n      })\n      .result();\n  }\n\n  /**\n   * Create a BitGo key\n   * @param params (empty)\n   */\n  async createBitGo(params: CreateBitGoOptions = {}): Promise<Keychain> {\n    params.source = 'bitgo';\n\n    this.baseCoin.preCreateBitGo(params as any);\n    return await this.add(params);\n  }\n\n  /**\n   * Create a backup key\n   * @param params\n   * @param params.provider (optional)\n   */\n  async createBackup(params: CreateBackupOptions = {}): Promise<Keychain> {\n    params.source = 'backup';\n\n    const isTssBackupKey = params.prv && (params.commonKeychain || params.commonPub);\n\n    if (_.isUndefined(params.provider) && !isTssBackupKey) {\n      // if the provider is undefined, we generate a local key and add the source details\n      const key = this.create();\n      _.extend(params, key);\n      if (params.passphrase !== undefined) {\n        _.extend(params, { encryptedPrv: this.bitgo.encrypt({ input: key.prv, password: params.passphrase }) });\n      }\n    }\n\n    const serverResponse = await this.add(params);\n    return _.extend({}, serverResponse, _.pick(params, ['prv', 'encryptedPrv', 'provider', 'source']));\n  }\n\n  /**\n   * Gets keys for signing from a wallet\n   * @param params\n   * @returns {Promise<Keychain[]>}\n   */\n  async getKeysForSigning(params: GetKeysForSigningOptions = {}): Promise<Keychain[]> {\n    if (!_.isObject(params.wallet)) {\n      throw new Error('missing required param wallet');\n    }\n    const wallet = params.wallet;\n    const reqId = params.reqId || new RequestTracer();\n    const ids = wallet.baseCoin.keyIdsForSigning();\n    const keychainQueriesBluebirds = ids.map((id) => this.get({ id: wallet.keyIds()[id], reqId }));\n    return Promise.all(keychainQueriesBluebirds);\n  }\n\n  /**\n   * Convenience function to create and store MPC keychains with BitGo.\n   * @param params passphrase used to encrypt secret materials\n   * @return {Promise<KeychainsTriplet>} newly created User, Backup, and BitGo keys\n   */\n  async createMpc(params: CreateMpcOptions): Promise<KeychainsTriplet> {\n    let MpcUtils;\n    let multisigTypeVersion: 'MPCv2' | undefined = undefined;\n    if (params.multisigType === 'tss' && this.baseCoin.getMPCAlgorithm() === 'ecdsa') {\n      const tssSettings: TssSettings = await this.bitgo\n        .get(this.bitgo.microservicesUrl('/api/v2/tss/settings'))\n        .result();\n      multisigTypeVersion =\n        tssSettings.coinSettings[this.baseCoin.getFamily()]?.walletCreationSettings?.multiSigTypeVersion;\n    }\n\n    switch (params.multisigType) {\n      case 'tss':\n        MpcUtils =\n          this.baseCoin.getMPCAlgorithm() === 'eddsa'\n            ? EDDSAUtils.default\n            : multisigTypeVersion === 'MPCv2'\n            ? ECDSAUtils.EcdsaMPCv2Utils\n            : ECDSAUtils.EcdsaUtils;\n        break;\n      default:\n        throw new Error('Unsupported multi-sig type');\n    }\n    const mpcUtils = new MpcUtils(this.bitgo, this.baseCoin);\n    return await mpcUtils.createKeychains({\n      passphrase: params.passphrase,\n      enterprise: params.enterprise,\n      originalPasscodeEncryptionCode: params.originalPasscodeEncryptionCode,\n      retrofit: params.retrofit,\n    });\n  }\n\n  async recreateMpc(params: RecreateMpcOptions): Promise<KeychainsTriplet> {\n    assert(params.coin, new Error('missing required param coin'));\n    assert(params.walletId, new Error('missing required param walletId'));\n    assert(params.otp, new Error('missing required param otp'));\n    assert(params.passphrase, new Error('missing required param passphrase'));\n\n    assert(\n      params.encryptedMaterial.encryptedWalletPassphrase,\n      new Error('missing required param encryptedWalletPassphrase')\n    );\n    assert(params.encryptedMaterial.encryptedUserKey, new Error('missing required param encryptedUserKey'));\n    assert(params.encryptedMaterial.encryptedBackupKey, new Error('missing required param encryptedBackupKey'));\n\n    await this.bitgo.post(this.bitgo.microservicesUrl('/api/v1/user/unlock')).send({ otp: params.otp }).result();\n    const { recoveryInfo } = await this.bitgo\n      .post(this.bitgo.microservicesUrl(`/api/v2/${params.coin}/wallet/${params.walletId}/passcoderecovery`))\n      .result();\n\n    if (!recoveryInfo || !('passcodeEncryptionCode' in recoveryInfo)) {\n      throw new Error('failed to get recovery info');\n    }\n\n    const decryptedWalletPassphrase = this.bitgo.decrypt({\n      input: params.encryptedMaterial.encryptedWalletPassphrase,\n      password: recoveryInfo.passcodeEncryptionCode,\n    });\n\n    const decryptedUserKey = this.bitgo.decrypt({\n      input: params.encryptedMaterial.encryptedUserKey,\n      password: decryptedWalletPassphrase,\n    });\n\n    const decryptedBackupKey = this.bitgo.decrypt({\n      input: params.encryptedMaterial.encryptedBackupKey,\n      password: decryptedWalletPassphrase,\n    });\n\n    return this.createMpc({\n      ...params,\n      multisigType: 'tss',\n      retrofit: {\n        decryptedUserKey,\n        decryptedBackupKey,\n        walletId: params.walletId,\n      },\n    });\n  }\n\n  /**\n   * It parses the JSON downloaded from the OVC for platform (BitGo),\n   * and creates a corresponding TSS BitGo key. It also returns the JSON that needs\n   * to be uploaded back to the OVCs containing the BitGo -> OVC shares.\n   * @param ovcOutputJson JSON format of the file downloaded from the OVC for platform\n   * @param enterprise Optional enterprise ID associated with the key\n   * @returns {BitGoKeyFromOvcShares}\n   */\n  async createTssBitGoKeyFromOvcShares(ovcOutputJson: unknown, enterprise?: string): Promise<BitGoKeyFromOvcShares> {\n    const decodedOvcOutput = decodeOrElse(OvcToBitGoJSON.name, OvcToBitGoJSON, ovcOutputJson, (errors) => {\n      throw new Error(`Error(s) parsing OVC JSON: ${errors}`);\n    });\n\n    if (decodedOvcOutput.state !== 1) {\n      throw new Error('State expected to be \"1\". Please complete the first two OVC operations');\n    }\n\n    // OVC-1 is responsible for the User key\n    const ovc1 = decodedOvcOutput.ovc[1];\n    // OVC-2 is responsible for the Backup key\n    const ovc2 = decodedOvcOutput.ovc[2];\n\n    const keyShares: ApiKeyShare[] = [\n      {\n        from: 'user',\n        to: 'bitgo',\n        publicShare: ovc1.ovcToBitgoShare.publicShare,\n        privateShare: ovc1.ovcToBitgoShare.privateShare,\n        privateShareProof: ovc1.ovcToBitgoShare.uSig.toString() ?? '',\n        vssProof: ovc1.ovcToBitgoShare.vssProof ?? '',\n      },\n      {\n        from: 'backup',\n        to: 'bitgo',\n        publicShare: ovc2.ovcToBitgoShare.publicShare,\n        privateShare: ovc2.ovcToBitgoShare.privateShare,\n        privateShareProof: ovc2.ovcToBitgoShare.uSig.toString() ?? '',\n        vssProof: ovc2.ovcToBitgoShare.vssProof ?? '',\n      },\n    ];\n\n    const key = await this.baseCoin.keychains().add({\n      source: 'bitgo',\n      keyShares,\n      keyType: 'tss',\n      userGPGPublicKey: ovc1.gpgPubKey,\n      backupGPGPublicKey: ovc2.gpgPubKey,\n      enterprise,\n    });\n    assert(key.keyShares);\n    assert(key.commonKeychain);\n    assert(key.walletHSMGPGPublicKeySigs);\n\n    const bitgoToUserShare = key.keyShares.find(\n      (value: { from: string; to: string }) => value.from === 'bitgo' && value.to === 'user'\n    );\n    assert(bitgoToUserShare);\n    assert(bitgoToUserShare.vssProof);\n    assert(bitgoToUserShare.paillierPublicKey);\n    const bitgoToBackupShare = key.keyShares.find(\n      (value: { from: string; to: string }) => value.from === 'bitgo' && value.to === 'backup'\n    );\n    assert(bitgoToBackupShare);\n    assert(bitgoToBackupShare.vssProof);\n    assert(bitgoToBackupShare.paillierPublicKey);\n\n    // Create JSON data with platform shares for OVC-1 and OVC-2\n    const bitgoToOvcOutput: BitGoToOvcJSON = {\n      wallet: {\n        ...decodedOvcOutput,\n        platform: {\n          commonKeychain: key.commonKeychain,\n          walletGpgPubKeySigs: key.walletHSMGPGPublicKeySigs,\n          ovc: {\n            // BitGo to User (OVC-1)\n            1: {\n              bitgoToOvcShare: {\n                i: 1,\n                j: 3,\n                publicShare: bitgoToUserShare.publicShare,\n                privateShare: bitgoToUserShare.privateShare,\n                paillierPublicKey: bitgoToUserShare.paillierPublicKey,\n                vssProof: bitgoToUserShare.vssProof,\n              },\n            },\n            // BitGo to Backup (OVC-2)\n            2: {\n              bitgoToOvcShare: {\n                i: 2,\n                j: 3,\n                publicShare: bitgoToBackupShare.publicShare,\n                privateShare: bitgoToBackupShare.privateShare,\n                paillierPublicKey: bitgoToBackupShare.paillierPublicKey,\n                vssProof: bitgoToBackupShare.vssProof,\n              },\n            },\n          },\n        },\n      },\n    };\n\n    // Mark it ready for next operation, should be 2\n    bitgoToOvcOutput.wallet.state += 1;\n\n    const output: BitGoKeyFromOvcShares = {\n      bitGoKeyId: key.id,\n      bitGoOutputJsonForOvc: bitgoToOvcOutput,\n    };\n\n    return decodeOrElse(BitGoKeyFromOvcShares.name, BitGoKeyFromOvcShares, output, (errors) => {\n      throw new Error(`Error producing the output: ${errors}`);\n    });\n  }\n\n  /**\n   * Create user keychain, encrypt the private key with the wallet passphrase and store it in BitGo.\n   * @param walletPassphrase\n   * @returns Keychain including the decrypted private key\n   */\n  async createUserKeychain(walletPassphrase: string): Promise<Keychain> {\n    const keychains = this.baseCoin.keychains();\n    const newKeychain = keychains.create();\n    const originalPasscodeEncryptionCode = generateRandomPassword(5);\n\n    const encryptedPrv = this.bitgo.encrypt({\n      password: walletPassphrase,\n      input: newKeychain.prv,\n    });\n\n    return {\n      ...(await keychains.add({\n        encryptedPrv,\n        originalPasscodeEncryptionCode,\n        pub: newKeychain.pub,\n        source: 'user',\n      })),\n      prv: newKeychain.prv,\n    };\n  }\n}\n"]}

Выполнить команду


Для локальной разработки. Не используйте в интернете!