mirror of
https://github.com/bjc/prosody.git
synced 2025-04-03 21:27:38 +03:00
This ensures that all interactions with storage happen inside an async thread, allowing async waiting to be performed in storage drivers.
79 lines
1.9 KiB
Lua
79 lines
1.9 KiB
Lua
module:set_global();
|
|
|
|
local async = require("prosody.util.async");
|
|
local datetime = require("prosody.util.datetime");
|
|
|
|
local periods = { hourly = 3600; daily = 86400; weekly = 7 * 86400 }
|
|
|
|
local active_hosts = {}
|
|
|
|
function module.add_host(host_module)
|
|
|
|
local last_run_times = host_module:open_store("cron", "map");
|
|
active_hosts[host_module.host] = true;
|
|
|
|
local function save_task(task, started_at)
|
|
last_run_times:set(nil, task.id, started_at);
|
|
end
|
|
|
|
local function restore_task(task)
|
|
if task.last == nil then
|
|
task.last = last_run_times:get(nil, task.id);
|
|
end
|
|
end
|
|
|
|
local function task_added(event)
|
|
local task = event.item;
|
|
if task.name == nil then
|
|
task.name = task.when;
|
|
end
|
|
if task.id == nil then
|
|
task.id = event.source.name .. "/" .. task.name:gsub("%W", "_"):lower();
|
|
end
|
|
task.restore = restore_task;
|
|
task.save = save_task;
|
|
module:log("debug", "%s task %s added", task.when, task.id);
|
|
return true
|
|
end
|
|
|
|
local function task_removed(event)
|
|
local task = event.item;
|
|
host_module:log("debug", "Task %s removed", task.id);
|
|
return true
|
|
end
|
|
|
|
host_module:handle_items("task", task_added, task_removed, true);
|
|
|
|
function host_module.unload()
|
|
active_hosts[host_module.host] = nil;
|
|
end
|
|
end
|
|
|
|
local function should_run(when, last)
|
|
return not last or last + periods[when] * 0.995 <= os.time()
|
|
end
|
|
|
|
local function run_task(task)
|
|
task:restore();
|
|
if not should_run(task.when, task.last) then
|
|
return
|
|
end
|
|
local started_at = os.time();
|
|
task:run(started_at);
|
|
task.last = started_at;
|
|
task:save(started_at);
|
|
end
|
|
|
|
local task_runner = async.runner(run_task);
|
|
scheduled = module:add_timer(1, function()
|
|
module:log("info", "Running periodic tasks");
|
|
local delay = 3600;
|
|
for host in pairs(active_hosts) do
|
|
module:log("debug", "Running periodic tasks for host %s", host);
|
|
for _, task in ipairs(module:context(host):get_host_items("task")) do
|
|
task_runner:run(task);
|
|
end
|
|
end
|
|
module:log("debug", "Wait %ds", delay);
|
|
return delay
|
|
end);
|