| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | package javascript | 
					
						
							| 
									
										
										
										
											2014-05-15 20:45:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:12 +02:00
										 |  |  | 	"io/ioutil" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"path" | 
					
						
							|  |  |  | 	"path/filepath" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-23 15:01:27 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum" | 
					
						
							| 
									
										
										
										
											2014-10-31 10:59:17 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/chain" | 
					
						
							| 
									
										
										
										
											2014-10-31 14:20:11 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/cmd/utils" | 
					
						
							| 
									
										
										
										
											2014-10-23 15:01:27 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/ethutil" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/event" | 
					
						
							| 
									
										
										
										
											2014-10-31 12:56:05 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/logger" | 
					
						
							| 
									
										
										
										
											2014-10-31 14:43:14 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/state" | 
					
						
							| 
									
										
										
										
											2014-10-31 14:30:08 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/xeth" | 
					
						
							| 
									
										
										
										
											2014-05-20 17:49:12 +02:00
										 |  |  | 	"github.com/obscuren/otto" | 
					
						
							| 
									
										
										
										
											2014-05-15 20:45:19 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-31 12:56:05 +01:00
										 |  |  | var jsrelogger = logger.NewLogger("JSRE") | 
					
						
							| 
									
										
										
										
											2014-06-23 11:39:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-17 15:15:46 +02:00
										 |  |  | type JSRE struct { | 
					
						
							| 
									
										
										
										
											2014-05-19 12:15:03 +02:00
										 |  |  | 	ethereum *eth.Ethereum | 
					
						
							| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | 	Vm       *otto.Otto | 
					
						
							| 
									
										
										
										
											2014-10-31 14:30:08 +01:00
										 |  |  | 	pipe     *xeth.JSXEth | 
					
						
							| 
									
										
										
										
											2014-05-19 12:15:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-29 03:00:29 +01:00
										 |  |  | 	events event.Subscription | 
					
						
							| 
									
										
										
										
											2014-05-19 12:15:03 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	objectCb map[string][]otto.Value | 
					
						
							| 
									
										
										
										
											2014-05-15 20:45:19 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-04 12:19:50 +02:00
										 |  |  | func (jsre *JSRE) LoadExtFile(path string) { | 
					
						
							|  |  |  | 	result, err := ioutil.ReadFile(path) | 
					
						
							|  |  |  | 	if err == nil { | 
					
						
							| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | 		jsre.Vm.Run(result) | 
					
						
							| 
									
										
										
										
											2014-06-04 12:19:50 +02:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | 		jsrelogger.Infoln("Could not load file:", path) | 
					
						
							| 
									
										
										
										
											2014-06-04 12:19:50 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (jsre *JSRE) LoadIntFile(file string) { | 
					
						
							| 
									
										
										
										
											2014-10-27 11:50:38 +01:00
										 |  |  | 	assetPath := path.Join(os.Getenv("GOPATH"), "src", "github.com", "ethereum", "go-ethereum", "cmd", "mist", "assets", "ext") | 
					
						
							| 
									
										
										
										
											2014-06-04 12:19:50 +02:00
										 |  |  | 	jsre.LoadExtFile(path.Join(assetPath, file)) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-17 15:15:46 +02:00
										 |  |  | func NewJSRE(ethereum *eth.Ethereum) *JSRE { | 
					
						
							| 
									
										
										
										
											2014-05-19 12:15:03 +02:00
										 |  |  | 	re := &JSRE{ | 
					
						
							|  |  |  | 		ethereum, | 
					
						
							|  |  |  | 		otto.New(), | 
					
						
							| 
									
										
										
										
											2014-10-31 14:30:08 +01:00
										 |  |  | 		xeth.NewJSXEth(ethereum), | 
					
						
							| 
									
										
										
										
											2014-10-14 19:38:38 +02:00
										 |  |  | 		nil, | 
					
						
							| 
									
										
										
										
											2014-05-19 12:15:03 +02:00
										 |  |  | 		make(map[string][]otto.Value), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-19 16:32:45 +02:00
										 |  |  | 	// Init the JS lib | 
					
						
							| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | 	re.Vm.Run(jsLib) | 
					
						
							| 
									
										
										
										
											2014-05-19 16:32:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-04 12:19:50 +02:00
										 |  |  | 	// Load extra javascript files | 
					
						
							|  |  |  | 	re.LoadIntFile("string.js") | 
					
						
							|  |  |  | 	re.LoadIntFile("big.js") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-15 01:13:39 +01:00
										 |  |  | 	// Subscribe to events | 
					
						
							| 
									
										
										
										
											2014-10-14 19:38:38 +02:00
										 |  |  | 	mux := ethereum.EventMux() | 
					
						
							| 
									
										
										
										
											2014-10-31 10:59:17 +01:00
										 |  |  | 	re.events = mux.Subscribe(chain.NewBlockEvent{}) | 
					
						
							| 
									
										
										
										
											2014-07-15 01:13:39 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-27 11:50:38 +01:00
										 |  |  | 	// We have to make sure that, whoever calls this, calls "Stop" | 
					
						
							|  |  |  | 	go re.mainLoop() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-16 01:38:24 +02:00
										 |  |  | 	re.Bind("eth", &JSEthereum{re.pipe, re.Vm, ethereum}) | 
					
						
							| 
									
										
										
										
											2014-05-19 12:15:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 12:48:34 +02:00
										 |  |  | 	re.initStdFuncs() | 
					
						
							| 
									
										
										
										
											2014-05-19 12:15:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-23 11:39:09 +01:00
										 |  |  | 	jsrelogger.Infoln("started") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 12:48:34 +02:00
										 |  |  | 	return re | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-05-19 12:15:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 12:48:34 +02:00
										 |  |  | func (self *JSRE) Bind(name string, v interface{}) { | 
					
						
							| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | 	self.Vm.Set(name, v) | 
					
						
							| 
									
										
										
										
											2014-05-20 12:48:34 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-05-17 15:15:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 12:48:34 +02:00
										 |  |  | func (self *JSRE) Run(code string) (otto.Value, error) { | 
					
						
							| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | 	return self.Vm.Run(code) | 
					
						
							| 
									
										
										
										
											2014-05-17 15:15:46 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 19:28:48 +02:00
										 |  |  | func (self *JSRE) Require(file string) error { | 
					
						
							|  |  |  | 	if len(filepath.Ext(file)) == 0 { | 
					
						
							|  |  |  | 		file += ".js" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fh, err := os.Open(file) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	content, _ := ioutil.ReadAll(fh) | 
					
						
							|  |  |  | 	self.Run("exports = {};(function() {" + string(content) + "})();") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-19 12:15:03 +02:00
										 |  |  | func (self *JSRE) Stop() { | 
					
						
							| 
									
										
										
										
											2014-10-14 19:38:38 +02:00
										 |  |  | 	self.events.Unsubscribe() | 
					
						
							| 
									
										
										
										
											2014-06-23 11:39:09 +01:00
										 |  |  | 	jsrelogger.Infoln("stopped") | 
					
						
							| 
									
										
										
										
											2014-05-19 12:15:03 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *JSRE) mainLoop() { | 
					
						
							| 
									
										
										
										
											2014-10-14 19:38:38 +02:00
										 |  |  | 	for _ = range self.events.Chan() { | 
					
						
							| 
									
										
										
										
											2014-05-19 12:15:03 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 12:48:34 +02:00
										 |  |  | func (self *JSRE) initStdFuncs() { | 
					
						
							| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | 	t, _ := self.Vm.Get("eth") | 
					
						
							| 
									
										
										
										
											2014-05-20 12:48:34 +02:00
										 |  |  | 	eth := t.Object() | 
					
						
							| 
									
										
										
										
											2014-05-20 19:28:48 +02:00
										 |  |  | 	eth.Set("watch", self.watch) | 
					
						
							|  |  |  | 	eth.Set("addPeer", self.addPeer) | 
					
						
							| 
									
										
										
										
											2014-05-20 22:12:42 +02:00
										 |  |  | 	eth.Set("require", self.require) | 
					
						
							| 
									
										
										
										
											2014-05-22 00:25:48 +02:00
										 |  |  | 	eth.Set("stopMining", self.stopMining) | 
					
						
							|  |  |  | 	eth.Set("startMining", self.startMining) | 
					
						
							| 
									
										
										
										
											2014-06-24 09:36:05 +02:00
										 |  |  | 	eth.Set("execBlock", self.execBlock) | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:12 +02:00
										 |  |  | 	eth.Set("dump", self.dump) | 
					
						
							| 
									
										
										
										
											2014-05-20 19:28:48 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* | 
					
						
							|  |  |  |  * The following methods are natively implemented javascript functions | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:12 +02:00
										 |  |  | func (self *JSRE) dump(call otto.FunctionCall) otto.Value { | 
					
						
							| 
									
										
										
										
											2014-10-31 14:43:14 +01:00
										 |  |  | 	var state *state.State | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:12 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if len(call.ArgumentList) > 0 { | 
					
						
							| 
									
										
										
										
											2014-10-31 10:59:17 +01:00
										 |  |  | 		var block *chain.Block | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:12 +02:00
										 |  |  | 		if call.Argument(0).IsNumber() { | 
					
						
							|  |  |  | 			num, _ := call.Argument(0).ToInteger() | 
					
						
							| 
									
										
										
										
											2014-10-20 12:03:31 +02:00
										 |  |  | 			block = self.ethereum.ChainManager().GetBlockByNumber(uint64(num)) | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:12 +02:00
										 |  |  | 		} else if call.Argument(0).IsString() { | 
					
						
							|  |  |  | 			hash, _ := call.Argument(0).ToString() | 
					
						
							| 
									
										
										
										
											2014-10-20 12:03:31 +02:00
										 |  |  | 			block = self.ethereum.ChainManager().GetBlock(ethutil.Hex2Bytes(hash)) | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:12 +02:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			fmt.Println("invalid argument for dump. Either hex string or number") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if block == nil { | 
					
						
							|  |  |  | 			fmt.Println("block not found") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return otto.UndefinedValue() | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		state = block.State() | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2014-11-04 10:57:02 +01:00
										 |  |  | 		state = self.ethereum.BlockManager().CurrentState() | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:12 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | 	v, _ := self.Vm.ToValue(state.Dump()) | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-06 10:05:34 +02:00
										 |  |  | 	return v | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:12 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-22 00:25:48 +02:00
										 |  |  | func (self *JSRE) stopMining(call otto.FunctionCall) otto.Value { | 
					
						
							| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | 	v, _ := self.Vm.ToValue(utils.StopMining(self.ethereum)) | 
					
						
							| 
									
										
										
										
											2014-05-22 00:25:48 +02:00
										 |  |  | 	return v | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *JSRE) startMining(call otto.FunctionCall) otto.Value { | 
					
						
							| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | 	v, _ := self.Vm.ToValue(utils.StartMining(self.ethereum)) | 
					
						
							| 
									
										
										
										
											2014-05-22 00:25:48 +02:00
										 |  |  | 	return v | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 19:28:48 +02:00
										 |  |  | // eth.watch | 
					
						
							| 
									
										
										
										
											2014-05-22 00:25:48 +02:00
										 |  |  | func (self *JSRE) watch(call otto.FunctionCall) otto.Value { | 
					
						
							| 
									
										
										
										
											2014-05-20 19:28:48 +02:00
										 |  |  | 	addr, _ := call.Argument(0).ToString() | 
					
						
							|  |  |  | 	var storageAddr string | 
					
						
							|  |  |  | 	var cb otto.Value | 
					
						
							|  |  |  | 	var storageCallback bool | 
					
						
							|  |  |  | 	if len(call.ArgumentList) > 2 { | 
					
						
							|  |  |  | 		storageCallback = true | 
					
						
							|  |  |  | 		storageAddr, _ = call.Argument(1).ToString() | 
					
						
							|  |  |  | 		cb = call.Argument(2) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		cb = call.Argument(1) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if storageCallback { | 
					
						
							|  |  |  | 		self.objectCb[addr+storageAddr] = append(self.objectCb[addr+storageAddr], cb) | 
					
						
							| 
									
										
										
										
											2014-05-15 20:45:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-14 19:38:38 +02:00
										 |  |  | 		// event := "storage:" + string(ethutil.Hex2Bytes(addr)) + ":" + string(ethutil.Hex2Bytes(storageAddr)) | 
					
						
							|  |  |  | 		// self.ethereum.EventMux().Subscribe(event, self.changeChan) | 
					
						
							| 
									
										
										
										
											2014-05-20 19:28:48 +02:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2014-05-20 12:48:34 +02:00
										 |  |  | 		self.objectCb[addr] = append(self.objectCb[addr], cb) | 
					
						
							| 
									
										
										
										
											2014-05-15 20:45:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-14 19:38:38 +02:00
										 |  |  | 		// event := "object:" + string(ethutil.Hex2Bytes(addr)) | 
					
						
							|  |  |  | 		// self.ethereum.EventMux().Subscribe(event, self.changeChan) | 
					
						
							| 
									
										
										
										
											2014-05-20 19:28:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return otto.UndefinedValue() | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-05-15 22:15:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 19:28:48 +02:00
										 |  |  | func (self *JSRE) addPeer(call otto.FunctionCall) otto.Value { | 
					
						
							|  |  |  | 	host, err := call.Argument(0).ToString() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return otto.FalseValue() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	self.ethereum.ConnectToPeer(host) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return otto.TrueValue() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *JSRE) require(call otto.FunctionCall) otto.Value { | 
					
						
							|  |  |  | 	file, err := call.Argument(0).ToString() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2014-05-15 22:15:14 +02:00
										 |  |  | 		return otto.UndefinedValue() | 
					
						
							| 
									
										
										
										
											2014-05-20 19:28:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if err := self.Require(file); err != nil { | 
					
						
							|  |  |  | 		fmt.Println("err:", err) | 
					
						
							|  |  |  | 		return otto.UndefinedValue() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-11 16:24:17 +02:00
										 |  |  | 	t, _ := self.Vm.Get("exports") | 
					
						
							| 
									
										
										
										
											2014-05-15 22:15:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-20 19:28:48 +02:00
										 |  |  | 	return t | 
					
						
							| 
									
										
										
										
											2014-05-15 20:45:19 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-06-23 11:28:05 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | func (self *JSRE) execBlock(call otto.FunctionCall) otto.Value { | 
					
						
							|  |  |  | 	hash, err := call.Argument(0).ToString() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return otto.UndefinedValue() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-29 18:32:48 +01:00
										 |  |  | 	err = utils.BlockDo(self.ethereum, ethutil.Hex2Bytes(hash)) | 
					
						
							| 
									
										
										
										
											2014-06-23 11:28:05 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		fmt.Println(err) | 
					
						
							|  |  |  | 		return otto.FalseValue() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return otto.TrueValue() | 
					
						
							|  |  |  | } |