quorum/ethereal/ui/ui_lib.go

205 lines
4.4 KiB
Go
Raw Normal View History

2014-02-25 01:54:15 -08:00
package ethui
import (
"bitbucket.org/kardianos/osext"
2014-04-11 09:50:31 -07:00
"fmt"
2014-02-25 01:54:15 -08:00
"github.com/ethereum/eth-go"
2014-04-11 09:50:31 -07:00
"github.com/ethereum/eth-go/ethchain"
2014-02-25 01:54:15 -08:00
"github.com/ethereum/eth-go/ethutil"
"github.com/go-qml/qml"
2014-03-16 10:34:34 -07:00
"os"
"path"
"path/filepath"
"runtime"
2014-02-25 01:54:15 -08:00
)
2014-04-11 21:12:10 -07:00
type memAddr struct {
Num string
Value string
}
2014-02-25 01:54:15 -08:00
// UI Library that has some basic functionality exposed
type UiLib struct {
engine *qml.Engine
eth *eth.Ethereum
connected bool
assetPath string
2014-04-11 09:50:31 -07:00
// The main application window
win *qml.Window
2014-04-11 21:12:10 -07:00
Db *Debugger
}
func NewUiLib(engine *qml.Engine, eth *eth.Ethereum, assetPath string) *UiLib {
if assetPath == "" {
assetPath = DefaultAssetPath()
}
return &UiLib{engine: engine, eth: eth, assetPath: assetPath}
2014-02-25 01:54:15 -08:00
}
// Opens a QML file (external application)
func (ui *UiLib) Open(path string) {
component, err := ui.engine.LoadFile(path[7:])
if err != nil {
ethutil.Config.Log.Debugln(err)
}
win := component.CreateWindow(nil)
go func() {
win.Show()
win.Wait()
}()
}
func (ui *UiLib) OpenHtml(path string) {
container := NewHtmlApplication(path, ui)
app := NewExtApplication(container, ui)
go app.run()
}
2014-04-27 15:25:01 -07:00
func (ui *UiLib) Muted(content string) {
component, err := ui.engine.LoadFile(ui.AssetPath("qml/muted.qml"))
if err != nil {
ethutil.Config.Log.Debugln(err)
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 03:18:19 -08:00
func (ui *UiLib) Connect(button qml.Object) {
2014-02-25 01:54:15 -08:00
if !ui.connected {
2014-05-09 05:51:02 -07:00
ui.eth.Start(true)
2014-02-28 03:18:19 -08:00
ui.connected = true
button.Set("enabled", false)
2014-02-25 01:54:15 -08:00
}
}
func (ui *UiLib) ConnectToPeer(addr string) {
ui.eth.ConnectToPeer(addr)
}
func (ui *UiLib) AssetPath(p string) string {
return path.Join(ui.assetPath, p)
}
func DefaultAssetPath() string {
var base string
2014-03-16 10:34:34 -07:00
// If the current working directory is the go-ethereum dir
// assume a debug build and use the source directory as
// asset directory.
pwd, _ := os.Getwd()
if pwd == path.Join(os.Getenv("GOPATH"), "src", "github.com", "ethereum", "go-ethereum", "ethereal") {
base = path.Join(pwd, "assets")
2014-03-16 10:34:34 -07:00
} else {
switch runtime.GOOS {
case "darwin":
// Get Binary Directory
exedir, _ := osext.ExecutableFolder()
base = filepath.Join(exedir, "../Resources")
case "linux":
base = "/usr/share/ethereal"
case "window":
fallthrough
default:
base = "."
}
}
return base
}
2014-04-11 09:50:31 -07:00
2014-04-11 10:36:25 -07:00
func (ui *UiLib) DebugTx(recipient, valueStr, gasStr, gasPriceStr, data string) {
2014-04-11 09:50:31 -07:00
state := ui.eth.BlockChain().CurrentBlock.State()
2014-05-26 12:11:38 -07:00
script, err := ethutil.Compile(data)
2014-04-11 09:50:31 -07:00
if err != nil {
2014-04-15 19:08:25 -07:00
ethutil.Config.Log.Debugln(err)
return
2014-04-11 09:50:31 -07:00
}
2014-05-26 12:11:38 -07:00
dis := ethchain.Disassemble(script)
2014-04-11 09:50:31 -07:00
ui.win.Root().Call("clearAsm")
2014-04-15 19:08:25 -07:00
2014-04-11 09:50:31 -07:00
for _, str := range dis {
ui.win.Root().Call("setAsm", str)
}
// Contract addr as test address
2014-05-14 04:55:08 -07:00
keyPair := ethutil.GetKeyRing().Get(0)
2014-05-26 12:11:38 -07:00
callerTx :=
ethchain.NewContractCreationTx(ethutil.Big(valueStr), ethutil.Big(gasStr), ethutil.Big(gasPriceStr), script)
callerTx.Sign(keyPair.PrivateKey)
2014-05-08 09:24:28 -07:00
account := ui.eth.StateManager().TransState().GetStateObject(keyPair.Address())
2014-05-26 12:11:38 -07:00
contract := ethchain.MakeContract(callerTx, state)
callerClosure := ethchain.NewClosure(account, contract, contract.Init(), state, ethutil.Big(gasStr), ethutil.Big(gasPriceStr))
2014-04-11 09:50:31 -07:00
block := ui.eth.BlockChain().CurrentBlock
vm := ethchain.NewVm(state, ui.eth.StateManager(), ethchain.RuntimeVars{
2014-04-11 09:50:31 -07:00
Origin: account.Address(),
BlockNumber: block.BlockInfo().Number,
PrevHash: block.PrevHash,
Coinbase: block.Coinbase,
Time: block.Time,
Diff: block.Difficulty,
})
2014-05-26 12:11:38 -07:00
ui.Db.done = false
2014-04-11 10:36:25 -07:00
go func() {
2014-05-26 12:11:38 -07:00
callerClosure.Call(vm, contract.Init(), ui.Db.halting)
2014-04-11 09:50:31 -07:00
2014-04-11 10:36:25 -07:00
state.Reset()
2014-05-26 12:11:38 -07:00
ui.Db.done = true
2014-04-11 10:36:25 -07:00
}()
2014-04-11 09:50:31 -07:00
}
2014-04-11 21:12:10 -07:00
func (ui *UiLib) Next() {
2014-05-26 12:11:38 -07:00
if !ui.Db.done {
ui.Db.Next()
}
2014-04-11 21:12:10 -07:00
}
type Debugger struct {
2014-05-26 12:11:38 -07:00
win *qml.Window
N chan bool
done bool
2014-04-11 21:12:10 -07:00
}
func (d *Debugger) halting(pc int, op ethchain.OpCode, mem *ethchain.Memory, stack *ethchain.Stack) {
d.win.Root().Call("setInstruction", pc)
d.win.Root().Call("clearMem")
d.win.Root().Call("clearStack")
addr := 0
for i := 0; i+32 <= mem.Len(); i += 32 {
d.win.Root().Call("setMem", memAddr{fmt.Sprintf("%03d", addr), fmt.Sprintf("% x", mem.Data()[i:i+32])})
addr++
}
for _, val := range stack.Data() {
d.win.Root().Call("setStack", val.String())
}
out:
for {
select {
case <-d.N:
break out
default:
}
}
}
func (d *Debugger) Next() {
d.N <- true
}