Await is a template now

This commit is contained in:
Yuriy Glukhov 2019-08-15 15:32:46 +02:00
parent 4cc1b42108
commit e3cb0d1a96
2 changed files with 41 additions and 98 deletions

View File

@ -395,7 +395,7 @@ proc `$`*(entries: seq[StackTraceEntry]): string =
if hint.len > 0: if hint.len > 0:
result.add(spaces(indent+2) & "## " & hint & "\n") result.add(spaces(indent+2) & "## " & hint & "\n")
proc injectStacktrace[T](future: Future[T]) = proc injectStacktrace(future: FutureBase) =
const header = "\nAsync traceback:\n" const header = "\nAsync traceback:\n"
var exceptionMsg = future.error.msg var exceptionMsg = future.error.msg
@ -418,6 +418,17 @@ proc injectStacktrace[T](future: Future[T]) =
# newMsg.add "\n" & $entry # newMsg.add "\n" & $entry
future.error.msg = newMsg future.error.msg = newMsg
proc internalCheckComplete*(fut: FutureBase) =
# For internal use only. Used in asyncmacro
if not(isNil(fut.error)):
injectStacktrace(fut)
raise fut.error
proc internalRead*[T](fut: Future[T] | FutureVar[T]): T {.inline.} =
# For internal use only. Used in asyncmacro
when T isnot void:
return fut.value
proc read*[T](future: Future[T] | FutureVar[T]): T = proc read*[T](future: Future[T] | FutureVar[T]): T =
## Retrieves the value of ``future``. Future must be finished otherwise ## Retrieves the value of ``future``. Future must be finished otherwise
## this function will fail with a ``ValueError`` exception. ## this function will fail with a ``ValueError`` exception.
@ -427,11 +438,8 @@ proc read*[T](future: Future[T] | FutureVar[T]): T =
let fut = Future[T](future) let fut = Future[T](future)
{.pop.} {.pop.}
if fut.finished(): if fut.finished():
if not(isNil(fut.error)): internalCheckComplete(future)
injectStacktrace(fut) internalRead(future)
raise fut.error
when T isnot void:
return fut.value
else: else:
# TODO: Make a custom exception type for this? # TODO: Make a custom exception type for this?
raise newException(ValueError, "Future still in progress.") raise newException(ValueError, "Future still in progress.")

View File

@ -65,45 +65,6 @@ template createCb(retFutureSym, iteratorNameSym,
identName() identName()
#{.pop.} #{.pop.}
template useVar(result: var NimNode, futureVarNode: NimNode, valueReceiver,
rootReceiver: untyped, fromNode: NimNode, isawait: bool) =
## Params:
## futureVarNode: The NimNode which is a symbol identifying the Future[T]
## variable to yield.
## fromNode: Used for better debug information (to give context).
## valueReceiver: The node which defines an expression that retrieves the
## future's value.
##
## rootReceiver: ??? TODO
if isawait:
# -> yield future<x>
result.add newNimNode(nnkYieldStmt, fromNode).add(futureVarNode)
# -> future<x>.read
valueReceiver = newDotExpr(futureVarNode, newIdentNode("read"))
result.add rootReceiver
else:
# -> yield future<x>
result.add newNimNode(nnkYieldStmt, fromNode).add(futureVarNode)
valueReceiver = futureVarNode
result.add rootReceiver
template createVar(result: var NimNode, futSymName: string,
asyncProc: NimNode,
valueReceiver, rootReceiver, retFutSym: untyped,
fromNode: NimNode, isawait: bool) =
result = newNimNode(nnkStmtList, fromNode)
var futSym = genSym(nskVar, "future")
result.add newVarStmt(futSym, asyncProc) # -> var future<x> = y
# retFuture.child = future<x>
result.add newAssignment(
newDotExpr(
newCall(newIdentNode("FutureBase"), copyNimNode(retFutSym)),
newIdentNode("child")
),
newCall(newIdentNode("FutureBase"), copyNimNode(futSym))
)
useVar(result, futSym, valueReceiver, rootReceiver, fromNode, isawait)
proc createFutureVarCompletions(futureVarIdents: seq[NimNode], proc createFutureVarCompletions(futureVarIdents: seq[NimNode],
fromNode: NimNode): NimNode {.compileTime.} = fromNode: NimNode): NimNode {.compileTime.} =
result = newNimNode(nnkStmtList, fromNode) result = newNimNode(nnkStmtList, fromNode)
@ -149,56 +110,6 @@ proc processBody(node, retFutureSym: NimNode,
result.add newNimNode(nnkReturnStmt, node).add(newNilLit()) result.add newNimNode(nnkReturnStmt, node).add(newNilLit())
return # Don't process the children of this return stmt return # Don't process the children of this return stmt
of nnkCommand, nnkCall:
if (node[0].eqIdent("await") or node[0].eqIdent("awaitne")):
case node[1].kind
of nnkIdent, nnkInfix, nnkDotExpr, nnkCall, nnkCommand:
# await x
# await x or y
# await foo(p, x)
# await foo p, x
var futureValue: NimNode
result.createVar("future" & $node[1][0].toStrLit, node[1], futureValue,
futureValue, retFutureSym, node,
node[0].eqIdent("await"))
else:
error("Invalid node kind in 'await', got: " & $node[1].kind)
elif node.len > 1 and node[1].kind == nnkCommand and
(node[1][0].eqIdent("await") or node[1][0].eqIdent("awaitne")):
# foo await x
var newCommand = node
result.createVar("future" & $node[0].toStrLit, node[1][1], newCommand[1],
newCommand, retFutureSym, node,
node[1][0].eqIdent("await"))
of nnkVarSection, nnkLetSection:
case node[0][2].kind
of nnkCommand:
if (node[0][2][0].eqIdent("await") or node[0][2][0].eqIdent("awaitne")):
# var x = await y
var newVarSection = node # TODO: Should this use copyNimNode?
result.createVar("future" & node[0][0].strVal, node[0][2][1],
newVarSection[0][2], newVarSection, retFutureSym, node,
node[0][2][0].eqIdent("await"))
else: discard
of nnkAsgn:
case node[1].kind
of nnkCommand:
if node[1][0].eqIdent("await") or node[1][0].eqIdent("awaitne"):
# x = await y
var newAsgn = node
result.createVar("future" & $node[0].toStrLit, node[1][1], newAsgn[1],
newAsgn, retFutureSym, node,
node[1][0].eqIdent("await"))
else: discard
of nnkDiscardStmt:
# discard await x
if node[0].kind == nnkCommand and
(node[0][0].eqIdent("await") or node[0][0].eqIdent("awaitne")):
var newDiscard = node
result.createVar("futureDiscard_" & $toStrLit(node[0][1]), node[0][1],
newDiscard[0], newDiscard, retFutureSym, node,
node[0][0].eqIdent("await"))
of RoutineNodes-{nnkTemplateDef}: of RoutineNodes-{nnkTemplateDef}:
# skip all the nested procedure definitions # skip all the nested procedure definitions
return node return node
@ -273,7 +184,7 @@ proc asyncSingleProc(prc: NimNode): NimNode {.compileTime.} =
var outerProcBody = newNimNode(nnkStmtList, prc.body) var outerProcBody = newNimNode(nnkStmtList, prc.body)
# -> var retFuture = newFuture[T]() # -> var retFuture = newFuture[T]()
var retFutureSym = genSym(nskVar, "retFuture") var retFutureSym = ident "chronosInternalRetFuture"
var subRetType = var subRetType =
if returnType.kind == nnkEmpty: newIdentNode("void") if returnType.kind == nnkEmpty: newIdentNode("void")
else: baseType else: baseType
@ -353,8 +264,32 @@ proc asyncSingleProc(prc: NimNode): NimNode {.compileTime.} =
#if prcName == "recvLineInto": #if prcName == "recvLineInto":
# echo(toStrLit(result)) # echo(toStrLit(result))
proc await*[T](x: T) = template await*[T](f: Future[T]): auto =
{.error: "Await only available within {.async.}".} when declared(chronosInternalRetFuture):
when not declared(chronosInternalTmpFuture):
var chronosInternalTmpFuture: FutureBase
chronosInternalTmpFuture = f
chronosInternalRetFuture.child = chronosInternalTmpFuture
yield chronosInternalTmpFuture
chronosInternalTmpFuture.internalCheckComplete()
chronosInternalRetFuture.child = nil
cast[type(f)](chronosInternalTmpFuture).internalRead()
else:
static:
assert(false, "Await only available within {.async.}")
template awaitne*[T](f: Future[T]): Future[T] =
when declared(chronosInternalRetFuture):
when not declared(chronosInternalTmpFuture):
var chronosInternalTmpFuture: FutureBase
chronosInternalTmpFuture = f
chronosInternalRetFuture.child = chronosInternalTmpFuture
yield chronosInternalTmpFuture
chronosInternalRetFuture.child = nil
cast[type(f)](chronosInternalTmpFuture)
else:
static:
assert(false, "Await only available within {.async.}")
macro async*(prc: untyped): untyped = macro async*(prc: untyped): untyped =
## Macro which processes async procedures into the appropriate ## Macro which processes async procedures into the appropriate