mirror of
https://github.com/status-im/status-go.git
synced 2025-01-12 07:35:02 +00:00
6a096607cf
This PR adds Fetch API and fixes #289 by using concurrency safe Otto VM wrapper wherever it's possible. This involves new package geth/jail/vm that is used by jail and by our forked ottoext/{fetch/timers/loop} packages. It also adds more tests that are supposed to be run with --race flag of go test.
155 lines
3.5 KiB
Go
155 lines
3.5 KiB
Go
package loop
|
|
|
|
import (
|
|
"fmt"
|
|
"sync"
|
|
"sync/atomic"
|
|
|
|
"github.com/status-im/status-go/geth/jail/internal/vm"
|
|
)
|
|
|
|
func formatTask(t Task) string {
|
|
if t == nil {
|
|
return "<nil>"
|
|
}
|
|
|
|
return fmt.Sprintf("<%T> %d", t, t.GetID())
|
|
}
|
|
|
|
// Task represents something that the event loop can schedule and run.
|
|
//
|
|
// Task describes two operations that will almost always be boilerplate,
|
|
// SetID and GetID. They exist so that the event loop can identify tasks
|
|
// after they're added.
|
|
//
|
|
// Execute is called when a task has been pulled from the "ready" queue.
|
|
//
|
|
// Cancel is called when a task is removed from the loop without being
|
|
// finalised.
|
|
type Task interface {
|
|
SetID(id int64)
|
|
GetID() int64
|
|
Execute(vm *vm.VM, l *Loop) error
|
|
Cancel()
|
|
}
|
|
|
|
// Loop encapsulates the event loop's state. This includes the vm on which the
|
|
// loop operates, a monotonically incrementing event id, a map of tasks that
|
|
// aren't ready yet, keyed by their ID, and a channel of tasks that are ready
|
|
// to finalise on the VM. The channel holding the tasks pending finalising can
|
|
// be buffered or unbuffered.
|
|
type Loop struct {
|
|
vm *vm.VM
|
|
id int64
|
|
lock sync.RWMutex
|
|
tasks map[int64]Task
|
|
ready chan Task
|
|
closed bool
|
|
}
|
|
|
|
// New creates a new Loop with an unbuffered ready queue on a specific VM.
|
|
func New(vm *vm.VM) *Loop {
|
|
return NewWithBacklog(vm, 0)
|
|
}
|
|
|
|
// NewWithBacklog creates a new Loop on a specific VM, giving it a buffered
|
|
// queue, the capacity of which being specified by the backlog argument.
|
|
func NewWithBacklog(vm *vm.VM, backlog int) *Loop {
|
|
return &Loop{
|
|
vm: vm,
|
|
tasks: make(map[int64]Task),
|
|
ready: make(chan Task, backlog),
|
|
}
|
|
}
|
|
|
|
// VM gets the JavaScript interpreter associated with the loop.
|
|
func (l *Loop) VM() *vm.VM {
|
|
return l.vm
|
|
}
|
|
|
|
// Add puts a task into the loop. This signals to the loop that this task is
|
|
// doing something outside of the JavaScript environment, and that at some
|
|
// point, it will become ready for finalising.
|
|
func (l *Loop) Add(t Task) {
|
|
l.lock.Lock()
|
|
t.SetID(atomic.AddInt64(&l.id, 1))
|
|
l.tasks[t.GetID()] = t
|
|
l.lock.Unlock()
|
|
}
|
|
|
|
// Remove takes a task out of the loop. This should not be called if a task
|
|
// has already become ready for finalising. Warranty void if constraint is
|
|
// broken.
|
|
func (l *Loop) Remove(t Task) {
|
|
l.remove(t)
|
|
go l.Ready(nil)
|
|
}
|
|
|
|
func (l *Loop) remove(t Task) {
|
|
l.removeByID(t.GetID())
|
|
}
|
|
|
|
func (l *Loop) removeByID(id int64) {
|
|
l.lock.Lock()
|
|
delete(l.tasks, id)
|
|
l.lock.Unlock()
|
|
}
|
|
|
|
// Ready signals to the loop that a task is ready to be finalised. This might
|
|
// block if the "ready channel" in the loop is at capacity.
|
|
func (l *Loop) Ready(t Task) {
|
|
if l.closed {
|
|
return
|
|
}
|
|
|
|
l.ready <- t
|
|
}
|
|
|
|
// Eval executes some code in the VM associated with the loop and returns an
|
|
// error if that execution fails.
|
|
func (l *Loop) Eval(s interface{}) error {
|
|
if _, err := l.vm.Run(s); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (l *Loop) processTask(t Task) error {
|
|
id := t.GetID()
|
|
|
|
if err := t.Execute(l.vm, l); err != nil {
|
|
l.lock.RLock()
|
|
for _, t := range l.tasks {
|
|
t.Cancel()
|
|
}
|
|
l.lock.RUnlock()
|
|
|
|
return err
|
|
}
|
|
|
|
l.removeByID(id)
|
|
|
|
return nil
|
|
}
|
|
|
|
// Run handles the task scheduling and finalisation.
|
|
// It runs infinitely waiting for new tasks.
|
|
func (l *Loop) Run() error {
|
|
for t := range l.ready {
|
|
if t == nil {
|
|
continue
|
|
}
|
|
|
|
err := l.processTask(t)
|
|
if err != nil {
|
|
// TODO(divan): do we need to report
|
|
// errors up to the caller?
|
|
// Ignoring for now, as loop
|
|
// should keep running.
|
|
continue
|
|
}
|
|
}
|
|
return nil
|
|
}
|