feat: extend systemd-email-notify module with optional ntfy support
Add configurable ntfy options (tokenFile, url, topic) to the shared emailOnFailure module. When tokenFile is set, a ntfy-failure@ template service is added alongside the existing email notifications. Systems without ntfy configured are unaffected. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
parent
90a62f40f3
commit
7df551d46f
2 changed files with 65 additions and 40 deletions
|
|
@ -32,11 +32,33 @@ let
|
|||
$(systemctl status --full "$2")
|
||||
ERRMAIL
|
||||
'';
|
||||
|
||||
onFailureUnits =
|
||||
[ "email@%n.service" ]
|
||||
++ optionals (cfg.ntfy.tokenFile != null) [ "ntfy-failure@%n.service" ];
|
||||
in
|
||||
{
|
||||
options = {
|
||||
metacfg.emailOnFailure = with types; {
|
||||
enable = mkBoolOpt false "Whether or not to send mails on failure.";
|
||||
|
||||
ntfy.tokenFile = mkOption {
|
||||
type = types.nullOr types.path;
|
||||
default = null;
|
||||
description = "Path to a file containing the ntfy bearer token. Enables ntfy notifications when set.";
|
||||
};
|
||||
|
||||
ntfy.url = mkOption {
|
||||
type = types.str;
|
||||
default = "http://127.0.0.1:2586";
|
||||
description = "Base URL of the ntfy server.";
|
||||
};
|
||||
|
||||
ntfy.topic = mkOption {
|
||||
type = types.str;
|
||||
default = "alerts";
|
||||
description = "Ntfy topic to publish failure notifications to.";
|
||||
};
|
||||
};
|
||||
|
||||
systemd.email-notify.mailTo = mkOption {
|
||||
|
|
@ -55,12 +77,13 @@ in
|
|||
type =
|
||||
with types;
|
||||
attrsOf (submodule {
|
||||
config.onFailure = [ "email@%n.service" ];
|
||||
config.onFailure = onFailureUnits;
|
||||
});
|
||||
};
|
||||
};
|
||||
|
||||
config = mkIf cfg.enable {
|
||||
config = mkIf cfg.enable (mkMerge [
|
||||
{
|
||||
systemd.services."email@" = {
|
||||
description = "Sends a status mail via sendmail on service failures.";
|
||||
onFailure = mkForce [ ];
|
||||
|
|
@ -74,5 +97,28 @@ in
|
|||
Type = "oneshot";
|
||||
};
|
||||
};
|
||||
}
|
||||
|
||||
(mkIf (cfg.ntfy.tokenFile != null) {
|
||||
systemd.services."ntfy-failure@" = {
|
||||
description = "Send ntfy notification on service failure";
|
||||
onFailure = mkForce [ ];
|
||||
serviceConfig = {
|
||||
Type = "oneshot";
|
||||
ExecStart = pkgs.writeShellScript "ntfy-failure-notify" ''
|
||||
TOKEN=$(cat ${cfg.ntfy.tokenFile})
|
||||
UNIT="$1"
|
||||
${pkgs.curl}/bin/curl -s \
|
||||
-H "Authorization: Bearer $TOKEN" \
|
||||
-H "Title: Service failed: $UNIT" \
|
||||
-H "Priority: urgent" \
|
||||
-H "Tags: rotating_light" \
|
||||
-d "$(systemctl status --full "$UNIT" 2>&1 | head -40)" \
|
||||
${cfg.ntfy.url}/${cfg.ntfy.topic}
|
||||
'';
|
||||
};
|
||||
scriptArgs = "%i";
|
||||
};
|
||||
})
|
||||
]);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
{ config, pkgs, lib, ... }:
|
||||
{ config, ... }:
|
||||
{
|
||||
services.ntfy-sh = {
|
||||
enable = true;
|
||||
|
|
@ -19,30 +19,9 @@
|
|||
};
|
||||
};
|
||||
|
||||
# Notify via ntfy on any service failure (alongside email)
|
||||
systemd.services."ntfy-failure@" = {
|
||||
description = "Send ntfy notification on service failure";
|
||||
onFailure = lib.mkForce [ ];
|
||||
serviceConfig = {
|
||||
Type = "oneshot";
|
||||
ExecStart = pkgs.writeShellScript "ntfy-failure-notify" ''
|
||||
TOKEN=$(cat ${config.sops.secrets.ntfy.path})
|
||||
UNIT="$1"
|
||||
${pkgs.curl}/bin/curl -s \
|
||||
-H "Authorization: Bearer $TOKEN" \
|
||||
-H "Title: Service failed: $UNIT" \
|
||||
-H "Priority: urgent" \
|
||||
-H "Tags: rotating_light" \
|
||||
-d "$(systemctl status --full "$UNIT" 2>&1 | head -40)" \
|
||||
http://127.0.0.1:2586/alerts
|
||||
'';
|
||||
};
|
||||
scriptArgs = "%i";
|
||||
};
|
||||
|
||||
systemd.services = lib.mkOption {
|
||||
type = lib.types.attrsOf (lib.types.submodule {
|
||||
config.onFailure = [ "ntfy-failure@%n.service" ];
|
||||
});
|
||||
metacfg.emailOnFailure.ntfy = {
|
||||
tokenFile = config.sops.secrets.ntfy.path;
|
||||
url = "http://127.0.0.1:2586";
|
||||
topic = "alerts";
|
||||
};
|
||||
}
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue