2020-10-21 12:01:48 +07:00
|
|
|
import
|
2020-10-22 12:09:34 +07:00
|
|
|
tables, typetraits, options,
|
2020-10-21 12:01:48 +07:00
|
|
|
serialization/[object_serialization, errors],
|
2020-10-22 12:09:34 +07:00
|
|
|
./utils, ./types
|
2020-10-21 12:01:48 +07:00
|
|
|
|
|
|
|
type
|
|
|
|
WinregReader* = object
|
|
|
|
hKey: HKEY
|
|
|
|
path: string
|
2020-10-21 16:07:25 +07:00
|
|
|
key: seq[string]
|
2020-10-21 12:01:48 +07:00
|
|
|
|
|
|
|
WinregReaderError* = object of WinregError
|
|
|
|
|
|
|
|
GenericWinregReaderError* = object of WinregReaderError
|
|
|
|
deserializedField*: string
|
|
|
|
innerException*: ref CatchableError
|
|
|
|
|
|
|
|
proc handleReadException*(r: WinregReader,
|
|
|
|
Record: type,
|
|
|
|
fieldName: string,
|
|
|
|
field: auto,
|
|
|
|
err: ref CatchableError) =
|
|
|
|
var ex = new GenericWinregReaderError
|
|
|
|
ex.deserializedField = fieldName
|
|
|
|
ex.innerException = err
|
|
|
|
raise ex
|
|
|
|
|
|
|
|
proc init*(T: type WinregReader,
|
|
|
|
hKey: HKEY, path: string): T =
|
|
|
|
result.hKey = hKey
|
|
|
|
result.path = path
|
|
|
|
|
2020-10-22 12:09:34 +07:00
|
|
|
proc readValue*[T](r: var WinregReader, value: var T) =
|
2020-10-21 12:01:48 +07:00
|
|
|
mixin readValue
|
2020-10-21 16:07:25 +07:00
|
|
|
# TODO: reduce allocation
|
2020-10-21 12:01:48 +07:00
|
|
|
|
2020-10-21 16:07:25 +07:00
|
|
|
when T is (SomePrimitives or range or string):
|
|
|
|
let path = constructPath(r.path, r.key)
|
|
|
|
discard getValue(r.hKey, path, r.key[^1], value)
|
2020-10-31 17:44:03 +07:00
|
|
|
|
2020-10-21 16:07:25 +07:00
|
|
|
elif T is Option:
|
2020-10-28 16:12:30 +07:00
|
|
|
template getUnderlyingType[T](_: Option[T]): untyped = T
|
|
|
|
type UT = getUnderlyingType(value)
|
2020-10-21 16:07:25 +07:00
|
|
|
let path = constructPath(r.path, r.key)
|
2020-10-28 16:12:30 +07:00
|
|
|
if pathExists(r.hKey, path, r.key[^1]):
|
|
|
|
value = some(r.readValue(UT))
|
2020-10-31 17:44:03 +07:00
|
|
|
|
2020-10-21 12:01:48 +07:00
|
|
|
elif T is (seq or array):
|
|
|
|
when uTypeIsPrimitives(T):
|
2020-10-21 16:07:25 +07:00
|
|
|
let path = constructPath(r.path, r.key)
|
|
|
|
discard getValue(r.hKey, path, r.key[^1], value)
|
2020-10-31 17:44:03 +07:00
|
|
|
|
|
|
|
else:
|
2020-10-21 16:07:25 +07:00
|
|
|
let key = r.key[^1]
|
|
|
|
for i in 0..<value.len:
|
|
|
|
r.key[^1] = key & $i
|
|
|
|
r.readValue(value[i])
|
2020-10-31 17:44:03 +07:00
|
|
|
|
2020-10-21 12:01:48 +07:00
|
|
|
elif T is (object or tuple):
|
2020-10-21 16:07:25 +07:00
|
|
|
type T = type(value)
|
|
|
|
when T.totalSerializedFields > 0:
|
|
|
|
let fields = T.fieldReadersTable(WinregReader)
|
|
|
|
var expectedFieldPos = 0
|
|
|
|
r.key.add ""
|
|
|
|
value.enumInstanceSerializedFields(fieldName, field):
|
|
|
|
when T is tuple:
|
|
|
|
r.key[^1] = $expectedFieldPos
|
|
|
|
var reader = fields[][expectedFieldPos].reader
|
|
|
|
expectedFieldPos += 1
|
2020-10-31 17:44:03 +07:00
|
|
|
|
2020-10-21 16:07:25 +07:00
|
|
|
else:
|
|
|
|
r.key[^1] = fieldName
|
|
|
|
var reader = findFieldReader(fields[], fieldName, expectedFieldPos)
|
|
|
|
|
|
|
|
if reader != nil:
|
|
|
|
reader(value, r)
|
|
|
|
discard r.key.pop()
|
2020-10-31 17:44:03 +07:00
|
|
|
|
2020-10-21 12:01:48 +07:00
|
|
|
else:
|
|
|
|
const typeName = typetraits.name(T)
|
|
|
|
{.fatal: "Failed to convert from Winreg an unsupported type: " & typeName.}
|