class Nanook
# The Nanook::Block class contains methods to discover
# publicly-available information about blocks on the nano network.
#
# A block is represented by a unique id like this:
#
# "FBF8B0E6623A31AB528EBD839EEAA91CAFD25C12294C46754E45FD017F7939EB"
#
# Initialize this class through the convenient Nanook#block method:
#
# nanook = Nanook.new
# block = nanook.block("FBF8B0E...")
#
# Or compose the longhand way like this:
#
# rpc_conn = Nanook::Rpc.new
# block = Nanook::Block.new(rpc_conn, "FBF8B0E...")
class Block
def initialize(rpc, block)
@rpc = rpc
@block = block
block_required! # All methods expect a block
end
# Returns the {Nanook::Account} of the block.
#
# ==== Example:
# block.account # => Nanook::Account
#
# @return [Nanook::Account] the account of the block
def account
Nanook::Account.new(@rpc, rpc(:block_account, :hash)[:account])
end
# Stop generating work for a block.
#
# ==== Example:
#
# block.cancel_work # => true
#
# @return [Boolean] signalling if the action was successful
def cancel_work
rpc(:work_cancel, :hash).empty?
end
# Returns a consecutive list of block hashes in the account chain
# starting at block back to count (direction from frontier back to
# open block, from newer blocks to older). Will list all blocks back
# to the open block of this chain when count is set to "-1".
# The requested block hash is included in the answer.
#
# See also #successors.
#
# ==== Example:
#
# block.chain(limit: 2)
#
# ==== Example reponse:
#
# [
# "36A0FB717368BA8CF8D255B63DC207771EABC6C6FFC22A7F455EC2209464897E",
# "FBF8B0E6623A31AB528EBD839EEAA91CAFD25C12294C46754E45FD017F7939EB"
# ]
#
# @param limit [Integer] maximum number of block hashes to return (default is 1000)
# @param offset [Integer] return the account chain block hashes offset by the specified number of blocks (default is 0)
def chain(limit: 1000, offset: 0)
response = rpc(:chain, :block, count: limit, offset: offset)[:blocks]
Nanook::Util.coerce_empty_string_to_type(response, Array)
end
alias_method :ancestors, :chain
# Request confirmation for a block from online representative nodes.
# Will return immediately with a boolean to indicate if the request for
# confirmation was successful. Note that this boolean does not indicate
# the confirmation status of the block. If confirmed, your block should
# appear in {Nanook::Node#confirmation_history} within a short amount of
# time, or you can use the convenience method {Nanook::Block#confirmed_recently?}
#
# ==== Example:
# block.confirm # => true
#
# @return [Boolean] if the confirmation request was sent successful
def confirm
rpc(:block_confirm, :hash)[:started] == 1
end
# This call is for internal diagnostics/debug purposes only. Do not
# rely on this interface being stable and do not use in a production system.
#
# Check if the block appears in the list of recently confirmed blocks by
# online representatives. The full list of blocks can be queried for with {Nanook::Node#confirmation_history}.
#
# This method can work in conjunction with {Nanook::Block#confirm},
# whereby you can send any block (old or new) out to online representatives to
# confirm. The confirmation process can take up to a couple of minutes.
#
# The method returning +false+ can indicate that the block is still in the process of being
# confirmed and that you should call the method again soon, or that it
# was confirmed earlier than the list available in {Nanook::Node#confirmation_history},
# or that it was not confirmed.
#
# ==== Example:
# block.confirmed_recently? # => true
#
# @return [Boolean] +true+ if the block has been recently confirmed by
# online representatives.
def confirmed_recently?
@rpc.call(:confirmation_history)[:confirmations].map{|h| h[:hash]}.include?(@block)
end
alias_method :recently_confirmed?, :confirmed_recently?
# Generate work for a block.
#
# ==== Example:
# block.generate_work # => "2bf29ef00786a6bc"
#
# @param use_peers [Boolean] if set to +true+, then the node will query
# its work peers (if it has any, see {Nanook::WorkPeer#list}).
# When +false+, the node will only generate work locally (default is +false+)
# @return [String] the work id of the work completed.
def generate_work(use_peers: false)
rpc(:work_generate, :hash, use_peers: use_peers)[:work]
end
# Returns Array of Hashes containing information about a chain of
# send/receive blocks, starting from this block.
#
# ==== Example:
#
# block.history(limit: 1)
#
# ==== Example response:
#
# [
# {
# :account=>"nano_3x7cjioqahgs5ppheys6prpqtb4rdknked83chf97bot1unrbdkaux37t31b",
# :amount=>539834279601145558517940224,
# :hash=>"36A0FB717368BA8CF8D255B63DC207771EABC6C6FFC22A7F455EC2209464897E",
# :type=>"send"
# }
# ]
#
# @param limit [Integer] maximum number of send/receive block hashes
# to return in the chain (default is 1000)
def history(limit: 1000)
rpc(:history, :hash, count: limit)[:history]
end
# Returns the block hash id.
#
# ==== Example:
#
# block.id #=> "FBF8B0E..."
#
# @return [String] the block hash id
def id
@block
end
# Returns a Hash of information about the block.
#
# ==== Examples:
#
# block.info
# block.info(allow_unchecked: true)
#
# ==== Example response:
#
# {
# :id=>"36A0FB717368BA8CF8D255B63DC207771EABC6C6FFC22A7F455EC2209464897E",
# :type=>"send",
# :previous=>"FBF8B0E6623A31AB528EBD839EEAA91CAFD25C12294C46754E45FD017F7939EB",
# :destination=>"nano_3x7cjioqahgs5ppheys6prpqtb4rdknked83chf97bot1unrbdkaux37t31b",
# :balance=>"00000000000000000000000000000000",
# :work=>"44cc24b60705083a",
# :signature=>"42ADFEFE7C3FFF188AE92A202F8A5734DE91779C454613E446EEC93D001D6C953E9FD16730AF32C891791BA8EDAECEB059A213E2FE1EEB7ADF9D5D0815464D06"
# }
#
# @param allow_unchecked [Boolean] (default is +false+). If +true+,
# information can be returned about blocks that are unchecked (unverified).
def info(allow_unchecked: false)
if allow_unchecked
response = rpc(:unchecked_get, :hash)
unless response.has_key?(:error)
return _parse_info_response(response)
end
# If unchecked not found, continue to checked block
end
response = rpc(:block, :hash)
_parse_info_response(response)
end
# ==== Example:
#
# block.is_valid_work?("2bf29ef00786a6bc") # => true
#
# @param work [String] the work id to check is valid
# @return [Boolean] signalling if work is valid for the block
def is_valid_work?(work)
response = rpc(:work_validate, :hash, work: work)
!response.empty? && response[:valid] == 1
end
# Republish blocks starting at this block up the account chain
# back to the nano network.
#
# @return [Array] block hashes that were republished
#
# ==== Example:
#
# block.republish
#
# ==== Example response:
#
# ["36A0FB717368BA8CF8D255B63DC207771EABC6C6FFC22A7F455EC2209464897E"]
def republish(destinations:nil, sources:nil)
if !destinations.nil? && !sources.nil?
raise ArgumentError.new("You must provide either destinations or sources but not both")
end
# Add in optional arguments
params = {}
params[:destinations] = destinations unless destinations.nil?
params[:sources] = sources unless sources.nil?
params[:count] = 1 unless params.empty?
rpc(:republish, :hash, params)[:blocks]
end
# ==== Example:
#
# block.pending? #=> false
#
# @return [Boolean] signalling if the block is a pending block.
def pending?
response = rpc(:pending_exists, :hash)
!response.empty? && response[:exists] == 1
end
# Publish the block to the nano network.
#
# Note, if block has previously been published, use #republish instead.
#
# ==== Examples:
#
# block.publish # => "FBF8B0E..."
#
# @return [String] the block hash, or false.
def publish
rpc(:process, :block)[:hash] || false
end
alias_method :process, :publish
# Returns an Array of block hashes in the account chain ending at
# this block.
#
# See also #chain.
#
# ==== Example:
#
# block.successors
#
# ==== Example response:
#
# ["36A0FB717368BA8CF8D255B63DC207771EABC6C6FFC22A7F455EC2209464897E"]
#
# @param limit [Integer] maximum number of send/receive block hashes
# to return in the chain (default is 1000)
# @param offset [Integer] return the account chain block hashes offset
# by the specified number of blocks (default is 0)
# @return [Array] block hashes in the account chain ending at this block
def successors(limit: 1000, offset: 0)
response = rpc(:successors, :block, count: limit, offset: offset)[:blocks]
Nanook::Util.coerce_empty_string_to_type(response, Array)
end
def inspect
"#{self.class.name}(id: \"#{id}\", object_id: \"#{"0x00%x" % (object_id << 1)}\")"
end
private
# Some RPC calls expect the param that represents the block to be named
# "hash", and others "block".
# The param_name argument allows us to specify which it should be for this call.
def rpc(action, param_name, params={})
p = @block.nil? ? {} : { param_name.to_sym => @block }
@rpc.call(action, p.merge(params))
end
def block_required!
if @block.nil?
raise ArgumentError.new("Block must be present")
end
end
def _parse_info_response(response)
# The contents is a stringified JSON
if response[:contents]
r = JSON.parse(response[:contents]).to_symbolized_hash
return r.merge(id: id)
end
response
end
end
end