-
Notifications
You must be signed in to change notification settings - Fork 1.3k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add basic event loop with an API to be used by modules
As well as cut down setTimeout implementation. A recent update to goja introduced support for ECMAScript Promise. The catch here is that Promise's then will only be called when goja exits executing js code and it has already been resolved. Also resolving and rejecting Promises needs to happen while no other js code is being executed as it will otherwise lead to a data race. This more or less necessitates adding an event loop. Additionally because a call to a k6 modules such as `k6/http` might make a promise to signal when an http request is made, but if (no changes were made) the iteration then finishes before the request completes, nothing would've stopped the start of a *new* iteration. That new iteration would then probably just again ask k6/http to make a new request with a Promise ... This might be a desirable behaviour for some cases but arguably will be very confusing so this commit also adds a way to RegisterCallback that will return a function to actually queue the callback on the event loop, but prevent the event loop from ending before the callback is queued and possible executed, once RegisterCallback is called. Additionally to that, some additional code was needed so there is an event loop for all special functions calls (setup, teardown, handleSummary, default) and the init context. This also adds handling of rejected promise which don't have a reject handler similar to what deno does. It also adds a per iteration context that gets canceled on the end of each iteration letting other code know that it needs to stop. This is particularly needed here as if an iteration gets aborted by a syntax error (or unhandled promise rejection), a new iteration will start right after that. But this means that any in-flight asynchronous operation (an http requests for example) will *not* get stopped. With a context that gets canceled every time module code can notice that and abort any operation. For this same reason the event loop needs wait to be *empty* before the iteration ends. This did lead to some ... not very nice code, but a whole package needs a big refactor which will likely happen once common.Bind and co gets removed. And finally, a basic setTimeout implementation was added. There is no way to currently cancel the setTimeout - no clearTimeout. This likely needs to be extended but this can definitely wait. Or we might decide to actually drop setTimeout altogether as it isn't particularly useful currently without any async APIs, it just makes testing the event loop functionality possible. fixes #882
- Loading branch information
Showing
9 changed files
with
561 additions
and
26 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,181 @@ | ||
package local | ||
|
||
import ( | ||
"context" | ||
"io/ioutil" | ||
"net/url" | ||
"testing" | ||
"time" | ||
|
||
"github.com/sirupsen/logrus" | ||
"github.com/stretchr/testify/require" | ||
"go.k6.io/k6/js" | ||
"go.k6.io/k6/lib" | ||
"go.k6.io/k6/lib/metrics" | ||
"go.k6.io/k6/lib/testutils" | ||
"go.k6.io/k6/lib/types" | ||
"go.k6.io/k6/loader" | ||
) | ||
|
||
func eventLoopTest(t *testing.T, script []byte, testHandle func(context.Context, lib.Runner, error, *testutils.SimpleLogrusHook)) { | ||
logger := logrus.New() | ||
logger.SetOutput(ioutil.Discard) | ||
logHook := &testutils.SimpleLogrusHook{HookedLevels: []logrus.Level{logrus.InfoLevel, logrus.WarnLevel, logrus.ErrorLevel}} | ||
logger.AddHook(logHook) | ||
|
||
registry := metrics.NewRegistry() | ||
builtinMetrics := metrics.RegisterBuiltinMetrics(registry) | ||
runner, err := js.New( | ||
logger, | ||
&loader.SourceData{ | ||
URL: &url.URL{Path: "/script.js"}, | ||
Data: script, | ||
}, | ||
nil, | ||
lib.RuntimeOptions{}, | ||
builtinMetrics, | ||
registry, | ||
) | ||
require.NoError(t, err) | ||
|
||
ctx, cancel, execScheduler, samples := newTestExecutionScheduler(t, runner, logger, | ||
lib.Options{ | ||
TeardownTimeout: types.NullDurationFrom(time.Second), | ||
SetupTimeout: types.NullDurationFrom(time.Second), | ||
}) | ||
defer cancel() | ||
|
||
errCh := make(chan error, 1) | ||
go func() { errCh <- execScheduler.Run(ctx, ctx, samples, builtinMetrics) }() | ||
|
||
select { | ||
case err := <-errCh: | ||
testHandle(ctx, runner, err, logHook) | ||
case <-time.After(10 * time.Second): | ||
t.Fatal("timed out") | ||
} | ||
} | ||
|
||
func TestEventLoop(t *testing.T) { | ||
t.Parallel() | ||
script := []byte(` | ||
setTimeout(()=> {console.log("initcontext setTimeout")}, 200) | ||
console.log("initcontext"); | ||
export default function() { | ||
setTimeout(()=> {console.log("default setTimeout")}, 200) | ||
console.log("default"); | ||
}; | ||
export function setup() { | ||
setTimeout(()=> {console.log("setup setTimeout")}, 200) | ||
console.log("setup"); | ||
}; | ||
export function teardown() { | ||
setTimeout(()=> {console.log("teardown setTimeout")}, 200) | ||
console.log("teardown"); | ||
}; | ||
export function handleSummary() { | ||
setTimeout(()=> {console.log("handleSummary setTimeout")}, 200) | ||
console.log("handleSummary"); | ||
}; | ||
`) | ||
eventLoopTest(t, script, func(ctx context.Context, runner lib.Runner, err error, logHook *testutils.SimpleLogrusHook) { | ||
require.NoError(t, err) | ||
_, err = runner.HandleSummary(ctx, &lib.Summary{RootGroup: &lib.Group{}}) | ||
require.NoError(t, err) | ||
entries := logHook.Drain() | ||
msgs := make([]string, len(entries)) | ||
for i, entry := range entries { | ||
msgs[i] = entry.Message | ||
} | ||
require.Equal(t, []string{ | ||
"initcontext", // first initialization | ||
"initcontext setTimeout", | ||
"initcontext", // for vu | ||
"initcontext setTimeout", | ||
"initcontext", // for setup | ||
"initcontext setTimeout", | ||
"setup", // setup | ||
"setup setTimeout", | ||
"default", // one iteration | ||
"default setTimeout", | ||
"initcontext", // for teardown | ||
"initcontext setTimeout", | ||
"teardown", // teardown | ||
"teardown setTimeout", | ||
"initcontext", // for handleSummary | ||
"initcontext setTimeout", | ||
"handleSummary", // handleSummary | ||
"handleSummary setTimeout", | ||
}, msgs) | ||
}) | ||
} | ||
|
||
func TestEventLoopCrossScenario(t *testing.T) { | ||
t.Parallel() | ||
script := []byte(` | ||
import exec from "k6/execution" | ||
export const options = { | ||
scenarios: { | ||
"first":{ | ||
executor: "shared-iterations", | ||
maxDuration: "1s", | ||
iterations: 1, | ||
vus: 1, | ||
gracefulStop:"1s", | ||
}, | ||
"second": { | ||
executor: "shared-iterations", | ||
maxDuration: "1s", | ||
iterations: 1, | ||
vus: 1, | ||
startTime: "3s", | ||
} | ||
} | ||
} | ||
export default function() { | ||
let i = exec.scenario.name | ||
setTimeout(()=> {console.log(i)}, 3000) | ||
} | ||
`) | ||
|
||
eventLoopTest(t, script, func(_ context.Context, _ lib.Runner, err error, logHook *testutils.SimpleLogrusHook) { | ||
require.NoError(t, err) | ||
entries := logHook.Drain() | ||
msgs := make([]string, len(entries)) | ||
for i, entry := range entries { | ||
msgs[i] = entry.Message | ||
} | ||
require.Equal(t, []string{"second"}, msgs) | ||
}) | ||
} | ||
|
||
func TestEventLoopDoesntCrossIterations(t *testing.T) { | ||
t.Parallel() | ||
script := []byte(` | ||
import { sleep } from "k6" | ||
export const options = { | ||
iterations: 2, | ||
vus: 1, | ||
} | ||
export default function() { | ||
let i = __ITER; | ||
setTimeout(()=> { console.log(i) }, 1000) | ||
if (__ITER == 0) { | ||
throw "just error" | ||
} else { | ||
sleep(1) | ||
} | ||
} | ||
`) | ||
|
||
eventLoopTest(t, script, func(_ context.Context, _ lib.Runner, err error, logHook *testutils.SimpleLogrusHook) { | ||
require.NoError(t, err) | ||
entries := logHook.Drain() | ||
msgs := make([]string, len(entries)) | ||
for i, entry := range entries { | ||
msgs[i] = entry.Message | ||
} | ||
require.Equal(t, []string{"just error\n\tat /script.js:12:4(13)\n\tat native\n", "1"}, msgs) | ||
}) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
/* | ||
Package js is the JavaScript implementation of the lib.Runner and relative concepts for | ||
executing concurrent-safe JavaScript code. | ||
*/ | ||
package js |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,166 @@ | ||
package js | ||
|
||
import ( | ||
"fmt" | ||
"sync" | ||
"time" | ||
|
||
"github.com/dop251/goja" | ||
"go.k6.io/k6/js/modules" | ||
) | ||
|
||
// eventLoop implements an event loop with a cut-down setTimeout and | ||
// handling of unhandled rejected promises. | ||
// | ||
// A specific thing about this event loop is that it will wait to return | ||
// not only until the queue is empty but until nothing is registered that it will run in the future. | ||
// This is in contrast with more common behaviours where it only returns on | ||
// a specific event/action or when the loop is empty. | ||
// This is required as in k6 iterations (for which event loop will be primary used) | ||
// are supposed to be independent and any work started in them needs to finish, | ||
// but also they need to end when all the instructions are done. | ||
// Additionally because of this on any error while the event loop will exit it's | ||
// required to wait on the event loop to be empty before the execution can continue. | ||
type eventLoop struct { | ||
lock sync.Mutex | ||
queue []func() error | ||
wakeupCh chan struct{} // maybe use sync.Cond ? | ||
registeredCallbacks int | ||
vu modules.VU | ||
|
||
// pendingPromiseRejections are rejected promises with no handler, | ||
// if there is something in this map at an end of an event loop then it will exit with an error. | ||
// It's similar to what Deno and Node do. | ||
pendingPromiseRejections map[*goja.Promise]struct{} | ||
} | ||
|
||
// newEventLoop returns a new event loop with a few helpers attached to it: | ||
// - adding setTimeout javascript implementation | ||
// - reporting (and aborting on) unhandled promise rejections | ||
func newEventLoop(vu modules.VU) *eventLoop { | ||
e := &eventLoop{ | ||
wakeupCh: make(chan struct{}, 1), | ||
pendingPromiseRejections: make(map[*goja.Promise]struct{}), | ||
vu: vu, | ||
} | ||
vu.Runtime().SetPromiseRejectionTracker(e.promiseRejectionTracker) | ||
e.addSetTimeout() | ||
|
||
return e | ||
} | ||
|
||
func (e *eventLoop) wakeup() { | ||
select { | ||
case e.wakeupCh <- struct{}{}: | ||
default: | ||
} | ||
} | ||
|
||
// registerCallback register that a callback will be invoked on the loop, preventing it from returning/finishing. | ||
// The returned function, upon invocation, will queue it's argument and wakeup the loop if needed. | ||
// If the eventLoop has since stopped, it will not be executed. | ||
// This function *must* be called from within running on the event loop, but its result can be called from anywhere | ||
func (e *eventLoop) registerCallback() func(func() error) { | ||
e.lock.Lock() | ||
e.registeredCallbacks++ | ||
e.lock.Unlock() | ||
|
||
return func(f func() error) { | ||
e.lock.Lock() | ||
e.queue = append(e.queue, f) | ||
e.registeredCallbacks-- | ||
e.lock.Unlock() | ||
e.wakeup() | ||
} | ||
} | ||
|
||
func (e *eventLoop) promiseRejectionTracker(p *goja.Promise, op goja.PromiseRejectionOperation) { | ||
// No locking necessary here as the goja runtime will call this synchronously | ||
// Read Notes on https://tc39.es/ecma262/#sec-host-promise-rejection-tracker | ||
if op == goja.PromiseRejectionReject { | ||
e.pendingPromiseRejections[p] = struct{}{} | ||
} else { // goja.PromiseRejectionHandle so a promise that was previously rejected without handler now got one | ||
delete(e.pendingPromiseRejections, p) | ||
} | ||
} | ||
|
||
func (e *eventLoop) popAll() (queue []func() error, awaiting bool) { | ||
e.lock.Lock() | ||
queue = e.queue | ||
e.queue = make([]func() error, 0, len(queue)) | ||
awaiting = e.registeredCallbacks != 0 | ||
e.lock.Unlock() | ||
return | ||
} | ||
|
||
// start will run the event loop until it's empty and there are no unvoked registered callbacks | ||
// or a queued function returns an error. The provided firstCallback will be the first thing executed. | ||
// After start returns the event loop can be reused as long as waitOnRegistered is called | ||
func (e *eventLoop) start(firstCallback func() error) error { | ||
e.queue = []func() error{firstCallback} | ||
for { | ||
queue, awaiting := e.popAll() | ||
|
||
if len(queue) == 0 { | ||
if !awaiting { | ||
return nil | ||
} | ||
<-e.wakeupCh | ||
continue | ||
} | ||
|
||
for _, f := range queue { | ||
if err := f(); err != nil { | ||
return err | ||
} | ||
} | ||
|
||
// This will get a random unhandled rejection instead of the first one, for example. | ||
// But that seems to be the case in other tools as well so it seems to not be that big of a problem. | ||
for promise := range e.pendingPromiseRejections { | ||
// TODO maybe throw the whole promise up and get make a better message outside of the event loop | ||
value := promise.Result() | ||
// try to get the stack without actually needing the runtime | ||
// this might break in the future :( | ||
if o, ok := promise.Result().(*goja.Object); ok { | ||
stack := o.Get("stack") | ||
if stack != nil { | ||
value = stack | ||
} | ||
} | ||
// this is the de facto wording in both firefox and deno at least | ||
return fmt.Errorf("Uncaught (in promise) %s", value) //nolint:stylecheck | ||
} | ||
} | ||
} | ||
|
||
// this exists so we can wait on all registered callbacks so we know nothing is still doing work | ||
func (e *eventLoop) waitOnRegistered() { | ||
for { | ||
_, awaiting := e.popAll() | ||
if !awaiting { | ||
return | ||
} | ||
<-e.wakeupCh | ||
} | ||
} | ||
|
||
func (e *eventLoop) addSetTimeout() { | ||
_ = e.vu.Runtime().Set("setTimeout", func(f func() error, t float64) { | ||
// TODO maybe really return something to use with `clearTimeout | ||
// TODO support arguments ... maybe | ||
runOnLoop := e.registerCallback() | ||
go func() { | ||
timer := time.NewTimer(time.Duration(t * float64(time.Millisecond))) | ||
select { | ||
case <-timer.C: | ||
runOnLoop(f) | ||
case <-e.vu.Context().Done(): | ||
// TODO log something? | ||
|
||
timer.Stop() | ||
runOnLoop(func() error { return nil }) | ||
} | ||
}() | ||
}) | ||
} |
Oops, something went wrong.