status-go/jail/internal/fetch/fetch.go

181 lines
3.6 KiB
Go

package fetch
//go:generate go-bindata -pkg fetch -o dist_fetch.go ./dist-fetch/
import (
"io"
"io/ioutil"
"net/http"
"net/http/httptest"
"strings"
"github.com/robertkrimen/otto"
"github.com/status-im/status-go/jail/internal/loop"
"github.com/status-im/status-go/jail/internal/promise"
"github.com/status-im/status-go/jail/internal/vm"
)
func mustValue(v otto.Value, err error) otto.Value {
if err != nil {
panic(err)
}
return v
}
type fetchTask struct {
id int64
jsReq, jsRes *otto.Object
cb otto.Value
err error
status int
statusText string
headers map[string][]string
body []byte
}
func (t *fetchTask) SetID(id int64) { t.id = id }
func (t *fetchTask) GetID() int64 { return t.id }
func (t *fetchTask) Execute(vm *vm.VM, l *loop.Loop) error {
var arguments []interface{}
if t.err != nil {
e, err := vm.Call(`new Error`, nil, t.err.Error())
if err != nil {
return err
}
arguments = append(arguments, e)
}
// We're locking on VM here because underlying otto's VM
// is not concurrently safe, and this function indirectly
// access vm's functions in cb.Call/h.Set.
vm.Lock()
defer vm.Unlock()
err := t.jsRes.Set("status", t.status)
if err != nil {
return err
}
err = t.jsRes.Set("statusText", t.statusText)
if err != nil {
return err
}
h := mustValue(t.jsRes.Get("headers")).Object()
for k, vs := range t.headers {
for _, v := range vs {
if _, err = h.Call("append", k, v); err != nil {
return err
}
}
}
err = t.jsRes.Set("_body", string(t.body))
if err != nil {
return err
}
_, err = t.cb.Call(otto.NullValue(), arguments...)
return err
}
func (t *fetchTask) Cancel() {
}
// Define fetch
func Define(vm *vm.VM, l *loop.Loop) error {
return DefineWithHandler(vm, l, nil)
}
//DefineWithHandler fetch with handler
func DefineWithHandler(vm *vm.VM, l *loop.Loop, h http.Handler) error {
if err := promise.Define(vm, l); err != nil {
return err
}
jsData := MustAsset("dist-fetch/bundle.js")
smData := MustAsset("dist-fetch/bundle.js.map")
s, err := vm.CompileWithSourceMap("fetch-bundle.js", jsData, smData)
if err != nil {
return err
}
_, err = vm.Run(s)
if err != nil {
return err
}
err = vm.Set("__private__fetch_execute", func(c otto.FunctionCall) otto.Value {
jsReq := c.Argument(0).Object()
jsRes := c.Argument(1).Object()
cb := c.Argument(2)
method := mustValue(jsReq.Get("method")).String()
urlStr := mustValue(jsReq.Get("url")).String()
jsBody := mustValue(jsReq.Get("body"))
var body io.Reader
if jsBody.IsString() {
body = strings.NewReader(jsBody.String())
}
t := &fetchTask{
jsReq: jsReq,
jsRes: jsRes,
cb: cb,
}
// If err is non-nil, then the loop is closed
// and we shouldn't do anymore with it.
if err := l.Add(t); err != nil {
return otto.UndefinedValue()
}
go func() {
defer l.Ready(t) // nolint: errcheck
req, rqErr := http.NewRequest(method, urlStr, body)
if rqErr != nil {
t.err = rqErr
return
}
if h != nil && urlStr[0] == '/' {
res := httptest.NewRecorder()
h.ServeHTTP(res, req)
t.status = res.Code
t.statusText = http.StatusText(res.Code)
t.headers = res.Header()
t.body = res.Body.Bytes()
} else {
res, e := http.DefaultClient.Do(req)
if e != nil {
t.err = e
return
}
d, e := ioutil.ReadAll(res.Body)
if e != nil {
t.err = e
return
}
t.status = res.StatusCode
t.statusText = res.Status
t.headers = res.Header
t.body = d
}
}()
return otto.UndefinedValue()
})
return err
}