forked from cerc-io/plugeth
283 lines
6.1 KiB
Go
283 lines
6.1 KiB
Go
/*
|
|
This file is part of go-ethereum
|
|
|
|
go-ethereum 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 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
go-ethereum 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 go-ethereum. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
/**
|
|
* @authors
|
|
* Jeffrey Wilcke <i@jev.io>
|
|
*/
|
|
package main
|
|
|
|
import (
|
|
"io/ioutil"
|
|
|
|
"path/filepath"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/ethereum/go-ethereum/core/types"
|
|
"github.com/ethereum/go-ethereum/eth"
|
|
"github.com/ethereum/go-ethereum/event/filter"
|
|
"github.com/ethereum/go-ethereum/xeth"
|
|
"github.com/obscuren/qml"
|
|
)
|
|
|
|
type memAddr struct {
|
|
Num string
|
|
Value string
|
|
}
|
|
|
|
// UI Library that has some basic functionality exposed
|
|
type UiLib struct {
|
|
*xeth.XEth
|
|
engine *qml.Engine
|
|
eth *eth.Ethereum
|
|
connected bool
|
|
assetPath string
|
|
// The main application window
|
|
win *qml.Window
|
|
|
|
filterCallbacks map[int][]int
|
|
filterManager *filter.FilterManager
|
|
}
|
|
|
|
func NewUiLib(engine *qml.Engine, eth *eth.Ethereum, assetPath, libPath string) *UiLib {
|
|
x := xeth.New(eth, nil)
|
|
lib := &UiLib{
|
|
XEth: x,
|
|
engine: engine,
|
|
eth: eth,
|
|
assetPath: assetPath,
|
|
filterCallbacks: make(map[int][]int),
|
|
}
|
|
lib.filterManager = filter.NewFilterManager(eth.EventMux())
|
|
go lib.filterManager.Start()
|
|
|
|
return lib
|
|
}
|
|
|
|
func (self *UiLib) Notef(args []interface{}) {
|
|
guilogger.Infoln(args...)
|
|
}
|
|
|
|
func (self *UiLib) ImportTx(rlpTx string) {
|
|
tx := types.NewTransactionFromBytes(common.Hex2Bytes(rlpTx))
|
|
err := self.eth.TxPool().Add(tx)
|
|
if err != nil {
|
|
guilogger.Infoln("import tx failed ", err)
|
|
}
|
|
}
|
|
|
|
func (ui *UiLib) Muted(content string) {
|
|
component, err := ui.engine.LoadFile(ui.AssetPath("qml/muted.qml"))
|
|
if err != nil {
|
|
guilogger.Debugln(err)
|
|
|
|
return
|
|
}
|
|
win := component.CreateWindow(nil)
|
|
go func() {
|
|
path := "file://" + ui.AssetPath("muted/index.html")
|
|
win.Set("url", path)
|
|
|
|
win.Show()
|
|
win.Wait()
|
|
}()
|
|
}
|
|
|
|
func (ui *UiLib) Connect(button qml.Object) {
|
|
if !ui.connected {
|
|
ui.eth.Start()
|
|
ui.connected = true
|
|
button.Set("enabled", false)
|
|
}
|
|
}
|
|
|
|
func (ui *UiLib) ConnectToPeer(nodeURL string) {
|
|
if err := ui.eth.AddPeer(nodeURL); err != nil {
|
|
guilogger.Infoln("AddPeer error: " + err.Error())
|
|
}
|
|
}
|
|
|
|
func (ui *UiLib) AssetPath(p string) string {
|
|
return filepath.Join(ui.assetPath, p)
|
|
}
|
|
|
|
func (self *UiLib) Transact(params map[string]interface{}) (string, error) {
|
|
object := mapToTxParams(params)
|
|
|
|
return self.XEth.Transact(
|
|
object["from"],
|
|
object["to"],
|
|
"",
|
|
object["value"],
|
|
object["gas"],
|
|
object["gasPrice"],
|
|
object["data"],
|
|
)
|
|
}
|
|
|
|
func (self *UiLib) Call(params map[string]interface{}) (string, error) {
|
|
object := mapToTxParams(params)
|
|
|
|
return self.XEth.Call(
|
|
object["from"],
|
|
object["to"],
|
|
object["value"],
|
|
object["gas"],
|
|
object["gasPrice"],
|
|
object["data"],
|
|
)
|
|
}
|
|
|
|
func (self *UiLib) AddLocalTransaction(to, data, gas, gasPrice, value string) int {
|
|
return 0
|
|
/*
|
|
return self.miner.AddLocalTx(&miner.LocalTx{
|
|
To: common.Hex2Bytes(to),
|
|
Data: common.Hex2Bytes(data),
|
|
Gas: gas,
|
|
GasPrice: gasPrice,
|
|
Value: value,
|
|
}) - 1
|
|
*/
|
|
}
|
|
|
|
func (self *UiLib) RemoveLocalTransaction(id int) {
|
|
//self.miner.RemoveLocalTx(id)
|
|
}
|
|
|
|
func (self *UiLib) ToggleMining() bool {
|
|
if !self.eth.IsMining() {
|
|
err := self.eth.StartMining(4)
|
|
return err == nil
|
|
} else {
|
|
self.eth.StopMining()
|
|
return false
|
|
}
|
|
}
|
|
|
|
func (self *UiLib) ToHex(data string) string {
|
|
return "0x" + common.Bytes2Hex([]byte(data))
|
|
}
|
|
|
|
func (self *UiLib) ToAscii(data string) string {
|
|
start := 0
|
|
if len(data) > 1 && data[0:2] == "0x" {
|
|
start = 2
|
|
}
|
|
return string(common.Hex2Bytes(data[start:]))
|
|
}
|
|
|
|
/// Ethereum filter methods
|
|
func (self *UiLib) NewFilter(object map[string]interface{}, view *qml.Common) (id int) {
|
|
/* TODO remove me
|
|
filter := qt.NewFilterFromMap(object, self.eth)
|
|
filter.MessageCallback = func(messages state.Messages) {
|
|
view.Call("messages", xeth.ToMessages(messages), id)
|
|
}
|
|
id = self.filterManager.InstallFilter(filter)
|
|
return id
|
|
*/
|
|
return 0
|
|
}
|
|
|
|
func (self *UiLib) NewFilterString(typ string, view *qml.Common) (id int) {
|
|
/* TODO remove me
|
|
filter := core.NewFilter(self.eth)
|
|
filter.BlockCallback = func(block *types.Block) {
|
|
view.Call("messages", "{}", id)
|
|
}
|
|
id = self.filterManager.InstallFilter(filter)
|
|
return id
|
|
*/
|
|
return 0
|
|
}
|
|
|
|
func (self *UiLib) Messages(id int) *common.List {
|
|
/* TODO remove me
|
|
filter := self.filterManager.GetFilter(id)
|
|
if filter != nil {
|
|
messages := xeth.ToMessages(filter.Find())
|
|
|
|
return messages
|
|
}
|
|
*/
|
|
|
|
return common.EmptyList()
|
|
}
|
|
|
|
func (self *UiLib) ReadFile(p string) string {
|
|
content, err := ioutil.ReadFile(self.AssetPath(filepath.Join("ext", p)))
|
|
if err != nil {
|
|
guilogger.Infoln("error reading file", p, ":", err)
|
|
}
|
|
return string(content)
|
|
}
|
|
|
|
func (self *UiLib) UninstallFilter(id int) {
|
|
self.filterManager.UninstallFilter(id)
|
|
}
|
|
|
|
func mapToTxParams(object map[string]interface{}) map[string]string {
|
|
// 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)
|
|
} else if str, ok := object["data"].(string); ok {
|
|
data = []string{str}
|
|
}
|
|
|
|
for _, str := range data {
|
|
if common.IsHex(str) {
|
|
str = str[2:]
|
|
|
|
if len(str) != 64 {
|
|
str = common.LeftPadString(str, 64)
|
|
}
|
|
} else {
|
|
str = common.Bytes2Hex(common.LeftPadBytes(common.Big(str).Bytes(), 32))
|
|
}
|
|
|
|
dataStr += str
|
|
}
|
|
object["data"] = dataStr
|
|
|
|
conv := make(map[string]string)
|
|
for key, value := range object {
|
|
if v, ok := value.(string); ok {
|
|
conv[key] = v
|
|
}
|
|
}
|
|
|
|
return conv
|
|
}
|