aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorGravatar Jarred Sumner <709451+Jarred-Sumner@users.noreply.github.com> 2022-11-02 00:14:32 -0700
committerGravatar Jarred Sumner <709451+Jarred-Sumner@users.noreply.github.com> 2022-11-02 00:14:32 -0700
commit86639fe62a8b94fe68c64846d18de4dcf16c09e6 (patch)
treeb3ac486256249fa9e2b15d33acb4c6312a1075c0 /src
parent54e7a6f57a80ac3a9e96348f0f796b219c958b73 (diff)
downloadbun-86639fe62a8b94fe68c64846d18de4dcf16c09e6.tar.gz
bun-86639fe62a8b94fe68c64846d18de4dcf16c09e6.tar.zst
bun-86639fe62a8b94fe68c64846d18de4dcf16c09e6.zip
Improve async function handling code in setTimeout and setInterval
Should fix
Diffstat (limited to 'src')
-rw-r--r--src/bun.js/api/bun.zig89
1 files changed, 70 insertions, 19 deletions
diff --git a/src/bun.js/api/bun.zig b/src/bun.js/api/bun.zig
index a343730f8..5189da1a7 100644
--- a/src/bun.js/api/bun.zig
+++ b/src/bun.js/api/bun.zig
@@ -2252,37 +2252,83 @@ pub const Timer = struct {
pub const Task = JSC.AnyTask.New(CallbackJob, perform);
- pub fn perform(this: *CallbackJob) void {
- var vm = this.globalThis.bunVM();
- var map: *TimeoutMap = if (this.repeat) &vm.timer.interval_map else &vm.timer.timeout_map;
+ pub export fn CallbackJob__onResolve(_: *JSC.JSGlobalObject, callframe: *JSC.CallFrame) callconv(.C) JSValue {
+ const args = callframe.arguments(2);
+ if (args.len < 2) {
+ return JSValue.jsUndefined();
+ }
- defer {
- this.callback.deinit();
- this.ref.unref(this.globalThis.bunVM());
- bun.default_allocator.destroy(this);
+ var this = args.ptr[1].asPtr(CallbackJob);
+ this.deinit();
+ return JSValue.jsUndefined();
+ }
+
+ pub export fn CallbackJob__onReject(globalThis: *JSC.JSGlobalObject, callframe: *JSC.CallFrame) callconv(.C) JSValue {
+ const args = callframe.arguments(2);
+ if (args.len < 2) {
+ return JSValue.jsUndefined();
}
+ var this = args.ptr[1].asPtr(CallbackJob);
+ globalThis.bunVM().runErrorHandler(args.ptr[0], null);
+ this.deinit();
+ return JSValue.jsUndefined();
+ }
+
+ pub fn deinit(this: *CallbackJob) void {
+ this.callback.deinit();
+ this.ref.unref(this.globalThis.bunVM());
+ bun.default_allocator.destroy(this);
+ }
+
+ pub fn perform(this: *CallbackJob) void {
+ var globalThis = this.globalThis;
+ var vm = globalThis.bunVM();
+ var map: *TimeoutMap = if (this.repeat) &vm.timer.interval_map else &vm.timer.timeout_map;
+
// This doesn't deinit the timer
// Timers are deinit'd separately
// We do need to handle when the timer is cancelled after the job has been enqueued
if (!this.repeat) {
if (map.fetchSwapRemove(this.id) == null) {
// if the timeout was cancelled, don't run the callback
+ this.deinit();
return;
}
} else {
if (!map.contains(this.id)) {
// if the interval was cancelled, don't run the callback
+ this.deinit();
return;
}
}
const callback = this.callback.get() orelse @panic("Expected CallbackJob to have a callback function");
- const result = callback.call(this.globalThis, &.{});
+ const result = callback.call(globalThis, &.{});
- if (result.isAnyError(this.globalThis)) {
+ if (result.isAnyError(globalThis)) {
vm.runErrorHandler(result, null);
+ this.deinit();
+ return;
+ }
+
+ if (result.asPromise()) |promise| {
+ switch (promise.status(globalThis.vm())) {
+ .Rejected => {
+ this.deinit();
+ vm.runErrorHandler(promise.result(globalThis.vm()), null);
+ },
+ .Fulfilled => {
+ this.deinit();
+
+ // get the value out of the promise
+ _ = promise.result(this.globalThis.vm());
+ },
+ .Pending => {
+ result.then(globalThis, this, CallbackJob__onResolve, CallbackJob__onReject);
+ },
+ }
}
}
};
@@ -2317,20 +2363,25 @@ pub const Timer = struct {
var this = this_ orelse
return;
+ var globalThis = this.globalThis;
+
var cb: CallbackJob = .{
.callback = if (repeats)
- JSC.Strong.create(this.callback.get() orelse {
- // if the callback was freed, that's an error
- if (comptime Environment.allow_assert)
- unreachable;
-
- this.deinit();
- _ = map.swapRemove(timer_id.id);
- return;
- }, this.globalThis)
+ JSC.Strong.create(
+ this.callback.get() orelse {
+ // if the callback was freed, that's an error
+ if (comptime Environment.allow_assert)
+ unreachable;
+
+ this.deinit();
+ _ = map.swapRemove(timer_id.id);
+ return;
+ },
+ globalThis,
+ )
else
this.callback,
- .globalThis = this.globalThis,
+ .globalThis = globalThis,
.id = timer_id.id,
.repeat = timer_id.repeat > 0,
};