2024-06-28 10:34:57 +00:00
|
|
|
{.push raises: [].}
|
2022-11-03 16:58:48 +00:00
|
|
|
|
|
|
|
import
|
|
|
|
std/[tables, typetraits, options, os],
|
|
|
|
serialization/object_serialization,
|
|
|
|
serialization/errors
|
2024-03-15 23:08:47 +00:00
|
|
|
import ./utils
|
2022-11-03 16:58:48 +00:00
|
|
|
|
|
|
|
type
|
|
|
|
EnvvarReader* = object
|
|
|
|
prefix: string
|
|
|
|
key: seq[string]
|
|
|
|
|
|
|
|
EnvvarError* = object of SerializationError
|
|
|
|
|
|
|
|
EnvvarReaderError* = object of EnvvarError
|
|
|
|
|
|
|
|
GenericEnvvarReaderError* = object of EnvvarReaderError
|
|
|
|
deserializedField*: string
|
|
|
|
innerException*: ref CatchableError
|
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
proc handleReadException*(
|
|
|
|
r: EnvvarReader,
|
|
|
|
Record: type,
|
|
|
|
fieldName: string,
|
|
|
|
field: auto,
|
|
|
|
err: ref CatchableError,
|
|
|
|
) {.raises: [GenericEnvvarReaderError].} =
|
2022-11-03 16:58:48 +00:00
|
|
|
var ex = new GenericEnvvarReaderError
|
|
|
|
ex.deserializedField = fieldName
|
|
|
|
ex.innerException = err
|
|
|
|
raise ex
|
|
|
|
|
|
|
|
proc init*(T: type EnvvarReader, prefix: string): T =
|
|
|
|
result.prefix = prefix
|
|
|
|
|
2023-09-21 11:12:14 +00:00
|
|
|
proc readValue*[T](r: var EnvvarReader, value: var T) {.raises: [SerializationError].} =
|
2022-11-03 16:58:48 +00:00
|
|
|
mixin readValue
|
|
|
|
|
|
|
|
when T is string:
|
|
|
|
let key = constructKey(r.prefix, r.key)
|
|
|
|
value = os.getEnv(key)
|
|
|
|
elif T is (SomePrimitives or range):
|
|
|
|
let key = constructKey(r.prefix, r.key)
|
2023-09-21 11:12:14 +00:00
|
|
|
try:
|
|
|
|
getValue(key, value)
|
|
|
|
except ValueError:
|
2024-03-15 23:08:47 +00:00
|
|
|
raise newException(
|
|
|
|
SerializationError,
|
|
|
|
"Couldn't getValue SomePrimitives: " & getCurrentExceptionMsg(),
|
|
|
|
)
|
2022-11-03 16:58:48 +00:00
|
|
|
elif T is Option:
|
2024-03-15 23:08:47 +00:00
|
|
|
template getUnderlyingType[T](_: Option[T]): untyped =
|
|
|
|
T
|
|
|
|
|
2022-11-03 16:58:48 +00:00
|
|
|
let key = constructKey(r.prefix, r.key)
|
|
|
|
if os.existsEnv(key):
|
|
|
|
type uType = getUnderlyingType(value)
|
|
|
|
when uType is string:
|
|
|
|
value = some(os.getEnv(key))
|
|
|
|
else:
|
2023-09-21 11:12:14 +00:00
|
|
|
try:
|
|
|
|
value = some(r.readValue(uType))
|
|
|
|
except ValueError, IOError:
|
2024-03-15 23:08:47 +00:00
|
|
|
raise newException(
|
|
|
|
SerializationError,
|
|
|
|
"Couldn't read Option value: " & getCurrentExceptionMsg(),
|
|
|
|
)
|
2022-11-03 16:58:48 +00:00
|
|
|
elif T is (seq or array):
|
|
|
|
when uTypeIsPrimitives(T):
|
|
|
|
let key = constructKey(r.prefix, r.key)
|
2024-06-05 13:32:35 +00:00
|
|
|
try:
|
|
|
|
getValue(key, value)
|
|
|
|
except ValueError:
|
|
|
|
raise newException(
|
|
|
|
SerializationError, "Couldn't get value: " & getCurrentExceptionMsg()
|
|
|
|
)
|
2022-11-03 16:58:48 +00:00
|
|
|
else:
|
|
|
|
let key = r.key[^1]
|
2024-03-15 23:08:47 +00:00
|
|
|
for i in 0 ..< value.len:
|
2022-11-03 16:58:48 +00:00
|
|
|
r.key[^1] = key & $i
|
|
|
|
r.readValue(value[i])
|
|
|
|
elif T is (object or tuple):
|
|
|
|
type T = type(value)
|
|
|
|
when T.totalSerializedFields > 0:
|
|
|
|
let fields = T.fieldReadersTable(EnvvarReader)
|
|
|
|
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
|
|
|
|
else:
|
|
|
|
r.key[^1] = fieldName
|
|
|
|
var reader = findFieldReader(fields[], fieldName, expectedFieldPos)
|
|
|
|
|
|
|
|
if reader != nil:
|
|
|
|
reader(value, r)
|
|
|
|
discard r.key.pop()
|
|
|
|
else:
|
|
|
|
const typeName = typetraits.name(T)
|
|
|
|
{.fatal: "Failed to convert from Envvar an unsupported type: " & typeName.}
|