aboutsummaryrefslogblamecommitdiffstats
path: root/xeth/xeth.go
blob: 690db51350baca631e27b62efc660c70803b8e0e (plain) (tree)
1
2
3
4
5
6
7
8
9
10
                    

            


                       
             
                  

              
 
                                                  
                                                


                                                    
                                               
                                                      
                                                
                                             
                                               
                                                 
 
 


                                                   

                                                                    
 




                                             
                                          

                             
                          
                           


                                 
                                 
                                  



                           
                        

 



















                                                                         
                  


                                           
                                        
                             
                               
 
                         








                                           

 


                                                              
 

                                                                                 
 



                                                          
                      


                                                     
                                                     
                                                          

                                                                        
                                         

                                                             
         
                            
                                               
         
                                                                   


                                     


                   
































                                                                                  


                                                                       

















                                                                      
                                                       
                                                           




                                                    






                                                      
                                                            
 
                                                      
                                       

                                                 
                              

 
                                                               
                                       




                                                 
                                                                        
                                                               





                                                          





                                                                        

 







                                                              
                                               
                                   
                                                   


                                                             
                                                   




                  
                                       



                                                           
                                                              

                               

 
                                   


                                   
                                   
                                  

 
                                                   
                                       
                                    

                                             

                                    
                                     
         
                                  

 
                                      


                                     
                                     
                                                     
                               

 
                                                        
                                
 
                                         

 
                                                              

                                                                        
                                            

 
                                                 


                                                            
                                                 
                                                         

 
                                                 
                                                                 

 
                                                   
                                                            

 
                                                      
                                                                  



                         
                                           

 


















































































































                                                                        




                                   
                                                   



                                            
                                                                                             









                                               
                                              
                                                       
 
                                          

 
                                                
                              


                             
                                                              

 
                                                 
                              


                             
                                                        

 
                                                            
                                                                      






                                                   
                                              
         
                                           
 
 
                                                                                                       

                                                                        





                                                                               
         







                                                 
                                                 
                                                                    
 
                                                                                            
                                     

 
                                                                                                           
             
                                       
                                       


                                                               
                                       


                                     


                                      










                                                                                                        
                                            
                                                            
                          
 
                                                          

                              
                                                         

                              
                                    




                                                            
 
                                     
                                                                     
         
                                           
 
 









                                                                                                
                                                






                                  
















                                                                
































                                                     
// eXtended ETHereum
package xeth

import (
    "bytes"
    "encoding/json"
    "fmt"
    "math/big"
    "sync"
    "time"

    "github.com/ethereum/go-ethereum/accounts"
    "github.com/ethereum/go-ethereum/common"
    "github.com/ethereum/go-ethereum/core"
    "github.com/ethereum/go-ethereum/core/types"
    "github.com/ethereum/go-ethereum/crypto"
    "github.com/ethereum/go-ethereum/event"
    "github.com/ethereum/go-ethereum/event/filter"
    "github.com/ethereum/go-ethereum/logger"
    "github.com/ethereum/go-ethereum/p2p"
    "github.com/ethereum/go-ethereum/state"
    "github.com/ethereum/go-ethereum/whisper"
)

var (
    pipelogger       = logger.NewLogger("XETH")
    filterTickerTime = 5 * time.Minute
    defaultGasPrice  = big.NewInt(10000000000000) //150000000000
    defaultGas       = big.NewInt(90000)          //500000
)

// to resolve the import cycle
type Backend interface {
    BlockProcessor() *core.BlockProcessor
    ChainManager() *core.ChainManager
    AccountManager() *accounts.Manager
    TxPool() *core.TxPool
    PeerCount() int
    IsListening() bool
    Peers() []*p2p.Peer
    BlockDb() common.Database
    StateDb() common.Database
    ExtraDb() common.Database
    EventMux() *event.TypeMux
    Whisper() *whisper.Whisper

    IsMining() bool
    StartMining() error
    StopMining()
    Version() string
}

// Frontend should be implemented by users of XEth. Its methods are
// called whenever XEth makes a decision that requires user input.
type Frontend interface {
    // UnlockAccount is called when a transaction needs to be signed
    // but the key corresponding to the transaction's sender is
    // locked.
    //
    // It should unlock the account with the given address and return
    // true if unlocking succeeded.
    UnlockAccount(address []byte) bool

    // This is called for all transactions inititated through
    // Transact. It should prompt the user to confirm the transaction
    // and return true if the transaction was acknowledged.
    //
    // ConfirmTransaction is not used for Call transactions
    // because they cannot change any state.
    ConfirmTransaction(tx *types.Transaction) bool
}

type XEth struct {
    eth            Backend
    blockProcessor *core.BlockProcessor
    chainManager   *core.ChainManager
    accountManager *accounts.Manager
    state          *State
    whisper        *Whisper

    frontend Frontend

    quit          chan struct{}
    filterManager *filter.FilterManager

    logMut sync.RWMutex
    logs   map[int]*logFilter

    messagesMut sync.RWMutex
    messages    map[int]*whisperFilter
}

// dummyFrontend is a non-interactive frontend that allows all
// transactions but cannot not unlock any keys.
type dummyFrontend struct{}

func (dummyFrontend) UnlockAccount([]byte) bool                  { return false }
func (dummyFrontend) ConfirmTransaction(*types.Transaction) bool { return true }

// New creates an XEth that uses the given frontend.
// If a nil Frontend is provided, a default frontend which
// confirms all transactions will be used.
func New(eth Backend, frontend Frontend) *XEth {
    xeth := &XEth{
        eth:            eth,
        blockProcessor: eth.BlockProcessor(),
        chainManager:   eth.ChainManager(),
        accountManager: eth.AccountManager(),
        whisper:        NewWhisper(eth.Whisper()),
        quit:           make(chan struct{}),
        filterManager:  filter.NewFilterManager(eth.EventMux()),
        frontend:       frontend,
        logs:           make(map[int]*logFilter),
        messages:       make(map[int]*whisperFilter),
    }
    if frontend == nil {
        xeth.frontend = dummyFrontend{}
    }
    xeth.state = NewState(xeth, xeth.chainManager.TransState())
    go xeth.start()
    go xeth.filterManager.Start()

    return xeth
}

func (self *XEth) start() {
    timer := time.NewTicker(2 * time.Second)
done:
    for {
        select {
        case <-timer.C:
            self.logMut.Lock()
            self.messagesMut.Lock()
            for id, filter := range self.logs {
                if time.Since(filter.timeout) > filterTickerTime {
                    self.filterManager.UninstallFilter(id)
                    delete(self.logs, id)
                }
            }

            for id, filter := range self.messages {
                if time.Since(filter.timeout) > filterTickerTime {
                    self.Whisper().Unwatch(id)
                    delete(self.messages, id)
                }
            }
            self.messagesMut.Unlock()
            self.logMut.Unlock()
        case <-self.quit:
            break done
        }
    }
}

func (self *XEth) stop() {
    close(self.quit)
}

func (self *XEth) DefaultGas() *big.Int      { return defaultGas }
func (self *XEth) DefaultGasPrice() *big.Int { return defaultGasPrice }

func (self *XEth) AtStateNum(num int64) *XEth {
    chain := self.Backend().ChainManager()
    var block *types.Block

    if num < 0 {
        num = chain.CurrentBlock().Number().Int64() + num + 1
    }
    block = chain.GetBlockByNumber(uint64(num))

    var st *state.StateDB
    if block != nil {
        st = state.New(block.Root(), self.Backend().StateDb())
    } else {
        st = chain.State()
    }
    return self.WithState(st)
}

func (self *XEth) Backend() Backend { return self.eth }
func (self *XEth) WithState(statedb *state.StateDB) *XEth {
    xeth := &XEth{
        eth:            self.eth,
        blockProcessor: self.blockProcessor,
        chainManager:   self.chainManager,
        whisper:        self.whisper,
    }

    xeth.state = NewState(xeth, statedb)
    return xeth
}
func (self *XEth) State() *State { return self.state }

func (self *XEth) Whisper() *Whisper { return self.whisper }

func (self *XEth) BlockByHash(strHash string) *Block {
    hash := common.FromHex(strHash)
    block := self.chainManager.GetBlock(hash)

    return NewBlock(block)
}

func (self *XEth) EthBlockByHash(strHash string) *types.Block {
    hash := common.FromHex(strHash)
    block := self.chainManager.GetBlock(hash)

    return block
}

func (self *XEth) EthTransactionByHash(hash string) *types.Transaction {
    data, _ := self.eth.ExtraDb().Get(common.FromHex(hash))
    if len(data) != 0 {
        return types.NewTransactionFromBytes(data)
    }
    return nil
}

func (self *XEth) BlockByNumber(num int64) *Block {
    if num == -1 {
        return NewBlock(self.chainManager.CurrentBlock())
    }

    return NewBlock(self.chainManager.GetBlockByNumber(uint64(num)))
}

func (self *XEth) EthBlockByNumber(num int64) *types.Block {
    if num == -1 {
        return self.chainManager.CurrentBlock()
    }

    return self.chainManager.GetBlockByNumber(uint64(num))
}

func (self *XEth) Block(v interface{}) *Block {
    if n, ok := v.(int32); ok {
        return self.BlockByNumber(int64(n))
    } else if str, ok := v.(string); ok {
        return self.BlockByHash(str)
    } else if f, ok := v.(float64); ok { // Don't ask ...
        return self.BlockByNumber(int64(f))
    }

    return nil
}

func (self *XEth) Accounts() []string {
    // TODO: check err?
    accounts, _ := self.eth.AccountManager().Accounts()
    accountAddresses := make([]string, len(accounts))
    for i, ac := range accounts {
        accountAddresses[i] = common.ToHex(ac.Address)
    }
    return accountAddresses
}

func (self *XEth) PeerCount() int {
    return self.eth.PeerCount()
}

func (self *XEth) IsMining() bool {
    return self.eth.IsMining()
}

func (self *XEth) SetMining(shouldmine bool) bool {
    ismining := self.eth.IsMining()
    if shouldmine && !ismining {
        err := self.eth.StartMining()
        return err == nil
    }
    if ismining && !shouldmine {
        self.eth.StopMining()
    }
    return self.eth.IsMining()
}

func (self *XEth) IsListening() bool {
    return self.eth.IsListening()
}

func (self *XEth) Coinbase() string {
    cb, _ := self.eth.AccountManager().Coinbase()
    return common.ToHex(cb)
}

func (self *XEth) NumberToHuman(balance string) string {
    b := common.Big(balance)

    return common.CurrencyToString(b)
}

func (self *XEth) StorageAt(addr, storageAddr string) string {
    storage := self.State().SafeGet(addr).StorageString(storageAddr)

    return common.ToHex(storage.Bytes())
}

func (self *XEth) BalanceAt(addr string) string {
    return self.State().SafeGet(addr).Balance().String()
}

func (self *XEth) TxCountAt(address string) int {
    return int(self.State().SafeGet(address).Nonce())
}

func (self *XEth) CodeAt(address string) string {
    return common.ToHex(self.State().SafeGet(address).Code())
}

func (self *XEth) IsContract(address string) bool {
    return len(self.State().SafeGet(address).Code()) > 0
}

func (self *XEth) SecretToAddress(key string) string {
    pair, err := crypto.NewKeyPairFromSec(common.FromHex(key))
    if err != nil {
        return ""
    }

    return common.ToHex(pair.Address())
}

func (self *XEth) RegisterFilter(args *core.FilterOptions) int {
    var id int
    filter := core.NewFilter(self.Backend())
    filter.SetOptions(args)
    filter.LogsCallback = func(logs state.Logs) {
        self.logMut.Lock()
        defer self.logMut.Unlock()

        self.logs[id].add(logs...)
    }
    id = self.filterManager.InstallFilter(filter)
    self.logs[id] = &logFilter{timeout: time.Now()}

    return id
}

func (self *XEth) UninstallFilter(id int) bool {
    if _, ok := self.logs[id]; ok {
        delete(self.logs, id)
        self.filterManager.UninstallFilter(id)
        return true
    }

    return false
}

func (self *XEth) NewFilterString(word string) int {
    var id int
    filter := core.NewFilter(self.Backend())

    callback := func(block *types.Block, logs state.Logs) {
        self.logMut.Lock()
        defer self.logMut.Unlock()

        for _, log := range logs {
            self.logs[id].add(log)
        }
        self.logs[id].add(&state.StateLog{})
    }

    switch word {
    case "pending":
        filter.PendingCallback = callback
    case "latest":
        filter.BlockCallback = callback
    }

    id = self.filterManager.InstallFilter(filter)
    self.logs[id] = &logFilter{timeout: time.Now()}

    return id
}

func (self *XEth) FilterChanged(id int) state.Logs {
    self.logMut.Lock()
    defer self.logMut.Unlock()

    if self.logs[id] != nil {
        return self.logs[id].get()
    }

    return nil
}

func (self *XEth) Logs(id int) state.Logs {
    self.logMut.Lock()
    defer self.logMut.Unlock()

    filter := self.filterManager.GetFilter(id)
    if filter != nil {
        return filter.Find()
    }

    return nil
}

func (self *XEth) AllLogs(args *core.FilterOptions) state.Logs {
    filter := core.NewFilter(self.Backend())
    filter.SetOptions(args)

    return filter.Find()
}

func (p *XEth) NewWhisperFilter(opts *Options) int {
    var id int
    opts.Fn = func(msg WhisperMessage) {
        p.messagesMut.Lock()
        defer p.messagesMut.Unlock()
        p.messages[id].add(msg) // = append(p.messages[id], msg)
    }
    id = p.Whisper().Watch(opts)
    p.messages[id] = &whisperFilter{timeout: time.Now()}
    return id
}

func (p *XEth) UninstallWhisperFilter(id int) bool {
    if _, ok := p.messages[id]; ok {
        delete(p.messages, id)
        return true
    }

    return false
}

func (self *XEth) MessagesChanged(id int) []WhisperMessage {
    self.messagesMut.Lock()
    defer self.messagesMut.Unlock()

    if self.messages[id] != nil {
        return self.messages[id].get()
    }

    return nil
}

type KeyVal struct {
    Key   string `json:"key"`
    Value string `json:"value"`
}

func (self *XEth) EachStorage(addr string) string {
    var values []KeyVal
    object := self.State().SafeGet(addr)
    it := object.Trie().Iterator()
    for it.Next() {
        values = append(values, KeyVal{common.ToHex(it.Key), common.ToHex(it.Value)})
    }

    valuesJson, err := json.Marshal(values)
    if err != nil {
        return ""
    }

    return string(valuesJson)
}

func (self *XEth) ToAscii(str string) string {
    padded := common.RightPadBytes([]byte(str), 32)

    return "0x" + common.ToHex(padded)
}

func (self *XEth) FromAscii(str string) string {
    if common.IsHex(str) {
        str = str[2:]
    }

    return string(bytes.Trim(common.FromHex(str), "\x00"))
}

func (self *XEth) FromNumber(str string) string {
    if common.IsHex(str) {
        str = str[2:]
    }

    return common.BigD(common.FromHex(str)).String()
}

func (self *XEth) PushTx(encodedTx string) (string, error) {
    tx := types.NewTransactionFromBytes(common.FromHex(encodedTx))
    err := self.eth.TxPool().Add(tx)
    if err != nil {
        return "", err
    }

    if tx.To() == nil {
        addr := core.AddressFromMessage(tx)
        return common.ToHex(addr), nil
    }
    return common.ToHex(tx.Hash()), nil
}

func (self *XEth) Call(fromStr, toStr, valueStr, gasStr, gasPriceStr, dataStr string) (string, error) {
    statedb := self.State().State() //self.chainManager.TransState()
    msg := callmsg{
        from:     statedb.GetOrNewStateObject(common.FromHex(fromStr)),
        to:       common.FromHex(toStr),
        gas:      common.Big(gasStr),
        gasPrice: common.Big(gasPriceStr),
        value:    common.Big(valueStr),
        data:     common.FromHex(dataStr),
    }
    if msg.gas.Cmp(big.NewInt(0)) == 0 {
        msg.gas = defaultGas
    }

    if msg.gasPrice.Cmp(big.NewInt(0)) == 0 {
        msg.gasPrice = defaultGasPrice
    }

    block := self.chainManager.CurrentBlock()
    vmenv := core.NewEnv(statedb, self.chainManager, msg, block)

    res, err := vmenv.Call(msg.from, msg.to, msg.data, msg.gas, msg.gasPrice, msg.value)
    return common.ToHex(res), err
}

func (self *XEth) Transact(fromStr, toStr, valueStr, gasStr, gasPriceStr, codeStr string) (string, error) {
    var (
        from             []byte
        to               []byte
        value            = common.NewValue(valueStr)
        gas              = common.NewValue(gasStr)
        price            = common.NewValue(gasPriceStr)
        data             []byte
        contractCreation bool
    )

    from = common.FromHex(fromStr)
    data = common.FromHex(codeStr)
    to = common.FromHex(toStr)
    if len(to) == 0 {
        contractCreation = true
    }

    var tx *types.Transaction
    if contractCreation {
        tx = types.NewContractCreationTx(value.BigInt(), gas.BigInt(), price.BigInt(), data)
    } else {
        tx = types.NewTransactionMessage(to, value.BigInt(), gas.BigInt(), price.BigInt(), data)
    }

    state := self.chainManager.TxState()
    nonce := state.NewNonce(from) //state.GetNonce(from)
    tx.SetNonce(nonce)

    if err := self.sign(tx, from, false); err != nil {
        return "", err
    }
    if err := self.eth.TxPool().Add(tx); err != nil {
        return "", err
    }
    //state.IncrementNonce(from)

    if contractCreation {
        addr := core.AddressFromMessage(tx)
        pipelogger.Infof("Contract addr %x\n", addr)
    }

    if types.IsContractAddr(to) {
        return common.ToHex(core.AddressFromMessage(tx)), nil
    }
    return common.ToHex(tx.Hash()), nil
}

func (self *XEth) sign(tx *types.Transaction, from []byte, didUnlock bool) error {
    sig, err := self.accountManager.Sign(accounts.Account{Address: from}, tx.Hash())
    if err == accounts.ErrLocked {
        if didUnlock {
            return fmt.Errorf("sender account still locked after successful unlock")
        }
        if !self.frontend.UnlockAccount(from) {
            return fmt.Errorf("could not unlock sender account")
        }
        // retry signing, the account should now be unlocked.
        return self.sign(tx, from, true)
    } else if err != nil {
        return err
    }
    tx.SetSignatureValues(sig)
    return nil
}

// callmsg is the message type used for call transations.
type callmsg struct {
    from          *state.StateObject
    to            []byte
    gas, gasPrice *big.Int
    value         *big.Int
    data          []byte
}

// accessor boilerplate to implement core.Message
func (m callmsg) From() []byte       { return m.from.Address() }
func (m callmsg) Nonce() uint64      { return m.from.Nonce() }
func (m callmsg) To() []byte         { return m.to }
func (m callmsg) GasPrice() *big.Int { return m.gasPrice }
func (m callmsg) Gas() *big.Int      { return m.gas }
func (m callmsg) Value() *big.Int    { return m.value }
func (m callmsg) Data() []byte       { return m.data }

type whisperFilter struct {
    messages []WhisperMessage
    timeout  time.Time
    id       int
}

func (w *whisperFilter) add(msgs ...WhisperMessage) {
    w.messages = append(w.messages, msgs...)
}
func (w *whisperFilter) get() []WhisperMessage {
    w.timeout = time.Now()
    tmp := w.messages
    w.messages = nil
    return tmp
}

type logFilter struct {
    logs    state.Logs
    timeout time.Time
    id      int
}

func (l *logFilter) add(logs ...state.Log) {
    l.logs = append(l.logs, logs...)
}

func (l *logFilter) get() state.Logs {
    l.timeout = time.Now()
    tmp := l.logs
    l.logs = nil
    return tmp
}