2018-12-18 17:39:39 +02:00
|
|
|
import
|
|
|
|
macros
|
|
|
|
|
|
|
|
export
|
|
|
|
macros
|
|
|
|
|
|
|
|
type
|
|
|
|
FieldDescription* = object
|
|
|
|
name*: NimNode
|
|
|
|
isPublic*: bool
|
|
|
|
typ*: NimNode
|
|
|
|
pragmas*: NimNode
|
2018-12-19 12:27:48 +02:00
|
|
|
caseField*: NimNode
|
|
|
|
caseBranch*: NimNode
|
|
|
|
|
|
|
|
const
|
|
|
|
nnkPragmaCallKinds = {nnkExprColonExpr, nnkCall, nnkCallStrLit}
|
|
|
|
|
|
|
|
proc findPragma*(pragmas: NimNode, pragmaSym: NimNode): NimNode =
|
|
|
|
for p in pragmas:
|
2019-06-05 03:14:54 +03:00
|
|
|
if p.kind in {nnkSym, nnkIdent} and eqIdent(p, pragmaSym):
|
2018-12-19 12:27:48 +02:00
|
|
|
return p
|
2019-06-05 03:14:54 +03:00
|
|
|
if p.kind in nnkPragmaCallKinds and p.len > 0 and eqIdent(p[0], pragmaSym):
|
2018-12-19 12:27:48 +02:00
|
|
|
return p
|
2018-12-18 17:39:39 +02:00
|
|
|
|
2019-07-03 02:30:30 +03:00
|
|
|
template readPragma*(field: FieldDescription, pragmaName: static string): NimNode =
|
2019-07-03 10:15:40 +03:00
|
|
|
let p = findPragma(field.pragmas, bindSym(pragmaName))
|
2019-07-03 02:30:30 +03:00
|
|
|
if p != nil and p.len == 2: p[1] else: p
|
|
|
|
|
2018-12-18 17:39:39 +02:00
|
|
|
iterator recordFields*(typeImpl: NimNode): FieldDescription =
|
|
|
|
# TODO: This doesn't support inheritance yet
|
|
|
|
let
|
|
|
|
objectType = typeImpl[2]
|
|
|
|
recList = objectType[2]
|
|
|
|
|
2018-12-19 12:27:48 +02:00
|
|
|
type
|
|
|
|
RecursionStackItem = tuple
|
|
|
|
currentNode: NimNode
|
|
|
|
currentChildItem: int
|
|
|
|
parentCaseField: NimNode
|
|
|
|
parentCaseBranch: NimNode
|
|
|
|
|
2018-12-18 17:39:39 +02:00
|
|
|
if recList.len > 0:
|
2018-12-19 12:27:48 +02:00
|
|
|
var traversalStack: seq[RecursionStackItem] = @[
|
|
|
|
(recList, 0, NimNode(nil), NimNode(nil))
|
|
|
|
]
|
|
|
|
|
|
|
|
template recuseInto(childNode: NimNode,
|
2019-03-26 01:16:39 +02:00
|
|
|
currentCaseField: NimNode = nil,
|
|
|
|
currentCaseBranch: NimNode = nil) =
|
2018-12-19 12:27:48 +02:00
|
|
|
traversalStack.add (childNode, 0, currentCaseField, currentCaseBranch)
|
|
|
|
|
2018-12-18 17:39:39 +02:00
|
|
|
while true:
|
2019-03-13 23:53:00 +01:00
|
|
|
doAssert traversalStack.len > 0
|
2018-12-18 17:39:39 +02:00
|
|
|
|
2018-12-19 12:27:48 +02:00
|
|
|
var stackTop = traversalStack[^1]
|
|
|
|
let recList = stackTop.currentNode
|
|
|
|
let idx = stackTop.currentChildItem
|
2018-12-18 17:39:39 +02:00
|
|
|
let n = recList[idx]
|
2018-12-19 12:27:48 +02:00
|
|
|
inc traversalStack[^1].currentChildItem
|
2018-12-18 17:39:39 +02:00
|
|
|
|
|
|
|
if idx == recList.len - 1:
|
|
|
|
discard traversalStack.pop
|
|
|
|
|
|
|
|
case n.kind
|
|
|
|
of nnkRecWhen:
|
|
|
|
for i in countdown(n.len - 1, 0):
|
|
|
|
let branch = n[i]
|
|
|
|
case branch.kind:
|
|
|
|
of nnkElifBranch:
|
2018-12-19 12:27:48 +02:00
|
|
|
recuseInto branch[1]
|
2018-12-18 17:39:39 +02:00
|
|
|
of nnkElse:
|
2018-12-19 12:27:48 +02:00
|
|
|
recuseInto branch[0]
|
2018-12-18 17:39:39 +02:00
|
|
|
else:
|
2019-03-13 23:53:00 +01:00
|
|
|
doAssert false
|
2018-12-18 17:39:39 +02:00
|
|
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
of nnkRecCase:
|
2019-03-13 23:53:00 +01:00
|
|
|
doAssert n.len > 0
|
2018-12-18 17:39:39 +02:00
|
|
|
for i in countdown(n.len - 1, 1):
|
|
|
|
let branch = n[i]
|
|
|
|
case branch.kind
|
|
|
|
of nnkOfBranch:
|
2018-12-19 12:27:48 +02:00
|
|
|
recuseInto branch[^1], n[0], branch
|
2018-12-18 17:39:39 +02:00
|
|
|
of nnkElse:
|
2018-12-19 12:27:48 +02:00
|
|
|
recuseInto branch[0], n[0], branch
|
2018-12-18 17:39:39 +02:00
|
|
|
else:
|
2019-03-13 23:53:00 +01:00
|
|
|
doAssert false
|
2018-12-18 17:39:39 +02:00
|
|
|
|
2019-03-26 01:16:39 +02:00
|
|
|
recuseInto newTree(nnkRecCase, n[0]), n[0]
|
2018-12-18 17:39:39 +02:00
|
|
|
continue
|
|
|
|
|
|
|
|
of nnkIdentDefs:
|
|
|
|
let fieldType = n[^2]
|
|
|
|
for i in 0 ..< n.len - 2:
|
|
|
|
var field: FieldDescription
|
|
|
|
field.name = n[i]
|
2018-12-19 12:27:48 +02:00
|
|
|
field.typ = fieldType
|
|
|
|
field.caseField = stackTop.parentCaseField
|
|
|
|
field.caseBranch = stackTop.parentCaseBranch
|
2018-12-18 17:39:39 +02:00
|
|
|
|
|
|
|
if field.name.kind == nnkPragmaExpr:
|
|
|
|
field.pragmas = field.name[1]
|
|
|
|
field.name = field.name[0]
|
|
|
|
|
2018-12-19 12:27:48 +02:00
|
|
|
if field.name.kind == nnkPostfix:
|
|
|
|
field.isPublic = true
|
|
|
|
field.name = field.name[1]
|
|
|
|
|
2018-12-18 17:39:39 +02:00
|
|
|
yield field
|
|
|
|
|
|
|
|
of nnkNilLit, nnkDiscardStmt, nnkCommentStmt, nnkEmpty:
|
|
|
|
discard
|
|
|
|
|
|
|
|
else:
|
2019-03-13 23:53:00 +01:00
|
|
|
doAssert false
|
2018-12-18 17:39:39 +02:00
|
|
|
|
|
|
|
if traversalStack.len == 0: break
|
|
|
|
|
2019-07-03 02:30:30 +03:00
|
|
|
macro field*(obj: typed, fieldName: static string): untyped =
|
|
|
|
newDotExpr(obj, ident fieldName)
|
|
|
|
|
2019-01-23 13:47:13 +02:00
|
|
|
proc skipPragma*(n: NimNode): NimNode =
|
|
|
|
if n.kind == nnkPragmaExpr: n[0]
|
|
|
|
else: n
|
|
|
|
|
2018-12-19 12:27:48 +02:00
|
|
|
# FIXED NewLit
|
|
|
|
|
|
|
|
proc newLitFixed*(c: char): NimNode {.compileTime.} =
|
|
|
|
## produces a new character literal node.
|
|
|
|
result = newNimNode(nnkCharLit)
|
|
|
|
result.intVal = ord(c)
|
|
|
|
|
|
|
|
proc newLitFixed*(i: int): NimNode {.compileTime.} =
|
|
|
|
## produces a new integer literal node.
|
|
|
|
result = newNimNode(nnkIntLit)
|
|
|
|
result.intVal = i
|
|
|
|
|
|
|
|
proc newLitFixed*(i: int8): NimNode {.compileTime.} =
|
|
|
|
## produces a new integer literal node.
|
|
|
|
result = newNimNode(nnkInt8Lit)
|
|
|
|
result.intVal = i
|
|
|
|
|
|
|
|
proc newLitFixed*(i: int16): NimNode {.compileTime.} =
|
|
|
|
## produces a new integer literal node.
|
|
|
|
result = newNimNode(nnkInt16Lit)
|
|
|
|
result.intVal = i
|
|
|
|
|
|
|
|
proc newLitFixed*(i: int32): NimNode {.compileTime.} =
|
|
|
|
## produces a new integer literal node.
|
|
|
|
result = newNimNode(nnkInt32Lit)
|
|
|
|
result.intVal = i
|
|
|
|
|
|
|
|
proc newLitFixed*(i: int64): NimNode {.compileTime.} =
|
|
|
|
## produces a new integer literal node.
|
|
|
|
result = newNimNode(nnkInt64Lit)
|
|
|
|
result.intVal = i
|
|
|
|
|
|
|
|
proc newLitFixed*(i: uint): NimNode {.compileTime.} =
|
|
|
|
## produces a new unsigned integer literal node.
|
|
|
|
result = newNimNode(nnkUIntLit)
|
|
|
|
result.intVal = BiggestInt(i)
|
|
|
|
|
|
|
|
proc newLitFixed*(i: uint8): NimNode {.compileTime.} =
|
|
|
|
## produces a new unsigned integer literal node.
|
|
|
|
result = newNimNode(nnkUInt8Lit)
|
|
|
|
result.intVal = BiggestInt(i)
|
|
|
|
|
|
|
|
proc newLitFixed*(i: uint16): NimNode {.compileTime.} =
|
|
|
|
## produces a new unsigned integer literal node.
|
|
|
|
result = newNimNode(nnkUInt16Lit)
|
|
|
|
result.intVal = BiggestInt(i)
|
|
|
|
|
|
|
|
proc newLitFixed*(i: uint32): NimNode {.compileTime.} =
|
|
|
|
## produces a new unsigned integer literal node.
|
|
|
|
result = newNimNode(nnkUInt32Lit)
|
|
|
|
result.intVal = BiggestInt(i)
|
|
|
|
|
|
|
|
proc newLitFixed*(i: uint64): NimNode {.compileTime.} =
|
|
|
|
## produces a new unsigned integer literal node.
|
|
|
|
result = newNimNode(nnkUInt64Lit)
|
|
|
|
result.intVal = BiggestInt(i)
|
|
|
|
|
|
|
|
proc newLitFixed*(b: bool): NimNode {.compileTime.} =
|
|
|
|
## produces a new boolean literal node.
|
|
|
|
result = if b: bindSym"true" else: bindSym"false"
|
|
|
|
|
|
|
|
proc newLitFixed*(f: float32): NimNode {.compileTime.} =
|
|
|
|
## produces a new float literal node.
|
|
|
|
result = newNimNode(nnkFloat32Lit)
|
|
|
|
result.floatVal = f
|
|
|
|
|
|
|
|
proc newLitFixed*(f: float64): NimNode {.compileTime.} =
|
|
|
|
## produces a new float literal node.
|
|
|
|
result = newNimNode(nnkFloat64Lit)
|
|
|
|
result.floatVal = f
|
|
|
|
|
|
|
|
proc newLitFixed*(s: string): NimNode {.compileTime.} =
|
|
|
|
## produces a new string literal node.
|
|
|
|
result = newNimNode(nnkStrLit)
|
|
|
|
result.strVal = s
|
|
|
|
|
|
|
|
proc newLitFixed*[N,T](arg: array[N,T]): NimNode {.compileTime.}
|
|
|
|
proc newLitFixed*[T](arg: seq[T]): NimNode {.compileTime.}
|
|
|
|
proc newLitFixed*(arg: tuple): NimNode {.compileTime.}
|
|
|
|
|
|
|
|
proc newLitFixed*(arg: object): NimNode {.compileTime.} =
|
|
|
|
result = nnkObjConstr.newTree(arg.type.getTypeInst[1])
|
|
|
|
for a, b in arg.fieldPairs:
|
|
|
|
result.add nnkExprColonExpr.newTree( newIdentNode(a), newLitFixed(b) )
|
|
|
|
|
|
|
|
proc newLitFixed*[N,T](arg: array[N,T]): NimNode {.compileTime.} =
|
|
|
|
result = nnkBracket.newTree
|
|
|
|
for x in arg:
|
|
|
|
result.add newLitFixed(x)
|
|
|
|
|
|
|
|
proc newLitFixed*[T](arg: seq[T]): NimNode {.compileTime.} =
|
|
|
|
var bracket = nnkBracket.newTree
|
|
|
|
for x in arg:
|
|
|
|
bracket.add newLitFixed(x)
|
|
|
|
|
|
|
|
result = nnkCall.newTree(
|
|
|
|
nnkBracketExpr.newTree(
|
|
|
|
nnkAccQuoted.newTree( bindSym"@" ),
|
|
|
|
getTypeInst( bindSym"T" )
|
|
|
|
),
|
|
|
|
bracket
|
|
|
|
)
|
|
|
|
|
|
|
|
proc newLitFixed*(arg: tuple): NimNode {.compileTime.} =
|
|
|
|
result = nnkPar.newTree
|
|
|
|
for a,b in arg.fieldPairs:
|
|
|
|
result.add nnkExprColonExpr.newTree(newIdentNode(a), newLitFixed(b))
|
|
|
|
|
2019-03-11 11:37:51 +02:00
|
|
|
iterator typedParams*(n: NimNode, skip = 0): (NimNode, NimNode) =
|
2019-05-22 09:50:49 +03:00
|
|
|
let params = n[3]
|
|
|
|
for i in (1 + skip) ..< params.len:
|
|
|
|
let paramNodes = params[i]
|
2019-03-11 11:37:51 +02:00
|
|
|
let paramType = paramNodes[^2]
|
|
|
|
|
|
|
|
for j in 0 ..< paramNodes.len - 2:
|
|
|
|
yield (paramNodes[j], paramType)
|
|
|
|
|