class Future {
#s;
#promise;
#onStartPromise;
#onTimeoutPromise;
#onCancelPromise;
static get [Symbol.species]() {
return Future;
}
get state() {
return this.#s.state;
}
constructor(computation) {
if (computation instanceof Future) {
const o = computation;
this.#s = o.#s;
this.#promise = o.#promise;
this.#onStartPromise = o.#onStartPromise;
this.#onTimeoutPromise = o.#onTimeoutPromise;
this.#onCancelPromise = o.#onCancelPromise;
}
this.#onStartPromise = new Promise(
(resolve, reject) => (this.#s.onStart = resolve)
);
this.#onTimeoutPromise = new Promise(
(resolve, reject) => (this.#s.onTimeout = resolve)
);
this.#promise = new Promise((resolve, reject) => {
this.#s.computation = () => {
this.#s.computation = null;
this.#s.state = "STARTED";
this.#s.startTime = Date.now();
this.#s.onStart(this.#s.startTime);
try {
resolve(computation(this));
} catch (e) {
// Not if this is already resolved.
if (this.#s.state === "STARTED") {
this.#s.exception = e;
}
reject(e);
}
};
}).then(
(v) => this.#resolved("FULFILLED", undefined, v),
(e) =>
e instanceof Timeout
? this.#resolved("TIMEOUT", this.#s.onTimeout, e)
: e instanceof Cancelled
? this.#resolved("CANCELLED", this.#s.onCancel, e)
: this.#resolved("REJECTED", undefined, e)
);
}
// Arrive at a final state.
#resolved(state, handler, v) {
this.#s.state = state;
this.#s.computation = this.#s.onCancel = null;
this.#s.onTimeout = this.#s.onStart = null;
this.#s.exception = v;
handler?.(v);
return v;
}
then(onFulfilled, onRejected) {
this.#s.computation?.();
const next = new Future(this);
next.#promise = next.#promise.then(onFulfilled ?? ((a) => a), onRejected);
}
catch(onRejected) {
const next = new Future(this);
next.#promise = next.#promise.catch(onRejected);
return next;
}
finally(onFinally) {
const next = new Future(this);
next.#promise = next.#promise.finally(onFinally);
return next;
}
when(onFulfilled, onRejected) {
const next = new Future(this);
next.#promise = next.#promise.then(onFulfilled, onRejected);
return next;
}
onStart(handler) {
const next = new Future(this);
next.#onStartPromise = next.#onStartPromise.then(handler);
return next;
}
start() {
this.#s.computation?.();
return this;
}
cancel(msg = "Cancelled") {
this.#resolved(
"Cancelled",
this.#s.onCancel,
new Cancelled(msg, this, this.#s.startTime)
);
return this;
}
onCancel(handler) {
const next = new Future(this);
next.#onCancelPromise = next.#onCancelPromise.catch(handler);
return next;
}
onTimeout(handler) {
const next = new Future(this);
next.#onTimeoutPromise = this.#onTimeoutPromise.catch(handler);
return next;
}
isCancelled() {
return this.#s.state !== "PENDING" || this.#s.state !== "STARTED";
}
check(continuation) {
switch (this.#s.state) {
case "PENDING":
throw new Error(
"Check is to be used as part of a running Future computation."
);
case "STARTED":
return continuation(this);
case "TIMEOUT":
case "CANCELLED":
throw this.#s.exception;
default:
throw new Error("Computation has already completed.");
}
}
static delay(delay) {
return (computation) => {
const p = new Promise((resolve, reject) => setTimeout(resolve, delay));
return new Future(() => p.then(computation));
};
}
static timeoutFromNow(timeout, msg = "Timeout") {
return (computation) => {
// Start the timer now.
const start = Date.now();
const future = new Future(() => {
const c = Promise.resolve(computation()).then(
(v) => ((future.#s.onTimeout = null), v)
);
const p = new Promise((resolve, reject) =>
setTimeout(() => reject(new Timeout(msg, future, start)), timeout)
).catch((e) => () => (future.#s.onTimeout?.(e), Throw(e)));
return Promise.race([p, c]);
});
return future;
};
}
static timeout(timeout, msg = "Timeout") {
const msg_dflt = msg;
return (computation, msg) => {
const tmsg = msg ?? msg_dflt;
const future = new Future(() => {
// Start the timer when the Future executes.
const start = Date.now();
const p = new Promise((resolve, reject) =>
setTimeout(() => reject(new Timeout(tmsg, future, start)), timeout)
).catch((e) => () => (future.#s.onTimeout?.(e), Throw(e)));
const c = Promise.resolve(computation()).then(
(v) => ((future.#s.onTimeout = null), v)
);
return Promise.race([p, c]);
});
return future;
};
}
}