2017-08-04 23:14:17 +07:00
|
|
|
package jail_test
|
|
|
|
|
|
|
|
import (
|
2017-09-08 13:55:17 +02:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
2017-08-04 23:14:17 +07:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/robertkrimen/otto"
|
|
|
|
)
|
|
|
|
|
|
|
|
func (s *JailTestSuite) TestJailTimeoutFailure() {
|
|
|
|
require := s.Require()
|
|
|
|
|
2017-09-02 20:04:23 +03:00
|
|
|
cell, err := s.jail.NewCell(testChatID)
|
2017-08-04 23:14:17 +07:00
|
|
|
require.NoError(err)
|
2017-09-01 22:17:34 +02:00
|
|
|
require.NotNil(cell)
|
2017-08-04 23:14:17 +07:00
|
|
|
|
2017-09-02 20:04:23 +03:00
|
|
|
// Attempt to run a timeout string against a Cell.
|
2017-09-01 22:17:34 +02:00
|
|
|
_, err = cell.Run(`
|
|
|
|
var timerCounts = 0;
|
|
|
|
setTimeout(function(n){
|
|
|
|
if (Date.now() - n < 50) {
|
|
|
|
throw new Error("Timed out");
|
|
|
|
}
|
|
|
|
|
|
|
|
timerCounts++;
|
|
|
|
}, 30, Date.now());
|
|
|
|
`)
|
2017-08-04 23:14:17 +07:00
|
|
|
require.NoError(err)
|
|
|
|
|
2017-09-01 22:17:34 +02:00
|
|
|
// wait at least 10x longer to decrease probability
|
|
|
|
// of false negatives as we using real clock here
|
|
|
|
time.Sleep(300 * time.Millisecond)
|
2017-08-04 23:14:17 +07:00
|
|
|
|
2017-09-01 22:17:34 +02:00
|
|
|
value, err := cell.Get("timerCounts")
|
2017-08-04 23:14:17 +07:00
|
|
|
require.NoError(err)
|
2017-09-01 22:17:34 +02:00
|
|
|
require.True(value.IsNumber())
|
|
|
|
require.Equal("0", value.String())
|
2017-08-04 23:14:17 +07:00
|
|
|
}
|
|
|
|
|
2017-09-01 22:17:34 +02:00
|
|
|
func (s *JailTestSuite) TestJailTimeout() {
|
2017-08-04 23:14:17 +07:00
|
|
|
require := s.Require()
|
|
|
|
|
2017-09-02 20:04:23 +03:00
|
|
|
cell, err := s.jail.NewCell(testChatID)
|
2017-08-04 23:14:17 +07:00
|
|
|
require.NoError(err)
|
2017-09-01 22:17:34 +02:00
|
|
|
require.NotNil(cell)
|
2017-08-04 23:14:17 +07:00
|
|
|
|
2017-09-02 20:04:23 +03:00
|
|
|
// Attempt to run a timeout string against a Cell.
|
2017-09-01 22:17:34 +02:00
|
|
|
_, err = cell.Run(`
|
|
|
|
var timerCounts = 0;
|
|
|
|
setTimeout(function(n){
|
|
|
|
if (Date.now() - n < 50) {
|
|
|
|
throw new Error("Timed out");
|
|
|
|
}
|
|
|
|
|
|
|
|
timerCounts++;
|
|
|
|
}, 50, Date.now());
|
|
|
|
`)
|
|
|
|
require.NoError(err)
|
2017-08-04 23:14:17 +07:00
|
|
|
|
2017-09-01 22:17:34 +02:00
|
|
|
// wait at least 10x longer to decrease probability
|
|
|
|
// of false negatives as we using real clock here
|
|
|
|
time.Sleep(300 * time.Millisecond)
|
2017-08-04 23:14:17 +07:00
|
|
|
|
2017-09-01 22:17:34 +02:00
|
|
|
value, err := cell.Get("timerCounts")
|
2017-08-04 23:14:17 +07:00
|
|
|
require.NoError(err)
|
2017-09-01 22:17:34 +02:00
|
|
|
require.True(value.IsNumber())
|
|
|
|
require.Equal("1", value.String())
|
2017-08-04 23:14:17 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *JailTestSuite) TestJailLoopInCall() {
|
|
|
|
require := s.Require()
|
|
|
|
|
|
|
|
// load Status JS and add test command to it
|
|
|
|
s.jail.BaseJS(baseStatusJSCode)
|
|
|
|
s.jail.Parse(testChatID, ``)
|
|
|
|
|
2017-09-02 20:04:23 +03:00
|
|
|
cell, err := s.jail.Cell(testChatID)
|
2017-08-04 23:14:17 +07:00
|
|
|
require.NoError(err)
|
|
|
|
require.NotNil(cell)
|
|
|
|
|
|
|
|
items := make(chan string)
|
|
|
|
|
|
|
|
err = cell.Set("__captureResponse", func(val string) otto.Value {
|
|
|
|
go func() { items <- val }()
|
|
|
|
return otto.UndefinedValue()
|
|
|
|
})
|
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
_, err = cell.Run(`
|
|
|
|
function callRunner(namespace){
|
|
|
|
console.log("Initiating callRunner for: ", namespace)
|
|
|
|
return setTimeout(function(){
|
|
|
|
__captureResponse(namespace);
|
|
|
|
}, 1000);
|
|
|
|
}
|
|
|
|
`)
|
|
|
|
require.NoError(err)
|
|
|
|
|
2017-09-01 22:17:34 +02:00
|
|
|
_, err = cell.Call("callRunner", nil, "softball")
|
2017-08-04 23:14:17 +07:00
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
select {
|
|
|
|
case received := <-items:
|
|
|
|
require.Equal(received, "softball")
|
|
|
|
case <-time.After(5 * time.Second):
|
|
|
|
require.Fail("Failed to received event response")
|
|
|
|
}
|
|
|
|
}
|
2017-09-08 13:55:17 +02:00
|
|
|
|
|
|
|
// TestJailLoopRace tests multiple setTimeout callbacks,
|
|
|
|
// supposed to be run with '-race' flag.
|
|
|
|
func (s *JailTestSuite) TestJailLoopRace() {
|
|
|
|
require := s.Require()
|
|
|
|
|
|
|
|
cell, err := s.jail.NewCell(testChatID)
|
|
|
|
require.NoError(err)
|
|
|
|
require.NotNil(cell)
|
|
|
|
|
|
|
|
items := make(chan struct{})
|
|
|
|
|
|
|
|
err = cell.Set("__captureResponse", func() otto.Value {
|
|
|
|
go func() { items <- struct{}{} }()
|
|
|
|
return otto.UndefinedValue()
|
|
|
|
})
|
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
_, err = cell.Run(`
|
|
|
|
function callRunner(){
|
|
|
|
return setTimeout(function(){
|
|
|
|
__captureResponse();
|
|
|
|
}, 1000);
|
|
|
|
}
|
|
|
|
`)
|
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
for i := 0; i < 100; i++ {
|
|
|
|
_, err = cell.Call("callRunner", nil)
|
|
|
|
require.NoError(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < 100; i++ {
|
|
|
|
select {
|
|
|
|
case <-items:
|
|
|
|
case <-time.After(5 * time.Second):
|
|
|
|
require.Fail("test timed out")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *JailTestSuite) TestJailFetchPromise() {
|
|
|
|
body := `{"key": "value"}`
|
|
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
w.Header().Add("Content-Type", "application/json")
|
|
|
|
w.Write([]byte(body))
|
|
|
|
}))
|
|
|
|
defer server.Close()
|
|
|
|
|
|
|
|
require := s.Require()
|
|
|
|
|
|
|
|
cell, err := s.jail.NewCell(testChatID)
|
|
|
|
require.NoError(err)
|
|
|
|
require.NotNil(cell)
|
|
|
|
|
|
|
|
dataCh := make(chan otto.Value, 1)
|
|
|
|
errCh := make(chan otto.Value, 1)
|
|
|
|
|
|
|
|
err = cell.Set("__captureSuccess", func(res otto.Value) { dataCh <- res })
|
|
|
|
require.NoError(err)
|
|
|
|
err = cell.Set("__captureError", func(res otto.Value) { errCh <- res })
|
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
// run JS code for fetching valid URL
|
|
|
|
_, err = cell.Run(`fetch('` + server.URL + `').then(function(r) {
|
|
|
|
return r.text()
|
|
|
|
}).then(function(data) {
|
|
|
|
__captureSuccess(data)
|
|
|
|
}).catch(function (e) {
|
|
|
|
__captureError(e)
|
|
|
|
})`)
|
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
select {
|
|
|
|
case data := <-dataCh:
|
|
|
|
require.True(data.IsString())
|
|
|
|
require.Equal(body, data.String())
|
|
|
|
case err := <-errCh:
|
|
|
|
require.Fail("request failed", err)
|
|
|
|
case <-time.After(1 * time.Second):
|
|
|
|
require.Fail("test timed out")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *JailTestSuite) TestJailFetchCatch() {
|
|
|
|
require := s.Require()
|
|
|
|
|
|
|
|
cell, err := s.jail.NewCell(testChatID)
|
|
|
|
require.NoError(err)
|
|
|
|
require.NotNil(cell)
|
|
|
|
|
|
|
|
dataCh := make(chan otto.Value, 1)
|
|
|
|
errCh := make(chan otto.Value, 1)
|
|
|
|
|
|
|
|
err = cell.Set("__captureSuccess", func(res otto.Value) { dataCh <- res })
|
|
|
|
require.NoError(err)
|
|
|
|
err = cell.Set("__captureError", func(res otto.Value) { errCh <- res })
|
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
// run JS code for fetching invalid URL
|
|
|
|
_, err = cell.Run(`fetch('http://👽/nonexistent').then(function(r) {
|
|
|
|
return r.text()
|
|
|
|
}).then(function(data) {
|
|
|
|
__captureSuccess(data)
|
|
|
|
}).catch(function (e) {
|
|
|
|
__captureError(e)
|
|
|
|
})`)
|
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
select {
|
|
|
|
case data := <-dataCh:
|
|
|
|
require.Fail("request should have failed, but returned", data)
|
|
|
|
case e := <-errCh:
|
|
|
|
require.True(e.IsObject())
|
|
|
|
name, err := e.Object().Get("name")
|
|
|
|
require.NoError(err)
|
|
|
|
require.Equal("Error", name.String())
|
|
|
|
_, err = e.Object().Get("message")
|
|
|
|
require.NoError(err)
|
|
|
|
case <-time.After(1 * time.Second):
|
|
|
|
require.Fail("test timed out")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestJailFetchRace tests multiple fetch callbacks,
|
|
|
|
// supposed to be run with '-race' flag.
|
|
|
|
func (s *JailTestSuite) TestJailFetchRace() {
|
|
|
|
body := `{"key": "value"}`
|
|
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
w.Header().Add("Content-Type", "application/json")
|
|
|
|
w.Write([]byte(body))
|
|
|
|
}))
|
|
|
|
defer server.Close()
|
|
|
|
require := s.Require()
|
|
|
|
|
|
|
|
cell, err := s.jail.NewCell(testChatID)
|
|
|
|
require.NoError(err)
|
|
|
|
require.NotNil(cell)
|
|
|
|
|
|
|
|
dataCh := make(chan otto.Value, 1)
|
|
|
|
errCh := make(chan otto.Value, 1)
|
|
|
|
|
|
|
|
err = cell.Set("__captureSuccess", func(res otto.Value) { dataCh <- res })
|
|
|
|
require.NoError(err)
|
|
|
|
err = cell.Set("__captureError", func(res otto.Value) { errCh <- res })
|
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
// run JS code for fetching valid URL
|
|
|
|
_, err = cell.Run(`fetch('` + server.URL + `').then(function(r) {
|
|
|
|
return r.text()
|
|
|
|
}).then(function(data) {
|
|
|
|
__captureSuccess(data)
|
|
|
|
}).catch(function (e) {
|
|
|
|
__captureError(e)
|
|
|
|
})`)
|
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
// run JS code for fetching invalid URL
|
|
|
|
_, err = cell.Run(`fetch('http://👽/nonexistent').then(function(r) {
|
|
|
|
return r.text()
|
|
|
|
}).then(function(data) {
|
|
|
|
__captureSuccess(data)
|
|
|
|
}).catch(function (e) {
|
|
|
|
__captureError(e)
|
|
|
|
})`)
|
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
for i := 0; i < 2; i++ {
|
|
|
|
select {
|
|
|
|
case data := <-dataCh:
|
|
|
|
require.True(data.IsString())
|
|
|
|
require.Equal(body, data.String())
|
|
|
|
case e := <-errCh:
|
|
|
|
require.True(e.IsObject())
|
|
|
|
name, err := e.Object().Get("name")
|
|
|
|
require.NoError(err)
|
|
|
|
require.Equal("Error", name.String())
|
|
|
|
_, err = e.Object().Get("message")
|
|
|
|
require.NoError(err)
|
|
|
|
case <-time.After(1 * time.Second):
|
|
|
|
require.Fail("test timed out")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|