2019-01-27 03:34:47 +01:00
|
|
|
with import <stockholm/lib>;
|
|
|
|
{ config, pkgs, ... }:
|
|
|
|
|
|
|
|
let
|
2019-01-27 16:55:47 +01:00
|
|
|
#for shared state directory
|
2019-01-27 03:34:47 +01:00
|
|
|
stateDir = config.krebs.reaktor2.r.stateDir;
|
|
|
|
|
|
|
|
generators = pkgs.reaktor2-plugins.generators;
|
|
|
|
hooks = pkgs.reaktor2-plugins.hooks;
|
|
|
|
commands = pkgs.reaktor2-plugins.commands;
|
|
|
|
|
2021-07-29 15:37:06 +02:00
|
|
|
taskRcFile = builtins.toFile "taskrc" ''
|
|
|
|
confirmation=no
|
|
|
|
'';
|
|
|
|
|
|
|
|
task = name: {
|
2020-06-23 09:53:21 +02:00
|
|
|
pattern = "^${name}-([a-z]+)(?::?\\s*(.*))?";
|
2019-01-27 16:55:47 +01:00
|
|
|
activate = "match";
|
|
|
|
command = 1;
|
|
|
|
arguments = [2];
|
2019-01-27 20:32:04 +01:00
|
|
|
env.TASKDATA = "${stateDir}/${name}";
|
2019-01-27 16:55:47 +01:00
|
|
|
commands = {
|
2019-01-27 20:32:04 +01:00
|
|
|
add.filename = pkgs.writeDash "${name}-task-add" ''
|
2021-07-29 15:37:06 +02:00
|
|
|
${pkgs.taskwarrior}/bin/task rc:${taskRcFile} add "$1"
|
2019-01-27 20:32:04 +01:00
|
|
|
'';
|
|
|
|
list.filename = pkgs.writeDash "${name}-task-list" ''
|
2021-07-29 15:37:06 +02:00
|
|
|
${pkgs.taskwarrior}/bin/task rc:${taskRcFile} export \
|
2019-01-27 20:32:04 +01:00
|
|
|
| ${pkgs.jq}/bin/jq -r '
|
|
|
|
.[] | select(.id != 0) | "\(.id) \(.description)"
|
|
|
|
'
|
|
|
|
'';
|
|
|
|
delete.filename = pkgs.writeDash "${name}-task-delete" ''
|
2021-07-29 15:37:06 +02:00
|
|
|
${pkgs.taskwarrior}/bin/task rc:${taskRcFile} delete "$1"
|
2019-01-27 20:32:04 +01:00
|
|
|
'';
|
|
|
|
done.filename = pkgs.writeDash "${name}-task-done" ''
|
2021-07-29 15:37:06 +02:00
|
|
|
${pkgs.taskwarrior}/bin/task rc:${taskRcFile} done "$1"
|
2019-01-27 20:32:04 +01:00
|
|
|
'';
|
2019-01-27 03:34:47 +01:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
systemPlugin = {
|
|
|
|
plugin = "system";
|
|
|
|
config = {
|
|
|
|
workdir = stateDir;
|
|
|
|
hooks.JOIN = [
|
|
|
|
{
|
|
|
|
activate = "always";
|
|
|
|
command = {
|
|
|
|
filename =
|
|
|
|
"${pkgs.Reaktor.src}/reaktor/commands/tell-on_join";
|
|
|
|
env = {
|
|
|
|
PATH = makeBinPath [
|
|
|
|
pkgs.coreutils # XXX env, touch
|
|
|
|
pkgs.jq # XXX sed
|
|
|
|
pkgs.utillinux # XXX flock
|
|
|
|
];
|
|
|
|
state_file = "${stateDir}/tell.json";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
];
|
|
|
|
hooks.PRIVMSG = [
|
|
|
|
{
|
2019-01-29 21:39:16 +01:00
|
|
|
pattern = "^bier bal(an(ce)?)?$";
|
2019-01-27 03:34:47 +01:00
|
|
|
activate = "match";
|
|
|
|
command = {
|
|
|
|
env = {
|
|
|
|
state_file = "${stateDir}/ledger";
|
|
|
|
};
|
2019-01-27 13:50:31 +01:00
|
|
|
filename = pkgs.writeDash "bier-balance" ''
|
2019-01-27 14:27:54 +01:00
|
|
|
${pkgs.hledger}/bin/hledger -f $state_file bal -N -O csv \
|
|
|
|
| ${pkgs.coreutils}/bin/tail +2 \
|
2019-07-15 18:22:35 +02:00
|
|
|
| ${pkgs.miller}/bin/mlr --icsv --opprint cat \
|
|
|
|
| ${pkgs.gnused}/bin/sed 's/^/the_/'
|
2019-01-27 03:34:47 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
}
|
|
|
|
{
|
2020-09-15 22:52:34 +02:00
|
|
|
pattern = ''^([\H-]*):?\s+([+-][1-9][0-9]*)\s+(\S+)$'';
|
2019-01-27 03:34:47 +01:00
|
|
|
activate = "match";
|
|
|
|
arguments = [1 2 3];
|
|
|
|
command = {
|
|
|
|
env = {
|
|
|
|
# TODO; get state as argument
|
|
|
|
state_file = "${stateDir}/ledger";
|
|
|
|
};
|
|
|
|
filename = pkgs.writeDash "ledger-add" ''
|
|
|
|
set -x
|
|
|
|
tonick=$1
|
|
|
|
amt=$2
|
|
|
|
unit=$3
|
|
|
|
printf '%s\n %s %d %s\n %s %d %s\n' "$(date -Id)" "$tonick" "$amt" "$unit" "$_from" "$(expr 0 - "''${amt#+}")" "$unit" >> $state_file
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
}
|
|
|
|
hooks.sed
|
|
|
|
(generators.command_hook {
|
2020-11-03 20:56:36 +01:00
|
|
|
inherit (commands) hello random-emoji nixos-version;
|
2019-01-27 03:34:47 +01:00
|
|
|
tell = {
|
|
|
|
filename =
|
|
|
|
"${pkgs.Reaktor.src}/reaktor/commands/tell-on_privmsg";
|
|
|
|
env = {
|
|
|
|
PATH = makeBinPath [
|
|
|
|
pkgs.coreutils # XXX date, env
|
|
|
|
pkgs.jq # XXX sed
|
|
|
|
pkgs.utillinux # XXX flock
|
|
|
|
];
|
|
|
|
state_file = "${stateDir}/tell.txt";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
})
|
2019-01-27 16:55:47 +01:00
|
|
|
(task "agenda")
|
|
|
|
];
|
2019-01-27 03:34:47 +01:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
in {
|
|
|
|
|
2019-04-23 20:05:03 +02:00
|
|
|
users.users.reaktor2 = {
|
|
|
|
uid = genid_uint31 "reaktor2";
|
|
|
|
home = stateDir;
|
2021-06-05 14:14:56 +02:00
|
|
|
isSystemUser = true;
|
2021-07-29 15:46:34 +02:00
|
|
|
extraGroups = [ "reaktor2" ];
|
2019-04-23 20:05:03 +02:00
|
|
|
};
|
|
|
|
|
2021-07-29 15:46:34 +02:00
|
|
|
users.groups.reaktor2 = {};
|
|
|
|
|
2021-07-29 15:37:06 +02:00
|
|
|
systemd.services.htgen-agenda.serviceConfig.StateDirectory = "reaktor2";
|
|
|
|
krebs.htgen.agenda = {
|
|
|
|
port = 8009;
|
|
|
|
user = {
|
|
|
|
name = "reaktor2";
|
|
|
|
home = stateDir;
|
|
|
|
};
|
|
|
|
script = ''. ${pkgs.writeDash "agenda" ''
|
|
|
|
echo "$Method $Request_URI" >&2
|
|
|
|
case "$Method" in
|
|
|
|
"GET")
|
|
|
|
printf 'HTTP/1.1 200 OK\r\n'
|
|
|
|
printf 'Connection: close\r\n'
|
|
|
|
printf '\r\n'
|
|
|
|
TASKDATA=/var/lib/reaktor2/agenda ${pkgs.taskwarrior}/bin/task rc:${taskRcFile} export
|
|
|
|
exit
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
''}'';
|
|
|
|
};
|
|
|
|
|
|
|
|
services.nginx = {
|
|
|
|
virtualHosts."agenda.r" = {
|
|
|
|
locations."= /index.html".extraConfig = ''
|
|
|
|
alias ${pkgs.writeText "agenda.html" ''
|
|
|
|
<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<title>Agenda</title>
|
|
|
|
<meta charset="utf-8" />
|
|
|
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|
|
|
<style>
|
|
|
|
html {
|
|
|
|
font-family: monospace;
|
|
|
|
}
|
|
|
|
|
|
|
|
dt {
|
|
|
|
float: left;
|
|
|
|
clear: left;
|
|
|
|
width: 30px;
|
|
|
|
text-align: right;
|
|
|
|
font-weight: bold;
|
|
|
|
}
|
|
|
|
|
|
|
|
dd {
|
|
|
|
margin: 0 0 0 40px;
|
|
|
|
padding: 0 0 0.5em 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
.date {
|
|
|
|
color: grey;
|
|
|
|
font-style: italic;
|
|
|
|
}
|
|
|
|
</style>
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<dl id="agenda"></dl>
|
|
|
|
<script>
|
|
|
|
const urlSearchParams = new URLSearchParams(window.location.search);
|
|
|
|
const params = Object.fromEntries(urlSearchParams.entries());
|
|
|
|
|
|
|
|
if (params.hasOwnProperty("style")) {
|
|
|
|
fetch(params["style"])
|
|
|
|
.then((response) =>
|
|
|
|
response.text().then((css) => {
|
|
|
|
const title = document.getElementsByTagName("title")[0];
|
|
|
|
const style = document.createElement("style");
|
|
|
|
style.appendChild(document.createTextNode(css));
|
|
|
|
title.appendChild(style);
|
|
|
|
})
|
|
|
|
)
|
|
|
|
.catch(console.log);
|
|
|
|
}
|
|
|
|
|
|
|
|
fetch("/agenda.json")
|
|
|
|
.then((response) => {
|
|
|
|
response.json().then((agenda) => {
|
|
|
|
const dl = document.getElementById("agenda");
|
|
|
|
for (const agendaItem of agenda) {
|
|
|
|
if (agendaItem.status !== "pending") continue;
|
|
|
|
// task warrior date format to ISO
|
|
|
|
const entryDate = agendaItem.entry.replace(
|
|
|
|
/(\d{4})(\d{2})(\d{2})T(\d{2})(\d{2})(\d{2})Z/,
|
|
|
|
"$1-$2-$3T$4:$5:$6Z"
|
|
|
|
);
|
|
|
|
|
|
|
|
const dt = document.createElement("dt");
|
|
|
|
dt.className = "id";
|
|
|
|
dt.appendChild(document.createTextNode(agendaItem.id.toString()));
|
|
|
|
dl.appendChild(dt);
|
|
|
|
|
|
|
|
const spanDate = document.createElement("span");
|
|
|
|
spanDate.className = "date";
|
|
|
|
spanDate.title = new Date(entryDate).toString();
|
|
|
|
spanDate.appendChild(document.createTextNode(entryDate));
|
|
|
|
|
|
|
|
const dd = document.createElement("dd");
|
|
|
|
dd.className = "description";
|
|
|
|
dd.appendChild(document.createTextNode(agendaItem.description));
|
|
|
|
dd.appendChild(document.createTextNode(" "));
|
|
|
|
dd.appendChild(spanDate);
|
|
|
|
|
|
|
|
dl.appendChild(dd);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
})
|
|
|
|
.then((data) => console.log(data));
|
|
|
|
</script>
|
|
|
|
</body>
|
|
|
|
</html>
|
|
|
|
''};
|
|
|
|
'';
|
|
|
|
locations."/agenda.json".extraConfig = ''
|
|
|
|
proxy_set_header Host $host;
|
|
|
|
proxy_pass http://localhost:8009;
|
|
|
|
'';
|
|
|
|
extraConfig = ''
|
|
|
|
add_header 'Access-Control-Allow-Origin' '*';
|
|
|
|
add_header 'Access-Control-Allow-Methods' 'GET, POST, OPTIONS';
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2021-07-29 15:46:34 +02:00
|
|
|
systemd.services.reaktor2-r.serviceConfig.DynamicUser = mkForce false;
|
|
|
|
systemd.services.reaktor2-hackint.serviceConfig.DynamicUser = mkForce false;
|
2019-01-27 03:34:47 +01:00
|
|
|
krebs.reaktor2 = {
|
2021-05-20 10:57:27 +02:00
|
|
|
hackint = {
|
|
|
|
hostname = "irc.hackint.org";
|
|
|
|
nick = "reaktor2|krebs";
|
|
|
|
plugins = [
|
|
|
|
{
|
|
|
|
plugin = "register";
|
|
|
|
config = {
|
|
|
|
channels = [
|
|
|
|
"#krebs"
|
|
|
|
];
|
|
|
|
};
|
|
|
|
}
|
|
|
|
systemPlugin
|
|
|
|
];
|
2021-05-20 12:54:30 +02:00
|
|
|
username = "reaktor2";
|
2021-05-20 10:57:27 +02:00
|
|
|
port = "6697";
|
|
|
|
};
|
2019-01-27 03:34:47 +01:00
|
|
|
r = {
|
|
|
|
nick = "reaktor2|krebs";
|
2019-01-29 21:39:16 +01:00
|
|
|
sendDelaySec = null;
|
2019-01-27 03:34:47 +01:00
|
|
|
plugins = [
|
|
|
|
{
|
|
|
|
plugin = "register";
|
|
|
|
config = {
|
|
|
|
channels = [
|
|
|
|
"#noise"
|
|
|
|
"#xxx"
|
|
|
|
];
|
|
|
|
};
|
|
|
|
}
|
|
|
|
systemPlugin
|
|
|
|
];
|
2019-01-27 20:06:06 +01:00
|
|
|
username = "reaktor2";
|
2019-01-27 03:34:47 +01:00
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|