Sha256: d1b9e12b1922f96ca060f68d3c6b8d42b96a2053cb87e739395199025dc48e91

Contents?: true

Size: 1.88 KB

Versions: 2

Compression:

Stored size: 1.88 KB

Contents

module Bitcoin
  module Protocol
    # https://en.bitcoin.it/wiki/Protocol_documentation#Merkle_Trees
    class PartialMerkleTree
      Node = Struct.new(:value, :left, :right, :width_idx)

      BIT_MASK = [0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80].freeze

      def initialize(total_txs, hashes, flags)
        @total_txs = total_txs
        @flags = flags
        @hashes = hashes.map(&:reverse_hth)
        @visit_idx = 0
      end

      def tx_hashes
        @leaves.reject { |n| n.value.nil? }.map(&:value)
      end

      def build_tree
        lay = @leaves = Array.new(@total_txs) { Node.new(nil, nil, nil) }
        while lay.size > 1
          lay = lay.each_slice(2).map do |left, right|
            Node.new(nil, left, right)
          end
        end
        lay[0]
      end

      def current_flag
        (@flags[@visit_idx / 8].ord & BIT_MASK[@visit_idx % 8]).zero?
      end

      def root
        @root ||= build_tree
      end

      def set_value(node = root) # rubocop:disable Naming/AccessorMethodName
        warn '[DEPRECATION] `PartialMerkleTree.set_value` is deprecated. ' \
          'Use `assign_value` instead.'
        assign_value(node)
      end

      def assign_value(node = root)
        if current_flag || (node.left.nil? && node.right.nil?)
          node.value = @hashes.shift
          return
        end

        if node.left
          @visit_idx += 1
          assign_value(node.left)
        end
        if node.right
          @visit_idx += 1
          assign_value(node.right)
        end

        right = node.right || node.left
        node.value = Bitcoin.bitcoin_mrkl(node.left.value, right.value)

        nil
      end

      def valid_tree?(mrkl_root_hash)
        return false unless @hashes.empty?
        return false if ((@visit_idx + 1) / 8.0).ceil != @flags.length
        return false if mrkl_root_hash != root.value
        true
      end
    end
  end
end

Version data entries

2 entries across 2 versions & 1 rubygems

Version Path
bitcoin-ruby-0.0.20 lib/bitcoin/protocol/partial_merkle_tree.rb
bitcoin-ruby-0.0.19 lib/bitcoin/protocol/partial_merkle_tree.rb