UNPKG

bitgo

Version:
164 lines • 28.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); const sdk_core_1 = require("@bitgo/sdk-core"); const sdk_test_1 = require("@bitgo/sdk-test"); const src_1 = require("../../../../../src"); const openpgp = require("openpgp"); const nock = require("nock"); const assert = require("assert"); class TestEcdsaMpcv2Utils extends sdk_core_1.ECDSAUtils.EcdsaMPCv2Utils { async testPickBitgoPubGpgKeyForSigning(isMpcv2, reqId, enterpriseId) { return this.pickBitgoPubGpgKeyForSigning(isMpcv2, reqId, enterpriseId); } } class TestEddsaMpcv1Utils extends sdk_core_1.EddsaUtils { async testPickBitgoPubGpgKeyForSigning(isMpcv2, reqId, enterpriseId) { return this.pickBitgoPubGpgKeyForSigning(isMpcv2, reqId, enterpriseId); } } describe('TSS MPC Pick BitGo GPG Pub Key Utils:', function () { const walletId = '5b34252f1bf349930e34020a00000000'; const enterpriseId = '6449153a6f6bc20006d66771cdbe15d3'; const ecdsaCoinName = 'hteth'; const eddsaCoinName = 'tsol'; const ecdsaWalletData = { id: walletId, enterprise: enterpriseId, coin: ecdsaCoinName, coinSpecific: {}, multisigType: 'tss', keys: ['key1', 'key2', 'key3'], }; const eddsaWalletData = { id: walletId, enterprise: enterpriseId, coin: eddsaCoinName, coinSpecific: {}, multisigType: 'tss', keys: ['key1', 'key2', 'key3'], }; const envs = ['test', 'staging', 'prod']; const ecdsaMpcv2Utils = []; const eddsaMpcv1Utils = []; before(async function () { nock.cleanAll(); for (const env of envs) { const bitgoInstance = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env }); bitgoInstance.initializeTestVars(); let coinInstance = bitgoInstance.coin(ecdsaCoinName); ecdsaMpcv2Utils.push(new TestEcdsaMpcv2Utils(bitgoInstance, coinInstance, new sdk_core_1.Wallet(bitgoInstance, coinInstance, ecdsaWalletData))); coinInstance = bitgoInstance.coin(eddsaCoinName); eddsaMpcv1Utils.push(new TestEddsaMpcv1Utils(bitgoInstance, coinInstance, new sdk_core_1.Wallet(bitgoInstance, coinInstance, eddsaWalletData))); } }); beforeEach(async function () { for (const env of envs) { const bgUrl = sdk_core_1.common.Environments[env].uri; nock(bgUrl).get(`/api/v2/${ecdsaCoinName}/key/key3`).times(envs.length).reply(200, { hsmType: 'onprem' }); nock(bgUrl).get(`/api/v2/${eddsaCoinName}/key/key3`).times(envs.length).reply(200, { hsmType: 'nitro' }); } }); envs.forEach(async function (env, index) { it(`should pick correct Mpcv2 BitGo GPG Pub Key for ${env} env`, async function () { const bitgoGpgPubKey = await ecdsaMpcv2Utils[index].testPickBitgoPubGpgKeyForSigning(true); bitgoGpgPubKey .armor() .should.equal(sdk_core_1.BitgoMpcGpgPubKeys.bitgoMpcGpgPubKeys['mpcv2']['onprem'][env === 'staging' ? 'test' : env]); }); }); envs.forEach(async function (env, index) { it(`should pick correct Mpcv1 BitGo GPG Pub Key for ${env} env`, async function () { const bitgoGpgPubKey = await eddsaMpcv1Utils[index].testPickBitgoPubGpgKeyForSigning(false); bitgoGpgPubKey .armor() .should.equal(sdk_core_1.BitgoMpcGpgPubKeys.bitgoMpcGpgPubKeys['mpcv1']['nitro'][env === 'staging' ? 'test' : env]); }); }); it(`should pick BitGo GPG Pub Key based on enterprise flag for mock env`, async function () { const bgUrl = sdk_core_1.common.Environments['mock'].uri; const testBitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' }); const testCoin = testBitgo.coin(ecdsaCoinName); const bitgoGPGKey = await openpgp.generateKey({ userIDs: [ { name: 'bitgo', email: 'bitgo@test.com', }, ], }); nock(bgUrl) .get(`/api/v2/${ecdsaCoinName}/tss/pubkey`) .query({ enterpriseId }) .reply(200, { mpcv2PublicKey: bitgoGPGKey.publicKey }); const ecdsaMpcv2Util = new TestEcdsaMpcv2Utils(testBitgo, testCoin, new sdk_core_1.Wallet(testBitgo, testCoin, ecdsaWalletData)); const bitgoGpgPubKey = await ecdsaMpcv2Util.testPickBitgoPubGpgKeyForSigning(true, undefined, enterpriseId); bitgoGpgPubKey.armor().should.equal(bitgoGPGKey.publicKey); }); it(`should pick BitGo GPG Pub Key based on constants api for mock env if enterprise flag based fetch fails`, async function () { nock.cleanAll(); const bgUrl = sdk_core_1.common.Environments['mock'].uri; const testBitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' }); const testCoin = testBitgo.coin(ecdsaCoinName); const bitgoGPGKey = await openpgp.generateKey({ userIDs: [ { name: 'bitgo', email: 'bitgo@test.com', }, ], }); const constants = { mpc: { bitgoMPCv2PublicKey: bitgoGPGKey.publicKey, bitgoPublicKey: bitgoGPGKey.publicKey, }, }; nock(bgUrl).get('/api/v1/client/constants').times(2).reply(200, { ttl: 3600, constants }); const ecdsaMpcv2Util = new TestEcdsaMpcv2Utils(testBitgo, testCoin, new sdk_core_1.Wallet(testBitgo, testCoin, ecdsaWalletData)); const bitgoGpgPubKey = await ecdsaMpcv2Util.testPickBitgoPubGpgKeyForSigning(true, undefined, enterpriseId); bitgoGpgPubKey.armor().should.equal(bitgoGPGKey.publicKey); }); it(`should throw an error if config is not available in one of test, staging, or prod`, async function () { nock.cleanAll(); const testBitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'test' }); const testCoin = testBitgo.coin(ecdsaCoinName); const ecdsaMpcv2Util = new TestEcdsaMpcv2Utils(testBitgo, testCoin, new sdk_core_1.Wallet(testBitgo, testCoin, ecdsaWalletData)); await assert.rejects(async () => await ecdsaMpcv2Util.testPickBitgoPubGpgKeyForSigning(true)); }); it('should select the correct GPG key for verifyWalletSignatures with hardcoded options', async function () { // Test environment setup const env = 'test'; const bitgoInstance = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env }); bitgoInstance.initializeTestVars(); const coinInstance = bitgoInstance.coin(eddsaCoinName); const eddsaMpcv1Util = new TestEddsaMpcv1Utils(bitgoInstance, coinInstance, new sdk_core_1.Wallet(bitgoInstance, coinInstance, eddsaWalletData)); // Get the key that would be selected when using hardcoded BitGo keys // mpcv1 and nitro are passed as options const gpgKey = await eddsaMpcv1Util.testPickBitgoPubGpgKeyForSigning(false, undefined, undefined); // Mock implementation of verifyWalletSignatures to capture the bitgoGpgKey that gets used let capturedKey; eddsaMpcv1Util.verifyWalletSignatures = async function (userGpgPub, backupGpgPub, bitgoKeychain, decryptedShare, verifierIndex, useHardcodedBitGoKeys) { // Save the key that would be used when specifying hardcoded options if (useHardcodedBitGoKeys) { const hardcodedKey = await openpgp.readKey({ armoredKey: sdk_core_1.BitgoMpcGpgPubKeys.bitgoMpcGpgPubKeys['mpcv1']['nitro']['test'], }); capturedKey = hardcodedKey.armor(); } // Not actually verifying in this test return; }; // Call with hardcoded key options await eddsaMpcv1Util.verifyWalletSignatures('mock-user-key', 'mock-backup-key', { commonKeychain: 'mock-keychain', walletHSMGPGPublicKeySigs: '', id: '', type: 'tss', }, 'decrypted-share', 1, { env: 'test', pubKeyType: 'nitro' }); // Verify the hardcoded key matches what we expect capturedKey.should.equal(sdk_core_1.BitgoMpcGpgPubKeys.bitgoMpcGpgPubKeys['mpcv1']['nitro']['test']); // Also verify it's the same as what's returned by testPickBitgoPubGpgKeyForSigning gpgKey.armor().should.equal(capturedKey); }); }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bitgoMpcGpgPubKeys.js","sourceRoot":"","sources":["../../../../../../test/v2/unit/internal/tssUtils/bitgoMpcGpgPubKeys.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BitgoMpcGpgPubKeys,\n  common,\n  ECDSAUtils,\n  EddsaUtils,\n  EnvironmentName,\n  IRequestTracer,\n  Wallet,\n} from '@bitgo/sdk-core';\nimport { TestBitGo } from '@bitgo/sdk-test';\n\nimport { BitGo } from '../../../../../src';\nimport * as openpgp from 'openpgp';\nimport nock = require('nock');\nimport assert = require('assert');\n\nclass TestEcdsaMpcv2Utils extends ECDSAUtils.EcdsaMPCv2Utils {\n  public async testPickBitgoPubGpgKeyForSigning(\n    isMpcv2: boolean,\n    reqId?: IRequestTracer,\n    enterpriseId?: string\n  ): Promise<openpgp.Key> {\n    return this.pickBitgoPubGpgKeyForSigning(isMpcv2, reqId, enterpriseId);\n  }\n}\n\nclass TestEddsaMpcv1Utils extends EddsaUtils {\n  public async testPickBitgoPubGpgKeyForSigning(\n    isMpcv2: boolean,\n    reqId?: IRequestTracer,\n    enterpriseId?: string\n  ): Promise<openpgp.Key> {\n    return this.pickBitgoPubGpgKeyForSigning(isMpcv2, reqId, enterpriseId);\n  }\n}\n\ndescribe('TSS MPC Pick BitGo GPG Pub Key Utils:', function () {\n  const walletId = '5b34252f1bf349930e34020a00000000';\n  const enterpriseId = '6449153a6f6bc20006d66771cdbe15d3';\n  const ecdsaCoinName = 'hteth';\n  const eddsaCoinName = 'tsol';\n  const ecdsaWalletData = {\n    id: walletId,\n    enterprise: enterpriseId,\n    coin: ecdsaCoinName,\n    coinSpecific: {},\n    multisigType: 'tss',\n    keys: ['key1', 'key2', 'key3'],\n  };\n  const eddsaWalletData = {\n    id: walletId,\n    enterprise: enterpriseId,\n    coin: eddsaCoinName,\n    coinSpecific: {},\n    multisigType: 'tss',\n    keys: ['key1', 'key2', 'key3'],\n  };\n  const envs: EnvironmentName[] = ['test', 'staging', 'prod'];\n  const ecdsaMpcv2Utils: TestEcdsaMpcv2Utils[] = [];\n  const eddsaMpcv1Utils: TestEddsaMpcv1Utils[] = [];\n\n  before(async function () {\n    nock.cleanAll();\n    for (const env of envs) {\n      const bitgoInstance = TestBitGo.decorate(BitGo, { env });\n      bitgoInstance.initializeTestVars();\n      let coinInstance = bitgoInstance.coin(ecdsaCoinName);\n      ecdsaMpcv2Utils.push(\n        new TestEcdsaMpcv2Utils(bitgoInstance, coinInstance, new Wallet(bitgoInstance, coinInstance, ecdsaWalletData))\n      );\n      coinInstance = bitgoInstance.coin(eddsaCoinName);\n      eddsaMpcv1Utils.push(\n        new TestEddsaMpcv1Utils(bitgoInstance, coinInstance, new Wallet(bitgoInstance, coinInstance, eddsaWalletData))\n      );\n    }\n  });\n\n  beforeEach(async function () {\n    for (const env of envs) {\n      const bgUrl = common.Environments[env].uri;\n      nock(bgUrl).get(`/api/v2/${ecdsaCoinName}/key/key3`).times(envs.length).reply(200, { hsmType: 'onprem' });\n      nock(bgUrl).get(`/api/v2/${eddsaCoinName}/key/key3`).times(envs.length).reply(200, { hsmType: 'nitro' });\n    }\n  });\n\n  envs.forEach(async function (env, index) {\n    it(`should pick correct Mpcv2 BitGo GPG Pub Key for ${env} env`, async function () {\n      const bitgoGpgPubKey = await ecdsaMpcv2Utils[index].testPickBitgoPubGpgKeyForSigning(true);\n      bitgoGpgPubKey\n        .armor()\n        .should.equal(BitgoMpcGpgPubKeys.bitgoMpcGpgPubKeys['mpcv2']['onprem'][env === 'staging' ? 'test' : env]);\n    });\n  });\n\n  envs.forEach(async function (env, index) {\n    it(`should pick correct Mpcv1 BitGo GPG Pub Key for ${env} env`, async function () {\n      const bitgoGpgPubKey = await eddsaMpcv1Utils[index].testPickBitgoPubGpgKeyForSigning(false);\n      bitgoGpgPubKey\n        .armor()\n        .should.equal(BitgoMpcGpgPubKeys.bitgoMpcGpgPubKeys['mpcv1']['nitro'][env === 'staging' ? 'test' : env]);\n    });\n  });\n\n  it(`should pick BitGo GPG Pub Key based on enterprise flag for mock env`, async function () {\n    const bgUrl = common.Environments['mock'].uri;\n    const testBitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    const testCoin = testBitgo.coin(ecdsaCoinName);\n    const bitgoGPGKey = await openpgp.generateKey({\n      userIDs: [\n        {\n          name: 'bitgo',\n          email: 'bitgo@test.com',\n        },\n      ],\n    });\n    nock(bgUrl)\n      .get(`/api/v2/${ecdsaCoinName}/tss/pubkey`)\n      .query({ enterpriseId })\n      .reply(200, { mpcv2PublicKey: bitgoGPGKey.publicKey });\n    const ecdsaMpcv2Util = new TestEcdsaMpcv2Utils(\n      testBitgo,\n      testCoin,\n      new Wallet(testBitgo, testCoin, ecdsaWalletData)\n    );\n    const bitgoGpgPubKey = await ecdsaMpcv2Util.testPickBitgoPubGpgKeyForSigning(true, undefined, enterpriseId);\n    bitgoGpgPubKey.armor().should.equal(bitgoGPGKey.publicKey);\n  });\n\n  it(`should pick BitGo GPG Pub Key based on constants api for mock env if enterprise flag based fetch fails`, async function () {\n    nock.cleanAll();\n    const bgUrl = common.Environments['mock'].uri;\n    const testBitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    const testCoin = testBitgo.coin(ecdsaCoinName);\n    const bitgoGPGKey = await openpgp.generateKey({\n      userIDs: [\n        {\n          name: 'bitgo',\n          email: 'bitgo@test.com',\n        },\n      ],\n    });\n    const constants = {\n      mpc: {\n        bitgoMPCv2PublicKey: bitgoGPGKey.publicKey,\n        bitgoPublicKey: bitgoGPGKey.publicKey,\n      },\n    };\n    nock(bgUrl).get('/api/v1/client/constants').times(2).reply(200, { ttl: 3600, constants });\n    const ecdsaMpcv2Util = new TestEcdsaMpcv2Utils(\n      testBitgo,\n      testCoin,\n      new Wallet(testBitgo, testCoin, ecdsaWalletData)\n    );\n    const bitgoGpgPubKey = await ecdsaMpcv2Util.testPickBitgoPubGpgKeyForSigning(true, undefined, enterpriseId);\n    bitgoGpgPubKey.armor().should.equal(bitgoGPGKey.publicKey);\n  });\n\n  it(`should throw an error if config is not available in one of test, staging, or prod`, async function () {\n    nock.cleanAll();\n    const testBitgo = TestBitGo.decorate(BitGo, { env: 'test' });\n    const testCoin = testBitgo.coin(ecdsaCoinName);\n    const ecdsaMpcv2Util = new TestEcdsaMpcv2Utils(\n      testBitgo,\n      testCoin,\n      new Wallet(testBitgo, testCoin, ecdsaWalletData)\n    );\n    await assert.rejects(async () => await ecdsaMpcv2Util.testPickBitgoPubGpgKeyForSigning(true));\n  });\n\n  it('should select the correct GPG key for verifyWalletSignatures with hardcoded options', async function () {\n    // Test environment setup\n    const env = 'test';\n    const bitgoInstance = TestBitGo.decorate(BitGo, { env });\n    bitgoInstance.initializeTestVars();\n    const coinInstance = bitgoInstance.coin(eddsaCoinName);\n    const eddsaMpcv1Util = new TestEddsaMpcv1Utils(\n      bitgoInstance,\n      coinInstance,\n      new Wallet(bitgoInstance, coinInstance, eddsaWalletData)\n    );\n\n    // Get the key that would be selected when using hardcoded BitGo keys\n    // mpcv1 and nitro are passed as options\n    const gpgKey = await eddsaMpcv1Util.testPickBitgoPubGpgKeyForSigning(false, undefined, undefined);\n\n    // Mock implementation of verifyWalletSignatures to capture the bitgoGpgKey that gets used\n    let capturedKey;\n    eddsaMpcv1Util.verifyWalletSignatures = async function (\n      userGpgPub,\n      backupGpgPub,\n      bitgoKeychain,\n      decryptedShare,\n      verifierIndex,\n      useHardcodedBitGoKeys\n    ) {\n      // Save the key that would be used when specifying hardcoded options\n      if (useHardcodedBitGoKeys) {\n        const hardcodedKey = await openpgp.readKey({\n          armoredKey: BitgoMpcGpgPubKeys.bitgoMpcGpgPubKeys['mpcv1']['nitro']['test'],\n        });\n        capturedKey = hardcodedKey.armor();\n      }\n      // Not actually verifying in this test\n      return;\n    };\n\n    // Call with hardcoded key options\n    await eddsaMpcv1Util.verifyWalletSignatures(\n      'mock-user-key',\n      'mock-backup-key',\n      {\n        commonKeychain: 'mock-keychain',\n        walletHSMGPGPublicKeySigs: '',\n        id: '',\n        type: 'tss',\n      },\n      'decrypted-share',\n      1,\n      { env: 'test', pubKeyType: 'nitro' }\n    );\n\n    // Verify the hardcoded key matches what we expect\n    capturedKey.should.equal(BitgoMpcGpgPubKeys.bitgoMpcGpgPubKeys['mpcv1']['nitro']['test']);\n    // Also verify it's the same as what's returned by testPickBitgoPubGpgKeyForSigning\n    gpgKey.armor().should.equal(capturedKey);\n  });\n});\n"]}