module Bitcoin module PSBT # Class for PSBTs which contains per output information class Output attr_accessor :redeem_script attr_accessor :witness_script attr_accessor :hd_key_paths attr_accessor :proprietaries attr_accessor :tap_internal_key attr_accessor :tap_tree attr_accessor :tap_bip32_derivations attr_accessor :unknowns def initialize @hd_key_paths = {} @proprietaries = [] @tap_bip32_derivations = {} @unknowns = {} end # parse PSBT output data form buffer. # @param [StringIO] buf psbt buffer. # @return [Bitcoin::PSBTOutput] psbt output. def self.parse_from_buf(buf) output = self.new found_sep = false until buf.eof? key_len = Bitcoin.unpack_var_int_from_io(buf) if key_len == 0 found_sep = true break end key_type = buf.read(1).unpack1('C') key = buf.read(key_len - 1) value = buf.read(Bitcoin.unpack_var_int_from_io(buf)) case key_type when PSBT_OUT_TYPES[:redeem_script] raise ArgumentError, 'Invalid output redeemScript typed key.' unless key_len == 1 raise ArgumentError, 'Duplicate Key, output redeemScript already provided' if output.redeem_script output.redeem_script = value when PSBT_OUT_TYPES[:witness_script] raise ArgumentError, 'Invalid output witnessScript typed key.' unless key_len == 1 raise ArgumentError, 'Duplicate Key, output witnessScript already provided' if output.witness_script output.witness_script = value when PSBT_OUT_TYPES[:bip32_derivation] raise ArgumentError, 'Duplicate Key, pubkey derivation path already provided' if output.hd_key_paths[key.bth] output.hd_key_paths[key.bth] = Bitcoin::PSBT::HDKeyPath.new(key, Bitcoin::PSBT::KeyOriginInfo.parse_from_payload(value)) when PSBT_OUT_TYPES[:proprietary] raise ArgumentError, 'Duplicate Key, key for proprietary value already provided.' if output.proprietaries.any?{|p| p.key == key} output.proprietaries << Proprietary.new(key, value) when PSBT_OUT_TYPES[:tap_internal_key] raise ArgumentError, 'Invalid output tap internal key typed key.' unless key_len == 1 raise ArgumentError, 'Invalid x-only public key size for the type tap internal key' unless value.bytesize == X_ONLY_PUBKEY_SIZE output.tap_internal_key = value.bth when PSBT_OUT_TYPES[:tap_tree] raise ArgumentError, 'Invalid output tap tree typed key.' unless key_len == 1 output.tap_tree = value.bth # TODO implement builder. when PSBT_OUT_TYPES[:tap_bip32_derivation] raise ArgumentError, 'Duplicate Key, key for tap bip32 derivation value already provided.' if output.tap_bip32_derivations[key.bth] raise ArgumentError, 'Size of key was not the expected size for the type tap bip32 derivation' unless key.bytesize == X_ONLY_PUBKEY_SIZE output.tap_bip32_derivations[key.bth] = value.bth else unknown_key = ([key_type].pack('C') + key).bth raise ArgumentError, 'Duplicate Key, key for unknown value already provided' if output.unknowns[unknown_key] output.unknowns[unknown_key] = value end end raise ArgumentError, 'Separator is missing at the end of an output map.' unless found_sep output end def to_payload payload = '' payload << PSBT.serialize_to_vector(PSBT_OUT_TYPES[:redeem_script], value: redeem_script) if redeem_script payload << PSBT.serialize_to_vector(PSBT_OUT_TYPES[:witness_script], value: witness_script) if witness_script payload << hd_key_paths.values.map{|v|v.to_payload(PSBT_OUT_TYPES[:bip32_derivation])}.join payload << proprietaries.map(&:to_payload).join payload << PSBT.serialize_to_vector(PSBT_OUT_TYPES[:tap_internal_key], value: tap_internal_key.htb) if tap_internal_key payload << PSBT.serialize_to_vector(PSBT_OUT_TYPES[:tap_tree], value: tap_tree.htb) if tap_tree payload << tap_bip32_derivations.map{|k, v|PSBT.serialize_to_vector(PSBT_OUT_TYPES[:tap_bip32_derivation], key: k.htb, value: v.htb)}.join payload << unknowns.map {|k,v|Bitcoin.pack_var_int(k.htb.bytesize) << k.htb << Bitcoin.pack_var_int(v.bytesize) << v}.join payload << PSBT_SEPARATOR.itb payload end # merge two PSBT outputs to create one PSBT. # @param [Bitcoin::PSBT::Output] psbo PSBT output to be combined which must have same property in PSBT Output. # @return [Bitcoin::PSBT::Output] combined object. def merge(psbo) raise ArgumentError, 'The argument psbt must be an instance of Bitcoin::PSBT::Output.' unless psbo.is_a?(Bitcoin::PSBT::Output) raise ArgumentError, 'The Partially Signed Output\'s redeem_script are different.' unless redeem_script == psbo.redeem_script raise ArgumentError, 'The Partially Signed Output\'s witness_script are different.' unless witness_script == psbo.witness_script combined = Bitcoin::PSBT::Output.new combined.redeem_script = redeem_script combined.witness_script = witness_script combined.unknowns = Hash[unknowns.merge(psbo.unknowns).sort] combined.hd_key_paths = hd_key_paths.merge(psbo.hd_key_paths) combined end end end end