## Personal Time Keeper ## ==================== ## ## Simple time keeping CLI import algorithm, docopt, json, langutils, logging, os, nre, sequtils, strutils, tempfile, terminal, times, timeutils, uuids import ptkutil type Mark* = tuple[id: UUID, time: TimeInfo, summary: string, notes: string, tags: seq[string]] Timeline* = tuple[name: string, marks: seq[Mark]] const STOP_MSG = "STOP" let NO_MARK: Mark = ( id: parseUUID("00000000-0000-0000-0000-000000000000"), time: fromSeconds(0).getLocalTime, summary: "", notes: "", tags: @[]) const ISO_TIME_FORMAT = "yyyy:MM:dd'T'HH:mm:ss" const TIME_FORMATS = @[ "H:mm", "HH:mm", "H:mm:ss", "HH:mm:ss", "yyyy:MM:dd'T'HH:mm:ss", "yyyy:MM:dd'T'HH:mm"] #proc `$`*(mark: Mark): string = #return (($mark.uuid)[ proc exitErr(msg: string): void = fatal "ptk: " & msg quit(QuitFailure) proc parseTime(timeStr: string): TimeInfo = for fmt in TIME_FORMATS: try: return parse(timeStr, fmt) except: discard nil raise newException(Exception, "unable to interpret as a date: " & timeStr) template `%`(mark: Mark): JsonNode = %* { "id": $(mark.id), "time": mark.time.format(ISO_TIME_FORMAT), "summary": mark.summary, "notes": mark.notes, "tags": mark.tags } template `%`(timeline: Timeline): JsonNode = %* { "name": timeline.name, "marks": timeline.marks } proc loadTimeline(filename: string): Timeline = var timelineJson: JsonNode try: timelineJson = parseFile(filename) except: raise newException(ValueError, "unable to parse the timeline file as JSON: " & filename) var timeline: Timeline = (name: timelineJson["name"].getStr(), marks: @[]) for markJson in timelineJson["marks"]: timeline.marks.add(( id: parseUUID(markJson["id"].getStr()), time: parse(markJson["time"].getStr(), ISO_TIME_FORMAT), summary: markJson["summary"].getStr(), notes: markJson["notes"].getStr(), tags: markJson["tags"].getElems(@[]).map(proc (t: JsonNode): string = t.getStr()))) timeline.marks = timeline.marks.sorted( proc(a, b: Mark): int = cmp(a.time, b.time)) return timeline proc saveTimeline(timeline: Timeline, location: string): void = var timelineFile: File try: timelineFile = open(location, fmWrite) timelineFile.writeLine(pretty(%timeline)) except: raise newException(IOError, "unable to save changes to " & location) finally: close(timelineFile) proc flexFormat(i: TimeInterval): string = let fmt = if i > 1.days: "d'd' H'h' m'm'" elif i >= 1.hours: "H'h' m'm'" elif i >= 1.minutes: "m'm' s's'" else: "s's'" return i.format(fmt) type WriteData = tuple[idx: int, mark: Mark, prefixLen: int, interval: TimeInterval] proc writeMarks(timeline: Timeline, indices: seq[int], includeNotes = false): void = let marks = timeline.marks let now = getLocalTime(getTime()) var idxs = indices.sorted( proc(a, b: int): int = cmp(marks[a].time, marks[b].time)) let timeFormat = if now - marks[idxs.first].time > 1.years: "yyyy-MM-dd HH:mm" elif now - marks[idxs.first].time > 7.days: "MMM dd HH:mm" elif now - marks[idxs.first].time > 1.days: "ddd HH:mm" else: "HH:mm" var toWrite: seq[WriteData] = @[] var longestPrefix = 0 for i in idxs: let interval: TimeInterval = 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 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, spaces(longestPrefix) & w.mark.notes) writeLine(stdout, "") proc formatMark(mark: Mark, nextMark = NO_MARK, timeFormat = ISO_TIME_FORMAT, includeNotes = false): string = let nextTime = if nextMark == NO_MARK: getLocalTime(getTime()) else: nextMark.time let duration = (nextTime - mark.time).flexFormat # TODO: pick up here calculating the time between marks let prefix = ($mark.id)[0..<8] & " " & mark.time.format(timeFormat) & " (" & duration & ") -- " let prefixLen = len(($mark.id)[0..<8] & " " & mark.time.format(timeFormat) & " (" & duration & ") -- ") result = prefix & mark.summary if includeNotes and len(mark.notes.strip()) > 0: let wrappedNotes = wordWrap(s = mark.notes, maxLineWidth = 80 - prefixLen) for line in splitLines(wrappedNotes): result &= "\x0D\x0A" & spaces(prefixLen) & line result &= "\x0D\x0A" proc findById(marks: seq[Mark], id: string): int = var idx = 0 for mark in marks: if startsWith($mark.id, id): return idx inc(idx) return -1 proc doInit(timelineLocation: string): void = 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: var Mark): void = 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.""") close(tempFile) discard os.execShellCmd "$EDITOR " & tempFileName & " /dev/tty" var markPart = Time for line in lines tempFileName: if strip(line)[0] == '#': continue elif markPart == Time: mark.time = parseTime(line); markPart = Summary elif markPart == Summary: mark.summary = line; markPart = Tags elif markPart == Tags: mark.tags = line.split({',', ';'}); markPart = Notes else: mark.notes &= line & "\x0D\x0A" finally: close(tempFile) proc filterMarkIndices(timeline: Timeline, args: Table[string, Value]): seq[int] = let marks = timeline.marks result = sequtils.toSeq(0.."]: let idx = marks.findById($args[""]) if idx > 0: result = result.filterIt(it >= idx) if args[""]: let idx = marks.findById($args[""]) if (idx > 0): result = result.filterIt(it <= idx) if args["--after"]: var startTime: TimeInfo try: startTime = parseTime($args["--after"]) except: raise newException(ValueError, "invalid value for --after: " & getCurrentExceptionMsg()) result = result.filterIt(marks[it].time > startTime) if args["--before"]: var endTime: TimeInfo try: endTime = parseTime($args["--before"]) except: raise newException(ValueError, "invalid value for --before: " & getCurrentExceptionMsg()) result = result.filterIt(marks[it].time < endTime) if args["--tags"]: let tags = (args["--tags"] ?: "").split({',', ';'}) result = result.filter(proc (i: int): bool = tags.allIt(marks[i].tags.contains(it))) if args["--remove-tags"]: let tags = (args["--remove-tags"] ?: "").split({',', ';'}) result = result.filter(proc (i: int): bool = not tags.allIt(marks[i].tags.contains(it))) if args["--matching"]: let pattern = re(args["--matching"] ?: "") result = result.filterIt(marks[it].summary.find(pattern).isSome) when isMainModule: try: let doc = """ Usage: ptk init [options] ptk add [options] ptk add [options] ptk amend [options] [] ptk stop [options] ptk continue ptk delete ptk (list | ls) [options] ptk sum-time --ids ... ptk sum-time [options] [] [] 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