## Personal Time Keeper ## ==================== ## ## Simple time keeping CLI import algorithm, docopt, json, langutils, logging, os, nre, std/wordwrap, sequtils, sets, strutils, sugar, tempfile, terminal, times, uuids import timeutils except `-` import private/util import private/api import private/models import private/version #proc `$`*(mark: Mark): string = #return (($mark.uuid)[ proc exitErr(msg: string): void = fatal "ptk: " & msg quit(QuitFailure) proc flexFormat(i: Duration): string = ## Pretty-format a time interval. let fmt = if i > initDuration(days = 1): "d'd' H'h' m'm'" elif i >= initDuration(hours = 1): "H'h' m'm'" elif i >= initDuration(minutes = 1): "m'm' s's'" else: "s's'" return i.format(fmt) type WriteData = tuple[idx: int, mark: Mark, prefixLen: int, interval: Duration] proc writeMarks(timeline: Timeline, indices: seq[int], includeNotes = false): void = ## Write a nicely-formatted list of Marks to stdout. let marks = timeline.marks let now = getTime().local if indices.len == 0: writeLine(stdout, "No marks match the given criteria.") return var idxs = indices.sorted((a, b) => cmp(marks[a].time, marks[b].time)) let largestInterval = now - marks[idxs.first].time let timeFormat = if largestInterval > initDuration(days = 365): "yyyy-MM-dd HH:mm" elif largestInterval > initDuration(days = 7): "MMM dd HH:mm" elif largestInterval > initDuration(days = 1): "ddd HH:mm" else: "HH:mm" var toWrite: seq[WriteData] = @[] var longestPrefix = 0 for i in idxs: let interval: Duration = if (i == marks.len - 1): now - marks[i].time else: marks[i + 1].time - marks[i].time prefix = ($marks[i].id)[0..<8] & " " & marks[i].time.format(timeFormat) & " (" & interval.flexFormat & ")" toWrite.add(( idx: i, mark: marks[i], prefixLen: prefix.len, interval: interval)) if prefix.len > longestPrefix: longestPrefix = prefix.len let colWidth = 80 let notesPrefixLen = 4 for w in toWrite: if w.mark.summary == STOP_MSG: continue setForegroundColor(stdout, fgBlack, true) write(stdout, ($w.mark.id)[0..<8]) setForegroundColor(stdout, fgYellow) write(stdout, " " & w.mark.time.format(timeFormat)) setForegroundColor(stdout, fgCyan) write(stdout, " (" & w.interval.flexFormat & ")") resetAttributes(stdout) write(stdout, spaces(longestPrefix - w.prefixLen) & " -- " & w.mark.summary) if w.mark.tags.len > 0: setForegroundColor(stdout, fgGreen) write(stdout, " (" & w.mark.tags.join(", ") & ")") resetAttributes(stdout) writeLine(stdout, "") if includeNotes and len(w.mark.notes.strip) > 0: writeLine(stdout, "") let wrappedNotes = wrapWords(s = w.mark.notes, maxLineWidth = colWidth) for line in splitLines(wrappedNotes): writeLine(stdout, spaces(notesPrefixLen) & line) writeLine(stdout, "") proc doInit(timelineLocation: string): void = ## Interactively initialize a new timeline at the given file path. stdout.write "Time log name [New Timeline]: " let name = stdin.readLine() let timeline = %* { "name": if name.strip.len > 0: name.strip else: "New Timeline", "marks": [] } #"createdAt": getLocalTime().format("yyyy-MM-dd'T'HH:mm:ss") } var timelineFile: File try: timelineFile = open(timelineLocation, fmWrite) timelineFile.write($timeline.pretty) finally: close(timelineFile) type ExpectedMarkPart = enum Time, Summary, Tags, Notes proc edit(mark: Mark): Mark = ## Interactively edit a mark using the editor named in the environment ## variable "EDITOR" result = mark var tempFile: File tempFileName: string try: (tempFile, tempFileName) = mkstemp("timestamp-mark-", ".txt", "", fmWrite) tempFile.writeLine( """# Edit the time, mark, tags, and notes below. Any lines starting with '#' will # be ignored. When done, save the file and close the editor.""") tempFile.writeLine(mark.time.format(ISO_TIME_FORMAT)) tempFile.writeLine(mark.summary) tempFile.writeLine(mark.tags.join(",")) tempFile.writeLine( """# Everything from the line below to the end of the file will be considered # notes for this timeline mark.""") tempFile.write(mark.notes) close(tempFile) tempFile = nil let editor = getEnv("EDITOR", "vim") discard os.execShellCmd editor & " " & tempFileName & " /dev/tty" var markPart = Time var notes: seq[string] = @[] for line in lines tempFileName: if strip(line).len > 0 and strip(line)[0] == '#': continue elif markPart == Time: result.time = parseTime(line); markPart = Summary elif markPart == Summary: result.summary = line; markPart = Tags elif markPart == Tags: result.tags = line.split({',', ';'}); markPart = Notes else: notes.add(line) result.notes = notes.join("\n") finally: close(tempFile) proc filterMarkIndices(timeline: Timeline, args: Table[string, Value]): seq[int] = ## Filter down a set of marks according to options provided in command line ## arguments. let marks = timeline.marks let now = getTime().local let allIndices = sequtils.toSeq(0.."]: let idx = marks.findById($args[""]) if idx > 0: selected = selected.filterMarks(mIdx >= idx) if args[""]: let idx = marks.findById($args[""]) if (idx > 0): selected = selected.filterMarks(mIdx <= idx) if args["--after"]: var startTime: DateTime try: startTime = parseTime($args["--after"]) except: raise newException(ValueError, "invalid value for --after: " & getCurrentExceptionMsg()) selected = selected.filterMarks(marks[mIdx].time > startTime) if args["--before"]: var endTime: DateTime try: endTime = parseTime($args["--before"]) except: raise newException(ValueError, "invalid value for --before: " & getCurrentExceptionMsg()) selected = selected.filterMarks(marks[mIdx].time < endTime) if args["--today"]: let b = now.startOfDay let e = b + 1.days selected = selected.filterMarks(marks[mIdx].time >= b and marks[mIdx].time < e) if args["--yesterday"]: let e = now.startOfDay let b = e - 1.days selected = selected.filterMarks(marks[mIdx].time >= b and marks[mIdx].time < e) if args["--this-week"]: let b = now.startOfWeek(dSun) let e = b + 7.days selected = selected.filterMarks(marks[mIdx].time >= b and marks[mIdx].time < e) if args["--last-week"]: let e = now.startOfWeek(dSun) let b = e - 7.days selected = selected.filterMarks(marks[mIdx].time >= b and marks[mIdx].time < e) if args["--tags"]: let tags = (args["--tags"] ?: "").split({',', ';'}) selected = selected.filterMarks(tags.allIt(marks[mIdx].tags.contains(it))) if args["--remove-tags"]: let tags = (args["--remove-tags"] ?: "").split({',', ';'}) selected = selected.filterMarks(not tags.allIt(marks[mIdx].tags.contains(it))) if args["--matching"]: let pattern = re("(?i)" & $(args["--matching"] ?: "")) selected = selected.filterMarks(marks[mIdx].summary.find(pattern).isSome) return sequtils.toSeq(selected.items).sorted(system.cmp) when isMainModule: try: let doc = """ Usage: ptk init [options] ptk (add | start) [options] ptk (add | start) [options] ptk resume [options] [] ptk amend [options] [] [] ptk merge [...] ptk stop [options] ptk continue ptk delete ptk (list | ls) [options] ptk (list | ls) tags ptk current ptk sum-time --ids ... ptk sum-time [options] [] [] ptk serve-api [--port ] ptk (-V | --version) ptk (-h | --help) Options: -E --echo-args Echo the program's understanding of it's arguments. -V --version Print the tool's version information. -a --after Restrict the selection to marks after . -b --before Restrict the selection to marks after . -c --config Use as configuration for the CLI. -e --edit Open the mark in an editor. -f --file Use the given timeline file. -g --tags Add the given tags (comma-separated) to the selected marks. -G --remove-tags Remove the given tag from the selected marks. -h --help Print this usage information. -m --matching Restric the selection to marks matching . -n --notes For add and amend, set the notes for a time mark. -t --time