mirror of
https://github.com/status-im/nim-chronos.git
synced 2025-01-23 17:51:26 +00:00
f56d286687
* introduce `asyncraises` to core future utilities Similar to the introduction of `raises` into a codebase, `asyncraises` needs to be introduced gradually across all functionality before deriving benefit. This is a first introduction along with utilities to manage raises lists and transform them at compile time. Several scenarios ensue: * for trivial cases, adding `asyncraises` is enough and the framework deduces the rest * some functions "add" new asyncraises (similar to what `raise` does in "normal" code) - for example `wait` may raise all exceptions of the future passed to it and additionally a few of its own - this requires extending the raises list * som functions "remove" raises (similar to what `try/except` does) such as `nocancel` with blocks cancellations and therefore reduce the raising set Both of the above cases are currently handled by a macro, but depending on the situation lead to code organisation issues around return types and pragma limitations - in particular, to keep `asyncraises` backwards-compatibility, some code needs to exist in two versions which somewhat complicates the implementation. * add `asyncraises` versions for several `asyncfutures` utilities * when assigning exceptions to a `Future` via `fail`, check at compile time if possible and at runtime if not that the exception matches constraints * fix `waitFor` comments * move async raises to separate module, implement `or`
522 lines
18 KiB
Nim
522 lines
18 KiB
Nim
#
|
|
#
|
|
# Nim's Runtime Library
|
|
# (c) Copyright 2015 Dominik Picheta
|
|
# (c) Copyright 2018-Present Status Research & Development GmbH
|
|
#
|
|
# See the file "copying.txt", included in this
|
|
# distribution, for details about the copyright.
|
|
#
|
|
|
|
import
|
|
std/[algorithm, macros, sequtils],
|
|
../[futures, config]
|
|
|
|
proc processBody(node, setResultSym, baseType: NimNode): NimNode {.compileTime.} =
|
|
case node.kind
|
|
of nnkReturnStmt:
|
|
# `return ...` -> `setResult(...); return`
|
|
let
|
|
res = newNimNode(nnkStmtList, node)
|
|
if node[0].kind != nnkEmpty:
|
|
res.add newCall(setResultSym, processBody(node[0], setResultSym, baseType))
|
|
res.add newNimNode(nnkReturnStmt, node).add(newEmptyNode())
|
|
|
|
res
|
|
of RoutineNodes-{nnkTemplateDef}:
|
|
# Skip nested routines since they have their own return value distinct from
|
|
# the Future we inject
|
|
node
|
|
else:
|
|
for i in 0 ..< node.len:
|
|
node[i] = processBody(node[i], setResultSym, baseType)
|
|
node
|
|
|
|
proc wrapInTryFinally(fut, baseType, body, raisesTuple: NimNode): NimNode {.compileTime.} =
|
|
# creates:
|
|
# try: `body`
|
|
# [for raise in raisesTuple]:
|
|
# except `raise`: closureSucceeded = false; `castFutureSym`.fail(exc)
|
|
# finally:
|
|
# if closureSucceeded:
|
|
# `castFutureSym`.complete(result)
|
|
#
|
|
# Calling `complete` inside `finally` ensures that all success paths
|
|
# (including early returns and code inside nested finally statements and
|
|
# defer) are completed with the final contents of `result`
|
|
let
|
|
closureSucceeded = genSym(nskVar, "closureSucceeded")
|
|
nTry = nnkTryStmt.newTree(body)
|
|
excName = ident"exc"
|
|
|
|
# Depending on the exception type, we must have at most one of each of these
|
|
# "special" exception handlers that are needed to implement cancellation and
|
|
# Defect propagation
|
|
var
|
|
hasDefect = false
|
|
hasCancelledError = false
|
|
hasCatchableError = false
|
|
|
|
template addDefect =
|
|
if not hasDefect:
|
|
hasDefect = true
|
|
# When a Defect is raised, the program is in an undefined state and
|
|
# continuing running other tasks while the Future completion sits on the
|
|
# callback queue may lead to further damage so we re-raise them eagerly.
|
|
nTry.add nnkExceptBranch.newTree(
|
|
nnkInfix.newTree(ident"as", ident"Defect", excName),
|
|
nnkStmtList.newTree(
|
|
nnkAsgn.newTree(closureSucceeded, ident"false"),
|
|
nnkRaiseStmt.newTree(excName)
|
|
)
|
|
)
|
|
template addCancelledError =
|
|
if not hasCancelledError:
|
|
hasCancelledError = true
|
|
nTry.add nnkExceptBranch.newTree(
|
|
ident"CancelledError",
|
|
nnkStmtList.newTree(
|
|
nnkAsgn.newTree(closureSucceeded, ident"false"),
|
|
newCall(ident "cancelAndSchedule", fut)
|
|
)
|
|
)
|
|
|
|
template addCatchableError =
|
|
if not hasCatchableError:
|
|
hasCatchableError = true
|
|
nTry.add nnkExceptBranch.newTree(
|
|
nnkInfix.newTree(ident"as", ident"CatchableError", excName),
|
|
nnkStmtList.newTree(
|
|
nnkAsgn.newTree(closureSucceeded, ident"false"),
|
|
newCall(ident "fail", fut, excName)
|
|
))
|
|
|
|
for exc in raisesTuple:
|
|
if exc.eqIdent("Exception"):
|
|
addCancelledError
|
|
addCatchableError
|
|
addDefect
|
|
|
|
# Because we store `CatchableError` in the Future, we cannot re-raise the
|
|
# original exception
|
|
nTry.add nnkExceptBranch.newTree(
|
|
nnkInfix.newTree(ident"as", ident"Exception", excName),
|
|
newCall(ident "fail", fut,
|
|
nnkStmtList.newTree(
|
|
nnkAsgn.newTree(closureSucceeded, ident"false"),
|
|
quote do: (ref ValueError)(msg: `excName`.msg, parent: `excName`)))
|
|
)
|
|
elif exc.eqIdent("CancelledError"):
|
|
addCancelledError
|
|
elif exc.eqIdent("CatchableError"):
|
|
# Ensure cancellations are re-routed to the cancellation handler even if
|
|
# not explicitly specified in the raises list
|
|
addCancelledError
|
|
addCatchableError
|
|
else:
|
|
nTry.add nnkExceptBranch.newTree(
|
|
nnkInfix.newTree(ident"as", exc, excName),
|
|
nnkStmtList.newTree(
|
|
nnkAsgn.newTree(closureSucceeded, ident"false"),
|
|
newCall(ident "fail", fut, excName)
|
|
))
|
|
|
|
nTry.add nnkFinally.newTree(
|
|
nnkIfStmt.newTree(
|
|
nnkElifBranch.newTree(
|
|
closureSucceeded,
|
|
if baseType.eqIdent("void"): # shortcut for non-generic void
|
|
newCall(ident "complete", fut)
|
|
else:
|
|
nnkWhenStmt.newTree(
|
|
nnkElifExpr.newTree(
|
|
nnkInfix.newTree(ident "is", baseType, ident "void"),
|
|
newCall(ident "complete", fut)
|
|
),
|
|
nnkElseExpr.newTree(
|
|
newCall(ident "complete", fut, ident "result")
|
|
)
|
|
)
|
|
)
|
|
)
|
|
)
|
|
|
|
nnkStmtList.newTree(
|
|
newVarStmt(closureSucceeded, ident"true"),
|
|
nTry
|
|
)
|
|
|
|
proc getName(node: NimNode): string {.compileTime.} =
|
|
case node.kind
|
|
of nnkSym:
|
|
return node.strVal
|
|
of nnkPostfix:
|
|
return node[1].strVal
|
|
of nnkIdent:
|
|
return node.strVal
|
|
of nnkEmpty:
|
|
return "anonymous"
|
|
else:
|
|
error("Unknown name.")
|
|
|
|
macro unsupported(s: static[string]): untyped =
|
|
error s
|
|
|
|
proc params2(someProc: NimNode): NimNode =
|
|
# until https://github.com/nim-lang/Nim/pull/19563 is available
|
|
if someProc.kind == nnkProcTy:
|
|
someProc[0]
|
|
else:
|
|
params(someProc)
|
|
|
|
proc cleanupOpenSymChoice(node: NimNode): NimNode {.compileTime.} =
|
|
# Replace every Call -> OpenSymChoice by a Bracket expr
|
|
# ref https://github.com/nim-lang/Nim/issues/11091
|
|
if node.kind in nnkCallKinds and
|
|
node[0].kind == nnkOpenSymChoice and node[0].eqIdent("[]"):
|
|
result = newNimNode(nnkBracketExpr)
|
|
for child in node[1..^1]:
|
|
result.add(cleanupOpenSymChoice(child))
|
|
else:
|
|
result = node.copyNimNode()
|
|
for child in node:
|
|
result.add(cleanupOpenSymChoice(child))
|
|
|
|
proc getAsyncCfg(prc: NimNode): tuple[raises: bool, async: bool, raisesTuple: NimNode] =
|
|
# reads the pragmas to extract the useful data
|
|
# and removes them
|
|
var
|
|
foundRaises = -1
|
|
foundAsync = -1
|
|
|
|
for index, pragma in pragma(prc):
|
|
if pragma.kind == nnkExprColonExpr and pragma[0] == ident "asyncraises":
|
|
foundRaises = index
|
|
elif pragma.eqIdent("async"):
|
|
foundAsync = index
|
|
elif pragma.kind == nnkExprColonExpr and pragma[0] == ident "raises":
|
|
warning("The raises pragma doesn't work on async procedure. " &
|
|
"Please remove it or use asyncraises instead")
|
|
|
|
result.raises = foundRaises >= 0
|
|
result.async = foundAsync >= 0
|
|
result.raisesTuple = nnkTupleConstr.newTree()
|
|
|
|
if foundRaises >= 0:
|
|
for possibleRaise in pragma(prc)[foundRaises][1]:
|
|
result.raisesTuple.add(possibleRaise)
|
|
if result.raisesTuple.len == 0:
|
|
result.raisesTuple = ident("void")
|
|
else:
|
|
when defined(chronosWarnMissingRaises):
|
|
warning("Async proc miss asyncraises")
|
|
const defaultException =
|
|
when defined(chronosStrictException): "CatchableError"
|
|
else: "Exception"
|
|
result.raisesTuple.add(ident(defaultException))
|
|
|
|
let toRemoveList = @[foundRaises, foundAsync].filterIt(it >= 0).sorted().reversed()
|
|
for toRemove in toRemoveList:
|
|
pragma(prc).del(toRemove)
|
|
|
|
proc isEmpty(n: NimNode): bool {.compileTime.} =
|
|
# true iff node recursively contains only comments or empties
|
|
case n.kind
|
|
of nnkEmpty, nnkCommentStmt: true
|
|
of nnkStmtList:
|
|
for child in n:
|
|
if not isEmpty(child): return false
|
|
true
|
|
else:
|
|
false
|
|
|
|
proc asyncSingleProc(prc: NimNode): NimNode {.compileTime.} =
|
|
## This macro transforms a single procedure into a closure iterator.
|
|
## The ``async`` macro supports a stmtList holding multiple async procedures.
|
|
if prc.kind notin {nnkProcTy, nnkProcDef, nnkLambda, nnkMethodDef, nnkDo}:
|
|
error("Cannot transform " & $prc.kind & " into an async proc." &
|
|
" proc/method definition or lambda node expected.", prc)
|
|
|
|
let returnType = cleanupOpenSymChoice(prc.params2[0])
|
|
|
|
# Verify that the return type is a Future[T]
|
|
let baseType =
|
|
if returnType.kind == nnkEmpty:
|
|
ident "void"
|
|
elif not (
|
|
returnType.kind == nnkBracketExpr and
|
|
(eqIdent(returnType[0], "Future") or eqIdent(returnType[0], "InternalRaisesFuture"))):
|
|
error(
|
|
"Expected return type of 'Future' got '" & repr(returnType) & "'", prc)
|
|
return
|
|
else:
|
|
returnType[1]
|
|
|
|
let
|
|
baseTypeIsVoid = baseType.eqIdent("void")
|
|
futureVoidType = nnkBracketExpr.newTree(ident "Future", ident "void")
|
|
(hasRaises, isAsync, raisesTuple) = getAsyncCfg(prc)
|
|
|
|
if hasRaises:
|
|
# Store `asyncraises` types in InternalRaisesFuture
|
|
prc.params2[0] = nnkBracketExpr.newTree(
|
|
newIdentNode("InternalRaisesFuture"),
|
|
baseType,
|
|
raisesTuple
|
|
)
|
|
elif baseTypeIsVoid:
|
|
# Adds the implicit Future[void]
|
|
prc.params2[0] =
|
|
newNimNode(nnkBracketExpr, prc).
|
|
add(newIdentNode("Future")).
|
|
add(newIdentNode("void"))
|
|
|
|
if prc.kind notin {nnkProcTy, nnkLambda}: # TODO: Nim bug?
|
|
prc.addPragma(newColonExpr(ident "stackTrace", ident "off"))
|
|
|
|
# The proc itself doesn't raise
|
|
prc.addPragma(
|
|
nnkExprColonExpr.newTree(newIdentNode("raises"), nnkBracket.newTree()))
|
|
|
|
# `gcsafe` isn't deduced even though we require async code to be gcsafe
|
|
# https://github.com/nim-lang/RFCs/issues/435
|
|
prc.addPragma(newIdentNode("gcsafe"))
|
|
|
|
if isAsync == false: # `asyncraises` without `async`
|
|
# type InternalRaisesFutureRaises {.used.} = `raisesTuple`
|
|
# `body`
|
|
prc.body = nnkStmtList.newTree(
|
|
nnkTypeSection.newTree(
|
|
nnkTypeDef.newTree(
|
|
nnkPragmaExpr.newTree(
|
|
ident"InternalRaisesFutureRaises",
|
|
nnkPragma.newTree(
|
|
newIdentNode("used")
|
|
)
|
|
),
|
|
newEmptyNode(),
|
|
raisesTuple
|
|
)
|
|
),
|
|
prc.body
|
|
)
|
|
|
|
return prc
|
|
|
|
if prc.kind in {nnkProcDef, nnkLambda, nnkMethodDef, nnkDo} and
|
|
not isEmpty(prc.body):
|
|
# don't do anything with forward bodies (empty)
|
|
let
|
|
prcName = prc.name.getName
|
|
setResultSym = ident "setResult"
|
|
procBody = prc.body.processBody(setResultSym, baseType)
|
|
internalFutureSym = ident "chronosInternalRetFuture"
|
|
internalFutureType =
|
|
if baseTypeIsVoid: futureVoidType
|
|
else: returnType
|
|
castFutureSym = nnkCast.newTree(internalFutureType, internalFutureSym)
|
|
resultIdent = ident "result"
|
|
|
|
resultDecl = nnkWhenStmt.newTree(
|
|
# when `baseType` is void:
|
|
nnkElifExpr.newTree(
|
|
nnkInfix.newTree(ident "is", baseType, ident "void"),
|
|
quote do:
|
|
template result: auto {.used.} =
|
|
{.fatal: "You should not reference the `result` variable inside" &
|
|
" a void async proc".}
|
|
),
|
|
# else:
|
|
nnkElseExpr.newTree(
|
|
newStmtList(
|
|
quote do: {.push warning[resultshadowed]: off.},
|
|
# var result {.used.}: `baseType`
|
|
# In the proc body, result may or may not end up being used
|
|
# depending on how the body is written - with implicit returns /
|
|
# expressions in particular, it is likely but not guaranteed that
|
|
# it is not used. Ideally, we would avoid emitting it in this
|
|
# case to avoid the default initializaiton. {.used.} typically
|
|
# works better than {.push.} which has a tendency to leak out of
|
|
# scope.
|
|
# TODO figure out if there's a way to detect `result` usage in
|
|
# the proc body _after_ template exapnsion, and therefore
|
|
# avoid creating this variable - one option is to create an
|
|
# addtional when branch witha fake `result` and check
|
|
# `compiles(procBody)` - this is not without cost though
|
|
nnkVarSection.newTree(nnkIdentDefs.newTree(
|
|
nnkPragmaExpr.newTree(
|
|
resultIdent,
|
|
nnkPragma.newTree(ident "used")),
|
|
baseType, newEmptyNode())
|
|
),
|
|
quote do: {.pop.},
|
|
)
|
|
)
|
|
)
|
|
|
|
# generates:
|
|
# template `setResultSym`(code: untyped) {.used.} =
|
|
# when typeof(code) is void: code
|
|
# else: `resultIdent` = code
|
|
#
|
|
# this is useful to handle implicit returns, but also
|
|
# to bind the `result` to the one we declare here
|
|
setResultDecl =
|
|
if baseTypeIsVoid: # shortcut for non-generic void
|
|
newEmptyNode()
|
|
else:
|
|
nnkTemplateDef.newTree(
|
|
setResultSym,
|
|
newEmptyNode(), newEmptyNode(),
|
|
nnkFormalParams.newTree(
|
|
newEmptyNode(),
|
|
nnkIdentDefs.newTree(
|
|
ident"code",
|
|
ident"untyped",
|
|
newEmptyNode(),
|
|
)
|
|
),
|
|
nnkPragma.newTree(ident"used"),
|
|
newEmptyNode(),
|
|
nnkWhenStmt.newTree(
|
|
nnkElifBranch.newTree(
|
|
nnkInfix.newTree(
|
|
ident"is", nnkTypeOfExpr.newTree(ident"code"), ident"void"),
|
|
ident"code"
|
|
),
|
|
nnkElse.newTree(
|
|
newAssignment(resultIdent, ident"code")
|
|
)
|
|
)
|
|
)
|
|
|
|
# Wrapping in try/finally ensures that early returns are handled properly
|
|
# and that `defer` is processed in the right scope
|
|
completeDecl = wrapInTryFinally(
|
|
castFutureSym, baseType,
|
|
if baseTypeIsVoid: procBody # shortcut for non-generic `void`
|
|
else: newCall(setResultSym, procBody),
|
|
raisesTuple
|
|
)
|
|
|
|
closureBody = newStmtList(resultDecl, setResultDecl, completeDecl)
|
|
|
|
internalFutureParameter = nnkIdentDefs.newTree(
|
|
internalFutureSym, newIdentNode("FutureBase"), newEmptyNode())
|
|
iteratorNameSym = genSym(nskIterator, $prcName)
|
|
closureIterator = newProc(
|
|
iteratorNameSym,
|
|
[newIdentNode("FutureBase"), internalFutureParameter],
|
|
closureBody, nnkIteratorDef)
|
|
|
|
outerProcBody = newNimNode(nnkStmtList, prc.body)
|
|
|
|
# Copy comment for nimdoc
|
|
if prc.body.len > 0 and prc.body[0].kind == nnkCommentStmt:
|
|
outerProcBody.add(prc.body[0])
|
|
|
|
iteratorNameSym.copyLineInfo(prc)
|
|
|
|
closureIterator.pragma = newNimNode(nnkPragma, lineInfoFrom=prc.body)
|
|
closureIterator.addPragma(newIdentNode("closure"))
|
|
|
|
# `async` code must be gcsafe
|
|
closureIterator.addPragma(newIdentNode("gcsafe"))
|
|
|
|
# Exceptions are caught inside the iterator and stored in the future
|
|
closureIterator.addPragma(nnkExprColonExpr.newTree(
|
|
newIdentNode("raises"),
|
|
nnkBracket.newTree()
|
|
))
|
|
|
|
outerProcBody.add(closureIterator)
|
|
|
|
# -> let resultFuture = newInternalRaisesFuture[T]()
|
|
# declared at the end to be sure that the closure
|
|
# doesn't reference it, avoid cyclic ref (#203)
|
|
let
|
|
retFutureSym = ident "resultFuture"
|
|
retFutureSym.copyLineInfo(prc)
|
|
# Do not change this code to `quote do` version because `instantiationInfo`
|
|
# will be broken for `newFuture()` call.
|
|
outerProcBody.add(
|
|
newLetStmt(
|
|
retFutureSym,
|
|
newCall(newTree(nnkBracketExpr, ident "newInternalRaisesFuture", baseType),
|
|
newLit(prcName))
|
|
)
|
|
)
|
|
# -> resultFuture.internalClosure = iterator
|
|
outerProcBody.add(
|
|
newAssignment(
|
|
newDotExpr(retFutureSym, newIdentNode("internalClosure")),
|
|
iteratorNameSym)
|
|
)
|
|
|
|
# -> futureContinue(resultFuture))
|
|
outerProcBody.add(
|
|
newCall(newIdentNode("futureContinue"), retFutureSym)
|
|
)
|
|
|
|
# -> return resultFuture
|
|
outerProcBody.add newNimNode(nnkReturnStmt, prc.body[^1]).add(retFutureSym)
|
|
|
|
prc.body = outerProcBody
|
|
|
|
when chronosDumpAsync:
|
|
echo repr prc
|
|
prc
|
|
|
|
template await*[T](f: Future[T]): untyped =
|
|
when declared(chronosInternalRetFuture):
|
|
chronosInternalRetFuture.internalChild = f
|
|
# `futureContinue` calls the iterator generated by the `async`
|
|
# transformation - `yield` gives control back to `futureContinue` which is
|
|
# responsible for resuming execution once the yielded future is finished
|
|
yield chronosInternalRetFuture.internalChild
|
|
# `child` released by `futureContinue`
|
|
cast[type(f)](chronosInternalRetFuture.internalChild).internalCheckComplete()
|
|
|
|
when T isnot void:
|
|
cast[type(f)](chronosInternalRetFuture.internalChild).value()
|
|
else:
|
|
unsupported "await is only available within {.async.}"
|
|
|
|
template awaitne*[T](f: Future[T]): Future[T] =
|
|
when declared(chronosInternalRetFuture):
|
|
chronosInternalRetFuture.internalChild = f
|
|
yield chronosInternalRetFuture.internalChild
|
|
cast[type(f)](chronosInternalRetFuture.internalChild)
|
|
else:
|
|
unsupported "awaitne is only available within {.async.}"
|
|
|
|
macro async*(prc: untyped): untyped =
|
|
## Macro which processes async procedures into the appropriate
|
|
## iterators and yield statements.
|
|
if prc.kind == nnkStmtList:
|
|
result = newStmtList()
|
|
for oneProc in prc:
|
|
oneProc.addPragma(ident"async")
|
|
result.add asyncSingleProc(oneProc)
|
|
else:
|
|
prc.addPragma(ident"async")
|
|
result = asyncSingleProc(prc)
|
|
|
|
macro asyncraises*(possibleExceptions, prc: untyped): untyped =
|
|
# Add back the pragma and let asyncSingleProc handle it
|
|
# Exerimental / subject to change and/or removal
|
|
if prc.kind == nnkStmtList:
|
|
result = newStmtList()
|
|
for oneProc in prc:
|
|
oneProc.addPragma(nnkExprColonExpr.newTree(
|
|
ident"asyncraises",
|
|
possibleExceptions
|
|
))
|
|
result.add asyncSingleProc(oneProc)
|
|
else:
|
|
prc.addPragma(nnkExprColonExpr.newTree(
|
|
ident"asyncraises",
|
|
possibleExceptions
|
|
))
|
|
result = asyncSingleProc(prc)
|