2021-11-08 16:39:17 +00:00
// Copyright 2017-2021 DERO Project. All rights reserved.
// Use of this source code in any form is governed by RESEARCH license.
// license can be found in the LICENSE file.
// GPG: 0F39 E425 8C65 3947 702A 8234 08B2 0360 A03A 9DE8
//
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
package explorerlib
// this file implements the explorer for DERO blockchain
// this needs only RPC access
// NOTE: Only use data exported from within the RPC interface, do direct use of exported variables fom packages
// NOTE: we can use structs defined within the RPCserver package
// TODO: error handling is non-existant ( as this was built up in hrs ). Add proper error handling
//
import "time"
import "fmt"
import "embed"
import "bytes"
import "unicode"
import "unsafe" // need to avoid this, but only used by byteviewer
import "strings"
import "strconv"
import "context"
import "encoding/hex"
import "net/http"
import "html/template"
//import "encoding/json"
//import "io/ioutil"
import "github.com/go-logr/logr"
import "github.com/deroproject/derohe/block"
import "github.com/deroproject/derohe/cryptography/crypto"
import "github.com/deroproject/derohe/globals"
import "github.com/deroproject/derohe/transaction"
import "github.com/deroproject/derohe/rpc"
import "github.com/deroproject/derohe/proof"
import "github.com/deroproject/derohe/glue/rwc"
import "github.com/creachadair/jrpc2"
import "github.com/creachadair/jrpc2/channel"
import "github.com/gorilla/websocket"
//go:embed templates/*.tmpl
var tpls embed . FS
//go:embed static/*
var static embed . FS
type Client struct {
WS * websocket . Conn
RPC * jrpc2 . Client
}
var rpc_client = & Client { }
var Connected bool = false
var mainnet = true
var endpoint string
var replacer = strings . NewReplacer ( "h" , ":" , "m" , ":" , "s" , "" )
var logger logr . Logger
func ( cli * Client ) Call ( method string , params interface { } , result interface { } ) error {
try := 0
try_again :
if cli == nil || ! cli . IsDaemonOnline ( ) {
go Connect ( )
time . Sleep ( time . Second )
try ++
if try < 3 {
goto try_again
}
return fmt . Errorf ( "client is offline or not connected" )
}
return cli . RPC . CallResult ( context . Background ( ) , method , params , result )
}
// this is as simple as it gets
// single threaded communication to get the daemon status and height
// this will tell whether the wallet can connection successfully to daemon or not
func ( cli * Client ) IsDaemonOnline ( ) bool {
if cli . WS == nil || cli . RPC == nil {
return false
}
return true
}
func ( cli * Client ) onlinecheck_and_get_online ( ) {
for {
if cli . IsDaemonOnline ( ) {
var result string
if err := cli . Call ( "DERO.Ping" , nil , & result ) ; err != nil {
logger . V ( 1 ) . Error ( err , "Ping failed:" )
cli . RPC . Close ( )
cli . WS = nil
cli . RPC = nil
Connect ( ) // try to connect again
} else {
//fmt.Printf("Ping Received %s\n", result)
}
}
time . Sleep ( time . Second )
}
}
// this is as simple as it gets
// single threaded communication to get the daemon status and height
// this will tell whether the wallet can connection successfully to daemon or not
func Connect ( ) ( err error ) {
// TODO enable socks support here
//rpc_conn, err = rpcc.Dial("ws://"+ w.Daemon_Endpoint + "/ws")
daemon_endpoint := endpoint
rpc_client . WS , _ , err = websocket . DefaultDialer . Dial ( "ws://" + daemon_endpoint + "/ws" , nil )
// notify user of any state change
// if daemon connection breaks or comes live again
if err == nil {
if ! Connected {
logger . V ( 1 ) . Info ( "Connection to RPC server successful" , "daemon_endpoint" , "ws://" + daemon_endpoint + "/ws" )
Connected = true
}
} else {
logger . Error ( err , "Error connecting to daemon" )
if Connected {
logger . Error ( err , "Connection to RPC server Failed " , "daemon_endpoint" , "ws://" + daemon_endpoint + "/ws" )
}
Connected = false
return
}
input_output := rwc . New ( rpc_client . WS )
rpc_client . RPC = jrpc2 . NewClient ( channel . RawJSON ( input_output , input_output ) , nil )
var result string
if err := rpc_client . Call ( "DERO.Ping" , nil , & result ) ; err != nil {
logger . V ( 1 ) . Error ( err , "Ping failed:" )
} else {
// fmt.Printf("Ping Received %s\n", result)
}
var info rpc . GetInfo_Result
// collect all the data afresh, execute rpc to service
if err = rpc_client . Call ( "DERO.GetInfo" , nil , & info ) ; err != nil {
logger . V ( 1 ) . Error ( err , "GetInfo failed:" )
} else {
mainnet = ! info . Testnet // inverse of testnet is mainnet
}
return nil
}
func StartServer ( loggerb logr . Logger , daemon_endpoint string , listen_address string ) ( err error ) {
logger = globals . Logger . WithName ( "explorer" )
endpoint = daemon_endpoint
logger . Info ( "Daemon RPC endpoint " , "endpoint" , endpoint )
logger . Info ( "Will listen " , "address" , listen_address )
// execute rpc to service
err = Connect ( )
if err == nil {
logger . Info ( "Connection to RPC server successful" )
} else {
logger . Error ( err , "Connection to RPC server Failed" )
return
}
go rpc_client . onlinecheck_and_get_online ( ) // keep connectingto server
all_templates , err = template . ParseFS ( tpls , "templates/*.tmpl" )
if err != nil {
logger . Error ( err , "error parsing templates." )
return
}
http . Handle ( "/static/" , http . StripPrefix ( "/static/" , http . FileServer ( http . FS ( static ) ) ) ) // include all static assets
http . HandleFunc ( "/search" , search_handler )
http . HandleFunc ( "/page/" , page_handler )
http . HandleFunc ( "/block/" , block_handler )
http . HandleFunc ( "/txpool/" , txpool_handler )
http . HandleFunc ( "/tx/" , tx_handler )
http . HandleFunc ( "/" , root_handler )
go func ( ) {
logger . Info ( "Listening for requests" )
err = http . ListenAndServe ( listen_address , nil )
logger . Error ( err , "ListenAndServe failed" )
} ( )
time . Sleep ( 50 * time . Millisecond )
return err
}
// all the tx info which ever needs to be printed
type txinfo struct {
Hex string // raw tx
Height string // height at which tx was mined
HeightBuilt uint64 // height at which tx was built
RootHash string // roothash which forms the basis for balance tree
TransactionType string // transaction type
Depth int64
Timestamp uint64 // timestamp
Age string // time diff from current time
Block_time string // UTC time from block header
Epoch uint64 // Epoch time
In_Pool bool // whether tx was in pool
Hash string // hash for hash
PrefixHash string // prefix hash
Version int // version of tx
Size string // size of tx in KB
Sizeuint64 uint64 // size of tx in bytes
Burn_Value string // value of burned amount
Fee string // fee in TX
Feeuint64 uint64 // fee in atomic units
In int // inputs counts
Out int // outputs counts
Amount string
CoinBase bool // is tx coin base
Extra string // extra within tx
Keyimages [ ] string // key images within tx
OutAddress [ ] string // contains output secret key
OutOffset [ ] uint64 // contains index offsets
Type string // ringct or ruffct ( bulletproof)
ValidBlock string // the tx is valid in which block
InvalidBlock [ ] string // the tx is invalid in which block
Skipped bool // this is only valid, when a block is being listed
Ring_size int
Ring [ ] [ ] string // contains entire ring in string form
TXpublickey string
PayID32 string // 32 byte payment ID
PayID8 string // 8 byte encrypted payment ID
Proof_address string // address agains which which the proving ran
Proof_index int64 // proof satisfied for which index
Proof_amount string // decoded amount
Proof_Payload_raw string // payload raw bytes
Proof_Payload string // if proof decoded, decoded , else decode error
Proof_error string // error if any while decoding proof
SC_TX_Available string //bool // whether this contains an SC TX
SC_Signer string // whether SC signer
SC_Signer_verified string // whether SC signer can be verified successfully
SC_Balance uint64 // SC SC_Balance in atomic units
SC_Balance_string string // SC_Balance in DERO
SC_Keys map [ string ] string // SC key value of
SC_Args rpc . Arguments // rpc.Arguments
SC_Code string // install SC
2021-11-10 11:04:22 +00:00
SC_State rpc . GetSC_Result // current SC state
SC_Install bool
2021-11-08 16:39:17 +00:00
Assets [ ] Asset
}
type Asset struct {
SCID string
Fees string
Burn string
Ring [ ] string
Ring_size int
}
// any information for block which needs to be printed
type block_info struct {
Block block . Block
Major_Version uint64
Minor_Version uint64
Height int64
TopoHeight int64
Depth int64
Timestamp uint64
Hash string
Tips [ ] string
Nonce uint64
Fees string
Reward string
Size string
Age string // time diff from current time
Block_time string // UTC time from block header
Epoch uint64 // Epoch time
Outputs string
Mtx txinfo
Txs [ ] txinfo
Orphan_Status bool
SyncBlock bool // whether the block is sync block
Tx_Count int
}
var all_templates * template . Template
// load and setup block_info from rpc
// if hash is less than 64 bytes then it is considered a height parameter
func load_block_from_rpc ( info * block_info , block_hash string , recursive bool ) ( err error ) {
var bl block . Block
var bresult rpc . GetBlock_Result
var block_height int
var block_bin [ ] byte
if len ( block_hash ) != 64 { // parameter is a height
fmt . Sscanf ( block_hash , "%d" , & block_height )
// user requested block height
logger . V ( 1 ) . Info ( "User requested block" , "topoheight" , block_height , "user input" , block_hash )
if err = rpc_client . Call ( "DERO.GetBlock" , rpc . GetBlock_Params { Height : uint64 ( block_height ) } , & bresult ) ; err != nil {
return fmt . Errorf ( "getblock rpc failed. err %s" , err )
}
} else { // parameter is the hex blob
logger . V ( 1 ) . Info ( "User requested block using hash" , "block_hash" , block_hash )
if err = rpc_client . Call ( "DERO.GetBlock" , rpc . GetBlock_Params { Hash : block_hash } , & bresult ) ; err != nil {
return fmt . Errorf ( "getblock rpc failed" )
}
}
// fmt.Printf("block %d %+v\n",i, bresult)
info . TopoHeight = bresult . Block_Header . TopoHeight
info . Height = bresult . Block_Header . Height
info . Depth = bresult . Block_Header . Depth
duration_millisecond := ( uint64 ( time . Now ( ) . UTC ( ) . UnixMilli ( ) ) - bresult . Block_Header . Timestamp )
info . Age = replacer . Replace ( ( time . Duration ( duration_millisecond ) * time . Millisecond ) . String ( ) )
info . Block_time = time . Unix ( 0 , int64 ( bresult . Block_Header . Timestamp * uint64 ( time . Millisecond ) ) ) . Format ( "2006-01-02 15:04:05" )
info . Epoch = bresult . Block_Header . Timestamp
info . Outputs = fmt . Sprintf ( "%.03f" , float32 ( bresult . Block_Header . Reward ) / 1000000000000.0 )
info . Size = "N/A"
info . Hash = bresult . Block_Header . Hash
//info.Prev_Hash = bresult.Block_Header.Prev_Hash
info . Tips = bresult . Block_Header . Tips
info . Orphan_Status = bresult . Block_Header . Orphan_Status
info . SyncBlock = bresult . Block_Header . SyncBlock
info . Nonce = bresult . Block_Header . Nonce
info . Major_Version = bresult . Block_Header . Major_Version
info . Minor_Version = bresult . Block_Header . Minor_Version
info . Reward = fmt . Sprintf ( "%.03f" , float32 ( bresult . Block_Header . Reward ) / 1000000000000.0 )
block_bin , _ = hex . DecodeString ( bresult . Blob )
//log.Infof("block %+v bresult %+v ", bl, bresult)
bl . Deserialize ( block_bin )
info . Block = bl
if recursive {
// fill in miner tx info
//err = load_tx_from_rpc(&info.Mtx, bl.Miner_TX.GetHash().String()) //TODO handle error
load_tx_info_from_tx ( & info . Mtx , & bl . Miner_TX )
// miner tx reward is calculated on runtime due to client protocol reasons in dero atlantis
// feed what is calculated by the daemon
reward := uint64 ( 0 )
if bl . Miner_TX . TransactionType == transaction . PREMINE {
reward += bl . Miner_TX . Value
}
info . Mtx . Amount = fmt . Sprintf ( "%.05f" , float64 ( reward + bresult . Block_Header . Reward ) / 100000 )
//logger.Error(err,"loading miner tx from rpc ", "txid", bl.Miner_TX.GetHash().String())
info . Tx_Count = len ( bl . Tx_hashes )
fees := uint64 ( 0 )
size := uint64 ( len ( bl . Serialize ( ) ) )
// if we have any other tx load them also
for i := 0 ; i < len ( bl . Tx_hashes ) ; i ++ {
var tx txinfo
err = load_tx_from_rpc ( & tx , bl . Tx_hashes [ i ] . String ( ) ) //TODO handle error
if err != nil {
logger . V ( 1 ) . Error ( err , "loading tx " , "txid" , bl . Tx_hashes [ i ] . String ( ) )
}
if tx . ValidBlock != bresult . Block_Header . Hash { // track skipped status
tx . Skipped = true
}
info . Txs = append ( info . Txs , tx )
fees += tx . Feeuint64
size += tx . Sizeuint64
}
info . Fees = fmt . Sprintf ( "%.03f" , float32 ( fees ) / 100000.0 )
info . Size = fmt . Sprintf ( "%.03f" , float32 ( size ) / 1024 )
}
return
}
// this will fill up the info struct from the tx
func load_tx_info_from_tx ( info * txinfo , tx * transaction . Transaction ) ( err error ) {
info . Hash = tx . GetHash ( ) . String ( )
//info.PrefixHash = tx.GetPrefixHash().String()
info . TransactionType = tx . TransactionType . String ( )
info . Size = fmt . Sprintf ( "%.03f" , float32 ( len ( tx . Serialize ( ) ) ) / 1024 )
info . Sizeuint64 = uint64 ( len ( tx . Serialize ( ) ) )
info . Version = int ( tx . Version )
//info.Extra = fmt.Sprintf("%x", tx.Extra)
if len ( tx . Payloads ) >= 1 {
info . RootHash = fmt . Sprintf ( "%x" , tx . Payloads [ 0 ] . Statement . Roothash [ : ] )
}
info . HeightBuilt = tx . Height
//info.In = len(tx.Vin)
//info.Out = len(tx.Vout)
if tx . TransactionType == transaction . BURN_TX {
info . Burn_Value = fmt . Sprintf ( " %.05f" , float64 ( tx . Value ) / 100000 )
}
switch tx . TransactionType {
case transaction . PREMINE :
var acckey crypto . Point
if err := acckey . DecodeCompressed ( tx . MinerAddress [ : ] ) ; err != nil {
panic ( err )
}
astring := rpc . NewAddressFromKeys ( & acckey )
astring . Mainnet = mainnet
info . OutAddress = append ( info . OutAddress , astring . String ( ) )
info . Amount = globals . FormatMoney ( tx . Value )
case transaction . REGISTRATION :
var acckey crypto . Point
if err := acckey . DecodeCompressed ( tx . MinerAddress [ : ] ) ; err != nil {
panic ( err )
}
astring := rpc . NewAddressFromKeys ( & acckey )
astring . Mainnet = mainnet
info . OutAddress = append ( info . OutAddress , astring . String ( ) )
case transaction . COINBASE :
info . CoinBase = true
info . In = 0
var acckey crypto . Point
if err := acckey . DecodeCompressed ( tx . MinerAddress [ : ] ) ; err != nil {
panic ( err )
}
astring := rpc . NewAddressFromKeys ( & acckey )
astring . Mainnet = mainnet
info . OutAddress = append ( info . OutAddress , astring . String ( ) )
case transaction . NORMAL , transaction . BURN_TX , transaction . SC_TX :
info . Fee = globals . FormatMoney ( tx . Fees ( ) )
info . Ring_size = int ( tx . Payloads [ 0 ] . Statement . RingSize )
}
if tx . TransactionType == transaction . SC_TX {
info . SC_Args = tx . SCDATA
}
// if outputs cannot be located, do not panic
// this will be the case for pool transactions
if len ( info . OutAddress ) != len ( info . OutOffset ) {
info . OutOffset = make ( [ ] uint64 , len ( info . OutAddress ) , len ( info . OutAddress ) )
}
switch 0 {
case 0 :
info . Type = "DERO_HOMOMORPHIC"
default :
panic ( "not implemented" )
}
if ! info . In_Pool && ! info . CoinBase && ( tx . TransactionType == transaction . NORMAL || tx . TransactionType == transaction . BURN_TX || tx . TransactionType == transaction . SC_TX ) { // find the age of block and other meta
var blinfo block_info
err := load_block_from_rpc ( & blinfo , fmt . Sprintf ( "%s" , info . Height ) , false ) // we only need block data and not data of txs
if err != nil {
return err
}
// fmt.Printf("Blinfo %+v height %d", blinfo, info.Height);
info . Age = blinfo . Age
info . Block_time = blinfo . Block_time
info . Epoch = blinfo . Epoch
info . Timestamp = blinfo . Epoch
info . Depth = blinfo . Depth
}
return nil
}
// load and setup txinfo from rpc
func load_tx_from_rpc ( info * txinfo , txhash string ) ( err error ) {
var tx_params rpc . GetTransaction_Params
var tx_result rpc . GetTransaction_Result
//fmt.Printf("Requesting tx data %s", txhash);
tx_params . Tx_Hashes = append ( tx_params . Tx_Hashes , txhash )
if err = rpc_client . Call ( "DERO.GetTransaction" , tx_params , & tx_result ) ; err != nil {
return fmt . Errorf ( "gettransa rpc failed err %s" , err )
}
//fmt.Printf("TX response %+v", tx_result)
if tx_result . Status != "OK" {
return fmt . Errorf ( "No Such TX RPC error status %s" , tx_result . Status )
}
var tx transaction . Transaction
if len ( tx_result . Txs_as_hex [ 0 ] ) < 50 {
return
}
info . Hex = tx_result . Txs_as_hex [ 0 ]
tx_bin , _ := hex . DecodeString ( tx_result . Txs_as_hex [ 0 ] )
tx . Deserialize ( tx_bin )
// fill as much info required from headers
if tx_result . Txs [ 0 ] . In_pool {
info . In_Pool = true
} else {
info . Height = fmt . Sprintf ( "%d" , tx_result . Txs [ 0 ] . Block_Height )
}
for x := range tx_result . Txs [ 0 ] . Output_Indices {
info . OutOffset = append ( info . OutOffset , tx_result . Txs [ 0 ] . Output_Indices [ x ] )
}
if tx . IsCoinbase ( ) { // fill miner tx reward from what the chain tells us
info . Amount = fmt . Sprintf ( "%.05f" , float64 ( uint64 ( tx_result . Txs [ 0 ] . Reward ) ) / 100000 )
}
info . ValidBlock = tx_result . Txs [ 0 ] . ValidBlock
info . InvalidBlock = tx_result . Txs [ 0 ] . InvalidBlock
info . Ring = tx_result . Txs [ 0 ] . Ring
if tx . TransactionType == transaction . NORMAL || tx . TransactionType == transaction . BURN_TX || tx . TransactionType == transaction . SC_TX {
for t := range tx . Payloads {
var a Asset
a . SCID = tx . Payloads [ t ] . SCID . String ( )
a . Fees = fmt . Sprintf ( "%.05f" , float64 ( tx . Payloads [ t ] . Statement . Fees ) / 100000 )
a . Burn = fmt . Sprintf ( "%.05f" , float64 ( tx . Payloads [ t ] . BurnValue ) / 100000 )
if len ( tx_result . Txs [ 0 ] . Ring ) == 0 {
continue
}
a . Ring_size = len ( tx_result . Txs [ 0 ] . Ring [ t ] )
a . Ring = tx_result . Txs [ 0 ] . Ring [ t ]
info . Assets = append ( info . Assets , a )
}
//fmt.Printf("assets now %+v\n", info.Assets)
}
info . SC_Balance = tx_result . Txs [ 0 ] . Balance
info . SC_Balance_string = fmt . Sprintf ( "%.05f" , float64 ( uint64 ( info . SC_Balance ) / 100000 ) )
info . SC_Code = tx_result . Txs [ 0 ] . Code
2021-11-10 11:04:22 +00:00
if tx . TransactionType == transaction . SC_TX && len ( info . SC_Code ) >= 1 {
if len ( info . SC_Code ) >= 1 {
info . SC_Install = true
}
var p = rpc . GetSC_Params { SCID : txhash , Variables : true }
var r rpc . GetSC_Result
if err = rpc_client . Call ( "DERO.GetSC" , p , & r ) ; err != nil {
return fmt . Errorf ( "gettransa rpc failed err %s" , err )
} else {
info . SC_State = r
}
}
2021-11-08 16:39:17 +00:00
//info.Ring = strings.Join(info.OutAddress, " ")
//fmt.Printf("tx_result %+v\n",tx_result.Txs)
// fmt.Printf("response contained tx %s \n", tx.GetHash())
return load_tx_info_from_tx ( info , & tx )
}
func block_handler ( w http . ResponseWriter , r * http . Request ) {
param := ""
fmt . Sscanf ( r . URL . EscapedPath ( ) , "/block/%s" , & param )
var blinfo block_info
err := load_block_from_rpc ( & blinfo , param , true )
_ = err
// execute template now
data := map [ string ] interface { } { }
fill_common_info ( data , false )
data [ "block" ] = blinfo
err = all_templates . ExecuteTemplate ( w , "block" , data )
if err != nil {
return
}
return
// fmt.Fprint(w, "This is a valid block")
}
func tx_handler ( w http . ResponseWriter , r * http . Request ) {
var info txinfo
tx_hex := ""
fmt . Sscanf ( r . URL . EscapedPath ( ) , "/tx/%s" , & tx_hex )
txhash := crypto . HashHexToHash ( tx_hex )
logger . V ( 1 ) . Info ( "user requested " , "txid" , tx_hex )
err := load_tx_from_rpc ( & info , txhash . String ( ) ) //TODO handle error
_ = err
// check whether user requested proof
tx_proof := r . PostFormValue ( "txproof" )
raw_tx_data := r . PostFormValue ( "raw_tx_data" )
if raw_tx_data != "" { // gives ability to prove transactions not in the blockchain
info . Hex = raw_tx_data
}
if tx_proof != "" {
logger . V ( 1 ) . Info ( "Proving TX" , "proof" , tx_proof , "tx_hex" , info . Hex , "ring" , info . Ring )
// there may be more than 1 amounts, only first one is shown
addresses , amounts , raw , decoded , err := proof . Prove ( tx_proof , info . Hex , info . Ring , mainnet )
if err == nil { //&& len(amounts) > 0 && len(indexes) > 0{
logger . V ( 1 ) . Info ( "Successfully proved transaction" , "txid" , tx_hex , "payload_count" , len ( decoded ) )
info . Proof_address = addresses [ 0 ]
info . Proof_amount = globals . FormatMoney ( amounts [ 0 ] )
info . Proof_Payload_raw = BytesViewer ( raw [ 0 ] ) . String ( ) // raw payload
info . Proof_Payload = decoded [ 0 ]
} else {
logger . V ( 1 ) . Error ( err , "err while proving" )
if err != nil {
info . Proof_error = err . Error ( )
}
}
}
// execute template now
data := map [ string ] interface { } { }
fill_common_info ( data , false )
data [ "info" ] = info
err = all_templates . ExecuteTemplate ( w , "tx" , data )
if err != nil {
return
}
return
}
func pool_handler ( w http . ResponseWriter , r * http . Request ) {
fmt . Fprint ( w , "This is a valid pool" )
}
// if there is any error, we return back empty
// if pos is wrong we return back
// pos is descending order
func fill_tx_structure ( pos int , size_in_blocks int ) ( data [ ] block_info ) {
i := pos
for ; i > ( pos - size_in_blocks ) && i >= 0 ; i -- { // query blocks by topo height
var blinfo block_info
if err := load_block_from_rpc ( & blinfo , fmt . Sprintf ( "%d" , i ) , true ) ; err == nil {
data = append ( data , blinfo )
} else {
logger . V ( 2 ) . Error ( err , "error loading block" , "i" , i )
}
}
if i == 0 {
var blinfo block_info
if err := load_block_from_rpc ( & blinfo , fmt . Sprintf ( "%d" , i ) , true ) ; err == nil {
data = append ( data , blinfo )
} else {
logger . V ( 2 ) . Error ( err , "error loading block" , "i" , i )
}
}
return
}
func show_page ( w http . ResponseWriter , page int ) {
data := map [ string ] interface { } { }
var info rpc . GetInfo_Result
var err error
if err = rpc_client . Call ( "DERO.GetInfo" , nil , & info ) ; err != nil {
goto exit_error
}
fill_common_info ( data , true )
if page == 0 { // use requested invalid page, give current page
page = int ( info . TopoHeight ) / 10
}
data [ "previous_page" ] = page - 1
if page <= 1 {
data [ "previous_page" ] = 1
}
data [ "current_page" ] = page
if ( int ( info . TopoHeight ) % 10 ) == 0 {
data [ "total_page" ] = ( int ( info . TopoHeight ) / 10 )
} else {
data [ "total_page" ] = ( int ( info . TopoHeight ) / 10 )
}
data [ "next_page" ] = page + 1
if ( page + 1 ) > data [ "total_page" ] . ( int ) {
data [ "next_page" ] = page
}
fill_tx_pool_info ( data , 25 )
if page == 1 { // page 1 has 11 blocks, it does not show genesis block
data [ "block_array" ] = fill_tx_structure ( int ( page * 10 ) , 12 )
} else {
if int ( info . TopoHeight ) - int ( page * 10 ) > 10 {
data [ "block_array" ] = fill_tx_structure ( int ( page * 10 ) , 10 )
} else {
data [ "block_array" ] = fill_tx_structure ( int ( info . TopoHeight ) , int ( info . TopoHeight ) - int ( ( page - 1 ) * 10 ) )
}
}
//fmt.Printf("page %+v\n", data)
err = all_templates . ExecuteTemplate ( w , "main" , data )
if err != nil {
goto exit_error
}
return
exit_error :
fmt . Fprintf ( w , "Error occurred err %s" , err )
}
func txpool_handler ( w http . ResponseWriter , r * http . Request ) {
data := map [ string ] interface { } { }
fill_common_info ( data , true )
fill_tx_pool_info ( data , 500 ) // show only 500 txs
var err error
if err = all_templates . ExecuteTemplate ( w , "txpool_page" , data ) ; err != nil {
goto exit_error
}
return
exit_error :
fmt . Fprintf ( w , "Error occurred err %s" , err )
}
// shows a page
func page_handler ( w http . ResponseWriter , r * http . Request ) {
page := 0
page_string := r . URL . EscapedPath ( )
fmt . Sscanf ( page_string , "/page/%d" , & page )
logger . V ( 1 ) . Info ( "user requested page" , "page" , page )
show_page ( w , page )
}
// root shows page 0
func root_handler ( w http . ResponseWriter , r * http . Request ) {
logger . V ( 1 ) . Info ( "Showing main page" )
show_page ( w , 0 )
}
// search handler, finds the items using rpc bruteforce
func search_handler ( w http . ResponseWriter , r * http . Request ) {
var info rpc . GetInfo_Result
var err error
logger . V ( 1 ) . Info ( "Showing search page" )
values , ok := r . URL . Query ( ) [ "value" ]
if ! ok || len ( values ) < 1 {
show_page ( w , 0 )
return
}
// Query()["key"] will return an array of items,
// we only want the single item.
value := strings . TrimSpace ( values [ 0 ] )
good := false
// collect all the data afresh, execute rpc to service
if err = rpc_client . Call ( "DERO.GetInfo" , nil , & info ) ; err != nil {
goto exit_error
}
if len ( value ) != 64 {
if s , err := strconv . ParseInt ( value , 10 , 64 ) ; err == nil && s >= 0 && s <= info . TopoHeight {
good = true
}
} else { // check whether the string can be hex decoded
t , err := hex . DecodeString ( value )
if err != nil || len ( t ) != 32 {
} else {
good = true
}
}
// value should be either 64 hex chars or a topoheight which should be less than current topoheight
if good {
// check whether the page is block or tx or height
var blinfo block_info
var tx txinfo
err := load_block_from_rpc ( & blinfo , value , false )
if err == nil {
logger . V ( 1 ) . Info ( "Redirecting user to block page" )
http . Redirect ( w , r , "/block/" + value , 302 )
return
}
err = load_tx_from_rpc ( & tx , value ) //TODO handle error
if err == nil {
logger . V ( 1 ) . Info ( "Redirecting user to tx page" )
http . Redirect ( w , r , "/tx/" + value , 302 )
return
}
}
{
data := map [ string ] interface { } { }
fill_common_info ( data , true )
if err = all_templates . ExecuteTemplate ( w , "notfound_page" , data ) ; err == nil {
return
}
}
exit_error :
show_page ( w , 0 )
return
}
func fill_common_info ( data map [ string ] interface { } , extra_data bool ) error {
var info rpc . GetInfo_Result
data [ "title" ] = "DERO HE BlockChain Explorer(v1)"
data [ "servertime" ] = time . Now ( ) . UTC ( ) . Format ( "2006-01-02 15:04:05" )
if ! extra_data {
return nil
}
// collect all the data afresh, execute rpc to service
if err := rpc_client . Call ( "DERO.GetInfo" , nil , & info ) ; err != nil {
return err
}
//fmt.Printf("get info %+v", info)
data [ "Network_Difficulty" ] = info . Difficulty
data [ "hash_rate" ] = fmt . Sprintf ( "%.03f" , float32 ( info . Difficulty / 1000000 ) / float32 ( info . Target ) )
data [ "txpool_size" ] = info . Tx_pool_size
data [ "testnet" ] = info . Testnet
data [ "network" ] = info . Network
data [ "fee_per_kb" ] = float64 ( info . Dynamic_fee_per_kb ) / 1000000000000
data [ "median_block_size" ] = fmt . Sprintf ( "%.02f" , float32 ( info . Median_Block_Size ) / 1024 )
data [ "total_supply" ] = info . Total_Supply
data [ "averageblocktime50" ] = info . AverageBlockTime50
return nil
}
// fill all the tx pool info as per requested
func fill_tx_pool_info ( data map [ string ] interface { } , max_count int ) error {
var err error
var txs [ ] txinfo
var txpool rpc . GetTxPool_Result
data [ "mempool" ] = txs // initialize with empty data
if err = rpc_client . Call ( "DERO.GetTxPool" , nil , & txpool ) ; err != nil {
return fmt . Errorf ( "gettxpool rpc failed" )
}
for i := range txpool . Tx_list {
var info txinfo
err := load_tx_from_rpc ( & info , txpool . Tx_list [ i ] ) //TODO handle error
if err != nil {
continue
}
txs = append ( txs , info )
if len ( txs ) >= max_count {
break
}
}
data [ "mempool" ] = txs
return nil
}
// BytesViewer bytes viewer
type BytesViewer [ ] byte
// String returns view in hexadecimal
func ( b BytesViewer ) String ( ) string {
if len ( b ) == 0 {
return "invlaid string"
}
const head = `
| Address | Hex | Text |
| -- -- -- - : | : -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- | : -- -- -- -- -- -- -- - |
`
const row = 16
result := make ( [ ] byte , 0 , len ( head ) / 2 * ( len ( b ) / 16 + 3 ) )
result = append ( result , head ... )
for i := 0 ; i < len ( b ) ; i += row {
result = append ( result , "| " ... )
result = append ( result , fmt . Sprintf ( "%08x" , i ) ... )
result = append ( result , " | " ... )
k := i + row
more := 0
if k >= len ( b ) {
more = k - len ( b )
k = len ( b )
}
for j := i ; j != k ; j ++ {
if b [ j ] < 16 {
result = append ( result , '0' )
}
result = strconv . AppendUint ( result , uint64 ( b [ j ] ) , 16 )
result = append ( result , ' ' )
}
for j := 0 ; j != more ; j ++ {
result = append ( result , " " ... )
}
result = append ( result , "| " ... )
buf := bytes . Map ( func ( r rune ) rune {
if unicode . IsSpace ( r ) {
return ' '
}
return r
} , b [ i : k ] )
result = append ( result , buf ... )
for j := 0 ; j != more ; j ++ {
result = append ( result , ' ' )
}
result = append ( result , " |\n" ... )
}
return * ( * string ) ( unsafe . Pointer ( & result ) )
}