import std/tables import monoucha/quickjs import monoucha/javascript import monoucha/jsutils import monoucha/jsopaque import monoucha/tojs import types/opt type PromiseState = enum psPending, psFulfilled, psRejected EmptyPromise* = ref object of RootObj cb: (proc()) next: EmptyPromise opaque: pointer state*: PromiseState Promise*[T] = ref object of EmptyPromise res: T get: GetValueProc[T] GetValueProc[T] = (proc(opaque: pointer; res: var T)) PromiseMap* = object tab: Table[int, EmptyPromise] opaque*: pointer proc newPromise*[T](): Promise[T] = return Promise[T]() proc newPromiseMap*(opaque: pointer): PromiseMap = return PromiseMap( opaque: opaque ) proc addPromise*[T](map: var PromiseMap; id: int; get: GetValueProc[T]): Promise[T] = let promise = Promise[T](get: get, opaque: map.opaque) map.tab[id] = promise return promise proc addEmptyPromise*(map: var PromiseMap; id: int): EmptyPromise = let promise = EmptyPromise(opaque: map.opaque) map.tab[id] = promise return promise proc resolve*(promise: EmptyPromise) = var promise = promise while true: if promise.cb != nil: promise.cb() promise.cb = nil promise.state = psFulfilled promise = promise.next if promise == nil: break promise.next = nil proc resolve*[T](promise: Promise[T]; res: T) = if promise.get != nil: promise.get(promise.opaque, promise.res) promise.get = nil promise.res = res promise.resolve() proc resolve*(map: var PromiseMap; promiseid: int) = var promise: EmptyPromise if map.tab.pop(promiseid, promise): promise.resolve() proc newResolvedPromise*(): EmptyPromise = let res = EmptyPromise() res.resolve() return res func empty*(map: PromiseMap): bool = map.tab.len == 0 proc then*(promise: EmptyPromise; cb: (proc())): EmptyPromise {.discardable.} = promise.cb = cb promise.next = EmptyPromise() if promise.state == psFulfilled: promise.resolve() return promise.next proc then*(promise: EmptyPromise; cb: (proc(): EmptyPromise)): EmptyPromise {.discardable.} = let next = EmptyPromise() promise.then(proc() = var p2 = cb() if p2 != nil: p2.then(proc() = next.resolve()) else: next.resolve()) return next proc then*[T](promise: Promise[T]; cb: (proc(x: T))): EmptyPromise {.discardable.} = return promise.then(proc() = if promise.get != nil: promise.get(promise.opaque, promise.res) promise.get = nil cb(promise.res)) proc then*[T](promise: EmptyPromise; cb: (proc(): Promise[T])): Promise[T] {.discardable.} = let next = Promise[T]() promise.then(proc() = var p2 = cb() if p2 != nil: p2.then(proc(x: T) = next.res = x next.resolve()) else: next.resolve()) return next proc then*[T](promise: Promise[T]; cb: (proc(x: T): EmptyPromise)): EmptyPromise {.discardable.} = let next = EmptyPromise() promise.then(proc(x: T) = let p2 = cb(x) if p2 != nil: p2.then(proc() = next.resolve()) else: next.resolve()) return next proc then*[T](promise: EmptyPromise; cb: (proc(): T)): Promise[T] {.discardable.} = let next = Promise[T]() promise.then(proc() = next.res = cb() next.resolve()) return next proc then*[T, U](promise: Promise[T]; cb: (proc(x: T): U)): Promise[U] {.discardable.} = let next = Promise[U]() promise.then(proc(x: T) = next.res = cb(x) next.resolve()) return next proc then*[T, U](promise: Promise[T]; cb: (proc(x: T): Promise[U])): Promise[U] {.discardable.} = let next = Promise[U]() promise.then(proc(x: T) = let p2 = cb(x) if p2 != nil: p2.then(proc(y: U) = next.res = y next.resolve()) else: next.resolve()) return next proc then*[T, U](promise: Promise[T]; cb: (proc(x: T): Opt[Promise[U]])): Promise[Opt[U]] {.discardable.} = let next = Promise[Opt[U]]() promise.then(proc(x: T) = let p2 = cb(x) if p2.isSome: p2.get.then(proc(y: U) = next.res = opt(y) next.resolve()) else: next.resolve()) return next proc all*(promises: seq[EmptyPromise]): EmptyPromise = let res = EmptyPromise() var i = 0 for promise in promises: promise.then(proc() = inc i if i == promises.len: res.resolve()) if promises.len == 0: res.resolve() return res # * Promise is converted to a JS promise which will be resolved when the Nim # promise is resolved. proc promiseThenCallback(ctx: JSContext; this_val: JSValue; argc: cint; argv: ptr UncheckedArray[JSValue]; magic: cint; func_data: ptr UncheckedArray[JSValue]): JSValue {.cdecl.} = let fun = func_data[0] let op = JS_GetOpaque(fun, JS_GetClassID(fun)) if op != nil: let p = cast[EmptyPromise](op) p.resolve() GC_unref(p) JS_SetOpaque(fun, nil) return JS_UNDEFINED proc fromJS*(ctx: JSContext; val: JSValue; res: var EmptyPromise): Opt[void] = if not JS_IsObject(val): JS_ThrowTypeError(ctx, "value is not an object") return err() res = EmptyPromise() GC_ref(res) let tmp = JS_NewObject(ctx) JS_SetOpaque(tmp, cast[pointer](res)) let fun = JS_NewCFunctionData(ctx, promiseThenCallback, 0, 0, 1, tmp.toJSValueArray()) JS_FreeValue(ctx, tmp) let val = JS_Invoke(ctx, val, ctx.getOpaque().strRefs[jstThen], 1, fun.toJSValueArray()) JS_FreeValue(ctx, fun) if JS_IsException(val): JS_FreeValue(ctx, val) GC_unref(res) res = nil return err() JS_FreeValue(ctx, val) return ok() proc toJS*(ctx: JSContext; promise: EmptyPromise): JSValue = if promise == nil: return JS_NULL var resolvingFuncs: array[2, JSValue] let jsPromise = JS_NewPromiseCapability(ctx, resolvingFuncs.toJSValueArray()) if JS_IsException(jsPromise): return JS_EXCEPTION promise.then(proc() = let res = JS_Call(ctx, resolvingFuncs[0], JS_UNDEFINED, 0, nil) JS_FreeValue(ctx, res) JS_FreeValue(ctx, resolvingFuncs[0]) JS_FreeValue(ctx, resolvingFuncs[1])) return jsPromise proc toJS*[T](ctx: JSContext; promise: Promise[T]): JSValue = if promise == nil: return JS_NULL var resolvingFuncs: array[2, JSValue] let jsPromise = JS_NewPromiseCapability(ctx, resolvingFuncs.toJSValueArray()) if JS_IsException(jsPromise): return JS_EXCEPTION promise.then(proc(x: T) = let x = toJS(ctx, x) let res = JS_Call(ctx, resolvingFuncs[0], JS_UNDEFINED, 1, x.toJSValueArray()) JS_FreeValue(ctx, res) JS_FreeValue(ctx, x) JS_FreeValue(ctx, resolvingFuncs[0]) JS_FreeValue(ctx, resolvingFuncs[1])) return jsPromise proc toJS*[T, E](ctx: JSContext; promise: Promise[Result[T, E]]): JSValue = if promise == nil: return JS_NULL var resolvingFuncs: array[2, JSValue] let jsPromise = JS_NewPromiseCapability(ctx, resolvingFuncs.toJSValueArray()) if JS_IsException(jsPromise): return JS_EXCEPTION promise.then(proc(x: Result[T, E]) = if x.isSome: let x = when T is void: JS_UNDEFINED else: toJS(ctx, x.get) let res = JS_Call(ctx, resolvingFuncs[0], JS_UNDEFINED, 1, x.toJSValueArray()) JS_FreeValue(ctx, res) JS_FreeValue(ctx, x) else: # err let x = when E is void: JS_UNDEFINED else: toJS(ctx, x.error) let res = JS_Call(ctx, resolvingFuncs[1], JS_UNDEFINED, 1, x.toJSValueArray()) JS_FreeValue(ctx, res) JS_FreeValue(ctx, x) JS_FreeValue(ctx, resolvingFuncs[0]) JS_FreeValue(ctx, resolvingFuncs[1])) return jsPromise