diff --git a/daily_notifier.nim b/daily_notifier.nim new file mode 100644 index 0000000..6c6b1e1 --- /dev/null +++ b/daily_notifier.nim @@ -0,0 +1,70 @@ +import docopt, os, nre, sequtils, strutils, times, timeutils + +type ParseState = enum BeforeHeading, ReadingPlans, AfterPlans + +type PlanItem* = tuple[time: TimeInfo, note: string] + +proc parseDailyPlan(filename: string): seq[PlanItem] = + + var planItems: seq[PlanItem] = @[] + var parseState = BeforeHeading + let planItemRe = re"\s*(\d{4})\s+(.*)" + let timeFmt = "HHmm" + + for line in lines filename: + case parseState + + of BeforeHeading: + if line.strip.startsWith("# Timeline"): parseState = ReadingPlans + + of ReadingPlans: + let match = line.find(planItemRe) + if match.isSome(): planItems.add(( + time: parse(match.get().captures[0], timeFmt), + note: match.get().captures[1])) + else: parseState = AfterPlans + + of AfterPlans: break + else: break + + return planItems + +when isMainModule: + + let doc = """ +Usage: + daily_notifier [options] + +Options: + + -d --plan-directory Directory to search for plan files. + + -f --plan-date-format Date pattern for identifying plan files. This + is used in conjunction with plan directory to + idenfity files that should be parsed as plan + files. + + -c --config Use as the source of configuration + for daily-notification. + + -h --help Print this usage information. +""" + let args = docopt(doc, version = "daily_notifier 0.1.0") + + if args["--help"]: + echo doc + quit() + + # Find and parse the .dailynotificationrc file + let rcLocations = @[ + if args["--config"]: $args[""] else:"", + ".dailynotificationrc", $getEnv("DAILY_NOTIFICATION_RC"), + $getEnv("HOME") & "/.dailynotificationrc"] + + var ptkrcFilename: string = + foldl(rcLocations, if len(a) > 0: a elif existsFile(b): b else: "") + + # Determine our plan directory and file template + # Start our daemon process (if needed) + # exit + diff --git a/daily_notifier.nimble b/daily_notifier.nimble index b85c420..a0506a9 100644 --- a/daily_notifier.nimble +++ b/daily_notifier.nimble @@ -4,8 +4,9 @@ version = "0.1.0" author = "Jonathan Bernard" description = "Little programs that reads my daily plan and notifies me of upcoming events." license = "MIT" +bin = @["daily_notifier", "deploy_plans_to_fastmail"] # Dependencies -requires "nim >= 0.15.0" +requires @["nim >= 0.15.0", "docopt", "timeutils", "tempfile"] diff --git a/deploy_plans_to_fastmail.nim b/deploy_plans_to_fastmail.nim new file mode 100644 index 0000000..13a7e1c --- /dev/null +++ b/deploy_plans_to_fastmail.nim @@ -0,0 +1,103 @@ +import json, nre, os, osproc, sequtils, sets, strutils, tempfile + +proc doStep(step, cmd: string): tuple[output: TaintedString, exitCode: int] = + echo "> " & cmd + result = execCmdEx(cmd, {poUsePath}) + if result.exitCode != 0: + writeLine(stderr, "Failed step [" & step & + "] Received error code: " & $result.exitCode) + quit(1) + echo "" + +when isMainModule: + let cfgFilePath = $getEnv("HOME") & "/.personal-planning.config.json" + + if not existsFile(cfgFilePath): + quit("Cannot find config file: " & cfgFilePath, 2) + + var cfg: JsonNode + try: cfg = parseFile(cfgFilePath) + except: quit("Could not parse as json: " & cfgFilePath, 3) + + let repoUrl = + if cfg.hasKey("repo"): cfg["repo"].str + else: "_git@git.jdb-labs.com:jdb/personal-planning.git" + + let ftpRoot = + if cfg.hasKey("ftp") and cfg["ftp"].hasKey("root"): cfg["ftp"]["root"].str + else: "ftp://ftp.fastmail.com/jonathan.jdbernard.com/files/personal-planning" + + let ftpUsername = + if cfg.hasKey("ftp") and cfg["ftp"].hasKey("username"): + cfg["ftp"]["username"].str + else: "jonathan@jdbernard.com" + + if not (cfg.hasKey("ftp") and cfg["ftp"].hasKey("password")): + quit("Could not find ftp.password in config file: " & cfgFilePath, 4) + + let ftpPassword = cfg["ftp"]["password"].str + + let htmlFileRe = + if cfg.hasKey("outputPattern"): re(cfg["outputPattern"].str) + else: re".* (\S+)\.html?$" + + let mdFileRe = + if cfg.hasKey("inputPattern"): re(cfg["inputPattern"].str) + else: re".*/(\S+-plan)\.md?$" + + const subDirs = ["daily", "weekly", "monthly", "yearly"] + + var output: string + var exitCode: int + + # Make a temprary directory + let tempdir = mkdtemp("personal-planning-") + + # Checkout personal development repository (local clone) + discard doStep("clone repo", "git clone " & repoUrl & " " & tempdir) + + for curDir in subDirs: + let fullDirPath = tempdir & "/" & curDir + let remoteOptions = " --user '" & + ftpUsername & ":" & ftpPassword & "' '" & + ftpRoot & "/" & curDir & "/' " + + # List the files on the server. + (output, exitCode) = doStep("read remote files (" & curDir & ")", + "curl " & remoteOptions) + + let remoteFiles = output.splitLines + .filterIt(it.find(htmlFileRe).isSome) + .mapIt(it.find(htmlFileRe).get().captures[0]) + .toSet + + # List the daily files locally. + let localFiles = + sequtils.toSeq(walkDir(fullDirPath)) + .filterIt(it.kind == pcFile and + it.path.find(mdFileRe).isSome) + .mapIt(it.path.find(mdFileRe).get().captures[0]) + .toSet + + # ID the files that are new (present locally but not remotely). + let newFiles = localFiles - remoteFiles + + for fileName in newFiles: + let tempPath = fullDirPath & "/temp.html" + let filePath = fullDirPath & "/" & fileName + + # Compile the markdown into HTML + discard doStep("compile plan file (" & fileName & ")", + "markdown " & filePath & ".md" & " > " & tempPath) + + # Concatenate the HTML template to create the final HTML + discard doStep("concatenate HTML template (" & fileName & ")", + "cat " & tempdir & "/code/start.html " & tempPath & " " & + tempdir & "/code/end.html > " & filePath & ".html") + # Upload the new file to FastMail + discard doStep("upload file to FastMail (" & fileName & ")", + "curl -T '" & filePath & ".html' " & remoteOptions) + + # Delete local temp repo + echo "Deleting " & tempdir + removeDir(tempdir)