Compare commits
	
		
			7 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| eaffb20df5 | |||
| 30c5720ab8 | |||
| 0a49618ab3 | |||
| b945534bd2 | |||
| 4bf35c336e | |||
| 751b3222db | |||
| 7af0acce68 | 
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,3 +1,2 @@ | ||||
| nimcache/ | ||||
| cliutils | ||||
| *.sw? | ||||
|   | ||||
							
								
								
									
										19
									
								
								cliutils.nim
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								cliutils.nim
									
									
									
									
									
								
							| @@ -1,13 +1,15 @@ | ||||
| import nre, terminal, strtabs, tables, unicode | ||||
| import std/[nre, terminal, strtabs, tables, unicode] | ||||
| import strutils except toUpper, toLower | ||||
|  | ||||
| import ./cliutils/config | ||||
| import ./cliutils/daemonize | ||||
| import ./cliutils/procutil | ||||
| import ./cliutils/queue_logger | ||||
|  | ||||
| export config | ||||
| export daemonize | ||||
| export procutil | ||||
| export queue_logger | ||||
|  | ||||
| type | ||||
|   TermColor = ForegroundColor or BackgroundColor | ||||
| @@ -27,7 +29,7 @@ proc stripAnsi*(str: string): string = | ||||
|   let STRIP_ANSI_REGEX = re"\x1B\[([0-9]{1,2}(;[0-9]{1,2})?)?[m|K]" | ||||
|   return str.replace(STRIP_ANSI_REGEX, "") | ||||
|  | ||||
| proc queryParamsToCliArgs*(queryParams: Table[string, string]): seq[string] = | ||||
| proc doParseQueryParams[T](queryParams: T): seq[string] = | ||||
|   result = @[] | ||||
|  | ||||
|   for k,v in queryParams: | ||||
| @@ -38,13 +40,8 @@ proc queryParamsToCliArgs*(queryParams: Table[string, string]): seq[string] = | ||||
|       result.add("--" & k) | ||||
|       if v != "true": result.add(v) # support things like ?verbose=true -> cmd --verbose | ||||
|  | ||||
| proc queryParamsToCliArgs*(queryParams: Table[string, string]): seq[string] = | ||||
|   doParseQueryParams(queryParams) | ||||
|  | ||||
| proc queryParamsToCliArgs*(queryParams: StringTableRef): seq[string] = | ||||
|   result = @[] | ||||
|  | ||||
|   for k,v in queryParams: | ||||
|     # support ?arg1=val1&arg2=val2 -> cmd val1 val2 | ||||
|     if k.startsWith("arg"): result.add(v) | ||||
|  | ||||
|     else : | ||||
|       result.add("--" & k) | ||||
|       if v != "true": result.add(v) # support things like ?verbose=true -> cmd --verbose | ||||
|   doParseQueryParams(queryParams) | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| # Package | ||||
|  | ||||
| version       = "0.8.0" | ||||
| version       = "0.9.1" | ||||
| author        = "Jonathan Bernard" | ||||
| description   = "Helper functions for writing command line interfaces." | ||||
| license       = "MIT" | ||||
|   | ||||
| @@ -1,10 +1,20 @@ | ||||
| import docopt, json, nre, os, strtabs, strutils | ||||
| import std/[json, nre, os, sequtils, strtabs, strutils] | ||||
| import docopt | ||||
|  | ||||
| type | ||||
|   CombinedConfig* = object | ||||
|     docopt*: Table[string, Value] | ||||
|     json*: JsonNode | ||||
|  | ||||
| proc keyNames(key: string): tuple[arg, env, json: string] {.raises: [KeyError].} = | ||||
|   try: | ||||
|     result = ( | ||||
|       "--" & key, | ||||
|       key.replace('-', '_').toUpper, | ||||
|       key.replace(re"(-\w)", proc (m: RegexMatch): string = ($m)[1..1].toUpper)) | ||||
|   except Exception: | ||||
|     raise newException(KeyError, "invalid config key: '" & key & "'") | ||||
|  | ||||
| template walkFieldDefs*(t: NimNode, body: untyped) = | ||||
|   let tTypeImpl = t.getTypeImpl | ||||
|  | ||||
| @@ -34,10 +44,25 @@ template walkFieldDefs*(t: NimNode, body: untyped) = | ||||
| #   cfgType.walkFieldDefs: | ||||
| #     let valLookup = quote do: `getVal`( | ||||
|  | ||||
| proc initCombinedConfig*( | ||||
|     filename: string, | ||||
|     docopt: Table[string, Value] = initTable[string, Value]() | ||||
|   ): CombinedConfig = | ||||
|  | ||||
|   result = CombinedConfig(docopt: docopt, json: parseFile(filename)) | ||||
|  | ||||
| proc findConfigFile*(name: string, otherLocations: seq[string] = @[]): string = | ||||
|   let cfgLocations = @[ | ||||
|     $getEnv(name.strip(chars = {'.'}).toUpper()), | ||||
|     $getEnv("HOME") / name, | ||||
|     "." / name ] & otherLocations | ||||
|  | ||||
|   result = cfgLocations.foldl(if fileExists(b): b else: a, "") | ||||
|   if result == "" or not fileExists(result): | ||||
|     raise newException(ValueError, "could not find configuration file") | ||||
|  | ||||
| proc getVal*(cfg: CombinedConfig, key: string): string = | ||||
|   let argKey = "--" & key | ||||
|   let envKey = key.replace('-', '_').toUpper | ||||
|   let jsonKey = key.replace(re"(-\w)", proc (m: RegexMatch): string = ($m)[1..1].toUpper) | ||||
|   let (argKey, envKey, jsonKey) = keyNames(key) | ||||
|  | ||||
|   if cfg.docopt.contains(argKey) and cfg.docopt[argKey]: return $cfg.docopt[argKey] | ||||
|   elif existsEnv(envKey): return getEnv(envKey) | ||||
| @@ -53,23 +78,41 @@ proc getVal*(cfg: CombinedConfig, key: string): string = | ||||
|     of JArray: return $node | ||||
|   else: raise newException(ValueError, "cannot find a configuration value for \"" & key & "\"") | ||||
|  | ||||
| proc getVal*(cfg: CombinedConfig, key, default: string): string = | ||||
| proc getVal*(cfg: CombinedConfig, key, default: string): string {.raises: [].} = | ||||
|   try: return getVal(cfg, key) | ||||
|   except: return default | ||||
|   except CatchableError: return default | ||||
|  | ||||
| proc getJson*(cfg: CombinedConfig, key: string): JsonNode = | ||||
|   let argKey = "--" & key | ||||
|   let envKey = key.replace('-', '_').toUpper | ||||
|   let jsonKey = key.replace(re"(-\w)", proc (m: RegexMatch): string = ($m)[1..1].toUpper) | ||||
| proc getJson*( | ||||
|     cfg: CombinedConfig, | ||||
|     key: string | ||||
|   ): JsonNode {.raises: [KeyError, ValueError, IOError, OSError]} = | ||||
|  | ||||
|   if cfg.docopt.contains(argKey) and cfg.docopt[argKey]: return parseJson($cfg.docopt[argKey]) | ||||
|   elif existsEnv(envKey): return parseJson(getEnv(envKey)) | ||||
|   elif cfg.json.hasKey(jsonKey): return cfg.json[jsonKey] | ||||
|   else: raise newException(ValueError, "cannot find a configuration value for \"" & key & "\"") | ||||
|   let (argKey, envKey, jsonKey) = keyNames(key) | ||||
|  | ||||
|   try: | ||||
|     if cfg.docopt.contains(argKey) and cfg.docopt[argKey]: return parseJson($cfg.docopt[argKey]) | ||||
|     elif existsEnv(envKey): return parseJson(getEnv(envKey)) | ||||
|     elif cfg.json.hasKey(jsonKey): return cfg.json[jsonKey] | ||||
|     else: raise newException(ValueError, "cannot find a configuration value for \"" & key & "\"") | ||||
|   except Exception: | ||||
|     raise newException(ValueError, "cannot parse value as JSON:" & getCurrentExceptionMsg()) | ||||
|  | ||||
| proc getJson*( | ||||
|     cfg: CombinedConfig, | ||||
|     key: string, | ||||
|     default: JsonNode | ||||
|   ) : JsonNode {.raises: []} = | ||||
|  | ||||
| proc getJson*(cfg: CombinedConfig, key: string, default: JsonNode): JsonNode = | ||||
|   try: return getJson(cfg, key) | ||||
|   except: return default | ||||
|   except CatchableError: return default | ||||
|  | ||||
| proc hasKey*(cfg: CombinedConfig, key: string): bool = | ||||
|   let (argKey, envKey, jsonKey) = keyNames(key) | ||||
|  | ||||
|   return | ||||
|     (cfg.docopt.contains(argKey) and cfg.docopt[argKey]) or | ||||
|     existsEnv(envKey) or | ||||
|     cfg.json.hasKey(jsonKey) | ||||
|  | ||||
| proc loadEnv*(): StringTableRef = | ||||
|   result = newStringTable() | ||||
|   | ||||
| @@ -19,27 +19,50 @@ proc daemonize*(pidfile, si, so, se: string, daemonMain: proc(): void): Pid = | ||||
|   if fileExists(pidfile): | ||||
|     raise newException(IOError, "pidfile " & pidfile & " already exists, daemon already running?") | ||||
|  | ||||
|   # We're about to do a little dance to end up with a process that is | ||||
|   # disconnected from the calling program's process group and is not a session | ||||
|   # leader (a daemon process). | ||||
|   # | ||||
|   # Remember, after a call to fork the code is now running in two processes, | ||||
|   # the parent process and the child process.  So the returned PID will be a | ||||
|   # positive value if we are still on the parent process, and represents the | ||||
|   # PID of the newly created process. If we are on the newly created process, | ||||
|   # the PID returned with be 0. | ||||
|  | ||||
|   let pid1 = fork() | ||||
|  | ||||
|   # Are we the child process? | ||||
|   if pid1 == 0: | ||||
|     # Yes, so let's get ready to execute the main code given to us. Note that | ||||
|     # this child process will not actually live long. As part of our setup, | ||||
|     # we're going to create a new process session, resulting in this child | ||||
|     # process being the process group owner and session leader. We don't our | ||||
|     # daemon process to be a session leader because that allows it to be | ||||
|     # connected to a terminal. So after we've setup the new process group we | ||||
|     # will fork again and actually run the logic we're daemonizing in the | ||||
|     # grandchild process. | ||||
|  | ||||
|     # Yes, so let's get ready to execute the main code given to us. | ||||
|     # Ignore whatever the original working directory was. | ||||
|     discard chdir("/") | ||||
|  | ||||
|     # Create a new session to decouple us from the original terminal. | ||||
|     discard setsid() | ||||
|  | ||||
|     # Set the process file mode creation mask to 0000 | ||||
|     discard umask(0) | ||||
|  | ||||
|     # Fork again... but I'm not sure why. | ||||
|     # Fork again so the grandchild process will not be the session leader. | ||||
|     let pid2 = fork() | ||||
|  | ||||
|     # We don't need the intermediate process, so if we are not the child this | ||||
|     # time, lets just quit | ||||
|     # If we're still on the child process, we're done. | ||||
|     if pid2 > 0: quit(QuitSuccess) | ||||
|  | ||||
|     # If we are the grandchild let's set up our environment. | ||||
|     # Otherwise, if we are the grandchild let's finish setting up our | ||||
|     # environment. | ||||
|     flushFile(stdout) | ||||
|     flushFile(stderr) | ||||
|  | ||||
|     # Setup our STDIN, STDERR, and STDOUT pipes (if given) | ||||
|     if si.len > 0: | ||||
|       fi = open(si, fmRead) | ||||
|       discard dup2(getFileHandle(fi), getFileHandle(stdin)) | ||||
| @@ -52,17 +75,20 @@ proc daemonize*(pidfile, si, so, se: string, daemonMain: proc(): void): Pid = | ||||
|       fe = open(se, fmAppend) | ||||
|       discard dup2(getFileHandle(fe), getFileHandle(stderr)) | ||||
|  | ||||
|     # Keep a record of the PID filename in our process memory so we can | ||||
|     # reference it in our cleanup hooks. | ||||
|     pidFileInner = pidfile | ||||
|  | ||||
|     # Add hooks to cleanup after ourselves when we're asked to die. | ||||
|     signal(SIGINT, onStop) | ||||
|     signal(SIGTERM, onStop) | ||||
|  | ||||
|     # Find out what our actual PID is and save it | ||||
|     # Find out what our actual PID is and save it to the PID file. | ||||
|     let childPid = getpid() | ||||
|     writeFile(pidfile, $childPid) | ||||
|  | ||||
|     # Finally, execute our main | ||||
|     # Finally, execute the logic given to us to daemonize | ||||
|     daemonMain() | ||||
|  | ||||
|   # If we're not the child proces, return the new PID of the daemonized process | ||||
|   return pid1 | ||||
|   | ||||
| @@ -47,7 +47,7 @@ proc exec*(command: string, workingDir: string = "", | ||||
|            args: openArray[string] = [], env: StringTableRef = nil, | ||||
|            options: set[ProcessOption] = {poUsePath}, | ||||
|            msgCB: HandleProcMsgCB = nil): int | ||||
|     {.tags: [ExecIOEffect, ReadIOEffect, RootEffect], gcsafe.} = | ||||
|     {.tags: [ExecIOEffect, ReadIOEffect, RootEffect].} = | ||||
|  | ||||
|   var p = startProcess(command, workingDir, args, env, options) | ||||
|   result = waitFor(p, msgCB, command) | ||||
|   | ||||
							
								
								
									
										25
									
								
								cliutils/queue_logger.nim
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								cliutils/queue_logger.nim
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| import std/[logging] | ||||
|  | ||||
| type | ||||
|   QueuedLogMessage* = tuple[lvl: Level, msg: string] | ||||
|  | ||||
|   QueueLogger* = ref object of Logger | ||||
|     queue: seq[QueuedLogMessage] | ||||
|  | ||||
| proc newQueueLogger*( | ||||
|   levelThreshold = lvlAll, | ||||
|   fmtStr = logging.defaultFmtStr): QueueLogger = | ||||
|  | ||||
|   new result | ||||
|   result.fmtStr = fmtStr | ||||
|   result.levelThreshold = levelThreshold | ||||
|   result.queue = newSeq[QueuedLogMessage]() | ||||
|  | ||||
| method log*(logger: QueueLogger, level: Level, args: varargs[string, `$`]) = | ||||
|   if level >= logger.levelThreshold: | ||||
|     logger.queue.add((level, substituteLog(logger.fmtStr, level, args))) | ||||
|  | ||||
| proc clearQueue*(logger: QueueLogger) = | ||||
|   logger.queue = newSeq[QueuedLogMessage]() | ||||
|  | ||||
| func messages*(logger: QueueLogger): seq[QueuedLogMessage] = logger.queue | ||||
		Reference in New Issue
	
	Block a user