nim-serialization/serialization.nim

165 lines
6.0 KiB
Nim
Raw Normal View History

import
2021-08-16 08:52:41 +02:00
std/typetraits,
stew/shims/macros, faststreams/[inputs, outputs],
./serialization/[object_serialization, errors, formats]
export
2021-08-16 08:52:41 +02:00
inputs, outputs, object_serialization, errors, formats
2021-03-18 12:55:50 +02:00
2018-12-18 01:00:00 +02:00
template encode*(Format: type, value: auto, params: varargs[untyped]): auto =
mixin init, Writer, writeValue, PreferredOutputType
{.noSideEffect.}:
# We assume that there is no side-effects here, because we are
# using a `memoryOutput`. The computed side-effects are coming
# from the fact that the dynamic dispatch mechanisms used in
# faststreams may be writing to a file or a network device.
try:
var s = memoryOutput()
type WriterType = Writer(Format)
var writer = unpackArgs(init, [WriterType, s, params])
writeValue writer, value
s.getOutput PreferredOutputType(Format)
except IOError:
raise (ref Defect)() # a memoryOutput cannot have an IOError
2018-12-18 01:00:00 +02:00
# TODO Nim cannot make sense of this initialization by var param?
proc readValue*(reader: var auto, T: type): T {.gcsafe, raises: [SerializationError, IOError].} =
{.warning[ProveInit]: off.}
2018-12-18 01:00:00 +02:00
mixin readValue
result = default(T)
2018-12-18 01:00:00 +02:00
reader.readValue(result)
{.warning[ProveInit]: on.}
2018-12-18 01:00:00 +02:00
template decode*(Format: distinct type,
input: string,
RecordType: distinct type,
params: varargs[untyped]): auto =
# TODO, this is dusplicated only due to a Nim bug:
2021-12-05 15:52:47 +01:00
# If `input` was `string|openArray[byte]`, it won't match `seq[byte]`
mixin init, Reader
{.noSideEffect.}:
# We assume that there are no side-effects here, because we are
# using a `memoryInput`. The computed side-effects are coming
# from the fact that the dynamic dispatch mechanisms used in
# faststreams may be reading from a file or a network device.
try:
2020-05-05 20:29:26 +03:00
var stream = unsafeMemoryInput(input)
type ReaderType = Reader(Format)
var reader = unpackArgs(init, [ReaderType, stream, params])
reader.readValue(RecordType)
except IOError:
2020-05-05 20:29:26 +03:00
raise (ref Defect)() # memory inputs cannot raise an IOError
template decode*(Format: distinct type,
2021-12-05 15:52:47 +01:00
input: openArray[byte],
RecordType: distinct type,
params: varargs[untyped]): auto =
# TODO, this is dusplicated only due to a Nim bug:
2021-12-05 15:52:47 +01:00
# If `input` was `string|openArray[byte]`, it won't match `seq[byte]`
mixin init, Reader
{.noSideEffect.}:
# We assume that there are no side-effects here, because we are
# using a `memoryInput`. The computed side-effects are coming
# from the fact that the dynamic dispatch mechanisms used in
# faststreams may be reading from a file or a network device.
try:
2020-05-05 20:29:26 +03:00
var stream = unsafeMemoryInput(input)
type ReaderType = Reader(Format)
var reader = unpackArgs(init, [ReaderType, stream, params])
reader.readValue(RecordType)
except IOError:
2020-05-05 20:29:26 +03:00
raise (ref Defect)() # memory inputs cannot raise an IOError
2018-12-18 01:00:00 +02:00
template loadFile*(Format: distinct type,
filename: string,
RecordType: distinct type,
params: varargs[untyped]): auto =
mixin init, Reader, readValue
var stream = memFileInput(filename)
try:
type ReaderType = Reader(Format)
var reader = unpackArgs(init, [ReaderType, stream, params])
reader.readValue(RecordType)
finally:
close stream
2018-12-18 01:00:00 +02:00
template loadFile*[RecordType](Format: type,
filename: string,
record: var RecordType,
params: varargs[untyped]) =
record = loadFile(Format, filename, RecordType, params)
template saveFile*(Format: type, filename: string, value: auto, params: varargs[untyped]) =
mixin init, Writer, writeValue
var stream = fileOutput(filename)
try:
type WriterType = Writer(Format)
var writer = unpackArgs(init, [WriterType, stream, params])
writer.writeValue(value)
finally:
close stream
2018-11-29 03:33:45 +02:00
template borrowSerialization*(Alias: type) {.dirty.} =
bind distinctBase
proc writeValue*[Writer](writer: var Writer, value: Alias) =
mixin writeValue
writeValue(writer, distinctBase value)
proc readValue*[Reader](reader: var Reader, value: var Alias) =
mixin readValue
value = Alias reader.readValue(distinctBase Alias)
2019-03-13 23:20:27 +02:00
template borrowSerialization*(Alias: distinct type,
OriginalType: distinct type) {.dirty.} =
proc writeValue*[Writer](writer: var Writer, value: Alias) =
mixin writeValue
writeValue(writer, OriginalType value)
proc readValue*[Reader](reader: var Reader, value: var Alias) =
mixin readValue
value = Alias reader.readValue(OriginalType)
template serializesAsBase*(SerializedType: distinct type,
Format: distinct type) =
mixin Reader, Writer
type ReaderType = Reader(Format)
type WriterType = Writer(Format)
template writeValue*(writer: var WriterType, value: SerializedType) =
mixin writeValue
writeValue(writer, distinctBase value)
template readValue*(reader: var ReaderType, value: var SerializedType) =
mixin readValue
value = SerializedType reader.readValue(distinctBase SerializedType)
macro serializesAsBaseIn*(SerializedType: type,
Formats: varargs[untyped]) =
result = newStmtList()
for Fmt in Formats:
result.add newCall(bindSym"serializesAsBase", SerializedType, Fmt)
2020-05-06 22:46:54 +03:00
template readValue*(stream: InputStream,
Format: type,
ValueType: type,
params: varargs[untyped]): untyped =
mixin Reader, init, readValue
type ReaderType = Reader(Format)
var reader = unpackArgs(init, [ReaderType, stream, params])
2020-05-06 22:46:54 +03:00
readValue reader, ValueType
template writeValue*(stream: OutputStream,
Format: type,
value: auto,
params: varargs[untyped]) =
mixin Writer, init, writeValue
type WriterType = Writer(Format)
var writer = unpackArgs(init, [WriterType, stream, params])
writeValue writer, value