xeth.go 7.76 KB
Newer Older
1 2 3 4 5
package xeth

/*
 * eXtended ETHereum
 */
obscuren's avatar
obscuren committed
6

obscuren's avatar
obscuren committed
7 8 9 10
import (
	"bytes"
	"encoding/json"

11
	"github.com/ethereum/go-ethereum/accounts"
obscuren's avatar
obscuren committed
12 13 14 15
	"github.com/ethereum/go-ethereum/core"
	"github.com/ethereum/go-ethereum/core/types"
	"github.com/ethereum/go-ethereum/crypto"
	"github.com/ethereum/go-ethereum/ethutil"
16
	"github.com/ethereum/go-ethereum/event"
obscuren's avatar
obscuren committed
17
	"github.com/ethereum/go-ethereum/logger"
18
	"github.com/ethereum/go-ethereum/miner"
19
	"github.com/ethereum/go-ethereum/p2p"
obscuren's avatar
obscuren committed
20
	"github.com/ethereum/go-ethereum/state"
21
	"github.com/ethereum/go-ethereum/whisper"
obscuren's avatar
obscuren committed
22
)
obscuren's avatar
obscuren committed
23

24
var pipelogger = logger.NewLogger("XETH")
obscuren's avatar
obscuren committed
25 26 27 28 29

// to resolve the import cycle
type Backend interface {
	BlockProcessor() *core.BlockProcessor
	ChainManager() *core.ChainManager
30
	AccountManager() *accounts.Manager
31 32
	TxPool() *core.TxPool
	PeerCount() int
obscuren's avatar
obscuren committed
33
	IsListening() bool
34
	Peers() []*p2p.Peer
obscuren's avatar
obscuren committed
35
	Db() ethutil.Database
36
	EventMux() *event.TypeMux
37
	Whisper() *whisper.Whisper
38
	Miner() *miner.Miner
obscuren's avatar
obscuren committed
39 40
}

41
type XEth struct {
obscuren's avatar
obscuren committed
42 43 44
	eth            Backend
	blockProcessor *core.BlockProcessor
	chainManager   *core.ChainManager
45
	accountManager *accounts.Manager
46
	state          *State
47
	whisper        *Whisper
48
	miner          *miner.Miner
obscuren's avatar
obscuren committed
49 50
}

51 52
func New(eth Backend) *XEth {
	xeth := &XEth{
obscuren's avatar
obscuren committed
53 54 55
		eth:            eth,
		blockProcessor: eth.BlockProcessor(),
		chainManager:   eth.ChainManager(),
56
		accountManager: eth.AccountManager(),
57
		whisper:        NewWhisper(eth.Whisper()),
58
		miner:          eth.Miner(),
obscuren's avatar
obscuren committed
59
	}
obscuren's avatar
obscuren committed
60
	xeth.state = NewState(xeth, xeth.chainManager.TransState())
obscuren's avatar
obscuren committed
61 62 63 64

	return xeth
}

obscuren's avatar
obscuren committed
65 66 67 68 69 70 71 72 73 74 75 76 77 78 79
func (self *XEth) Backend() Backend { return self.eth }
func (self *XEth) UseState(statedb *state.StateDB) *XEth {
	xeth := &XEth{
		eth:            self.eth,
		blockProcessor: self.blockProcessor,
		chainManager:   self.chainManager,
		whisper:        self.whisper,
		miner:          self.miner,
	}

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

80 81
func (self *XEth) Whisper() *Whisper   { return self.whisper }
func (self *XEth) Miner() *miner.Miner { return self.miner }
obscuren's avatar
obscuren committed
82

83
func (self *XEth) BlockByHash(strHash string) *Block {
obscuren's avatar
obscuren committed
84 85 86
	hash := fromHex(strHash)
	block := self.chainManager.GetBlock(hash)

87
	return NewBlock(block)
obscuren's avatar
obscuren committed
88 89
}

90
func (self *XEth) BlockByNumber(num int32) *Block {
obscuren's avatar
obscuren committed
91
	if num == -1 {
92
		return NewBlock(self.chainManager.CurrentBlock())
obscuren's avatar
obscuren committed
93 94
	}

95
	return NewBlock(self.chainManager.GetBlockByNumber(uint64(num)))
obscuren's avatar
obscuren committed
96 97
}

98
func (self *XEth) Block(v interface{}) *Block {
obscuren's avatar
obscuren committed
99 100 101 102 103 104 105 106 107 108 109
	if n, ok := v.(int32); ok {
		return self.BlockByNumber(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(int32(f))
	}

	return nil
}

110
func (self *XEth) Accounts() []string {
111 112 113 114 115 116 117
	// TODO: check err?
	accounts, _ := self.eth.AccountManager().Accounts()
	accountAddresses := make([]string, len(accounts))
	for i, ac := range accounts {
		accountAddresses[i] = toHex(ac.Address)
	}
	return accountAddresses
obscuren's avatar
obscuren committed
118 119
}

120
func (self *XEth) PeerCount() int {
obscuren's avatar
obscuren committed
121 122 123
	return self.eth.PeerCount()
}

124
func (self *XEth) IsMining() bool {
125
	return self.miner.Mining()
obscuren's avatar
obscuren committed
126 127
}

128 129 130 131 132 133 134 135 136 137 138
func (self *XEth) SetMining(shouldmine bool) bool {
	ismining := self.miner.Mining()
	if shouldmine && !ismining {
		self.miner.Start()
	}
	if ismining && !shouldmine {
		self.miner.Stop()
	}
	return self.miner.Mining()
}

139
func (self *XEth) IsListening() bool {
obscuren's avatar
obscuren committed
140 141 142
	return self.eth.IsListening()
}

143
func (self *XEth) Coinbase() string {
144 145
	cb, _ := self.eth.AccountManager().Coinbase()
	return toHex(cb)
obscuren's avatar
obscuren committed
146 147
}

148
func (self *XEth) NumberToHuman(balance string) string {
obscuren's avatar
obscuren committed
149 150 151 152 153
	b := ethutil.Big(balance)

	return ethutil.CurrencyToString(b)
}

154
func (self *XEth) StorageAt(addr, storageAddr string) string {
obscuren's avatar
obscuren committed
155 156 157 158 159
	storage := self.State().SafeGet(addr).StorageString(storageAddr)

	return toHex(storage.Bytes())
}

160
func (self *XEth) BalanceAt(addr string) string {
obscuren's avatar
obscuren committed
161 162 163
	return self.State().SafeGet(addr).Balance().String()
}

164
func (self *XEth) TxCountAt(address string) int {
165
	return int(self.State().SafeGet(address).Nonce())
obscuren's avatar
obscuren committed
166 167
}

168
func (self *XEth) CodeAt(address string) string {
169
	return toHex(self.State().SafeGet(address).Code())
obscuren's avatar
obscuren committed
170 171
}

172
func (self *XEth) IsContract(address string) bool {
173
	return len(self.State().SafeGet(address).Code()) > 0
obscuren's avatar
obscuren committed
174 175
}

176
func (self *XEth) SecretToAddress(key string) string {
obscuren's avatar
obscuren committed
177 178 179 180 181 182 183 184
	pair, err := crypto.NewKeyPairFromSec(fromHex(key))
	if err != nil {
		return ""
	}

	return toHex(pair.Address())
}

185
func (self *XEth) Execute(addr, value, gas, price, data string) (string, error) {
obscuren's avatar
obscuren committed
186 187 188 189 190 191 192 193
	return "", nil
}

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

194
func (self *XEth) EachStorage(addr string) string {
obscuren's avatar
obscuren committed
195 196 197 198 199 200 201 202 203 204 205 206 207 208 209
	var values []KeyVal
	object := self.State().SafeGet(addr)
	it := object.Trie().Iterator()
	for it.Next() {
		values = append(values, KeyVal{toHex(it.Key), toHex(it.Value)})
	}

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

	return string(valuesJson)
}

210
func (self *XEth) ToAscii(str string) string {
obscuren's avatar
obscuren committed
211 212 213 214 215
	padded := ethutil.RightPadBytes([]byte(str), 32)

	return "0x" + toHex(padded)
}

216
func (self *XEth) FromAscii(str string) string {
obscuren's avatar
obscuren committed
217 218 219 220 221 222 223
	if ethutil.IsHex(str) {
		str = str[2:]
	}

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

224
func (self *XEth) FromNumber(str string) string {
obscuren's avatar
obscuren committed
225 226 227 228 229 230 231
	if ethutil.IsHex(str) {
		str = str[2:]
	}

	return ethutil.BigD(fromHex(str)).String()
}

232
func (self *XEth) PushTx(encodedTx string) (string, error) {
obscuren's avatar
obscuren committed
233 234 235 236 237 238 239 240 241 242 243 244
	tx := types.NewTransactionFromBytes(fromHex(encodedTx))
	err := self.eth.TxPool().Add(tx)
	if err != nil {
		return "", err
	}

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

246 247 248 249 250 251 252 253
func (self *XEth) Call(toStr, valueStr, gasStr, gasPriceStr, dataStr string) (string, error) {
	if len(gasStr) == 0 {
		gasStr = "100000"
	}
	if len(gasPriceStr) == 0 {
		gasPriceStr = "1"
	}

254 255 256 257
	acct, err := self.accountManager.Default()
	if err != nil {
		return "", err
	}
258
	var (
obscuren's avatar
obscuren committed
259
		statedb = self.State().State() //self.chainManager.TransState()
260
		from    = statedb.GetOrNewStateObject(acct.Address)
261 262 263 264 265 266
		block   = self.chainManager.CurrentBlock()
		to      = statedb.GetOrNewStateObject(fromHex(toStr))
		data    = fromHex(dataStr)
		gas     = ethutil.Big(gasStr)
		price   = ethutil.Big(gasPriceStr)
		value   = ethutil.Big(valueStr)
267 268
	)

269
	msg := types.NewTransactionMessage(fromHex(toStr), value, gas, price, data)
270 271 272 273 274
	sig, err := self.accountManager.Sign(acct, msg.Hash())
	if err != nil {
		return "", err
	}
	msg.SetSignatureValues(sig)
275 276
	vmenv := core.NewEnv(statedb, self.chainManager, msg, block)
	res, err := vmenv.Call(from, to.Address(), data, gas, price, value)
277 278 279 280 281 282 283
	if err != nil {
		return "", err
	}

	return toHex(res), nil
}

284 285
func (self *XEth) Transact(fromStr, toStr, valueStr, gasStr, gasPriceStr, codeStr string) (string, error) {

286
	var (
287
		from             []byte
288 289 290 291 292 293 294 295
		to               []byte
		value            = ethutil.NewValue(valueStr)
		gas              = ethutil.NewValue(gasStr)
		price            = ethutil.NewValue(gasPriceStr)
		data             []byte
		contractCreation bool
	)

296
	from = fromHex(fromStr)
297 298 299 300 301 302 303 304 305 306 307 308 309
	data = fromHex(codeStr)
	to = 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)
	}

310 311
	state := self.chainManager.TransState()
	nonce := state.GetNonce(from)
312 313

	tx.SetNonce(nonce)
314
	sig, err := self.accountManager.Sign(accounts.Account{Address: from}, tx.Hash())
315 316 317 318
	if err != nil {
		return "", err
	}
	tx.SetSignatureValues(sig)
319

320
	err = self.eth.TxPool().Add(tx)
321 322 323
	if err != nil {
		return "", err
	}
324 325 326 327 328 329
	state.SetNonce(from, nonce+1)

	if contractCreation {
		addr := core.AddressFromMessage(tx)
		pipelogger.Infof("Contract addr %x\n", addr)
	}
330 331 332 333 334 335 336

	if types.IsContractAddr(to) {
		return toHex(core.AddressFromMessage(tx)), nil
	}

	return toHex(tx.Hash()), nil
}