2022-11-03 17:58:48 +01:00
|
|
|
when (NimMajor, NimMinor) < (1, 4):
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
else:
|
|
|
|
{.push raises: [].}
|
|
|
|
|
|
|
|
import
|
|
|
|
std/[os, strutils],
|
2023-03-31 15:24:04 +02:00
|
|
|
stew/byteutils,
|
|
|
|
stew/ptrops
|
2022-11-03 17:58:48 +01:00
|
|
|
|
|
|
|
|
|
|
|
type
|
|
|
|
SomePrimitives* = SomeInteger | enum | bool | SomeFloat | char
|
|
|
|
|
|
|
|
proc setValue*[T: SomePrimitives](key: string, val: openArray[T]) =
|
|
|
|
os.putEnv(key, byteutils.toHex(makeOpenArray(val[0].unsafeAddr, byte, val.len*sizeof(T))))
|
|
|
|
|
|
|
|
proc setValue*(key: string, val: SomePrimitives) =
|
|
|
|
os.putEnv(key, byteutils.toHex(makeOpenArray(val.unsafeAddr, byte, sizeof(val))))
|
|
|
|
|
|
|
|
proc decodePaddedHex(hex: string, res: ptr UncheckedArray[byte], outputLen: int) {.raises: [ValueError].} =
|
|
|
|
# make it an even length
|
|
|
|
let
|
|
|
|
inputLen = hex.len and not 0x01
|
|
|
|
numHex = inputLen div 2
|
|
|
|
maxLen = min(outputLen, numHex)
|
|
|
|
|
|
|
|
var
|
|
|
|
offI = hex.len - maxLen * 2
|
|
|
|
offO = outputLen - maxLen
|
|
|
|
|
|
|
|
for i in 0 ..< maxLen:
|
|
|
|
res[i + offO] = hex[2*i + offI].readHexChar shl 4 or hex[2*i + 1 + offI].readHexChar
|
|
|
|
|
|
|
|
# write single nibble from odd length hex
|
|
|
|
if (offO > 0) and (offI > 0):
|
|
|
|
res[offO-1] = hex[offI-1].readHexChar
|
|
|
|
|
|
|
|
proc getValue*(key: string, outVal: var string) {.raises: [ValueError].} =
|
|
|
|
let hex = os.getEnv(key)
|
|
|
|
let size = (hex.len div 2) + (hex.len and 0x01)
|
|
|
|
outVal.setLen(size)
|
|
|
|
decodePaddedHex(hex, cast[ptr UncheckedArray[byte]](outVal[0].addr), size)
|
|
|
|
|
|
|
|
proc getValue*[T: SomePrimitives](key: string, outVal: var seq[T]) =
|
|
|
|
let hex = os.getEnv(key)
|
|
|
|
let byteSize = (hex.len div 2) + (hex.len and 0x01)
|
|
|
|
let size = (byteSize + sizeof(T) - 1) div sizeof(T)
|
|
|
|
outVal.setLen(size)
|
|
|
|
decodePaddedHex(hex, cast[ptr UncheckedArray[byte]](outVal[0].addr), size * sizeof(T))
|
|
|
|
|
|
|
|
proc getValue*[N, T: SomePrimitives](key: string, outVal: var array[N, T]) =
|
|
|
|
let hex = os.getEnv(key)
|
|
|
|
decodePaddedHex(hex, cast[ptr UncheckedArray[byte]](outVal[0].addr), sizeof(outVal))
|
|
|
|
|
|
|
|
proc getValue*(key: string, outVal: var SomePrimitives) {.raises: [ValueError].} =
|
|
|
|
let hex = os.getEnv(key)
|
|
|
|
decodePaddedHex(hex, cast[ptr UncheckedArray[byte]](outVal.addr), sizeof(outVal))
|
|
|
|
|
|
|
|
template uTypeIsPrimitives*[T](_: type seq[T]): bool =
|
|
|
|
when T is SomePrimitives:
|
|
|
|
true
|
|
|
|
else:
|
|
|
|
false
|
|
|
|
|
|
|
|
template uTypeIsPrimitives*[N, T](_: type array[N, T]): bool =
|
|
|
|
when T is SomePrimitives:
|
|
|
|
true
|
|
|
|
else:
|
|
|
|
false
|
|
|
|
|
|
|
|
template uTypeIsPrimitives*[T](_: type openArray[T]): bool =
|
|
|
|
when T is SomePrimitives:
|
|
|
|
true
|
|
|
|
else:
|
|
|
|
false
|
|
|
|
|
|
|
|
template uTypeIsRecord*(_: typed): bool =
|
|
|
|
false
|
|
|
|
|
|
|
|
template uTypeIsRecord*[T](_: type seq[T]): bool =
|
|
|
|
when T is (object or tuple):
|
|
|
|
true
|
|
|
|
else:
|
|
|
|
false
|
|
|
|
|
|
|
|
template uTypeIsRecord*[N, T](_: type array[N, T]): bool =
|
|
|
|
when T is (object or tuple):
|
|
|
|
true
|
|
|
|
else:
|
|
|
|
false
|
|
|
|
|
|
|
|
|
|
|
|
func constructKey*(prefix: string, keys: openArray[string]): string =
|
|
|
|
var newKey: string
|
|
|
|
|
|
|
|
let envvarPrefix = prefix.strip().toUpper().multiReplace(("-", "_"), (" ", "_"))
|
|
|
|
newKey.add(envvarPrefix)
|
|
|
|
|
|
|
|
|
|
|
|
for k in keys:
|
|
|
|
newKey.add("_")
|
|
|
|
|
|
|
|
let envvarKey = k.toUpper().multiReplace(("-", "_"), (" ", "_"))
|
|
|
|
newKey.add(envvarKey)
|
|
|
|
|
|
|
|
newKey
|