Improve task notifier script

This commit is contained in:
Talia 2024-08-12 14:54:22 +02:00
parent e562e3680f
commit 4b86dfdfcc
1 changed files with 107 additions and 30 deletions

View File

@ -3,31 +3,89 @@
local json = require "cjson" local json = require "cjson"
local arrr = require "arrr" local arrr = require "arrr"
local shapeshift = require "shapeshift" local shapeshift = require "shapeshift"
local lumber = require("lumber")
local log = lumber.new {
level = lumber.levels.WARN;
format = require "lumber.format.term";
filter = function(message)
if type(message) == "string" then
return message
else
return require("inspect")(message)
end
end
}
local default_config = {
{ notify = "zenity", minutes = 60 };
{ notify = "notify", minutes = 180 };
}
local function zulu_offset() local function zulu_offset()
local current = os.date("*t") local current = os.date("*t")
current.isdst = false current.isdst = false
local zulu = os.date("!*t") local zulu = os.date("!*t")
--- @cast zulu -string
--- @cast current -string
return os.difftime(os.time(current), os.time(zulu)) return os.difftime(os.time(current), os.time(zulu))
end end
local config_file do
local xdg_home = os.getenv("XDG_CONFIG_HOME")
local home = os.getenv("HOME")
if xdg_home then
config_file = xdg_home .. "/task/tasknotif.lua"
elseif home then
config_file = home .. "/.config/task/tasknotif.lua"
else
config_file = "tasknotif.lua"
end
end
local params do local params do
local parse = arrr { local parse = arrr {
{ "Time or something", "--minutes", "-m", "number" }; { "Configuration to use (default: "..config_file..")", "--config", "-c" };
{ "Use Zenity to display a window instead of sending a notification", "--zenity", "-z" } { "Sets the log level", "--log", nil, true };
{ "Ignore anything that's already due soon at program start", "--pre-check" };
} }
local validate = shapeshift.table { local validate = shapeshift.table {
__extra = "keep"; __extra = "keep";
minutes = shapeshift.default(30, shapeshift.is.number); precheck = shapeshift.default(false, shapeshift.is.boolean);
} }
params = select(2, validate(parse(arg))) params = select(2, validate(parse(arg)))
end end
local done = {} if params.log then
log.level = lumber.levels[string.upper(params.log)]
end
while true do log:debug("Params", params)
log:info "Starting taskwarrior notifier"
--- @type table
local config if params.config then
config = assert(loadfile(params.config))()
else
local chunk = loadfile(config_file)
config = chunk and chunk() or default_config
end
local done = {}
for severity in ipairs(config) do
done[severity] = {}
end
--- @param run boolean Whether there should be any notification at all
local function check(run)
-- Save handled tasks in the current loop so the same task doesn't get several notififations at once
local handled = {}
if run == nil then run = true end
for severity, condition in ipairs(config) do
log:debug(condition)
local data = json.decode(io.popen("task export"):read("*a")) local data = json.decode(io.popen("task export"):read("*a"))
for _, task in ipairs(data) do for _, task in ipairs(data) do
if task.status == "pending" and task.due then if task.status == "pending" and task.due then
@ -41,27 +99,46 @@ while true do
-- If anything gets postponed outside of the warning time, -- If anything gets postponed outside of the warning time,
-- remove it from the done list -- remove it from the done list
if done[task.uuid] then if done[severity][task.uuid] then
if os.difftime(task.due, os.time()) > 60*params.minutes then if os.difftime(task.due, os.time()) > 60 * condition.minutes then
done[task.uuid] = nil done[severity][task.uuid] = nil
end end
end end
if os.difftime(task.due, os.time()) < 60*params.minutes then if os.difftime(task.due, os.time()) < 60 * condition.minutes then
if not done[task.uuid] then if handled[task.uuid] then
done[task.uuid] = task log:info("Skipping task "..task.uuid..": already handled in this loop")
print("Notifying:", task.uuid, task.description) end
if params.zenity then if not done[severity][task.uuid] then
done[severity][task.uuid] = task
if run and not handled[task.uuid] then
log:info("Notifying:", task.uuid, task.description)
if condition.notify == "zenity" then
os.execute(string.format("zenity --warning --title '%s' --text '%s'", "Task due soon", task.description)) os.execute(string.format("zenity --warning --title '%s' --text '%s'", "Task due soon", task.description))
else elseif condition.notify == "notify" then
os.execute("notify-send 'Task due soon' '"..task.description:gsub([[']], [['"'"']]).."'") os.execute("notify-send 'Task due soon' '"..task.description:gsub([[']], [['"'"']]).."'")
else
error("Unknown notification type: " .. tostring(condition.notify))
end
handled[task.uuid] = task
end
end
end end
end end
end end
end end
end end
if not os.execute("sleep 5") then if params.precheck then
log:info "Doing initial pre-scan"
check(false)
end
log:info "Starting scan loop"
while true do
check(true)
if not os.execute("sleep 30") then
log:info "Exiting taskwarrior notifier"
os.exit() os.exit()
end end
end end