Bitcoin.require_dependency :sequel, message: "Note: You will also need an adapter for your database like sqlite3, mysql2, postgresql" module Bitcoin::Storage::Backends # Storage backend using Sequel to connect to arbitrary SQL databases. # Inherits from StoreBase and implements its interface. class UtxoStore < StoreBase # possible script types SCRIPT_TYPES = [:unknown, :pubkey, :hash160, :multisig, :p2sh] if Bitcoin.namecoin? [:name_new, :name_firstupdate, :name_update].each {|n| SCRIPT_TYPES << n } end # sequel database connection attr_accessor :db DEFAULT_CONFIG = { # cache head block; it is only updated when new block comes in, # so this should only be used by the store receiving new blocks. cache_head: false, # cache this many utxo records before syncing to disk. # this should only be enabled during initial sync, because # with it the store cannot reorg properly. utxo_cache: 250, # cache this many blocks. # NOTE: this is also the maximum number of blocks the store can reorg. block_cache: 120, # keep an index of utxos for all addresses, not just the ones # we are explicitly told about. index_all_addrs: false } # create sequel store with given +config+ def initialize config, *args super config, *args @spent_outs, @new_outs, @watched_addrs = [], [], [] @deleted_utxos, @tx_cache, @block_cache = {}, {}, {} end # connect to database def connect super load_watched_addrs # rescan end # reset database; delete all data def reset [:blk, :utxo, :addr, :addr_txout].each {|table| @db[table].delete } @head = nil end # persist given block +blk+ to storage. def persist_block blk, chain, depth, prev_work = 0 load_watched_addrs @db.transaction do attrs = { :hash => blk.hash.htb.blob, :depth => depth, :chain => chain, :version => blk.ver, :prev_hash => blk.prev_block.reverse.blob, :mrkl_root => blk.mrkl_root.reverse.blob, :time => blk.time, :bits => blk.bits, :nonce => blk.nonce, :blk_size => blk.payload.bytesize, :work => (prev_work + blk.block_work).to_s } existing = @db[:blk].filter(:hash => blk.hash.htb.blob) if existing.any? existing.update attrs block_id = existing.first[:id] else block_id = @db[:blk].insert(attrs) end if @config[:block_cache] > 0 @block_cache.shift if @block_cache.size > @config[:block_cache] @deleted_utxos.shift if @deleted_utxos.size > @config[:block_cache] @block_cache[blk.hash] = blk end if chain == MAIN persist_transactions(blk.tx, block_id, depth) @tx_cache = {} @head = wrap_block(attrs.merge(id: block_id)) if chain == MAIN end return depth, chain end end def persist_transactions txs, block_id, depth txs.each.with_index do |tx, tx_blk_idx| tx.in.each.with_index do |txin, txin_tx_idx| next if txin.coinbase? size = @new_outs.size @new_outs.delete_if {|o| o[0][:tx_hash] == txin.prev_out.reverse.hth && o[0][:tx_idx] == txin.prev_out_index } @spent_outs << { tx_hash: txin.prev_out.reverse.hth.to_sequel_blob, tx_idx: txin.prev_out_index } if @new_outs.size == size end tx.out.each.with_index do |txout, txout_tx_idx| _, a, n = *parse_script(txout, txout_tx_idx) @new_outs << [{ :tx_hash => tx.hash.blob, :tx_idx => txout_tx_idx, :blk_id => block_id, :pk_script => txout.pk_script.blob, :value => txout.value }, @config[:index_all_addrs] ? a : a.select {|a| @watched_addrs.include?(a[1]) }, Bitcoin.namecoin? ? n : [] ] end end flush_spent_outs(depth) if @spent_outs.size > @config[:utxo_cache] flush_new_outs(depth) if @new_outs.size > @config[:utxo_cache] end def reorg new_side, new_main new_side.each do |block_hash| raise "trying to remove non-head block!" unless get_head.hash == block_hash depth = get_depth blk = @db[:blk][hash: block_hash.htb.blob] delete_utxos = @db[:utxo].where(blk_id: blk[:id]) @db[:addr_txout].where("txout_id IN ?", delete_utxos.map{|o|o[:id]}).delete delete_utxos.delete (@deleted_utxos[depth] || []).each do |utxo| utxo[:pk_script] = utxo[:pk_script].to_sequel_blob utxo_id = @db[:utxo].insert(utxo) addrs = Bitcoin::Script.new(utxo[:pk_script]).get_addresses addrs.each do |addr| hash160 = Bitcoin.hash160_from_address(addr) store_addr(utxo_id, hash160) end end @db[:blk].where(id: blk[:id]).update(chain: SIDE) end new_main.each do |block_hash| block = @db[:blk][hash: block_hash.htb.blob] blk = @block_cache[block_hash] persist_transactions(blk.tx, block[:id], block[:depth]) @db[:blk].where(id: block[:id]).update(chain: MAIN) end end def flush_spent_outs depth log.time "flushed #{@spent_outs.size} spent txouts in %.4fs" do if @spent_outs.any? @spent_outs.each_slice(250) do |slice| if @db.adapter_scheme == :postgres condition = slice.map {|o| "(tx_hash = '#{o[:tx_hash]}' AND tx_idx = #{o[:tx_idx]})" }.join(" OR ") else condition = slice.map {|o| "(tx_hash = X'#{o[:tx_hash].hth}' AND tx_idx = #{o[:tx_idx]})" }.join(" OR ") end @db["DELETE FROM addr_txout WHERE EXISTS (SELECT 1 FROM utxo WHERE utxo.id = addr_txout.txout_id AND (#{condition}));"].all @db["DELETE FROM utxo WHERE #{condition};"].first end end @spent_outs = [] end end def flush_new_outs depth log.time "flushed #{@new_outs.size} new txouts in %.4fs" do new_utxo_ids = @db[:utxo].insert_multiple(@new_outs.map{|o|o[0]}) @new_outs.each.with_index do |d, idx| d[1].each do |i, hash160| next unless i && hash160 store_addr(new_utxo_ids[idx], hash160) end end @new_outs.each.with_index do |d, idx| d[2].each do |i, script| next unless i && script store_name(script, new_utxo_ids[idx]) end end @new_outs = [] end end def add_watched_address address hash160 = Bitcoin.hash160_from_address(address) @db[:addr].insert(hash160: hash160) unless @db[:addr][hash160: hash160] @watched_addrs << hash160 unless @watched_addrs.include?(hash160) end def load_watched_addrs @watched_addrs = @db[:addr].all.map{|a| a[:hash160] } unless @config[:index_all_addrs] end def rescan load_watched_addrs @rescan_lock ||= Monitor.new @rescan_lock.synchronize do log.info { "Rescanning #{@db[:utxo].count} utxos for #{@watched_addrs.size} addrs" } count = @db[:utxo].count; n = 100_000 @db[:utxo].order(:id).each_slice(n).with_index do |slice, index| log.debug { "rescan progress: %.2f%" % (100.0 / count * (index*n)) } slice.each do |utxo| next if utxo[:pk_script].bytesize >= 10_000 hash160 = Bitcoin::Script.new(utxo[:pk_script]).get_hash160 if @config[:index_all_addrs] || @watched_addrs.include?(hash160) log.info { "Found utxo for address #{Bitcoin.hash160_to_address(hash160)}: " + "#{utxo[:tx_hash][0..8]}:#{utxo[:tx_idx]} (#{utxo[:value]})" } addr = @db[:addr][hash160: hash160] addr_utxo = {addr_id: addr[:id], txout_id: utxo[:id]} @db[:addr_txout].insert(addr_utxo) unless @db[:addr_txout][addr_utxo] end end end end end # check if block +blk_hash+ exists def has_block(blk_hash) !!@db[:blk].where(:hash => blk_hash.htb.blob).get(1) end # check if transaction +tx_hash+ exists def has_tx(tx_hash) !!@db[:utxo].where(:tx_hash => tx_hash.blob).get(1) end # get head block (highest block from the MAIN chain) def get_head (@config[:cache_head] && @head) ? @head : @head = wrap_block(@db[:blk].filter(:chain => MAIN).order(:depth).last) end # get depth of MAIN chain def get_depth return -1 unless get_head get_head.depth end # get block for given +blk_hash+ def get_block(blk_hash) wrap_block(@db[:blk][:hash => blk_hash.htb.blob]) end # get block by given +depth+ def get_block_by_depth(depth) wrap_block(@db[:blk][:depth => depth, :chain => MAIN]) end # get block by given +prev_hash+ def get_block_by_prev_hash(prev_hash) wrap_block(@db[:blk][:prev_hash => prev_hash.htb.blob, :chain => MAIN]) end # get block by given +tx_hash+ def get_block_by_tx(tx_hash) block_id = @db[:utxo][tx_hash: tx_hash.blob][:blk_id] get_block_by_id(block_id) end # get block by given +id+ def get_block_by_id(block_id) wrap_block(@db[:blk][:id => block_id]) end # get transaction for given +tx_hash+ def get_tx(tx_hash) @tx_cache[tx_hash] ||= wrap_tx(tx_hash) end # get transaction by given +tx_id+ def get_tx_by_id(tx_id) get_tx(tx_id) end def get_txout_by_id(id) wrap_txout(@db[:utxo][id: id]) end # get corresponding Models::TxOut for +txin+ def get_txout_for_txin(txin) wrap_txout(@db[:utxo][tx_hash: txin.prev_out.reverse.hth.blob, tx_idx: txin.prev_out_index]) end # get all Models::TxOut matching given +script+ def get_txouts_for_pk_script(script) utxos = @db[:utxo].filter(pk_script: script.blob).order(:blk_id) utxos.map {|utxo| wrap_txout(utxo) } end # get all Models::TxOut matching given +hash160+ def get_txouts_for_hash160(hash160, unconfirmed = false) addr = @db[:addr][hash160: hash160] return [] unless addr @db[:addr_txout].where(addr_id: addr[:id]).map {|ao| wrap_txout(@db[:utxo][id: ao[:txout_id]]) }.compact end def get_balance hash160 get_txouts_for_hash160(hash160).map(&:value).inject(:+) || 0 end # wrap given +block+ into Models::Block def wrap_block(block) return nil unless block data = {:id => block[:id], :depth => block[:depth], :chain => block[:chain], :work => block[:work].to_i, :hash => block[:hash].hth} blk = Bitcoin::Storage::Models::Block.new(self, data) blk.ver = block[:version] blk.prev_block = block[:prev_hash].reverse blk.mrkl_root = block[:mrkl_root].reverse blk.time = block[:time].to_i blk.bits = block[:bits] blk.nonce = block[:nonce] if cached = @block_cache[block[:hash].hth] blk.tx = cached.tx end blk.recalc_block_hash blk end # wrap given +transaction+ into Models::Transaction def wrap_tx(tx_hash) utxos = @db[:utxo].where(tx_hash: tx_hash.blob) return nil unless utxos.any? data = { blk_id: utxos.first[:blk_id] } tx = Bitcoin::Storage::Models::Tx.new(self, data) tx.hash = tx_hash # utxos.first[:tx_hash].hth utxos.each {|u| tx.out[u[:tx_idx]] = wrap_txout(u) } return tx end # wrap given +output+ into Models::TxOut def wrap_txout(utxo) return nil unless utxo data = {id: utxo[:id], tx_id: utxo[:tx_hash], tx_idx: utxo[:tx_idx]} txout = Bitcoin::Storage::Models::TxOut.new(self, data) txout.value = utxo[:value] txout.pk_script = utxo[:pk_script] txout end def check_consistency(*args) log.warn { "Utxo store doesn't support consistency check" } end end end