UNPKG

@c4tplatform/caminojs

Version:
226 lines 29 kB
"use strict"; /** * @packageDocumentation * @module Common-MultisigKeyChain */ var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.MultisigKeyChain = exports.MultisigKeyPair = exports.SignatureError = void 0; const buffer_1 = require("buffer/"); const _1 = require("."); const utils_1 = require("../utils"); const bintools_1 = __importDefault(require("../utils/bintools")); const keychain_1 = require("./keychain"); const secp256k1_1 = require("./secp256k1"); class SignatureError extends Error { } exports.SignatureError = SignatureError; const NotImplemented = new Error("not implemented in MultiSig KeyPair"); const TooManySignatures = new SignatureError("too many signatures"); const serialization = utils_1.Serialization.getInstance(); const bintools = bintools_1.default.getInstance(); const MaxSignatures = 256; /** * Class for representing a generic multi signature key. */ class MultisigKeyPair extends keychain_1.StandardKeyPair { generateKey() { throw NotImplemented; } importKey(_) { return false; } sign(_) { return this.privk; } recover(msg, sig) { throw NotImplemented; } verify(msg, sig) { throw NotImplemented; } getAddress() { return this.pubk; } getAddressString() { const addr = secp256k1_1.SECP256k1KeyPair.addressFromPublicKey(this.pubk); const type = "bech32"; return serialization.bufferToType(addr, type, this.keyChain.getHRP(), this.keyChain.getChainID()); } create(...args) { if (args.length == 3) { return new MultisigKeyPair(args[0], args[1], args[2]); } return new MultisigKeyPair(this.keyChain, this.pubk, this.privk); } clone() { return new MultisigKeyPair(this.keyChain, this.pubk, this.privk); } getPrivateKeyString() { return bintools.cb58Encode(this.privk); } getPublicKeyString() { return bintools.cb58Encode(this.pubk); } constructor(keyChain, address, signature) { super(); this.keyChain = keyChain; this.pubk = buffer_1.Buffer.from(address); this.privk = buffer_1.Buffer.from(signature); } } exports.MultisigKeyPair = MultisigKeyPair; /** * Class for representing a multisig keyChain in Camino. * * @typeparam MultisigKeyChain Class extending [[StandardKeyChain]] */ class MultisigKeyChain extends keychain_1.StandardKeyChain { getHRP() { return this.hrp; } getChainID() { return this.chainID; } create(...args) { if (args.length == 4) { return new MultisigKeyChain(args[0], args[1], args[2], args[4]); } return new MultisigKeyChain(this.hrp, this.chainID, this.signedBytes, this.credTypeID); } clone() { const newkc = new MultisigKeyChain(this.hrp, this.chainID, this.signedBytes, this.credTypeID); for (let k in this.keys) { newkc.addKey(this.keys[`${k}`].clone()); } newkc.txOwners = new Array(this.txOwners.length); this.txOwners.forEach((txo, index) => newkc.txOwners[index].fromBuffer(txo.toBuffer())); return newkc; } union(kc) { if (kc.chainID !== this.chainID || kc.hrp != this.hrp || kc.signedBytes.compare(this.signedBytes) != 0) { throw new Error("keychains do not match"); } const newkc = kc.clone(); Object.assign(newkc.keys, kc.keys); return newkc; } // Visit every txOutputOwner and try to verify with keys. // Traverse into msig aliases. Throw if one cannot be fulfilled buildSignatureIndices() { this.sigIdxs = []; for (const o of this.txOwners) this._traverseOwner(o); } getCredentials() { const result = []; for (const sigSet of this.sigIdxs) { const cred = new _1.SECPMultisigCredential(this.credTypeID); for (const sigIdx of sigSet) { cred.addSSignatureIndex(sigIdx); const sig = new _1.Signature(); sig.fromBuffer(this.getKey(sigIdx.getSource()).getPrivateKey()); cred.addSignature(sig); } result.push(cred); } return result; } _traverseOwner(owner) { var addrVisited = 0; var addrVerified = 0; const cycleCheck = new Set(); const stack = [ { index: 0, verified: 0, addrVerifiedTotal: 0, parentVerified: false, owners: owner } ]; const sigIdxs = []; const helper = buffer_1.Buffer.alloc(4); Stack: while (stack.length > 0) { // get head var currentStack = stack[stack.length - 1]; while (currentStack.index < currentStack.owners.getAddressesLength()) { // get the next address to check const addr = currentStack.owners.getAddress(currentStack.index); const addrStr = addr.toString("hex"); currentStack.index++; // Is it a multi-sig address ? const alias = this.msigAliases.get(addrStr); if (alias !== undefined) { if (stack.length > MaxSignatures) { throw TooManySignatures; } if (cycleCheck.has(addrStr)) { throw new Error("Cyclic multisig alias"); } cycleCheck.add(addrStr); stack.push({ index: 0, verified: 0, addrVerifiedTotal: addrVerified, parentVerified: currentStack.parentVerified || currentStack.verified >= currentStack.owners.getThreshold(), owners: alias }); continue Stack; } else { if (!currentStack.parentVerified && currentStack.verified < currentStack.owners.getThreshold()) { if (this.hasKey(addr)) { if (addrVisited > MaxSignatures) { throw TooManySignatures; } const sigIdx = new _1.SigIdx(); sigIdx.setSource(addr); helper.writeUIntBE(addrVisited, 0, 4); sigIdx.fromBuffer(helper); sigIdxs.push(sigIdx); currentStack.verified++; addrVerified++; } } addrVisited++; } } // remove head stack.pop(); // verify current level if (currentStack.verified < currentStack.owners.getThreshold()) { if (stack.length == 0) { throw new SignatureError("Not enough signatures"); } // We recover to previous state addrVerified = currentStack.addrVerifiedTotal; sigIdxs.splice(addrVerified); } else if (stack.length > 0) { currentStack = stack[stack.length - 1]; if (currentStack.verified < currentStack.owners.getThreshold()) { // apply child verification currentStack.verified++; } } } this.sigIdxs.push(sigIdxs); } constructor(hrp, chainID, signedBytes, credTypeID, txOwners, msigAliases) { super(); this.hrp = hrp; this.chainID = chainID; this.signedBytes = buffer_1.Buffer.from(signedBytes); (this.credTypeID = credTypeID), (this.txOwners = txOwners !== null && txOwners !== void 0 ? txOwners : []); this.msigAliases = msigAliases !== null && msigAliases !== void 0 ? msigAliases : new Map(); } } exports.MultisigKeyChain = MultisigKeyChain; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multisigkeychain.js","sourceRoot":"","sources":["../../src/common/multisigkeychain.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @packageDocumentation\n * @module Common-MultisigKeyChain\n */\n\nimport { Buffer } from \"buffer/\"\nimport {\n  Credential,\n  OutputOwners,\n  SECPMultisigCredential,\n  SigIdx,\n  Signature\n} from \".\"\n\nimport { Serialization, SerializedType } from \"../utils\"\nimport BinTools from \"../utils/bintools\"\nimport { StandardKeyChain, StandardKeyPair } from \"./keychain\"\nimport { SECP256k1KeyPair } from \"./secp256k1\"\n\nexport class SignatureError extends Error {}\nconst NotImplemented = new Error(\"not implemented in MultiSig KeyPair\")\nconst TooManySignatures = new SignatureError(\"too many signatures\")\n\nconst serialization: Serialization = Serialization.getInstance()\nconst bintools: BinTools = BinTools.getInstance()\nconst MaxSignatures = 256\n\n/**\n * Class for representing a generic multi signature key.\n */\nexport class MultisigKeyPair extends StandardKeyPair {\n  // The keychain required for address generation\n  protected keyChain: MultisigKeyChain\n\n  generateKey() {\n    throw NotImplemented\n  }\n\n  importKey(_: Buffer): boolean {\n    return false\n  }\n\n  sign(_: Buffer): Buffer {\n    return this.privk\n  }\n\n  recover(msg: Buffer, sig: Buffer): Buffer {\n    throw NotImplemented\n  }\n\n  verify(msg: Buffer, sig: Buffer): boolean {\n    throw NotImplemented\n  }\n\n  getAddress(): Buffer {\n    return this.pubk\n  }\n\n  getAddressString(): string {\n    const addr: Buffer = SECP256k1KeyPair.addressFromPublicKey(this.pubk)\n    const type: SerializedType = \"bech32\"\n    return serialization.bufferToType(\n      addr,\n      type,\n      this.keyChain.getHRP(),\n      this.keyChain.getChainID()\n    )\n  }\n\n  create(...args: any[]): this {\n    if (args.length == 3) {\n      return new MultisigKeyPair(args[0], args[1], args[2]) as this\n    }\n    return new MultisigKeyPair(this.keyChain, this.pubk, this.privk) as this\n  }\n\n  clone(): this {\n    return new MultisigKeyPair(this.keyChain, this.pubk, this.privk) as this\n  }\n\n  getPrivateKeyString(): string {\n    return bintools.cb58Encode(this.privk)\n  }\n\n  getPublicKeyString(): string {\n    return bintools.cb58Encode(this.pubk)\n  }\n\n  constructor(keyChain: MultisigKeyChain, address: Buffer, signature: Buffer) {\n    super()\n    this.keyChain = keyChain\n    this.pubk = Buffer.from(address)\n    this.privk = Buffer.from(signature)\n  }\n}\n\n/**\n * Class for representing a multisig keyChain in Camino.\n *\n * @typeparam MultisigKeyChain Class extending [[StandardKeyChain]]\n */\nexport class MultisigKeyChain extends StandardKeyChain<MultisigKeyPair> {\n  // The HRP required for address generation\n  protected hrp: string\n  // The chain ID required for address generation\n  protected chainID: string\n  // The bytes which are signed by this txID\n  protected signedBytes: Buffer\n  // the OutputOwners of all inputs and Auths inside the message\n  protected txOwners: OutputOwners[]\n  // the multisig aliases which take part in evaluation\n  protected msigAliases: Map<string, OutputOwners>\n  // Credentials for all the txOwners\n  protected sigIdxs: SigIdx[][]\n  // The CredentialID used for SECPMultisigCredential\n  protected credTypeID: number\n\n  getHRP(): string {\n    return this.hrp\n  }\n\n  getChainID(): string {\n    return this.chainID\n  }\n\n  create(...args: any[]): this {\n    if (args.length == 4) {\n      return new MultisigKeyChain(args[0], args[1], args[2], args[4]) as this\n    }\n    return new MultisigKeyChain(\n      this.hrp,\n      this.chainID,\n      this.signedBytes,\n      this.credTypeID\n    ) as this\n  }\n\n  clone(): this {\n    const newkc = new MultisigKeyChain(\n      this.hrp,\n      this.chainID,\n      this.signedBytes,\n      this.credTypeID\n    ) as this\n    for (let k in this.keys) {\n      newkc.addKey(this.keys[`${k}`].clone())\n    }\n    newkc.txOwners = new Array(this.txOwners.length)\n    this.txOwners.forEach((txo, index) =>\n      newkc.txOwners[index].fromBuffer(txo.toBuffer())\n    )\n    return newkc\n  }\n\n  union(kc: this): this {\n    if (\n      kc.chainID !== this.chainID ||\n      kc.hrp != this.hrp ||\n      kc.signedBytes.compare(this.signedBytes) != 0\n    ) {\n      throw new Error(\"keychains do not match\")\n    }\n    const newkc = kc.clone()\n    Object.assign(newkc.keys, kc.keys)\n\n    return newkc\n  }\n\n  // Visit every txOutputOwner and try to verify with keys.\n  // Traverse into msig aliases. Throw if one cannot be fulfilled\n  buildSignatureIndices() {\n    this.sigIdxs = []\n    for (const o of this.txOwners) this._traverseOwner(o)\n  }\n\n  getCredentials(): Credential[] {\n    const result: SECPMultisigCredential[] = []\n    for (const sigSet of this.sigIdxs) {\n      const cred = new SECPMultisigCredential(this.credTypeID)\n      for (const sigIdx of sigSet) {\n        cred.addSSignatureIndex(sigIdx)\n        const sig = new Signature()\n        sig.fromBuffer(this.getKey(sigIdx.getSource()).getPrivateKey())\n        cred.addSignature(sig)\n      }\n      result.push(cred)\n    }\n    return result\n  }\n\n  protected _traverseOwner(owner: OutputOwners): void {\n    var addrVisited = 0\n    var addrVerified = 0\n\n    type stackItem = {\n      index: number\n      verified: number\n      addrVerifiedTotal: number\n      parentVerified: boolean\n      owners: OutputOwners\n    }\n\n    const cycleCheck = new Set<string>()\n    const stack: stackItem[] = [\n      {\n        index: 0,\n        verified: 0,\n        addrVerifiedTotal: 0,\n        parentVerified: false,\n        owners: owner\n      }\n    ]\n\n    const sigIdxs: SigIdx[] = []\n    const helper = Buffer.alloc(4)\n\n    Stack: while (stack.length > 0) {\n      // get head\n      var currentStack = stack[stack.length - 1]\n      while (currentStack.index < currentStack.owners.getAddressesLength()) {\n        // get the next address to check\n        const addr = currentStack.owners.getAddress(currentStack.index)\n        const addrStr = addr.toString(\"hex\")\n        currentStack.index++\n        // Is it a multi-sig address ?\n        const alias = this.msigAliases.get(addrStr)\n        if (alias !== undefined) {\n          if (stack.length > MaxSignatures) {\n            throw TooManySignatures\n          }\n          if (cycleCheck.has(addrStr)) {\n            throw new Error(\"Cyclic multisig alias\")\n          }\n          cycleCheck.add(addrStr)\n          stack.push({\n            index: 0,\n            verified: 0,\n            addrVerifiedTotal: addrVerified,\n            parentVerified:\n              currentStack.parentVerified ||\n              currentStack.verified >= currentStack.owners.getThreshold(),\n            owners: alias\n          })\n          continue Stack\n        } else {\n          if (\n            !currentStack.parentVerified &&\n            currentStack.verified < currentStack.owners.getThreshold()\n          ) {\n            if (this.hasKey(addr)) {\n              if (addrVisited > MaxSignatures) {\n                throw TooManySignatures\n              }\n\n              const sigIdx = new SigIdx()\n              sigIdx.setSource(addr)\n              helper.writeUIntBE(addrVisited, 0, 4)\n              sigIdx.fromBuffer(helper)\n              sigIdxs.push(sigIdx)\n\n              currentStack.verified++\n              addrVerified++\n            }\n          }\n          addrVisited++\n        }\n      }\n\n      // remove head\n      stack.pop()\n      // verify current level\n      if (currentStack.verified < currentStack.owners.getThreshold()) {\n        if (stack.length == 0) {\n          throw new SignatureError(\"Not enough signatures\")\n        }\n        // We recover to previous state\n        addrVerified = currentStack.addrVerifiedTotal\n        sigIdxs.splice(addrVerified)\n      } else if (stack.length > 0) {\n        currentStack = stack[stack.length - 1]\n        if (currentStack.verified < currentStack.owners.getThreshold()) {\n          // apply child verification\n          currentStack.verified++\n        }\n      }\n    }\n\n    this.sigIdxs.push(sigIdxs)\n  }\n\n  constructor(\n    hrp: string,\n    chainID: string,\n    signedBytes: Buffer,\n    credTypeID: number,\n    txOwners?: OutputOwners[],\n    msigAliases?: Map<string, OutputOwners>\n  ) {\n    super()\n    this.hrp = hrp\n    this.chainID = chainID\n    this.signedBytes = Buffer.from(signedBytes)\n    ;(this.credTypeID = credTypeID), (this.txOwners = txOwners ?? [])\n    this.msigAliases = msigAliases ?? new Map<string, OutputOwners>()\n  }\n}\n"]}