2021-12-04 16:42:11 +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 walletapi
// this file needs serious improvements but have extremely limited time
/ * this file handles communication with the daemon
* this includes receiving output information
*
* *
* /
//import "io"
//import "os"
import "fmt"
import "time"
import "sync"
import "bytes"
import "math/big"
//import "bufio"
import "strings"
import "context"
//import "runtime"
//import "compress/gzip"
import "encoding/hex"
import "runtime/debug"
//import "github.com/vmihailenco/msgpack"
//import "github.com/gorilla/websocket"
//import "github.com/mafredri/cdp/rpcc"
import "github.com/deroproject/derohe/rpc"
import "github.com/deroproject/derohe/block"
import "github.com/deroproject/derohe/config"
import "github.com/deroproject/derohe/globals"
import "github.com/deroproject/derohe/cryptography/crypto"
import "github.com/deroproject/derohe/errormsg"
import "github.com/deroproject/derohe/transaction"
import "github.com/deroproject/derohe/cryptography/bn256"
import "github.com/creachadair/jrpc2"
// this global variable should be within wallet structure
var Connected bool = false
var daemon_height int64
var daemon_topoheight int64
// return daemon height
func Get_Daemon_Height ( ) int64 {
return daemon_height
}
// return topoheight of daemon
func Get_Daemon_TopoHeight ( ) int64 {
return daemon_topoheight
}
var simulator bool // turns on simulator, which has 0 fees
// there should be no global variables, so multiple wallets can run at the same time with different assset
var endpoint string
var output_lock sync . Mutex
var NotifyNewBlock * sync . Cond = sync . NewCond ( & sync . Mutex { } )
var NotifyHeightChange * sync . Cond = sync . NewCond ( & sync . Mutex { } )
// this function will wait n goroutines to wait for new block
func WaitNewBlock ( ) {
NotifyNewBlock . L . Lock ( )
NotifyNewBlock . Wait ( )
NotifyNewBlock . L . Unlock ( )
}
// this function will wait n goroutines to wait till height changes
func WaitNewHeightBlock ( ) {
NotifyHeightChange . L . Lock ( )
NotifyHeightChange . Wait ( )
NotifyHeightChange . L . Unlock ( )
}
func Notify_broadcaster ( req * jrpc2 . Request ) {
timer . Reset ( timeout ) // connection is alive
switch req . Method ( ) {
case "Block" :
NotifyNewBlock . L . Lock ( )
NotifyNewBlock . Broadcast ( )
NotifyNewBlock . L . Unlock ( )
case "Height" :
NotifyHeightChange . L . Lock ( )
NotifyHeightChange . Broadcast ( )
NotifyHeightChange . L . Unlock ( )
go test_connectivity ( )
case "MiniBlock" : // we can skip this
default :
logger . V ( 1 ) . Info ( "Notification received" , "method" , req . Method ( ) )
}
}
var Daemon_Endpoint string
var Daemon_Endpoint_Active string
func get_daemon_address ( ) string {
if globals . Arguments [ "--remote" ] == true && globals . IsMainnet ( ) {
Daemon_Endpoint_Active = config . REMOTE_DAEMON
}
// if user provided endpoint has error, use default
if Daemon_Endpoint_Active == "" {
Daemon_Endpoint_Active = "127.0.0.1:" + fmt . Sprintf ( "%d" , config . Mainnet . RPC_Default_Port )
if ! globals . IsMainnet ( ) {
Daemon_Endpoint_Active = "127.0.0.1:" + fmt . Sprintf ( "%d" , config . Testnet . RPC_Default_Port )
}
}
if globals . Arguments [ "--daemon-address" ] != nil {
Daemon_Endpoint_Active = globals . Arguments [ "--daemon-address" ] . ( string )
}
return Daemon_Endpoint_Active
}
// tests connectivity when connectivity to daemon
func test_connectivity ( ) ( err error ) {
var result string
// Issue a call with a response.
if err = rpc_client . Call ( "DERO.Echo" , [ ] string { "hello" , "world" } , & result ) ; err != nil {
logger . V ( 1 ) . Error ( err , "DERO.Echo Call failed:" )
Connected = false
return
}
//fmt.Println(result)
var info rpc . GetInfo_Result
// Issue a call with a response.
if err = rpc_client . Call ( "DERO.GetInfo" , nil , & info ) ; err != nil {
logger . V ( 1 ) . Error ( err , "DERO.GetInfo Call failed:" )
Connected = false
return
}
// detect whether both are in different modes
// daemon is in testnet and wallet in mainnet or
// daemon
if info . Testnet != ! globals . IsMainnet ( ) {
err = fmt . Errorf ( "Mainnet/TestNet is different between wallet/daemon.Please run daemon/wallet without --testnet" )
logger . Error ( err , "Mainnet/Testnet mismatch" )
2022-02-06 07:06:32 +00:00
fmt . Printf ( "Mainnet/Testnet mismatch\n" )
2021-12-04 16:42:11 +00:00
return
}
if strings . ToLower ( info . Network ) == "simulator" {
simulator = true
}
daemon_height = info . Height
daemon_topoheight = info . TopoHeight
logger . Info ( "successfully connected to daemon" )
return nil
}
// triggers syncing with wallet every 5 seconds
func ( w * Wallet_Memory ) sync_loop ( ) {
for {
select {
case <- w . Quit :
break
default :
}
2022-02-06 07:06:32 +00:00
if IsDaemonOnline ( ) && test_connectivity ( ) != nil {
time . Sleep ( timeout ) // wait 5 seconds
continue
}
var zerohash crypto . Hash
if len ( w . account . EntriesNative ) == 0 {
err := w . Sync_Wallet_Memory_With_Daemon ( )
logger . V ( 1 ) . Error ( err , "wallet syncing err" , err )
} else {
for k := range w . account . EntriesNative {
err := w . Sync_Wallet_Memory_With_Daemon_internal ( k )
if k == zerohash && err != nil {
logger . V ( 1 ) . Error ( err , "wallet syncing err" , err )
}
}
}
2021-12-04 16:42:11 +00:00
time . Sleep ( timeout ) // wait 5 seconds
}
}
func ( cli * Client ) Call ( method string , params interface { } , result interface { } ) error {
return cli . RPC . CallResult ( context . Background ( ) , method , params , result )
}
// returns whether wallet was online some time ago
func ( w * Wallet_Memory ) IsDaemonOnlineCached ( ) bool {
return Connected
}
// currently process url with compatibility for older ip address
func buildurl ( endpoint string ) string {
if strings . IndexAny ( endpoint , "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ" ) >= 0 { // url is already complete
return strings . TrimSuffix ( endpoint , "/" )
} else {
return "http://" + endpoint
}
}
// 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 IsDaemonOnline ( ) bool {
if rpc_client . WS == nil || rpc_client . RPC == nil {
return false
}
return true
}
// sync the wallet with daemon, this is instantaneous and can be done with a single call
// we have now the apis to avoid polling
func ( w * Wallet_Memory ) Sync_Wallet_Memory_With_Daemon_internal ( scid crypto . Hash ) ( err error ) {
if ! IsDaemonOnline ( ) {
daemon_height = 0
daemon_topoheight = 0
return fmt . Errorf ( "Daemon is offline" )
} else {
//w.random_ring_members()
//rlog.Debugf("wallet topo height %d daemon online topo height %d\n", w.account.TopoHeight, w.Daemon_TopoHeight)
previous := w . getEncryptedBalanceresult ( scid ) . Data
if _ , _ , _ , e , err := w . GetEncryptedBalanceAtTopoHeight ( scid , - 1 , w . GetAddress ( ) . String ( ) ) ; err == nil {
//fmt.Printf("data '%s' previous '%s' scid %s\n",w.account.Balance_Result[scid].Data , previous,scid)
if w . getEncryptedBalanceresult ( scid ) . Data != previous {
b := w . DecodeEncryptedBalanceNow ( e ) // try to decode balance
if scid . IsZero ( ) {
w . account . Balance_Mature = b
}
w . account . Balance [ scid ] = b
w . SyncHistory ( scid ) // also update statement
} else {
}
w . save_if_disk ( ) // save wallet
} else {
return err
}
}
return
}
func ( w * Wallet_Memory ) Sync_Wallet_Memory_With_Daemon ( ) ( err error ) {
var scid crypto . Hash
return w . Sync_Wallet_Memory_With_Daemon_internal ( scid )
}
func ( w * Wallet_Memory ) NameToAddress ( name string ) ( addr string , err error ) {
if name == "" {
return addr , fmt . Errorf ( "empty string is not a valid address" )
}
if ! IsDaemonOnline ( ) {
err = fmt . Errorf ( "offline or not connected. cannot translate name to address" )
return
}
var result rpc . NameToAddress_Result
if err = rpc_client . Call ( "DERO.NameToAddress" , rpc . NameToAddress_Params { Name : name , TopoHeight : - 1 } , & result ) ; err != nil {
return
}
if result . Status == "OK" {
addr = result . Address
return
} else {
err = fmt . Errorf ( "Err %s" , result . Status )
return
}
}
// this is as simple as it gets
// single threaded communication to relay TX to daemon
// if this is successful, then daemon is in control
func ( w * Wallet_Memory ) SendTransaction ( tx * transaction . Transaction ) ( err error ) {
if tx == nil {
return fmt . Errorf ( "Can not send nil transaction" )
}
if ! IsDaemonOnline ( ) {
return fmt . Errorf ( "offline or not connected. cannot send transaction." )
}
params := rpc . SendRawTransaction_Params { Tx_as_hex : hex . EncodeToString ( tx . Serialize ( ) ) }
var result rpc . SendRawTransaction_Result
if err := rpc_client . Call ( "DERO.SendRawTransaction" , params , & result ) ; err != nil {
return err
}
if result . Status == "OK" {
return nil
} else {
err = fmt . Errorf ( "Err %s" , result . Status )
}
return
}
// decode encrypted balance now
// it may take a long time, its currently sing threaded, need to parallelize
func ( w * Wallet_Memory ) DecodeEncryptedBalanceNow ( el * crypto . ElGamal ) uint64 {
balance_point := new ( bn256 . G1 ) . Add ( el . Left , new ( bn256 . G1 ) . Neg ( new ( bn256 . G1 ) . ScalarMult ( el . Right , w . account . Keys . Secret . BigInt ( ) ) ) )
return Balance_lookup_table . Lookup ( balance_point , w . account . Balance_Mature )
}
func ( w * Wallet_Memory ) GetSelfEncryptedBalanceAtTopoHeight ( scid crypto . Hash , topoheight int64 ) ( r rpc . GetEncryptedBalance_Result , err error ) {
defer func ( ) {
if r := recover ( ) ; r != nil {
logger . V ( 1 ) . Error ( nil , "Recovered while GetSelfEncryptedBalanceAtTopoHeight" , "r" , r , "stack" , debug . Stack ( ) )
err = fmt . Errorf ( "Recovered while GetSelfEncryptedBalanceAtTopoHeight r %s stack %s" , r , string ( debug . Stack ( ) ) )
}
} ( )
err = rpc_client . Call ( "DERO.GetEncryptedBalance" , rpc . GetEncryptedBalance_Params { SCID : scid , Address : w . GetAddress ( ) . String ( ) , TopoHeight : topoheight } , & r )
return
}
// this is as simple as it gets
// single threaded communication gets whether the the key image is spent in pool or in blockchain
// this can leak informtion which keyimage belongs to us
// TODO in order to stop privacy leaks we must guess this information somehow on client side itself
// maybe the server can broadcast a bloomfilter or something else from the mempool keyimages
//
func ( w * Wallet_Memory ) GetEncryptedBalanceAtTopoHeight ( scid crypto . Hash , topoheight int64 , accountaddr string ) ( bits int , lastused uint64 , blid crypto . Hash , e * crypto . ElGamal , err error ) {
defer func ( ) {
if r := recover ( ) ; r != nil {
logger . V ( 1 ) . Error ( nil , "Recovered while GetEncryptedBalanceAtTopoHeight" , "r" , r , "stack" , debug . Stack ( ) )
err = fmt . Errorf ( "Recovered while GetEncryptedBalanceAtTopoHeight r %s stack %s" , r , debug . Stack ( ) )
}
} ( )
if ! w . GetMode ( ) { // if wallet is in offline mode , we cannot do anything
err = fmt . Errorf ( "wallet is in offline mode" )
return
}
if ! IsDaemonOnline ( ) {
err = fmt . Errorf ( "offline or not connected" )
return
}
//var params rpc.GetEncryptedBalance_Params
var result rpc . GetEncryptedBalance_Result
// Issue a call with a response.
if err = rpc_client . Call ( "DERO.GetEncryptedBalance" , rpc . GetEncryptedBalance_Params { SCID : scid , Address : accountaddr , TopoHeight : topoheight } , & result ) ; err != nil {
logger . Error ( err , "DERO.GetEncryptedBalance Call failed:" )
if strings . Contains ( strings . ToLower ( err . Error ( ) ) , strings . ToLower ( errormsg . ErrAccountUnregistered . Error ( ) ) ) && accountaddr == w . GetAddress ( ) . String ( ) && scid . IsZero ( ) {
w . Error = errormsg . ErrAccountUnregistered
//fmt.Printf("setting unregisterd111 err %s scid %s topoheight %d\n",err,scid, topoheight)
//fmt.Printf("debug stack %s\n",debug.Stack())
return
}
// all SCID users are considered registered and their balance is assumed zero
if ! scid . IsZero ( ) {
if strings . Contains ( strings . ToLower ( err . Error ( ) ) , strings . ToLower ( errormsg . ErrAccountUnregistered . Error ( ) ) ) {
if addr , err1 := rpc . NewAddress ( accountaddr ) ; err1 != nil {
err = err1
return
} else {
e = crypto . ConstructElGamal ( addr . PublicKey . G1 ( ) , crypto . ElGamal_BASE_G ) // init zero balance
bits = 0 // since this is an SC we can
err = nil
return
}
}
}
return
}
// fmt.Printf("GetEncryptedBalance result %+v\n", result)
if scid . IsZero ( ) && accountaddr == w . GetAddress ( ) . String ( ) {
if result . Status == errormsg . ErrAccountUnregistered . Error ( ) {
w . Error = errormsg . ErrAccountUnregistered
w . account . Registered = false
} else {
w . account . Registered = true
}
}
// fmt.Printf("status '%s' err '%s' %+v %+v \n", result.Status , w.Error , result.Status == errormsg.ErrAccountUnregistered.Error() , accountaddr == w.account.GetAddress().String())
if scid . IsZero ( ) && result . Status == errormsg . ErrAccountUnregistered . Error ( ) {
err = fmt . Errorf ( "%s" , result . Status )
return
}
if topoheight == - 1 {
daemon_height = result . DHeight
daemon_topoheight = result . DTopoheight
w . Merkle_Balance_TreeHash = result . DMerkle_Balance_TreeHash
}
if topoheight == - 1 && accountaddr == w . GetAddress ( ) . String ( ) {
//fmt.Printf("topoheight %d accountaddr '%s' waddress '%s'\n ",topoheight,accountaddr,w.GetAddress().String())
w . setEncryptedBalanceresult ( scid , result )
w . account . TopoHeight = result . Topoheight
}
if scid . IsZero ( ) && result . Status != "OK" {
err = fmt . Errorf ( "%s" , result . Status )
return
}
hexdecoded , err := hex . DecodeString ( result . Data )
if err != nil {
return
}
if accountaddr == w . GetAddress ( ) . String ( ) && scid . IsZero ( ) {
w . Error = nil
}
var nb crypto . NonceBalance
nb . Unmarshal ( hexdecoded )
return result . Bits , nb . NonceHeight , result . BlockHash , nb . Balance , nil
}
func ( w * Wallet_Memory ) DecodeEncryptedBalance_Memory ( el * crypto . ElGamal , hint uint64 ) ( balance uint64 ) {
var balance_point bn256 . G1
balance_point . Add ( el . Left , new ( bn256 . G1 ) . Neg ( new ( bn256 . G1 ) . ScalarMult ( el . Right , w . account . Keys . Secret . BigInt ( ) ) ) )
return Balance_lookup_table . Lookup ( & balance_point , hint )
}
func ( w * Wallet_Memory ) GetDecryptedBalanceAtTopoHeight ( scid crypto . Hash , topoheight int64 , accountaddr string ) ( balance uint64 , noncetopo uint64 , err error ) {
_ , noncetopo , _ , encrypted_balance , err := w . GetEncryptedBalanceAtTopoHeight ( scid , topoheight , accountaddr )
if err != nil {
return 0 , 0 , err
}
2022-02-06 07:06:32 +00:00
if w . account . EntriesNative != nil {
if _ , ok := w . account . EntriesNative [ scid ] ; ! ok { //if we could obtain something, try tracking
w . account . EntriesNative [ scid ] = [ ] rpc . Entry { }
}
}
2021-12-04 16:42:11 +00:00
return w . DecodeEncryptedBalance_Memory ( encrypted_balance , 0 ) , noncetopo , nil
}
// sync history of wallet from blockchain
func ( w * Wallet_Memory ) Random_ring_members ( scid crypto . Hash ) ( alist [ ] string ) {
var result rpc . GetRandomAddress_Result
//fmt.Printf("getting ring members %s %s\n",scid.String(), debug.Stack())
// Issue a call with a response.
if err := rpc_client . Call ( "DERO.GetRandomAddress" , rpc . GetRandomAddress_Params { SCID : scid } , & result ) ; err != nil {
logger . V ( 1 ) . Error ( err , "DERO.GetRandomAddress Call failed:" )
return
}
//fmt.Printf("getting ring members %d\n",len(result.Address))
for _ , k := range result . Address {
if k != w . GetAddress ( ) . String ( ) {
alist = append ( alist , k )
}
}
//fmt.Printf("got ring members %d\n",len(result.Address))
return
}
// sync history of wallet from blockchain
func ( w * Wallet_Memory ) SyncHistory ( scid crypto . Hash ) ( balance uint64 ) {
defer func ( ) {
if r := recover ( ) ; r != nil {
logger . V ( 1 ) . Error ( nil , "Recovered while syncing connecting" , "r" , r , "stack" , debug . Stack ( ) )
}
} ( )
if w . getEncryptedBalanceresult ( scid ) . Registration < 0 { // unregistered so skip
return
}
last_topo_height := int64 ( - 1 )
//fmt.Printf("finding sync point ( Registration point %d)\n", w.getEncryptedBalanceresult(scid).Registration)
entries := w . account . EntriesNative [ scid ]
// we need to find a sync point, to minimize traffic
for i := len ( entries ) - 1 ; i >= 0 ; {
// below condition will trigger if chain got pruned on server
if w . getEncryptedBalanceresult ( scid ) . Registration >= entries [ i ] . TopoHeight { // keep old history if chain got pruned
break
}
if last_topo_height == entries [ i ] . TopoHeight {
i --
} else {
last_topo_height = entries [ i ] . TopoHeight
var result rpc . GetBlockHeaderByHeight_Result
// Issue a call with a response.
if err := rpc_client . Call ( "DERO.GetBlockHeaderByTopoHeight" , rpc . GetBlockHeaderByTopoHeight_Params { TopoHeight : uint64 ( entries [ i ] . TopoHeight ) } , & result ) ; err != nil {
logger . V ( 1 ) . Error ( err , "DERO.GetBlockHeaderByTopoHeight Call failed:" )
return 0
}
if entries [ i ] . BlockHash != result . Block_Header . Hash {
if i >= 1 && last_topo_height == entries [ i - 1 ] . TopoHeight { // skipping any entries withing same block
for ; i >= 1 ; i -- {
if last_topo_height == entries [ i - 1 ] . TopoHeight {
entries = entries [ : i ]
w . account . EntriesNative [ scid ] = entries
}
}
}
}
if i == 0 {
w . account . EntriesNative [ scid ] = entries [ : 0 ] // discard all entries
break
}
// we have found a matching block hash, start syncing from here
if result . Status == "OK" && result . Block_Header . Hash == entries [ i ] . BlockHash {
w . synchistory_internal ( scid , entries [ i ] . TopoHeight + 1 , w . getEncryptedBalanceresult ( scid ) . Topoheight )
return
}
}
}
//fmt.Printf("syncing loop using Registration %+v\n",w.getEncryptedBalanceresult(scid).Registration)
// if we reached here, means we should sync from scratch
w . synchistory_internal ( scid , w . getEncryptedBalanceresult ( scid ) . Registration , w . getEncryptedBalanceresult ( scid ) . Topoheight )
//if w.account.Registration >= 0 {
// err :=
// err = w.synchistory_internal(w.account.Registration,6)
// }
// fmt.Printf("syncing err %s\n",err)
// fmt.Printf("entries %+v\n", w.account.Entries)
return 0
}
// sync history
func ( w * Wallet_Memory ) synchistory_internal ( scid crypto . Hash , start_topo , end_topo int64 ) error {
var err error
var start_balance_e * crypto . ElGamal
if start_topo == w . getEncryptedBalanceresult ( scid ) . Registration {
start_balance_e = crypto . ConstructElGamal ( w . account . Keys . Public . G1 ( ) , crypto . ElGamal_BASE_G )
} else {
_ , _ , _ , start_balance_e , err = w . GetEncryptedBalanceAtTopoHeight ( scid , start_topo , w . GetAddress ( ) . String ( ) )
if err != nil {
return err
}
}
_ , _ , _ , end_balance_e , err := w . GetEncryptedBalanceAtTopoHeight ( scid , end_topo , w . GetAddress ( ) . String ( ) )
if err != nil {
return err
}
return w . synchistory_internal_binary_search ( 0 , scid , start_topo , start_balance_e , end_topo , end_balance_e )
}
func ( w * Wallet_Memory ) synchistory_internal_binary_search ( level int , scid crypto . Hash , start_topo int64 , start_balance_e * crypto . ElGamal , end_topo int64 , end_balance_e * crypto . ElGamal ) error {
var err error
//defer fmt.Printf("end %d start %d err %s\n", end_topo, start_topo, err)
if end_topo < 0 {
return fmt . Errorf ( "done" )
}
/ * if bytes . Compare ( start_balance_e . Serialize ( ) , end_balance_e . Serialize ( ) ) == 0 {
return nil
}
* /
defer globals . Recover ( 0 )
//for start_topo <= end_topo{
{
median := ( start_topo + end_topo ) / 2
//fmt.Printf("%slevel %d low %d high %d median %d\n", strings.Repeat("\t", level), level, start_topo, end_topo, median)
if start_topo == median {
if err = w . synchistory_block ( scid , start_topo ) ; err != nil {
return err
}
}
if end_topo - start_topo <= 1 {
err = w . synchistory_block ( scid , end_topo )
return err
}
_ , _ , _ , median_balance_e , err := w . GetEncryptedBalanceAtTopoHeight ( scid , median , w . GetAddress ( ) . String ( ) )
if err != nil {
fmt . Printf ( "getting block err %s\n" , err )
return err
}
// check if there is a change in lower section, if yes process more
//fmt.Printf("%slevel %d checking lower\n", strings.Repeat("\t", level), level)
if start_topo == w . getEncryptedBalanceresult ( scid ) . Registration || bytes . Compare ( start_balance_e . Serialize ( ) , median_balance_e . Serialize ( ) ) != 0 {
err = w . synchistory_internal_binary_search ( level + 1 , scid , start_topo , start_balance_e , median , median_balance_e )
if err != nil {
return err
}
}
// check if there is a change in higher section, if yes process more
//fmt.Printf("%slevel %d checking higher\n", strings.Repeat("\t", level), level)
if bytes . Compare ( median_balance_e . Serialize ( ) , end_balance_e . Serialize ( ) ) != 0 {
err = w . synchistory_internal_binary_search ( level + 1 , scid , median , median_balance_e , end_topo , end_balance_e )
if err != nil {
return err
}
}
}
return nil
}
// extract history from a single block
// first get a block, then get all the txs
// Todo we should expose an API to get all txs which have the specific address as ring member
// for a particular block
// for the entire chain
func ( w * Wallet_Memory ) synchistory_block ( scid crypto . Hash , topo int64 ) ( err error ) {
var local_entries [ ] rpc . Entry
compressed_address := w . account . Keys . Public . EncodeCompressed ( )
var previous_balance_e , current_balance_e * crypto . ElGamal
var previous_balance , current_balance , total_sent , total_received uint64
if topo <= 0 || w . getEncryptedBalanceresult ( scid ) . Registration == topo {
previous_balance_e = crypto . ConstructElGamal ( w . account . Keys . Public . G1 ( ) , crypto . ElGamal_BASE_G )
} else {
_ , _ , _ , previous_balance_e , err = w . GetEncryptedBalanceAtTopoHeight ( scid , topo - 1 , w . GetAddress ( ) . String ( ) )
if err != nil {
return err
}
}
_ , _ , _ , current_balance_e , err = w . GetEncryptedBalanceAtTopoHeight ( scid , topo , w . GetAddress ( ) . String ( ) )
if err != nil {
return err
}
var bl block . Block
var bresult rpc . GetBlock_Result
if err = rpc_client . Call ( "DERO.GetBlock" , rpc . GetBlock_Params { Height : uint64 ( topo ) } , & bresult ) ; err != nil {
return fmt . Errorf ( "getblock rpc failed" )
}
if bresult . Block_Header . SideBlock && w . getEncryptedBalanceresult ( scid ) . Registration != topo {
return nil
}
EWData := fmt . Sprintf ( "%x" , current_balance_e . Serialize ( ) )
previous_balance = w . DecodeEncryptedBalance_Memory ( previous_balance_e , 0 )
current_balance = w . DecodeEncryptedBalance_Memory ( current_balance_e , 0 )
// we can skip some check if both balances are equal ( means we are ring members in this block)
// this check will also fail if we total spend == total receivein the block
// currently it is not implmented, and we bruteforce everything
block_bin , err := hex . DecodeString ( bresult . Blob )
if err != nil {
return err
}
if err = bl . Deserialize ( block_bin ) ; err != nil {
return err
}
if ! bresult . Block_Header . SideBlock && len ( bl . Tx_hashes ) >= 1 {
for i := range bl . Tx_hashes {
var tx transaction . Transaction
var tx_params rpc . GetTransaction_Params
var tx_result rpc . GetTransaction_Result
tx_params . Tx_Hashes = append ( tx_params . Tx_Hashes , bl . Tx_hashes [ i ] . String ( ) )
//fmt.Printf("Requesting tx data %s\n", bl.Tx_hashes[i].String())
if err = rpc_client . Call ( "DERO.GetTransaction" , tx_params , & tx_result ) ; err != nil {
return fmt . Errorf ( "gettransa rpc failed %s" , err )
}
tx_bin , err := hex . DecodeString ( tx_result . Txs_as_hex [ 0 ] )
if err != nil {
return err
}
if err = tx . Deserialize ( tx_bin ) ; err != nil {
logger . V ( 1 ) . Error ( err , "Error deserialing tx" , "txid" , bl . Tx_hashes [ i ] . String ( ) , "incoming bytes" , tx_result . Txs_as_hex [ 0 ] )
continue
}
if tx . TransactionType == transaction . REGISTRATION {
continue
}
// if daemon was syncing/or disk corrupption, it may not give data, so skip
if len ( tx_result . Txs ) == 0 {
return fmt . Errorf ( "Daemon did not expandd tx %s" , bl . Tx_hashes [ i ] . String ( ) )
}
// since balance might change with tx, we track within tx using this
previous_balance_e_tx := new ( crypto . ElGamal ) . Deserialize ( previous_balance_e . Serialize ( ) )
for t := range tx . Payloads {
if len ( tx_result . Txs ) == 0 {
continue
}
if len ( tx_result . Txs [ 0 ] . Ring ) == 0 {
continue
}
_ = int ( tx . Payloads [ t ] . Statement . RingSize )
_ = tx_result . Txs [ 0 ]
_ = tx_result . Txs [ 0 ] . Ring
_ = tx_result . Txs [ 0 ] . Ring [ t ]
if int ( tx . Payloads [ t ] . Statement . RingSize ) != len ( tx_result . Txs [ 0 ] . Ring [ t ] ) {
logger . V ( 1 ) . Error ( fmt . Errorf ( "missing ring members" ) , "missing ring members" , "txid" , bl . Tx_hashes [ i ] . String ( ) , "expected" , int ( tx . Payloads [ t ] . Statement . RingSize ) , "got" , len ( tx_result . Txs [ t ] . Ring ) )
continue
}
if tx . Payloads [ t ] . SCID != scid { // skip tokens in which we are not interested
continue
}
previous_balance = w . DecodeEncryptedBalanceNow ( previous_balance_e_tx )
for j := 0 ; j < int ( tx . Payloads [ t ] . Statement . RingSize ) ; j ++ { // first fill in all the ring members
var addr * rpc . Address
if addr , err = rpc . NewAddress ( tx_result . Txs [ 0 ] . Ring [ t ] [ j ] ) ; err != nil {
panic ( err )
}
var buf [ 33 ] byte
copy ( buf [ : ] , addr . PublicKey . EncodeCompressed ( ) )
tx . Payloads [ t ] . Statement . Publickeylist_compressed = append ( tx . Payloads [ t ] . Statement . Publickeylist_compressed , buf )
}
for j := 0 ; j < int ( tx . Payloads [ t ] . Statement . RingSize ) ; j ++ { // check whether statement has public key
// check whether our address is a ring member if yes, process it as ours
if bytes . Compare ( compressed_address , tx . Payloads [ t ] . Statement . Publickeylist_compressed [ j ] [ : ] ) == 0 {
// this tx contains us either as a ring member, or sender or receiver, so add all members as ring members for future
// keep collecting ring members to make things exponentially complex
for k := range tx . Payloads [ t ] . Statement . Publickeylist_compressed {
var p bn256 . G1
if err = p . DecodeCompressed ( tx . Payloads [ t ] . Statement . Publickeylist_compressed [ k ] [ : ] ) ; err != nil {
fmt . Printf ( "key could not be decompressed" )
} else {
tx . Payloads [ t ] . Statement . Publickeylist = append ( tx . Payloads [ t ] . Statement . Publickeylist , & p )
}
}
/ * for k := range tx . Statement . Publickeylist_compressed {
if j != k {
ringmember := address . NewAddressFromKeys ( ( * crypto . Point ) ( tx . Statement . Publickeylist [ k ] ) )
ringmember . Mainnet = w . GetNetwork ( )
w . account . RingMembers [ ringmember . String ( ) ] = 1
}
} * /
changes := crypto . ConstructElGamal ( tx . Payloads [ t ] . Statement . C [ j ] , tx . Payloads [ t ] . Statement . D )
changed_balance_e := previous_balance_e_tx . Add ( changes )
previous_balance_e_tx = new ( crypto . ElGamal ) . Deserialize ( changed_balance_e . Serialize ( ) )
changed_balance := w . DecodeEncryptedBalance_Memory ( changed_balance_e , previous_balance )
//fmt.Printf("%d changed_balance %d previous_balance %d len payload %d\n", t, changed_balance, previous_balance, len(tx.Payloads[t].RPCPayload))
entry := rpc . Entry { Height : bl . Height , Pos : t , TopoHeight : topo , BlockHash : bl . GetHash ( ) . String ( ) , TransactionPos : i , TXID : tx . GetHash ( ) . String ( ) , Time : time . UnixMilli ( int64 ( bl . Timestamp ) ) , Fees : tx . Fees ( ) }
entry . EWData = EWData
ring_member := false
switch {
case previous_balance == changed_balance : //ring member/* handle 0 value tx but fees is deducted */
//fmt.Printf("Anon Ring Member in TX %s\n", bl.Tx_hashes[i].String())
ring_member = true
case previous_balance > changed_balance : // we generated this tx
entry . Burn = tx . Payloads [ t ] . BurnValue
entry . Amount = previous_balance - changed_balance - ( tx . Payloads [ t ] . Statement . Fees )
entry . Fees = tx . Payloads [ t ] . Statement . Fees
entry . Status = 1 // mark it as spend
total_sent += entry . Amount + entry . Fees // burn is in amount
rinputs := append ( [ ] byte { } , tx . Payloads [ t ] . Statement . Roothash [ : ] ... )
for l := range tx . Payloads [ t ] . Statement . Publickeylist_compressed {
rinputs = append ( rinputs , tx . Payloads [ t ] . Statement . Publickeylist_compressed [ l ] [ : ] ... )
}
rencrypted := new ( bn256 . G1 ) . ScalarMult ( crypto . HashToPoint ( crypto . HashtoNumber ( append ( [ ] byte ( crypto . PROTOCOL_CONSTANT ) , rinputs ... ) ) ) , w . account . Keys . Secret . BigInt ( ) )
r := crypto . ReducedHash ( rencrypted . EncodeCompressed ( ) )
//fmt.Printf("t %d r calculated %s value amount %d burn %d\n", t, r.Text(16), entry.Amount,entry.Burn)
parity := tx . Payloads [ t ] . Proof . Parity ( )
// lets separate ring members
for k := range tx . Payloads [ t ] . Statement . C {
if ( k % 2 == 0 ) == parity { // ignore senders self,this condition is well thought out and works good enough
continue
}
// we need to brute force receiver in this case, if amount sent is zero
if entry . Amount == entry . Burn && tx . Payloads [ t ] . RPCType == transaction . ENCRYPTED_DEFAULT_PAYLOAD_CBOR {
shared_key := crypto . GenerateSharedSecret ( r , tx . Payloads [ t ] . Statement . Publickeylist [ k ] )
var data_copy [ ] byte
data_copy = append ( data_copy , tx . Payloads [ t ] . RPCPayload ... )
crypto . EncryptDecryptUserData ( crypto . Keccak256 ( shared_key [ : ] , tx . Payloads [ t ] . Statement . Publickeylist [ k ] . EncodeCompressed ( ) ) , data_copy )
var args rpc . Arguments
if err = args . UnmarshalBinary ( data_copy [ 1 : ] ) ; err != nil {
//fmt.Printf("k %d len(data_copy) %d err %s data_copy %x\n",k, len(data_copy),err, data_copy)
continue
}
// we have found one which could be decoded, fall through
}
var x bn256 . G1
x . ScalarMult ( crypto . G , new ( big . Int ) . SetInt64 ( int64 ( entry . Amount - entry . Burn ) ) )
x . Add ( new ( bn256 . G1 ) . Set ( & x ) , new ( bn256 . G1 ) . ScalarMult ( tx . Payloads [ t ] . Statement . Publickeylist [ k ] , r ) )
if x . String ( ) == tx . Payloads [ t ] . Statement . C [ k ] . String ( ) {
var x bn256 . G1
x . ScalarMult ( crypto . G , new ( big . Int ) . SetInt64 ( int64 ( 0 - entry . Amount ) ) )
x . Add ( new ( bn256 . G1 ) . Set ( & x ) , tx . Payloads [ t ] . Statement . C [ k ] ) // get the blinder
blinder := & x
shared_key := crypto . GenerateSharedSecret ( r , tx . Payloads [ t ] . Statement . Publickeylist [ k ] )
// proof is blinder + amount transferred, it will recover the encrypted rpc payload also
// enable sender side proofs
proof := rpc . NewAddressFromKeys ( ( * crypto . Point ) ( blinder ) )
proof . Proof = true
proof . Arguments = rpc . Arguments { { Name : "H" , DataType : rpc . DataHash , Value : crypto . Hash ( shared_key ) } , { Name : rpc . RPC_VALUE_TRANSFER , DataType : rpc . DataUint64 , Value : uint64 ( entry . Amount - entry . Burn ) } }
entry . Proof = proof . String ( )
entry . PayloadType = tx . Payloads [ t ] . RPCType
switch tx . Payloads [ t ] . RPCType {
case transaction . ENCRYPTED_DEFAULT_PAYLOAD_CBOR :
crypto . EncryptDecryptUserData ( crypto . Keccak256 ( shared_key [ : ] , tx . Payloads [ t ] . Statement . Publickeylist [ k ] . EncodeCompressed ( ) ) , tx . Payloads [ t ] . RPCPayload )
//fmt.Printf("decoded plaintext payload t %d %x\n",t,tx.Payloads[t].RPCPayload)
//sender_idx := uint(tx.Payloads[t].RPCPayload[0])
addr := rpc . NewAddressFromKeys ( ( * crypto . Point ) ( w . account . Keys . Public . G1 ( ) ) )
addr . Mainnet = w . GetNetwork ( )
entry . Sender = addr . String ( )
entry . Payload = append ( entry . Payload , tx . Payloads [ t ] . RPCPayload [ 1 : ] ... )
entry . Data = append ( entry . Data , tx . Payloads [ t ] . RPCPayload [ : ] ... )
args , _ := entry . ProcessPayload ( )
_ = args
// fmt.Printf("data received %s idx %d arguments %s\n", string(entry.Payload), sender_idx, args)
default :
entry . PayloadError = fmt . Sprintf ( "unknown payload type %d" , tx . Payloads [ t ] . RPCType )
entry . Payload = tx . Payloads [ t ] . RPCPayload
}
//paymentID := binary.BigEndian.Uint64(payment_id_encrypted_bytes[:]) // get decrypted payment id
addr := rpc . NewAddressFromKeys ( ( * crypto . Point ) ( tx . Payloads [ t ] . Statement . Publickeylist [ k ] ) )
addr . Mainnet = w . GetNetwork ( )
entry . Destination = addr . String ( )
//fmt.Printf("t %d height %d Sent funds to %s entry %+v\n",t, tx.Height, addr.String(), entry)
break
}
}
case previous_balance < changed_balance : // someone sentus this amount
entry . Amount = changed_balance - previous_balance
entry . Incoming = true
// we should decode the payment id
var x bn256 . G1
x . ScalarMult ( crypto . G , new ( big . Int ) . SetInt64 ( 0 - int64 ( entry . Amount ) ) ) // decrease amounts
x . Add ( new ( bn256 . G1 ) . Set ( & x ) , tx . Payloads [ t ] . Statement . C [ j ] ) // get the blinder
blinder := & x
shared_key := crypto . GenerateSharedSecret ( w . account . Keys . Secret . BigInt ( ) , tx . Payloads [ t ] . Statement . D )
// enable receiver side proofs
proof := rpc . NewAddressFromKeys ( ( * crypto . Point ) ( blinder ) )
proof . Proof = true
proof . Arguments = rpc . Arguments { { Name : "H" , DataType : rpc . DataHash , Value : crypto . Hash ( shared_key ) } , { Name : rpc . RPC_VALUE_TRANSFER , DataType : rpc . DataUint64 , Value : uint64 ( entry . Amount ) } }
entry . Proof = proof . String ( )
entry . PayloadType = tx . Payloads [ t ] . RPCType
switch tx . Payloads [ t ] . RPCType {
case 0 :
//fmt.Printf("decoding encrypted payload %x\n",tx.Payloads[t].RPCPayload)
crypto . EncryptDecryptUserData ( crypto . Keccak256 ( shared_key [ : ] , w . GetAddress ( ) . PublicKey . EncodeCompressed ( ) ) , tx . Payloads [ t ] . RPCPayload )
//fmt.Printf("decoded plaintext payload %x\n",tx.Payloads[t].RPCPayload)
sender_idx := uint ( tx . Payloads [ t ] . RPCPayload [ 0 ] )
if sender_idx <= uint ( tx . Payloads [ t ] . Statement . RingSize ) {
addr := rpc . NewAddressFromKeys ( ( * crypto . Point ) ( tx . Payloads [ t ] . Statement . Publickeylist [ sender_idx ] ) )
addr . Mainnet = w . GetNetwork ( )
entry . Sender = addr . String ( )
}
entry . Payload = append ( entry . Payload , tx . Payloads [ t ] . RPCPayload [ 1 : ] ... )
entry . Data = append ( entry . Data , tx . Payloads [ t ] . RPCPayload [ : ] ... )
args , _ := entry . ProcessPayload ( )
_ = args
// fmt.Printf("data received %s idx %d arguments %s\n", string(entry.Payload), sender_idx, args)
default :
entry . PayloadError = fmt . Sprintf ( "unknown payload type %d" , tx . Payloads [ t ] . RPCType )
entry . Payload = tx . Payloads [ t ] . RPCPayload
}
//fmt.Printf("Received %s amount in TX(%d) %s payment id %x Src_ID %s data %s\n", globals.FormatMoney(changed_balance-previous_balance), tx.Height, bl.Tx_hashes[i].String(), entry.PaymentID, tx.Src_ID, tx.Data)
//fmt.Printf("Received amount in TX(%d) %s payment id %x Src_ID %s data %s\n", tx.Height, bl.Tx_hashes[i].String(), entry.PaymentID, tx.SrcID, tx.Data)
total_received += ( changed_balance - previous_balance )
}
if ! ring_member { // do not book keep ring members
local_entries = append ( local_entries , entry )
}
//break // this tx has been processed so skip it
}
}
}
//fmt.Printf("block %d %+v\n", topo, tx_result)
}
}
previous_balance = w . DecodeEncryptedBalance_Memory ( previous_balance_e , 0 )
coinbase_reward := current_balance - ( previous_balance - total_sent + total_received )
//fmt.Printf("ht %d coinbase_reward %d curent balance %d previous_balance %d sent %d received %d\n", bl.Height, coinbase_reward, current_balance, previous_balance, total_sent, total_received)
if bytes . Compare ( compressed_address , bl . Miner_TX . MinerAddress [ : ] ) == 0 || coinbase_reward > 0 { // wallet user has minted a block
entry := rpc . Entry { Height : bl . Height , TopoHeight : topo , BlockHash : bl . GetHash ( ) . String ( ) , TransactionPos : - 1 , Time : time . UnixMilli ( int64 ( bl . Timestamp ) ) }
entry . EWData = EWData
entry . Amount = current_balance - ( previous_balance - total_sent + total_received )
entry . Coinbase = true
local_entries = append ( [ ] rpc . Entry { entry } , local_entries ... )
//fmt.Printf("Coinbase Reward %s for block %d\n", globals.FormatMoney(current_balance-(previous_balance-total_sent+total_received)), topo)
}
for _ , e := range local_entries {
w . InsertReplace ( scid , e )
}
if len ( local_entries ) >= 1 {
w . save_if_disk ( ) // save wallet()
// w.db.Sync()
}
return nil
}