2020-04-24 23:36:52 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
let
|
|
|
|
|
|
|
|
cfg = config.services.smokeping;
|
|
|
|
smokepingHome = "/var/lib/smokeping";
|
|
|
|
smokepingPidDir = "/run";
|
|
|
|
configFile =
|
|
|
|
if cfg.config == null
|
2023-07-15 17:15:38 +00:00
|
|
|
then
|
|
|
|
''
|
|
|
|
*** General ***
|
|
|
|
cgiurl = ${cfg.cgiUrl}
|
|
|
|
contact = ${cfg.ownerEmail}
|
|
|
|
datadir = ${smokepingHome}/data
|
|
|
|
imgcache = ${smokepingHome}/cache
|
|
|
|
imgurl = ${cfg.imgUrl}
|
|
|
|
linkstyle = ${cfg.linkStyle}
|
|
|
|
${lib.optionalString (cfg.mailHost != "") "mailhost = ${cfg.mailHost}"}
|
|
|
|
owner = ${cfg.owner}
|
|
|
|
pagedir = ${smokepingHome}/cache
|
|
|
|
piddir = ${smokepingPidDir}
|
|
|
|
${lib.optionalString (cfg.sendmail != null) "sendmail = ${cfg.sendmail}"}
|
|
|
|
smokemail = ${cfg.smokeMailTemplate}
|
|
|
|
*** Presentation ***
|
|
|
|
template = ${cfg.presentationTemplate}
|
|
|
|
${cfg.presentationConfig}
|
|
|
|
*** Alerts ***
|
|
|
|
${cfg.alertConfig}
|
|
|
|
*** Database ***
|
|
|
|
${cfg.databaseConfig}
|
|
|
|
*** Probes ***
|
|
|
|
${cfg.probeConfig}
|
|
|
|
*** Targets ***
|
|
|
|
${cfg.targetConfig}
|
|
|
|
${cfg.extraConfig}
|
|
|
|
''
|
|
|
|
else
|
|
|
|
cfg.config;
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
configPath = pkgs.writeText "smokeping.conf" configFile;
|
|
|
|
cgiHome = pkgs.writeScript "smokeping.fcgi" ''
|
|
|
|
#!${pkgs.bash}/bin/bash
|
2023-04-12 12:48:02 +00:00
|
|
|
${cfg.package}/bin/smokeping_cgi /etc/smokeping.conf
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
in
|
|
|
|
|
|
|
|
{
|
2024-05-15 15:35:15 +00:00
|
|
|
imports = [
|
|
|
|
(mkRemovedOptionModule [ "services" "smokeping" "port" ] ''
|
|
|
|
The smokeping web service is now served by nginx.
|
|
|
|
In order to change the port, you need to change the nginx configuration under `services.nginx.virtualHosts.smokeping.listen.*.port`.
|
|
|
|
'')
|
|
|
|
];
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
options = {
|
|
|
|
services.smokeping = {
|
2024-04-21 15:54:59 +00:00
|
|
|
enable = mkEnableOption "smokeping service";
|
2022-11-21 17:40:18 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
alertConfig = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = ''
|
|
|
|
to = root@localhost
|
|
|
|
from = smokeping@localhost
|
|
|
|
'';
|
2021-10-06 13:57:05 +00:00
|
|
|
example = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
to = alertee@address.somewhere
|
|
|
|
from = smokealert@company.xy
|
|
|
|
|
|
|
|
+someloss
|
|
|
|
type = loss
|
|
|
|
# in percent
|
|
|
|
pattern = >0%,*12*,>0%,*12*,>0%
|
|
|
|
comment = loss 3 times in a row;
|
|
|
|
'';
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Configuration for alerts.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
cgiUrl = mkOption {
|
|
|
|
type = types.str;
|
2024-05-15 15:35:15 +00:00
|
|
|
default = "http://${cfg.hostName}/smokeping.cgi";
|
|
|
|
defaultText = literalExpression ''"http://''${hostName}/smokeping.cgi"'';
|
2020-04-24 23:36:52 +00:00
|
|
|
example = "https://somewhere.example.com/smokeping.cgi";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "URL to the smokeping cgi.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
config = mkOption {
|
|
|
|
type = types.nullOr types.lines;
|
|
|
|
default = null;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2022-09-09 14:08:57 +00:00
|
|
|
Full smokeping config supplied by the user. Overrides
|
|
|
|
and replaces any other configuration supplied.
|
|
|
|
'';
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
databaseConfig = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = ''
|
|
|
|
step = 300
|
|
|
|
pings = 20
|
|
|
|
# consfn mrhb steps total
|
|
|
|
AVERAGE 0.5 1 1008
|
|
|
|
AVERAGE 0.5 12 4320
|
|
|
|
MIN 0.5 12 4320
|
|
|
|
MAX 0.5 12 4320
|
|
|
|
AVERAGE 0.5 144 720
|
|
|
|
MAX 0.5 144 720
|
|
|
|
MIN 0.5 144 720
|
|
|
|
|
|
|
|
'';
|
2021-10-06 13:57:05 +00:00
|
|
|
example = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
# near constant pings.
|
|
|
|
step = 30
|
|
|
|
pings = 20
|
|
|
|
# consfn mrhb steps total
|
|
|
|
AVERAGE 0.5 1 10080
|
|
|
|
AVERAGE 0.5 12 43200
|
|
|
|
MIN 0.5 12 43200
|
|
|
|
MAX 0.5 12 43200
|
|
|
|
AVERAGE 0.5 144 7200
|
|
|
|
MAX 0.5 144 7200
|
|
|
|
MIN 0.5 144 7200
|
|
|
|
'';
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''Configure the ping frequency and retention of the rrd files.
|
2020-04-24 23:36:52 +00:00
|
|
|
Once set, changing the interval will require deletion or migration of all
|
|
|
|
the collected data.'';
|
|
|
|
};
|
|
|
|
extraConfig = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = "";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Any additional customization not already included.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
hostName = mkOption {
|
|
|
|
type = types.str;
|
2021-02-05 17:12:51 +00:00
|
|
|
default = config.networking.fqdn;
|
2021-10-06 13:57:05 +00:00
|
|
|
defaultText = literalExpression "config.networking.fqdn";
|
2020-04-24 23:36:52 +00:00
|
|
|
example = "somewhere.example.com";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "DNS name for the urls generated in the cgi.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
imgUrl = mkOption {
|
|
|
|
type = types.str;
|
2021-12-06 16:07:01 +00:00
|
|
|
default = "cache";
|
|
|
|
defaultText = literalExpression ''"cache"'';
|
2020-04-24 23:36:52 +00:00
|
|
|
example = "https://somewhere.example.com/cache";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2021-12-06 16:07:01 +00:00
|
|
|
Base url for images generated in the cgi.
|
|
|
|
|
|
|
|
The default is a relative URL to ensure it works also when e.g. forwarding
|
|
|
|
the GUI port via SSH.
|
|
|
|
'';
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
linkStyle = mkOption {
|
2023-07-15 17:15:38 +00:00
|
|
|
type = types.enum [ "original" "absolute" "relative" ];
|
2020-04-24 23:36:52 +00:00
|
|
|
default = "relative";
|
|
|
|
example = "absolute";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "DNS name for the urls generated in the cgi.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
mailHost = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
|
|
|
example = "localhost";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Use this SMTP server to send alerts";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
owner = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "nobody";
|
2022-08-12 12:06:08 +00:00
|
|
|
example = "Bob Foobawr";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Real name of the owner of the instance";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
ownerEmail = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "no-reply@${cfg.hostName}";
|
2021-10-06 13:57:05 +00:00
|
|
|
defaultText = literalExpression ''"no-reply@''${hostName}"'';
|
2020-04-24 23:36:52 +00:00
|
|
|
example = "no-reply@yourdomain.com";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Email contact for owner";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
2024-01-02 11:29:13 +00:00
|
|
|
package = mkPackageOption pkgs "smokeping" { };
|
2021-12-06 16:07:01 +00:00
|
|
|
host = mkOption {
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
default = "localhost";
|
|
|
|
example = "192.0.2.1"; # rfc5737 example IP for documentation
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2021-12-06 16:07:01 +00:00
|
|
|
Host/IP to bind to for the web server.
|
|
|
|
|
2022-08-12 12:06:08 +00:00
|
|
|
Setting it to `null` skips passing the -h option to thttpd,
|
2021-12-06 16:07:01 +00:00
|
|
|
which makes it bind to all interfaces.
|
|
|
|
'';
|
|
|
|
};
|
2020-04-24 23:36:52 +00:00
|
|
|
presentationConfig = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = ''
|
|
|
|
+ charts
|
|
|
|
menu = Charts
|
|
|
|
title = The most interesting destinations
|
|
|
|
++ stddev
|
|
|
|
sorter = StdDev(entries=>4)
|
|
|
|
title = Top Standard Deviation
|
|
|
|
menu = Std Deviation
|
|
|
|
format = Standard Deviation %f
|
|
|
|
++ max
|
|
|
|
sorter = Max(entries=>5)
|
|
|
|
title = Top Max Roundtrip Time
|
|
|
|
menu = by Max
|
|
|
|
format = Max Roundtrip Time %f seconds
|
|
|
|
++ loss
|
|
|
|
sorter = Loss(entries=>5)
|
|
|
|
title = Top Packet Loss
|
|
|
|
menu = Loss
|
|
|
|
format = Packets Lost %f
|
|
|
|
++ median
|
|
|
|
sorter = Median(entries=>5)
|
|
|
|
title = Top Median Roundtrip Time
|
|
|
|
menu = by Median
|
|
|
|
format = Median RTT %f seconds
|
|
|
|
+ overview
|
|
|
|
width = 600
|
|
|
|
height = 50
|
|
|
|
range = 10h
|
|
|
|
+ detail
|
|
|
|
width = 600
|
|
|
|
height = 200
|
|
|
|
unison_tolerance = 2
|
|
|
|
"Last 3 Hours" 3h
|
|
|
|
"Last 30 Hours" 30h
|
|
|
|
"Last 10 Days" 10d
|
|
|
|
"Last 360 Days" 360d
|
|
|
|
'';
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "presentation graph style";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
presentationTemplate = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "${pkgs.smokeping}/etc/basepage.html.dist";
|
2021-10-06 13:57:05 +00:00
|
|
|
defaultText = literalExpression ''"''${pkgs.smokeping}/etc/basepage.html.dist"'';
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Default page layout for the web UI.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
probeConfig = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = ''
|
|
|
|
+ FPing
|
|
|
|
binary = ${config.security.wrapperDir}/fping
|
|
|
|
'';
|
2021-12-06 16:07:01 +00:00
|
|
|
defaultText = literalExpression ''
|
|
|
|
'''
|
|
|
|
+ FPing
|
|
|
|
binary = ''${config.security.wrapperDir}/fping
|
|
|
|
'''
|
|
|
|
'';
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Probe configuration";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
sendmail = mkOption {
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
default = null;
|
|
|
|
example = "/run/wrappers/bin/sendmail";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Use this sendmail compatible script to deliver alerts";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
smokeMailTemplate = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "${cfg.package}/etc/smokemail.dist";
|
2021-10-06 13:57:05 +00:00
|
|
|
defaultText = literalExpression ''"''${package}/etc/smokemail.dist"'';
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Specify the smokemail template for alerts.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
targetConfig = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = ''
|
2021-02-05 17:12:51 +00:00
|
|
|
probe = FPing
|
|
|
|
menu = Top
|
|
|
|
title = Network Latency Grapher
|
|
|
|
remark = Welcome to the SmokePing website of xxx Company. \
|
|
|
|
Here you will learn all about the latency of our network.
|
|
|
|
+ Local
|
|
|
|
menu = Local
|
|
|
|
title = Local Network
|
|
|
|
++ LocalMachine
|
|
|
|
menu = Local Machine
|
|
|
|
title = This host
|
|
|
|
host = localhost
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Target configuration";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
user = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "smokeping";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "User that runs smokeping and (optionally) thttpd. A group of the same name will be created as well.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
webService = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Enable a smokeping web interface";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
|
|
|
assertions = [
|
|
|
|
{
|
|
|
|
assertion = !(cfg.sendmail != null && cfg.mailHost != "");
|
|
|
|
message = "services.smokeping: sendmail and Mailhost cannot both be enabled.";
|
|
|
|
}
|
|
|
|
];
|
|
|
|
security.wrappers = {
|
2021-09-22 15:38:15 +00:00
|
|
|
fping =
|
2023-07-15 17:15:38 +00:00
|
|
|
{
|
|
|
|
setuid = true;
|
2021-09-22 15:38:15 +00:00
|
|
|
owner = "root";
|
|
|
|
group = "root";
|
|
|
|
source = "${pkgs.fping}/bin/fping";
|
|
|
|
};
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
2023-04-12 12:48:02 +00:00
|
|
|
environment.etc."smokeping.conf".source = configPath;
|
2020-04-24 23:36:52 +00:00
|
|
|
environment.systemPackages = [ pkgs.fping ];
|
|
|
|
users.users.${cfg.user} = {
|
|
|
|
isNormalUser = false;
|
|
|
|
isSystemUser = true;
|
2021-09-18 10:52:07 +00:00
|
|
|
group = cfg.user;
|
2020-04-24 23:36:52 +00:00
|
|
|
description = "smokeping daemon user";
|
|
|
|
home = smokepingHome;
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
|
|
|
|
users.users.${config.services.nginx.user} = mkIf cfg.webService {
|
|
|
|
extraGroups = [
|
|
|
|
cfg.user ## user == group in this module
|
|
|
|
];
|
|
|
|
};
|
|
|
|
|
2023-04-12 12:48:02 +00:00
|
|
|
users.groups.${cfg.user} = { };
|
2024-09-19 14:19:46 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
systemd.services.smokeping = {
|
2023-04-12 12:48:02 +00:00
|
|
|
reloadTriggers = [ configPath ];
|
|
|
|
requiredBy = [ "multi-user.target" ];
|
2020-04-24 23:36:52 +00:00
|
|
|
serviceConfig = {
|
|
|
|
User = cfg.user;
|
|
|
|
Restart = "on-failure";
|
2023-04-12 12:48:02 +00:00
|
|
|
ExecStart = "${cfg.package}/bin/smokeping --config=/etc/smokeping.conf --nodaemon";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
preStart = ''
|
|
|
|
${cfg.package}/bin/smokeping --check --config=${configPath}
|
|
|
|
${cfg.package}/bin/smokeping --static --config=${configPath}
|
|
|
|
'';
|
|
|
|
};
|
2024-05-15 15:35:15 +00:00
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
systemd.tmpfiles.rules = [
|
|
|
|
# create cache and data directories
|
|
|
|
"d ${smokepingHome}/cache 0750 ${cfg.user} ${cfg.user}"
|
|
|
|
"d ${smokepingHome}/data 0750 ${cfg.user} ${cfg.user}"
|
|
|
|
# create symlings
|
|
|
|
"L+ ${smokepingHome}/css - - - - ${cfg.package}/htdocs/css"
|
|
|
|
"L+ ${smokepingHome}/js - - - - ${cfg.package}/htdocs/js"
|
|
|
|
"L+ ${smokepingHome}/smokeping.fcgi - - - - ${cgiHome}"
|
|
|
|
# recursively adjust access mode and ownership (in case config change)
|
|
|
|
"Z ${smokepingHome} 0750 ${cfg.user} ${cfg.user}"
|
|
|
|
];
|
|
|
|
|
2024-05-15 15:35:15 +00:00
|
|
|
# use nginx to serve the smokeping web service
|
2024-07-31 10:19:44 +00:00
|
|
|
services.fcgiwrap.instances.smokeping = mkIf cfg.webService {
|
2024-07-27 06:49:29 +00:00
|
|
|
process.user = cfg.user;
|
|
|
|
process.group = cfg.user;
|
|
|
|
socket = { inherit (config.services.nginx) user group; };
|
|
|
|
};
|
2024-05-15 15:35:15 +00:00
|
|
|
services.nginx = mkIf cfg.webService {
|
|
|
|
enable = true;
|
|
|
|
virtualHosts."smokeping" = {
|
|
|
|
serverName = mkDefault cfg.host;
|
|
|
|
locations."/" = {
|
|
|
|
root = smokepingHome;
|
|
|
|
index = "smokeping.fcgi";
|
|
|
|
};
|
|
|
|
locations."/smokeping.fcgi" = {
|
|
|
|
extraConfig = ''
|
|
|
|
include ${config.services.nginx.package}/conf/fastcgi_params;
|
2024-07-31 10:19:44 +00:00
|
|
|
fastcgi_pass unix:${config.services.fcgiwrap.instances.smokeping.socket.address};
|
2024-05-15 15:35:15 +00:00
|
|
|
fastcgi_param SCRIPT_FILENAME ${smokepingHome}/smokeping.fcgi;
|
|
|
|
fastcgi_param DOCUMENT_ROOT ${smokepingHome};
|
|
|
|
'';
|
|
|
|
};
|
2021-12-06 16:07:01 +00:00
|
|
|
};
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-10-23 06:41:50 +00:00
|
|
|
meta.maintainers = with lib.maintainers; [ nh2 ];
|
2020-04-24 23:36:52 +00:00
|
|
|
}
|