nim-namespaced-logging/src/namespaced_logging.nim

77 lines
2.7 KiB
Nim

import std/[logging, options, sequtils, strutils, tables]
export logging
type
LoggingNamespace* = ref object
name: string
level*: Level
msgPrefix*: string
var knownNamespacesInst {.threadvar.}: TableRef[string, LoggingNamespace]
template knownNamespaces(): TableRef[string, LoggingNamespace] =
if knownNamespacesInst == nil:
knownNamespacesInst = newTable[string, LoggingNamespace]()
knownNamespacesInst
proc initLoggingNamespace(
name: string,
level = lvlInfo,
msgPrefix: string
): LoggingNamespace {.raises: [].} =
result = LoggingNamespace(
name: name,
level: level,
msgPrefix: msgPrefix)
knownNamespaces[name] = result
proc getLoggerForNamespace*(
namespace: string,
level = lvlInfo,
msgPrefix: Option[string] = none[string]()
): LoggingNamespace {.raises: [].} =
## Get a LogginNamesapce for the given namespace. The first time this is
## called for a given name space a new logger will be created. In that case,
## the optional `level` and `msgPrefix` will be used to configure the logger.
## In all other cases, these paratmers are ignored and the existing namespace
## instance is returned
if knownNamespaces.hasKey(namespace):
try: return knownNamespaces[namespace]
except KeyError:
try: error "namespaced_logging: Impossible error. " &
"knownNamespaces contains " & namespace & " but raised a KeyError " &
"trying to access it."
except: discard
else:
if msgPrefix.isSome:
return initLoggingNamespace(namespace, level, msgPrefix.get)
else:
return initLoggingNamespace(namespace, level, namespace & ": ")
proc setLevelForNamespace*(namespace: string, lvl: Level, recursive = false) {.raises: [] .} =
if recursive:
for k, v in knownNamespaces.pairs:
if k.startsWith(namespace):
v.level = lvl
else: getLoggerForNamespace(namespace).level = lvl
proc name*(ns: LoggingNamespace): string = ns.name
proc log*(ns: LoggingNamespace, level: Level, args: varargs[string, `$`]) {.raises: [] .} =
try:
if level >= ns.level:
if not ns.msgPrefix.isEmptyOrWhitespace:
log(level, args.mapIt(ns.msgPrefix & it))
else: log(level, args)
except: discard
proc debug*(ns: LoggingNamespace, args: varargs[string, `$`]) = log(ns, lvlDebug, args)
proc info*(ns: LoggingNamespace, args: varargs[string, `$`]) = log(ns, lvlInfo, args)
proc notice*(ns: LoggingNamespace, args: varargs[string, `$`]) = log(ns, lvlNotice, args)
proc warn*(ns: LoggingNamespace, args: varargs[string, `$`]) = log(ns, lvlWarn, args)
proc error*(ns: LoggingNamespace, args: varargs[string, `$`]) = log(ns, lvlError, args)
proc fatal*(ns: LoggingNamespace, args: varargs[string, `$`]) = log(ns, lvlFatal, args)