2017-09-01 20:17:34 +00:00
|
|
|
package fetch
|
2017-07-13 11:04:47 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/GeertJohan/go.rice"
|
|
|
|
"github.com/robertkrimen/otto"
|
|
|
|
|
2017-09-08 11:55:17 +00:00
|
|
|
"github.com/status-im/status-go/geth/jail/internal/loop"
|
|
|
|
"github.com/status-im/status-go/geth/jail/internal/promise"
|
|
|
|
"github.com/status-im/status-go/geth/jail/internal/vm"
|
2017-07-13 11:04:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
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 }
|
|
|
|
|
2017-09-08 11:55:17 +00:00
|
|
|
func (t *fetchTask) Execute(vm *vm.VM, l *loop.Loop) error {
|
2017-07-13 11:04:47 +00:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2017-09-08 11:55:17 +00:00
|
|
|
// 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()
|
|
|
|
|
2017-07-13 11:04:47 +00:00
|
|
|
t.jsRes.Set("status", t.status)
|
|
|
|
t.jsRes.Set("statusText", t.statusText)
|
|
|
|
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
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
t.jsRes.Set("_body", string(t.body))
|
|
|
|
|
|
|
|
if _, err := t.cb.Call(otto.NullValue(), arguments...); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *fetchTask) Cancel() {
|
|
|
|
}
|
|
|
|
|
2017-09-08 11:55:17 +00:00
|
|
|
func Define(vm *vm.VM, l *loop.Loop) error {
|
2017-07-13 11:04:47 +00:00
|
|
|
return DefineWithHandler(vm, l, nil)
|
|
|
|
}
|
|
|
|
|
2017-09-08 11:55:17 +00:00
|
|
|
func DefineWithHandler(vm *vm.VM, l *loop.Loop, h http.Handler) error {
|
2017-07-13 11:04:47 +00:00
|
|
|
if err := promise.Define(vm, l); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
jsData := rice.MustFindBox("dist-fetch").MustString("bundle.js")
|
|
|
|
smData := rice.MustFindBox("dist-fetch").MustString("bundle.js.map")
|
|
|
|
|
|
|
|
s, err := vm.CompileWithSourceMap("fetch-bundle.js", jsData, smData)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := vm.Run(s); err != nil {
|
|
|
|
return 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,
|
|
|
|
}
|
|
|
|
|
|
|
|
l.Add(t)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
defer l.Ready(t)
|
|
|
|
|
|
|
|
req, err := http.NewRequest(method, urlStr, body)
|
|
|
|
if err != nil {
|
|
|
|
t.err = err
|
|
|
|
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, err := http.DefaultClient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
t.err = err
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
d, err := ioutil.ReadAll(res.Body)
|
|
|
|
if err != nil {
|
|
|
|
t.err = err
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
t.status = res.StatusCode
|
|
|
|
t.statusText = res.Status
|
|
|
|
t.headers = res.Header
|
|
|
|
t.body = d
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return otto.UndefinedValue()
|
|
|
|
})
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|