2014-10-23 16:48:53 +03:00
|
|
|
// Copyright (c) 2013-2014, Jeffrey Wilcke. All rights reserved.
|
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2.1 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this library; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
|
|
|
// MA 02110-1301 USA
|
|
|
|
|
2014-07-02 01:13:50 +03:00
|
|
|
package main
|
2014-02-25 11:54:15 +02:00
|
|
|
|
|
|
|
import (
|
2014-08-15 14:45:34 +03:00
|
|
|
"bytes"
|
2014-08-14 01:18:37 +03:00
|
|
|
"fmt"
|
2014-08-07 17:35:47 +03:00
|
|
|
"path"
|
2014-08-15 14:45:34 +03:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
2014-08-07 17:35:47 +03:00
|
|
|
|
2014-10-23 16:01:27 +03:00
|
|
|
"github.com/ethereum/go-ethereum"
|
2014-10-31 11:59:17 +02:00
|
|
|
"github.com/ethereum/go-ethereum/chain"
|
2014-10-31 13:37:43 +02:00
|
|
|
"github.com/ethereum/go-ethereum/crypto"
|
2014-10-23 16:01:27 +03:00
|
|
|
"github.com/ethereum/go-ethereum/ethutil"
|
2014-08-12 13:16:21 +03:00
|
|
|
"github.com/ethereum/go-ethereum/javascript"
|
2014-11-07 13:18:48 +02:00
|
|
|
"github.com/ethereum/go-ethereum/miner"
|
2014-10-31 15:43:14 +02:00
|
|
|
"github.com/ethereum/go-ethereum/state"
|
2014-10-23 16:01:27 +03:00
|
|
|
"github.com/ethereum/go-ethereum/ui/qt"
|
2014-10-31 15:30:08 +02:00
|
|
|
"github.com/ethereum/go-ethereum/xeth"
|
2014-08-15 14:27:43 +03:00
|
|
|
"gopkg.in/qml.v1"
|
2014-02-25 11:54:15 +02:00
|
|
|
)
|
|
|
|
|
2014-04-12 07:12:10 +03:00
|
|
|
type memAddr struct {
|
|
|
|
Num string
|
|
|
|
Value string
|
|
|
|
}
|
|
|
|
|
2014-02-25 11:54:15 +02:00
|
|
|
// UI Library that has some basic functionality exposed
|
|
|
|
type UiLib struct {
|
2014-10-31 15:30:08 +02:00
|
|
|
*xeth.JSXEth
|
2014-02-25 11:54:15 +02:00
|
|
|
engine *qml.Engine
|
|
|
|
eth *eth.Ethereum
|
|
|
|
connected bool
|
2014-03-27 12:14:04 +02:00
|
|
|
assetPath string
|
2014-04-11 19:50:31 +03:00
|
|
|
// The main application window
|
2014-05-28 16:48:17 +03:00
|
|
|
win *qml.Window
|
|
|
|
Db *Debugger
|
|
|
|
DbWindow *DebuggerWindow
|
2014-08-12 13:16:21 +03:00
|
|
|
|
|
|
|
jsEngine *javascript.JSRE
|
2014-08-20 11:00:02 +03:00
|
|
|
|
|
|
|
filterCallbacks map[int][]int
|
2014-11-07 13:18:48 +02:00
|
|
|
|
|
|
|
miner *miner.Miner
|
2014-03-27 12:14:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewUiLib(engine *qml.Engine, eth *eth.Ethereum, assetPath string) *UiLib {
|
2014-11-07 13:18:48 +02:00
|
|
|
lib := &UiLib{JSXEth: xeth.NewJSXEth(eth), engine: engine, eth: eth, assetPath: assetPath, jsEngine: javascript.NewJSRE(eth), filterCallbacks: make(map[int][]int)} //, filters: make(map[int]*xeth.JSFilter)}
|
|
|
|
lib.miner = miner.New(eth.KeyManager().Address(), eth)
|
|
|
|
|
|
|
|
return lib
|
2014-08-12 13:16:21 +03:00
|
|
|
}
|
|
|
|
|
2014-09-14 01:13:47 +03:00
|
|
|
func (self *UiLib) Notef(args []interface{}) {
|
2014-10-31 13:56:05 +02:00
|
|
|
guilogger.Infoln(args...)
|
2014-09-08 01:50:25 +03:00
|
|
|
}
|
|
|
|
|
2014-08-15 14:45:34 +03:00
|
|
|
func (self *UiLib) LookupDomain(domain string) string {
|
|
|
|
world := self.World()
|
|
|
|
|
|
|
|
if len(domain) > 32 {
|
2014-10-31 13:37:43 +02:00
|
|
|
domain = string(crypto.Sha3([]byte(domain)))
|
2014-08-15 14:45:34 +03:00
|
|
|
}
|
|
|
|
data := world.Config().Get("DnsReg").StorageString(domain).Bytes()
|
|
|
|
|
|
|
|
// Left padded = A record, Right padded = CNAME
|
2014-08-23 16:30:23 +03:00
|
|
|
if len(data) > 0 && data[0] == 0 {
|
2014-08-15 14:45:34 +03:00
|
|
|
data = bytes.TrimLeft(data, "\x00")
|
|
|
|
var ipSlice []string
|
|
|
|
for _, d := range data {
|
|
|
|
ipSlice = append(ipSlice, strconv.Itoa(int(d)))
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.Join(ipSlice, ".")
|
|
|
|
} else {
|
|
|
|
data = bytes.TrimRight(data, "\x00")
|
|
|
|
|
|
|
|
return string(data)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-25 11:33:05 +03:00
|
|
|
func (self *UiLib) LookupName(addr string) string {
|
|
|
|
var (
|
|
|
|
nameReg = self.World().Config().Get("NameReg")
|
|
|
|
lookup = nameReg.Storage(ethutil.Hex2Bytes(addr))
|
|
|
|
)
|
|
|
|
|
|
|
|
if lookup.Len() != 0 {
|
|
|
|
return strings.Trim(lookup.Str(), "\x00")
|
|
|
|
}
|
|
|
|
|
|
|
|
return addr
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *UiLib) LookupAddress(name string) string {
|
|
|
|
var (
|
|
|
|
nameReg = self.World().Config().Get("NameReg")
|
|
|
|
lookup = nameReg.Storage(ethutil.RightPadBytes([]byte(name), 32))
|
|
|
|
)
|
|
|
|
|
|
|
|
if lookup.Len() != 0 {
|
|
|
|
return ethutil.Bytes2Hex(lookup.Bytes())
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2014-09-17 16:58:44 +03:00
|
|
|
func (self *UiLib) PastPeers() *ethutil.List {
|
|
|
|
return ethutil.NewList(eth.PastPeers())
|
|
|
|
}
|
|
|
|
|
2014-08-12 13:16:21 +03:00
|
|
|
func (self *UiLib) ImportTx(rlpTx string) {
|
2014-10-31 11:59:17 +02:00
|
|
|
tx := chain.NewTransactionFromBytes(ethutil.Hex2Bytes(rlpTx))
|
2014-08-12 13:16:21 +03:00
|
|
|
self.eth.TxPool().QueueTransaction(tx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *UiLib) EvalJavascriptFile(path string) {
|
|
|
|
self.jsEngine.LoadExtFile(path[7:])
|
2014-02-25 11:54:15 +02:00
|
|
|
}
|
|
|
|
|
2014-08-14 01:18:37 +03:00
|
|
|
func (self *UiLib) EvalJavascriptString(str string) string {
|
|
|
|
value, err := self.jsEngine.Run(str)
|
|
|
|
if err != nil {
|
|
|
|
return err.Error()
|
|
|
|
}
|
|
|
|
|
|
|
|
return fmt.Sprintf("%v", value)
|
|
|
|
}
|
|
|
|
|
2014-06-23 17:25:57 +03:00
|
|
|
func (ui *UiLib) OpenQml(path string) {
|
|
|
|
container := NewQmlApplication(path[7:], ui)
|
|
|
|
app := NewExtApplication(container, ui)
|
2014-02-25 11:54:15 +02:00
|
|
|
|
2014-06-23 17:25:57 +03:00
|
|
|
go app.run()
|
2014-02-25 11:54:15 +02:00
|
|
|
}
|
|
|
|
|
2014-04-21 01:57:57 +03:00
|
|
|
func (ui *UiLib) OpenHtml(path string) {
|
2014-04-30 02:44:02 +03:00
|
|
|
container := NewHtmlApplication(path, ui)
|
|
|
|
app := NewExtApplication(container, ui)
|
2014-04-23 12:51:48 +03:00
|
|
|
|
2014-04-30 02:44:02 +03:00
|
|
|
go app.run()
|
|
|
|
}
|
2014-04-23 12:51:48 +03:00
|
|
|
|
2014-08-07 17:35:47 +03:00
|
|
|
func (ui *UiLib) OpenBrowser() {
|
|
|
|
ui.OpenHtml("file://" + ui.AssetPath("ext/home.html"))
|
|
|
|
}
|
|
|
|
|
2014-04-28 01:25:01 +03:00
|
|
|
func (ui *UiLib) Muted(content string) {
|
|
|
|
component, err := ui.engine.LoadFile(ui.AssetPath("qml/muted.qml"))
|
|
|
|
if err != nil {
|
2014-10-31 13:56:05 +02:00
|
|
|
guilogger.Debugln(err)
|
2014-04-28 01:25:01 +03:00
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
win := component.CreateWindow(nil)
|
|
|
|
go func() {
|
|
|
|
path := "file://" + ui.AssetPath("muted/index.html")
|
|
|
|
win.Set("url", path)
|
|
|
|
|
|
|
|
win.Show()
|
|
|
|
win.Wait()
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2014-02-28 13:18:19 +02:00
|
|
|
func (ui *UiLib) Connect(button qml.Object) {
|
2014-02-25 11:54:15 +02:00
|
|
|
if !ui.connected {
|
2014-05-09 15:51:02 +03:00
|
|
|
ui.eth.Start(true)
|
2014-02-28 13:18:19 +02:00
|
|
|
ui.connected = true
|
|
|
|
button.Set("enabled", false)
|
2014-02-25 11:54:15 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ui *UiLib) ConnectToPeer(addr string) {
|
|
|
|
ui.eth.ConnectToPeer(addr)
|
|
|
|
}
|
2014-03-03 01:54:41 +02:00
|
|
|
|
|
|
|
func (ui *UiLib) AssetPath(p string) string {
|
2014-03-27 12:14:04 +02:00
|
|
|
return path.Join(ui.assetPath, p)
|
2014-03-03 01:54:41 +02:00
|
|
|
}
|
2014-06-23 14:20:59 +03:00
|
|
|
|
2014-05-28 16:48:17 +03:00
|
|
|
func (self *UiLib) StartDbWithContractAndData(contractHash, data string) {
|
|
|
|
dbWindow := NewDebuggerWindow(self)
|
2014-11-04 11:57:02 +02:00
|
|
|
object := self.eth.BlockManager().CurrentState().GetStateObject(ethutil.Hex2Bytes(contractHash))
|
2014-07-24 13:30:41 +03:00
|
|
|
if len(object.Code) > 0 {
|
|
|
|
dbWindow.SetCode("0x" + ethutil.Bytes2Hex(object.Code))
|
2014-05-28 16:48:17 +03:00
|
|
|
}
|
2014-05-28 19:11:27 +03:00
|
|
|
dbWindow.SetData("0x" + data)
|
2014-05-28 16:48:17 +03:00
|
|
|
|
|
|
|
dbWindow.Show()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *UiLib) StartDbWithCode(code string) {
|
|
|
|
dbWindow := NewDebuggerWindow(self)
|
|
|
|
dbWindow.SetCode("0x" + code)
|
|
|
|
dbWindow.Show()
|
|
|
|
}
|
2014-03-03 01:54:41 +02:00
|
|
|
|
2014-05-27 11:42:37 +03:00
|
|
|
func (self *UiLib) StartDebugger() {
|
|
|
|
dbWindow := NewDebuggerWindow(self)
|
|
|
|
|
|
|
|
dbWindow.Show()
|
|
|
|
}
|
2014-08-20 11:00:02 +03:00
|
|
|
|
2014-10-08 20:16:03 +03:00
|
|
|
func (self *UiLib) NewFilter(object map[string]interface{}) (id int) {
|
2014-10-18 14:20:35 +03:00
|
|
|
filter := qt.NewFilterFromMap(object, self.eth)
|
2014-10-31 15:43:14 +02:00
|
|
|
filter.MessageCallback = func(messages state.Messages) {
|
2014-10-31 15:30:08 +02:00
|
|
|
self.win.Root().Call("invokeFilterCallback", xeth.ToJSMessages(messages), id)
|
2014-08-20 11:00:02 +03:00
|
|
|
}
|
2014-10-08 20:16:03 +03:00
|
|
|
id = self.eth.InstallFilter(filter)
|
2014-09-14 01:13:47 +03:00
|
|
|
return id
|
2014-08-20 13:42:49 +03:00
|
|
|
}
|
|
|
|
|
2014-10-08 20:16:03 +03:00
|
|
|
func (self *UiLib) NewFilterString(typ string) (id int) {
|
2014-10-31 11:59:17 +02:00
|
|
|
filter := chain.NewFilter(self.eth)
|
|
|
|
filter.BlockCallback = func(block *chain.Block) {
|
2014-09-14 01:13:47 +03:00
|
|
|
self.win.Root().Call("invokeFilterCallback", "{}", id)
|
2014-08-20 13:42:49 +03:00
|
|
|
}
|
2014-10-08 20:16:03 +03:00
|
|
|
id = self.eth.InstallFilter(filter)
|
2014-09-14 01:13:47 +03:00
|
|
|
return id
|
2014-08-20 11:00:02 +03:00
|
|
|
}
|
|
|
|
|
2014-09-14 01:13:47 +03:00
|
|
|
func (self *UiLib) Messages(id int) *ethutil.List {
|
|
|
|
filter := self.eth.GetFilter(id)
|
2014-08-20 11:00:02 +03:00
|
|
|
if filter != nil {
|
2014-10-31 15:30:08 +02:00
|
|
|
messages := xeth.ToJSMessages(filter.Find())
|
2014-09-14 01:13:47 +03:00
|
|
|
|
2014-09-24 21:40:18 +03:00
|
|
|
return messages
|
2014-08-20 11:00:02 +03:00
|
|
|
}
|
2014-09-14 01:13:47 +03:00
|
|
|
|
|
|
|
return ethutil.EmptyList()
|
2014-08-20 11:00:02 +03:00
|
|
|
}
|
|
|
|
|
2014-09-14 01:13:47 +03:00
|
|
|
func (self *UiLib) UninstallFilter(id int) {
|
|
|
|
self.eth.UninstallFilter(id)
|
2014-08-20 11:00:02 +03:00
|
|
|
}
|
2014-08-20 13:42:49 +03:00
|
|
|
|
2014-09-22 15:54:27 +03:00
|
|
|
func mapToTxParams(object map[string]interface{}) map[string]string {
|
2014-08-20 13:42:49 +03:00
|
|
|
// Default values
|
|
|
|
if object["from"] == nil {
|
|
|
|
object["from"] = ""
|
|
|
|
}
|
|
|
|
if object["to"] == nil {
|
|
|
|
object["to"] = ""
|
|
|
|
}
|
|
|
|
if object["value"] == nil {
|
|
|
|
object["value"] = ""
|
|
|
|
}
|
|
|
|
if object["gas"] == nil {
|
|
|
|
object["gas"] = ""
|
|
|
|
}
|
|
|
|
if object["gasPrice"] == nil {
|
|
|
|
object["gasPrice"] = ""
|
|
|
|
}
|
|
|
|
|
|
|
|
var dataStr string
|
|
|
|
var data []string
|
|
|
|
if list, ok := object["data"].(*qml.List); ok {
|
|
|
|
list.Convert(&data)
|
2014-09-22 15:54:27 +03:00
|
|
|
} else if str, ok := object["data"].(string); ok {
|
|
|
|
data = []string{str}
|
2014-08-20 13:42:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, str := range data {
|
|
|
|
if ethutil.IsHex(str) {
|
|
|
|
str = str[2:]
|
|
|
|
|
|
|
|
if len(str) != 64 {
|
|
|
|
str = ethutil.LeftPadString(str, 64)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
str = ethutil.Bytes2Hex(ethutil.LeftPadBytes(ethutil.Big(str).Bytes(), 32))
|
|
|
|
}
|
|
|
|
|
|
|
|
dataStr += str
|
|
|
|
}
|
2014-09-22 15:54:27 +03:00
|
|
|
object["data"] = dataStr
|
|
|
|
|
|
|
|
conv := make(map[string]string)
|
|
|
|
for key, value := range object {
|
|
|
|
if v, ok := value.(string); ok {
|
|
|
|
conv[key] = v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return conv
|
|
|
|
}
|
|
|
|
|
2014-10-31 15:30:08 +02:00
|
|
|
func (self *UiLib) Transact(params map[string]interface{}) (*xeth.JSReceipt, error) {
|
2014-09-22 15:54:27 +03:00
|
|
|
object := mapToTxParams(params)
|
2014-08-20 13:42:49 +03:00
|
|
|
|
2014-10-31 15:30:08 +02:00
|
|
|
return self.JSXEth.Transact(
|
2014-09-22 15:54:27 +03:00
|
|
|
object["from"],
|
|
|
|
object["to"],
|
|
|
|
object["value"],
|
|
|
|
object["gas"],
|
|
|
|
object["gasPrice"],
|
|
|
|
object["data"],
|
2014-08-20 13:42:49 +03:00
|
|
|
)
|
|
|
|
}
|
2014-09-19 23:42:55 +03:00
|
|
|
|
|
|
|
func (self *UiLib) Compile(code string) (string, error) {
|
|
|
|
bcode, err := ethutil.Compile(code, false)
|
|
|
|
if err != nil {
|
|
|
|
return err.Error(), err
|
|
|
|
}
|
|
|
|
|
|
|
|
return ethutil.Bytes2Hex(bcode), err
|
|
|
|
}
|
2014-09-22 15:54:27 +03:00
|
|
|
|
|
|
|
func (self *UiLib) Call(params map[string]interface{}) (string, error) {
|
|
|
|
object := mapToTxParams(params)
|
|
|
|
|
2014-10-31 15:30:08 +02:00
|
|
|
return self.JSXEth.Execute(
|
2014-09-22 15:54:27 +03:00
|
|
|
object["to"],
|
|
|
|
object["value"],
|
|
|
|
object["gas"],
|
|
|
|
object["gasPrice"],
|
|
|
|
object["data"],
|
|
|
|
)
|
|
|
|
}
|
2014-11-07 13:18:48 +02:00
|
|
|
|
|
|
|
func (self *UiLib) AddLocalTransaction(to, data, gas, gasPrice, value string) int {
|
|
|
|
return self.miner.AddLocalTx(&miner.LocalTx{
|
|
|
|
To: ethutil.Hex2Bytes(to),
|
|
|
|
Data: ethutil.Hex2Bytes(data),
|
|
|
|
Gas: gas,
|
|
|
|
GasPrice: gasPrice,
|
|
|
|
Value: value,
|
|
|
|
}) - 1
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *UiLib) RemoveLocalTransaction(id int) {
|
|
|
|
self.miner.RemoveLocalTx(id)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *UiLib) SetGasPrice(price string) {
|
|
|
|
self.miner.MinAcceptedGasPrice = ethutil.Big(price)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *UiLib) ToggleMining() bool {
|
|
|
|
if !self.miner.Mining() {
|
|
|
|
self.miner.Start()
|
|
|
|
|
|
|
|
return true
|
|
|
|
} else {
|
|
|
|
self.miner.Stop()
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|