Compare commits
9 Commits
Author | SHA1 | Date | |
---|---|---|---|
5140afa671 | |||
56be47f7e1 | |||
75f74c3a0a | |||
e5c6e6187c | |||
81d326c5c8 | |||
72c332fa45 | |||
4a878026d8 | |||
ee733957c6 | |||
d75f5607c2 |
136
ptk.nim
136
ptk.nim
@ -9,7 +9,7 @@ import algorithm, docopt, json, langutils, logging, os, nre, sequtils,
|
||||
import private/ptkutil
|
||||
|
||||
type
|
||||
Mark* = tuple[id: UUID, time: TimeInfo, summary: string, notes: string, tags: seq[string]]
|
||||
Mark* = tuple[id: UUID, time: DateTime, summary: string, notes: string, tags: seq[string]]
|
||||
## Representation of a single mark on the timeline.
|
||||
|
||||
Timeline* = tuple[name: string, marks: seq[Mark]]
|
||||
@ -19,7 +19,7 @@ const STOP_MSG = "STOP"
|
||||
|
||||
let NO_MARK: Mark = (
|
||||
id: parseUUID("00000000-0000-0000-0000-000000000000"),
|
||||
time: fromSeconds(0).getLocalTime,
|
||||
time: fromUnix(0).local,
|
||||
summary: "", notes: "", tags: @[])
|
||||
|
||||
const ISO_TIME_FORMAT = "yyyy-MM-dd'T'HH:mm:ss"
|
||||
@ -40,7 +40,7 @@ proc exitErr(msg: string): void =
|
||||
fatal "ptk: " & msg
|
||||
quit(QuitFailure)
|
||||
|
||||
proc parseTime(timeStr: string): TimeInfo =
|
||||
proc parseTime(timeStr: string): DateTime =
|
||||
## Helper to parse time strings trying multiple known formats.
|
||||
for fmt in TIME_FORMATS:
|
||||
try: return parse(timeStr, fmt)
|
||||
@ -78,7 +78,7 @@ proc loadTimeline(filename: string): Timeline =
|
||||
# TODO: an incorrect time format that was used in version 0.6 and prior.
|
||||
# Version 0.7 between 1.0 support this format on read only and will write
|
||||
# out the correct format (so they can be used to convert older timelines).
|
||||
var time: TimeInfo
|
||||
var time: DateTime
|
||||
try: time = parse(markJson["time"].getStr(), ISO_TIME_FORMAT)
|
||||
except: time = parse(markJson["time"].getStr(), "yyyy:MM:dd'T'HH:mm:ss")
|
||||
|
||||
@ -121,7 +121,11 @@ proc writeMarks(timeline: Timeline, indices: seq[int], includeNotes = false): vo
|
||||
## Write a nicely-formatted list of Marks to stdout.
|
||||
|
||||
let marks = timeline.marks
|
||||
let now = getLocalTime(getTime())
|
||||
let now = getTime().local
|
||||
|
||||
if indices.len == 0:
|
||||
writeLine(stdout, "No marks match the given criteria.")
|
||||
return
|
||||
|
||||
var idxs = indices.sorted(
|
||||
proc(a, b: int): int = cmp(marks[a].time, marks[b].time))
|
||||
@ -154,6 +158,9 @@ proc writeMarks(timeline: Timeline, indices: seq[int], includeNotes = false): vo
|
||||
|
||||
if prefix.len > longestPrefix: longestPrefix = prefix.len
|
||||
|
||||
let colWidth = 80
|
||||
let notesPrefixLen = 4
|
||||
|
||||
for w in toWrite:
|
||||
if w.mark.summary == STOP_MSG: continue
|
||||
|
||||
@ -174,7 +181,11 @@ proc writeMarks(timeline: Timeline, indices: seq[int], includeNotes = false): vo
|
||||
writeLine(stdout, "")
|
||||
|
||||
if includeNotes and len(w.mark.notes.strip) > 0:
|
||||
writeLine(stdout, spaces(longestPrefix) & w.mark.notes)
|
||||
writeLine(stdout, "")
|
||||
let wrappedNotes = wordWrap(s = w.mark.notes,
|
||||
maxLineWidth = colWidth)
|
||||
for line in splitLines(wrappedNotes):
|
||||
writeLine(stdout, spaces(notesPrefixLen) & line)
|
||||
writeLine(stdout, "")
|
||||
|
||||
proc formatMark(mark: Mark, nextMark = NO_MARK, timeFormat = ISO_TIME_FORMAT, includeNotes = false): string =
|
||||
@ -183,20 +194,19 @@ proc formatMark(mark: Mark, nextMark = NO_MARK, timeFormat = ISO_TIME_FORMAT, in
|
||||
## the Mark's notes in the output.
|
||||
|
||||
let nextTime =
|
||||
if nextMark == NO_MARK: getLocalTime(getTime())
|
||||
if nextMark == NO_MARK: getTime().local
|
||||
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)
|
||||
let wrappedNotes = wordWrap(s = mark.notes, maxLineWidth = 80 - prefix.len)
|
||||
for line in splitLines(wrappedNotes):
|
||||
result &= "\x0D\x0A" & spaces(prefixLen) & line
|
||||
result &= "\x0D\x0A" & spaces(prefix.len) & line
|
||||
result &= "\x0D\x0A"
|
||||
|
||||
proc findById(marks: seq[Mark], id: string): int =
|
||||
@ -277,69 +287,88 @@ proc filterMarkIndices(timeline: Timeline, args: Table[string, Value]): seq[int]
|
||||
## arguments.
|
||||
|
||||
let marks = timeline.marks
|
||||
result = sequtils.toSeq(0..<marks.len).filterIt(marks[it].summary != STOP_MSG)
|
||||
let now = getTime().local
|
||||
let allIndices = sequtils.toSeq(0..<marks.len).filterIt(marks[it].summary != STOP_MSG).toSet
|
||||
let union = args["--or"]
|
||||
|
||||
var selected =
|
||||
if union: initSet[int]()
|
||||
else: allIndices
|
||||
|
||||
template filterMarks(curSet: HashSet[int], pred: untyped): untyped =
|
||||
var res: HashSet[int] = initSet[int]()
|
||||
if union:
|
||||
for mIdx {.inject.} in allIndices:
|
||||
if pred: res.incl(mIdx)
|
||||
res = res + curSet
|
||||
else:
|
||||
for mIdx {.inject.} in curSet:
|
||||
if pred: res.incl(mIdx)
|
||||
res
|
||||
|
||||
if args["<firstId>"]:
|
||||
let idx = marks.findById($args["<firstId>"])
|
||||
if idx > 0: result = result.filterIt(it >= idx)
|
||||
if idx > 0: selected = selected.filterMarks(mIdx >= idx)
|
||||
|
||||
if args["<lastId>"]:
|
||||
let idx = marks.findById($args["<lastId>"])
|
||||
if (idx > 0): result = result.filterIt(it <= idx)
|
||||
if (idx > 0): selected = selected.filterMarks(mIdx <= idx)
|
||||
|
||||
if args["--after"]:
|
||||
var startTime: TimeInfo
|
||||
var startTime: DateTime
|
||||
try: startTime = parseTime($args["--after"])
|
||||
except: raise newException(ValueError,
|
||||
"invalid value for --after: " & getCurrentExceptionMsg())
|
||||
result = result.filterIt(marks[it].time > startTime)
|
||||
selected = selected.filterMarks(marks[mIdx].time > startTime)
|
||||
|
||||
if args["--before"]:
|
||||
var endTime: TimeInfo
|
||||
var endTime: DateTime
|
||||
try: endTime = parseTime($args["--before"])
|
||||
except: raise newException(ValueError,
|
||||
"invalid value for --before: " & getCurrentExceptionMsg())
|
||||
result = result.filterIt(marks[it].time < endTime)
|
||||
selected = selected.filterMarks(marks[mIdx].time < endTime)
|
||||
|
||||
if args["--today"]:
|
||||
let now = getLocalTime(getTime())
|
||||
let b = now.startOfDay
|
||||
let e = b + 1.days
|
||||
result = result.filterIt(marks[it].time >= b and marks[it].time < e)
|
||||
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 now = getLocalTime(getTime())
|
||||
let b = now.startOfWeek(dSun)
|
||||
let e = b + 7.days
|
||||
result = result.filterIt(marks[it].time >= b and marks[it].time < e)
|
||||
selected = selected.filterMarks(marks[mIdx].time >= b and marks[mIdx].time < e)
|
||||
|
||||
if args["--last-week"]:
|
||||
let now = getLocalTime(getTime())
|
||||
let e = now.startOfWeek(dSun)
|
||||
let b = e - 7.days
|
||||
result = result.filterIt(marks[it].time >= b and marks[it].time < e)
|
||||
selected = selected.filterMarks(marks[mIdx].time >= b and marks[mIdx].time < e)
|
||||
|
||||
if args["--tags"]:
|
||||
let tags = (args["--tags"] ?: "").split({',', ';'})
|
||||
result = result.filter(proc (i: int): bool =
|
||||
tags.allIt(marks[i].tags.contains(it)))
|
||||
selected = selected.filterMarks(tags.allIt(marks[mIdx].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)))
|
||||
selected = selected.filterMarks(not tags.allIt(marks[mIdx].tags.contains(it)))
|
||||
|
||||
if args["--matching"]:
|
||||
let pattern = re(args["--matching"] ?: "")
|
||||
result = result.filterIt(marks[it].summary.find(pattern).isSome)
|
||||
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 [options]
|
||||
ptk add [options] <summary>
|
||||
ptk (add | start) [options]
|
||||
ptk (add | start) [options] <summary>
|
||||
ptk resume [options] [<id>]
|
||||
ptk amend [options] [<id>] [<summary>]
|
||||
ptk merge <timeline> [<timeline>...]
|
||||
@ -348,6 +377,7 @@ Usage:
|
||||
ptk delete <id>
|
||||
ptk (list | ls) [options]
|
||||
ptk (list | ls) tags
|
||||
ptk current
|
||||
ptk sum-time --ids <ids>...
|
||||
ptk sum-time [options] [<firstId>] [<lastId>]
|
||||
ptk (-V | --version)
|
||||
@ -369,18 +399,21 @@ Options:
|
||||
-n --notes <notes> For add and amend, set the notes for a time mark.
|
||||
-t --time <time> For add and amend, use this time instead of the current time.
|
||||
-T --today Restrict the selection to marks during today.
|
||||
-Y --yesterday Restrict the selection to marks during yesterday.
|
||||
-w --this-week Restrict the selection to marks during this week.
|
||||
-W --last-week Restrict the selection to marks during the last week.
|
||||
-O --or Create a union from the time conditionals, not an intersection
|
||||
(e.g. --today --or --yesterday)
|
||||
-v --verbose Include notes in timeline entry output.
|
||||
"""
|
||||
|
||||
# TODO: add ptk delete [options]
|
||||
|
||||
logging.addHandler(newConsoleLogger())
|
||||
let now = getLocalTime(getTime())
|
||||
let now = getTime().local
|
||||
|
||||
# Parse arguments
|
||||
let args = docopt(doc, version = "ptk 0.10.0")
|
||||
let args = docopt(doc, version = "ptk 0.12.3")
|
||||
|
||||
if args["--echo-args"]: echo $args
|
||||
|
||||
@ -460,26 +493,30 @@ Options:
|
||||
|
||||
if args["stop"]:
|
||||
|
||||
if timeline.marks.last.summary == STOP_MSG:
|
||||
echo "ptk: no current task, nothing to stop"
|
||||
quit(0)
|
||||
|
||||
let newMark: Mark = (
|
||||
id: genUUID(),
|
||||
time: if args["--time"]: parseTime($args["--time"]) else: now,
|
||||
summary: STOP_MSG,
|
||||
notes: args["--notes"] ?: "",
|
||||
tags: (args["--tags"] ?: "").split({',', ';'}))
|
||||
tags: (args["--tags"] ?: "").split({',', ';'}).filterIt(not it.isNilOrWhitespace))
|
||||
|
||||
timeline.marks.add(newMark)
|
||||
|
||||
timeline.writeMarks(
|
||||
indices = @[timeline.marks.len - 2],
|
||||
includeNotes = args["--verbose"])
|
||||
echo "stopped timer"
|
||||
echo "ptk: stopped timer"
|
||||
|
||||
saveTimeline(timeline, timelineLocation)
|
||||
|
||||
if args["continue"]:
|
||||
|
||||
if timeline.marks.last.summary != STOP_MSG:
|
||||
echo "There is already something in progress:"
|
||||
echo "ptk: there is already something in progress:"
|
||||
timeline.writeMarks(
|
||||
indices = @[timeline.marks.len - 1],
|
||||
includeNotes = args["--verbose"])
|
||||
@ -500,21 +537,22 @@ Options:
|
||||
|
||||
saveTimeline(timeline, timelineLocation)
|
||||
|
||||
if args["add"]:
|
||||
if args["add"] or args["start"]:
|
||||
|
||||
var newMark: Mark = (
|
||||
id: genUUID(),
|
||||
time: if args["--time"]: parseTime($args["--time"]) else: now,
|
||||
summary: args["<summary>"] ?: "",
|
||||
notes: args["--notes"] ?: "",
|
||||
tags: (args["--tags"] ?: "").split({',', ';'}))
|
||||
tags: (args["--tags"] ?: "").split({',', ';'}).filterIt(not it.isNilOrWhitespace))
|
||||
|
||||
if args["--edit"]: edit(newMark)
|
||||
|
||||
let prevLastIdx = timeline.marks.getLastIndex()
|
||||
timeline.marks.add(newMark)
|
||||
timeline.writeMarks(
|
||||
indices = @[prevLastIdx, timeline.marks.len - 1],
|
||||
indices = if prevLastIdx < 0: @[0]
|
||||
else: @[prevLastIdx, timeline.marks.len - 1],
|
||||
includeNotes = args["--verbose"])
|
||||
|
||||
saveTimeline(timeline, timelineLocation)
|
||||
@ -526,7 +564,9 @@ Options:
|
||||
if args["<id>"]:
|
||||
markToResumeIdx = timeline.marks.findById($args["<id>"])
|
||||
if markToResumeIdx == -1: exitErr "Cannot find a mark matching " & $args["<id>"]
|
||||
else: markToResumeIdx = timeline.marks.getLastIndex()
|
||||
else:
|
||||
markToResumeIdx = timeline.marks.getLastIndex()
|
||||
if markToResumeIdx < 0: exitErr "No mark to resume."
|
||||
var markToResume = timeline.marks[markToResumeIdx]
|
||||
|
||||
var newMark: Mark = (
|
||||
@ -553,7 +593,9 @@ Options:
|
||||
if args["<id>"]:
|
||||
markIdx = timeline.marks.findById($args["<id>"])
|
||||
if markIdx == -1: exitErr "Cannot find a mark matching " & $args["<id>"]
|
||||
else: markIdx = timeline.marks.getLastIndex()
|
||||
else:
|
||||
markIdx = timeline.marks.getLastIndex()
|
||||
if markIdx < 0: exitErr "No mark to amend."
|
||||
|
||||
var mark = timeline.marks[markIdx]
|
||||
|
||||
@ -601,7 +643,17 @@ Options:
|
||||
|
||||
timeline.writeMarks(
|
||||
indices = selectedIndices,
|
||||
includeNotes = args["--version"])
|
||||
includeNotes = args["--verbose"])
|
||||
|
||||
if args["current"]:
|
||||
|
||||
let idx = timeline.marks.len - 1
|
||||
if timeline.marks[idx].summary == STOP_MSG:
|
||||
echo "ptk: no current task"
|
||||
else:
|
||||
timeline.writeMarks(
|
||||
indices = @[idx],
|
||||
includeNotes = true)
|
||||
|
||||
if args["sum-time"]:
|
||||
|
||||
|
@ -1,6 +1,6 @@
|
||||
# Package
|
||||
|
||||
version = "0.10.0"
|
||||
version = "0.12.3"
|
||||
author = "Jonathan Bernard"
|
||||
description = "Personal Time Keeper"
|
||||
license = "MIT"
|
||||
@ -8,5 +8,5 @@ bin = @["ptk"]
|
||||
|
||||
# Dependencies
|
||||
|
||||
requires @["nim >= 0.15.0", "docopt >= 0.6.4", "uuids", "langutils", "tempfile", "timeutils >= 0.2.0", "isaac >= 0.1.2"]
|
||||
requires @["nim >= 0.18.0", "docopt >= 0.6.4", "uuids", "langutils", "tempfile", "timeutils >= 0.2.2", "isaac >= 0.1.2"]
|
||||
|
||||
|
Reference in New Issue
Block a user