2018-12-18 01:00:00 +02:00
|
|
|
import
|
2019-07-07 11:50:26 +02:00
|
|
|
stew/shims/macros
|
2018-11-11 13:40:19 +02:00
|
|
|
|
|
|
|
template dontSerialize* {.pragma.}
|
|
|
|
## Specifies that a certain field should be ignored for
|
|
|
|
## the purposes of serialization
|
|
|
|
|
2018-12-18 01:00:00 +02:00
|
|
|
type
|
|
|
|
FieldMarkerImpl*[name: static string] = object
|
|
|
|
|
|
|
|
FieldReader*[RecordType, Reader] = tuple[
|
|
|
|
fieldName: string,
|
|
|
|
reader: proc (rec: var RecordType, reader: var Reader) {.nimcall.}
|
|
|
|
]
|
|
|
|
|
|
|
|
FieldReadersTable*[RecordType, Reader] = openarray[FieldReader[RecordType, Reader]]
|
2018-11-11 13:40:19 +02:00
|
|
|
|
|
|
|
template eachSerializedFieldImpl*[T](x: T, op: untyped) =
|
2018-11-29 03:34:06 +02:00
|
|
|
when false:
|
|
|
|
static: echo treeRepr(T.getTypeImpl)
|
|
|
|
|
2018-11-11 13:40:19 +02:00
|
|
|
for k, v in fieldPairs(x):
|
2018-11-29 03:34:06 +02:00
|
|
|
when true: # not hasCustomPragma(v, dontSerialize):
|
2018-11-11 13:40:19 +02:00
|
|
|
op(k, v)
|
|
|
|
|
|
|
|
proc totalSerializedFieldsImpl(T: type): int =
|
|
|
|
mixin eachSerializedFieldImpl
|
|
|
|
|
|
|
|
proc helper: int =
|
|
|
|
var dummy: T
|
2018-12-28 03:01:24 +02:00
|
|
|
template countFields(k, v) = inc result
|
2018-11-11 13:40:19 +02:00
|
|
|
eachSerializedFieldImpl(dummy, countFields)
|
|
|
|
|
|
|
|
const res = helper()
|
|
|
|
return res
|
|
|
|
|
|
|
|
template totalSerializedFields*(T: type): int =
|
|
|
|
(static(totalSerializedFieldsImpl(T)))
|
|
|
|
|
|
|
|
macro serialziedFields*(T: typedesc, fields: varargs[untyped]): untyped =
|
|
|
|
var body = newStmtList()
|
|
|
|
let
|
|
|
|
ins = genSym(nskParam, "instance")
|
|
|
|
op = genSym(nskParam, "op")
|
|
|
|
|
|
|
|
for field in fields:
|
|
|
|
body.add quote do: `op`(`ins`.`field`)
|
|
|
|
|
|
|
|
result = quote do:
|
|
|
|
template eachSerializedFieldImpl*(`ins`: `T`, `op`: untyped) {.inject.} =
|
|
|
|
`body`
|
|
|
|
|
|
|
|
template serializeFields*(value: auto, fieldName, fieldValue, body: untyped) =
|
|
|
|
# TODO: this would be nicer as a for loop macro
|
|
|
|
mixin eachSerializedFieldImpl
|
2018-12-18 01:00:00 +02:00
|
|
|
|
2018-11-11 13:40:19 +02:00
|
|
|
template op(fieldName, fieldValue: untyped) = body
|
|
|
|
eachSerializedFieldImpl(value, op)
|
|
|
|
|
2019-03-11 11:34:04 +02:00
|
|
|
template deserializeFields*(value: auto, fieldName, fieldValue, body: untyped) =
|
|
|
|
# TODO: this would be nicer as a for loop macro
|
|
|
|
mixin eachSerializedFieldImpl
|
|
|
|
|
|
|
|
template op(fieldName, fieldValue: untyped) = body
|
|
|
|
eachSerializedFieldImpl(value, op)
|
|
|
|
|
2018-12-18 01:00:00 +02:00
|
|
|
macro customSerialization*(field: untyped, definition): untyped =
|
|
|
|
discard
|
|
|
|
|
|
|
|
proc hasDontSerialize(pragmas: NimNode): bool =
|
|
|
|
if pragmas == nil: return false
|
|
|
|
let dontSerialize = bindSym "dontSerialize"
|
|
|
|
for p in pragmas:
|
|
|
|
if p == dontSerialize:
|
|
|
|
return true
|
|
|
|
|
|
|
|
macro makeFieldReadersTable(RecordType, Reader: distinct type): untyped =
|
|
|
|
var obj = RecordType.getType[1].getImpl
|
|
|
|
|
|
|
|
result = newTree(nnkBracket)
|
|
|
|
|
|
|
|
for field in recordFields(obj):
|
2019-03-20 03:16:27 +02:00
|
|
|
let
|
|
|
|
fieldIdent = field.name
|
|
|
|
fieldName = newLit($fieldIdent)
|
2018-12-18 01:00:00 +02:00
|
|
|
if not hasDontSerialize(field.pragmas):
|
|
|
|
var handler = quote do:
|
|
|
|
return proc (obj: var `RecordType`, reader: var `Reader`) {.nimcall.} =
|
2019-03-20 03:16:27 +02:00
|
|
|
try:
|
|
|
|
reader.readValue(obj.`fieldIdent`)
|
|
|
|
except SerializationError:
|
|
|
|
raise
|
|
|
|
except CatchableError as err:
|
|
|
|
reader.readValueFailed(`RecordType`, `fieldName`, obj.`fieldIdent`, err)
|
|
|
|
|
|
|
|
result.add newTree(nnkTupleConstr, fieldName, handler[0])
|
2018-12-18 01:00:00 +02:00
|
|
|
|
|
|
|
proc fieldReadersTable*(RecordType, Reader: distinct type):
|
|
|
|
ptr seq[FieldReader[RecordType, Reader]] {.gcsafe.} =
|
|
|
|
mixin readValue
|
|
|
|
var tbl {.global.} = @(makeFieldReadersTable(RecordType, Reader))
|
|
|
|
{.gcsafe.}:
|
|
|
|
return addr(tbl)
|
|
|
|
|
|
|
|
proc findFieldReader*(fieldsTable: FieldReadersTable,
|
|
|
|
fieldName: string,
|
|
|
|
expectedFieldPos: var int): auto =
|
|
|
|
for i in expectedFieldPos ..< fieldsTable.len:
|
|
|
|
if fieldsTable[i].fieldName == fieldName:
|
|
|
|
expectedFieldPos = i + 1
|
|
|
|
return fieldsTable[i].reader
|
|
|
|
|
|
|
|
for i in 0 ..< expectedFieldPos:
|
|
|
|
if fieldsTable[i].fieldName == fieldName:
|
|
|
|
return fieldsTable[i].reader
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|