diff options
Diffstat (limited to 'jsre/jsre.go')
-rw-r--r-- | jsre/jsre.go | 322 |
1 files changed, 0 insertions, 322 deletions
diff --git a/jsre/jsre.go b/jsre/jsre.go deleted file mode 100644 index 59730bc0d..000000000 --- a/jsre/jsre.go +++ /dev/null @@ -1,322 +0,0 @@ -// Copyright 2015 The go-ethereum Authors -// This file is part of the go-ethereum library. -// -// The go-ethereum library is free software: you can redistribute it and/or modify -// it under the terms of the GNU Lesser General Public License as published by -// the Free Software Foundation, either version 3 of the License, or -// (at your option) any later version. -// -// The go-ethereum 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 Lesser General Public License for more details. -// -// You should have received a copy of the GNU Lesser General Public License -// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. - -// Package jsre provides execution environment for JavaScript. -package jsre - -import ( - crand "crypto/rand" - "encoding/binary" - "fmt" - "io/ioutil" - "math/rand" - "sync" - "time" - - "github.com/ethereum/go-ethereum/common" - "github.com/robertkrimen/otto" -) - -/* -JSRE is a generic JS runtime environment embedding the otto JS interpreter. -It provides some helper functions to -- load code from files -- run code snippets -- require libraries -- bind native go objects -*/ -type JSRE struct { - assetPath string - evalQueue chan *evalReq - stopEventLoop chan bool - loopWg sync.WaitGroup -} - -// jsTimer is a single timer instance with a callback function -type jsTimer struct { - timer *time.Timer - duration time.Duration - interval bool - call otto.FunctionCall -} - -// evalReq is a serialized vm execution request processed by runEventLoop. -type evalReq struct { - fn func(vm *otto.Otto) - done chan bool -} - -// runtime must be stopped with Stop() after use and cannot be used after stopping -func New(assetPath string) *JSRE { - re := &JSRE{ - assetPath: assetPath, - evalQueue: make(chan *evalReq), - stopEventLoop: make(chan bool), - } - re.loopWg.Add(1) - go re.runEventLoop() - re.Set("loadScript", re.loadScript) - re.Set("inspect", prettyPrintJS) - return re -} - -// randomSource returns a pseudo random value generator. -func randomSource() *rand.Rand { - bytes := make([]byte, 8) - seed := time.Now().UnixNano() - if _, err := crand.Read(bytes); err == nil { - seed = int64(binary.LittleEndian.Uint64(bytes)) - } - - src := rand.NewSource(seed) - return rand.New(src) -} - -// This function runs the main event loop from a goroutine that is started -// when JSRE is created. Use Stop() before exiting to properly stop it. -// The event loop processes vm access requests from the evalQueue in a -// serialized way and calls timer callback functions at the appropriate time. - -// Exported functions always access the vm through the event queue. You can -// call the functions of the otto vm directly to circumvent the queue. These -// functions should be used if and only if running a routine that was already -// called from JS through an RPC call. -func (self *JSRE) runEventLoop() { - vm := otto.New() - r := randomSource() - vm.SetRandomSource(r.Float64) - - registry := map[*jsTimer]*jsTimer{} - ready := make(chan *jsTimer) - - newTimer := func(call otto.FunctionCall, interval bool) (*jsTimer, otto.Value) { - delay, _ := call.Argument(1).ToInteger() - if 0 >= delay { - delay = 1 - } - timer := &jsTimer{ - duration: time.Duration(delay) * time.Millisecond, - call: call, - interval: interval, - } - registry[timer] = timer - - timer.timer = time.AfterFunc(timer.duration, func() { - ready <- timer - }) - - value, err := call.Otto.ToValue(timer) - if err != nil { - panic(err) - } - return timer, value - } - - setTimeout := func(call otto.FunctionCall) otto.Value { - _, value := newTimer(call, false) - return value - } - - setInterval := func(call otto.FunctionCall) otto.Value { - _, value := newTimer(call, true) - return value - } - - clearTimeout := func(call otto.FunctionCall) otto.Value { - timer, _ := call.Argument(0).Export() - if timer, ok := timer.(*jsTimer); ok { - timer.timer.Stop() - delete(registry, timer) - } - return otto.UndefinedValue() - } - vm.Set("_setTimeout", setTimeout) - vm.Set("_setInterval", setInterval) - vm.Run(`var setTimeout = function(args) { - if (arguments.length < 1) { - throw TypeError("Failed to execute 'setTimeout': 1 argument required, but only 0 present."); - } - return _setTimeout.apply(this, arguments); - }`) - vm.Run(`var setInterval = function(args) { - if (arguments.length < 1) { - throw TypeError("Failed to execute 'setInterval': 1 argument required, but only 0 present."); - } - return _setInterval.apply(this, arguments); - }`) - vm.Set("clearTimeout", clearTimeout) - vm.Set("clearInterval", clearTimeout) - - var waitForCallbacks bool - -loop: - for { - select { - case timer := <-ready: - // execute callback, remove/reschedule the timer - var arguments []interface{} - if len(timer.call.ArgumentList) > 2 { - tmp := timer.call.ArgumentList[2:] - arguments = make([]interface{}, 2+len(tmp)) - for i, value := range tmp { - arguments[i+2] = value - } - } else { - arguments = make([]interface{}, 1) - } - arguments[0] = timer.call.ArgumentList[0] - _, err := vm.Call(`Function.call.call`, nil, arguments...) - if err != nil { - fmt.Println("js error:", err, arguments) - } - - _, inreg := registry[timer] // when clearInterval is called from within the callback don't reset it - if timer.interval && inreg { - timer.timer.Reset(timer.duration) - } else { - delete(registry, timer) - if waitForCallbacks && (len(registry) == 0) { - break loop - } - } - case req := <-self.evalQueue: - // run the code, send the result back - req.fn(vm) - close(req.done) - if waitForCallbacks && (len(registry) == 0) { - break loop - } - case waitForCallbacks = <-self.stopEventLoop: - if !waitForCallbacks || (len(registry) == 0) { - break loop - } - } - } - - for _, timer := range registry { - timer.timer.Stop() - delete(registry, timer) - } - - self.loopWg.Done() -} - -// Do executes the given function on the JS event loop. -func (self *JSRE) Do(fn func(*otto.Otto)) { - done := make(chan bool) - req := &evalReq{fn, done} - self.evalQueue <- req - <-done -} - -// stops the event loop before exit, optionally waits for all timers to expire -func (self *JSRE) Stop(waitForCallbacks bool) { - self.stopEventLoop <- waitForCallbacks - self.loopWg.Wait() -} - -// Exec(file) loads and runs the contents of a file -// if a relative path is given, the jsre's assetPath is used -func (self *JSRE) Exec(file string) error { - code, err := ioutil.ReadFile(common.AbsolutePath(self.assetPath, file)) - if err != nil { - return err - } - var script *otto.Script - self.Do(func(vm *otto.Otto) { - script, err = vm.Compile(file, code) - if err != nil { - return - } - _, err = vm.Run(script) - }) - return err -} - -// Bind assigns value v to a variable in the JS environment -// This method is deprecated, use Set. -func (self *JSRE) Bind(name string, v interface{}) error { - return self.Set(name, v) -} - -// Run runs a piece of JS code. -func (self *JSRE) Run(code string) (v otto.Value, err error) { - self.Do(func(vm *otto.Otto) { v, err = vm.Run(code) }) - return v, err -} - -// Get returns the value of a variable in the JS environment. -func (self *JSRE) Get(ns string) (v otto.Value, err error) { - self.Do(func(vm *otto.Otto) { v, err = vm.Get(ns) }) - return v, err -} - -// Set assigns value v to a variable in the JS environment. -func (self *JSRE) Set(ns string, v interface{}) (err error) { - self.Do(func(vm *otto.Otto) { err = vm.Set(ns, v) }) - return err -} - -// loadScript executes a JS script from inside the currently executing JS code. -func (self *JSRE) loadScript(call otto.FunctionCall) otto.Value { - file, err := call.Argument(0).ToString() - if err != nil { - // TODO: throw exception - return otto.FalseValue() - } - file = common.AbsolutePath(self.assetPath, file) - source, err := ioutil.ReadFile(file) - if err != nil { - // TODO: throw exception - return otto.FalseValue() - } - if _, err := compileAndRun(call.Otto, file, source); err != nil { - // TODO: throw exception - fmt.Println("err:", err) - return otto.FalseValue() - } - // TODO: return evaluation result - return otto.TrueValue() -} - -// EvalAndPrettyPrint evaluates code and pretty prints the result to -// standard output. -func (self *JSRE) EvalAndPrettyPrint(code string) (err error) { - self.Do(func(vm *otto.Otto) { - var val otto.Value - val, err = vm.Run(code) - if err != nil { - return - } - prettyPrint(vm, val) - fmt.Println() - }) - return err -} - -// Compile compiles and then runs a piece of JS code. -func (self *JSRE) Compile(filename string, src interface{}) (err error) { - self.Do(func(vm *otto.Otto) { _, err = compileAndRun(vm, filename, src) }) - return err -} - -func compileAndRun(vm *otto.Otto, filename string, src interface{}) (otto.Value, error) { - script, err := vm.Compile(filename, src) - if err != nil { - return otto.Value{}, err - } - return vm.Run(script) -} |