171 lines
5.4 KiB
Nix
171 lines
5.4 KiB
Nix
|
# Global configuration for atop.
|
||
|
|
||
|
{ config, lib, pkgs, ... }:
|
||
|
|
||
|
let cfg = config.programs.atop;
|
||
|
|
||
|
in
|
||
|
{
|
||
|
###### interface
|
||
|
|
||
|
options = {
|
||
|
|
||
|
programs.atop = rec {
|
||
|
|
||
|
enable = lib.mkEnableOption "Atop, a tool for monitoring system resources";
|
||
|
|
||
|
package = lib.mkPackageOption pkgs "atop" { };
|
||
|
|
||
|
netatop = {
|
||
|
enable = lib.mkOption {
|
||
|
type = lib.types.bool;
|
||
|
default = false;
|
||
|
description = ''
|
||
|
Whether to install and enable the netatop kernel module.
|
||
|
Note: this sets the kernel taint flag "O" for loading out-of-tree modules.
|
||
|
'';
|
||
|
};
|
||
|
package = lib.mkOption {
|
||
|
type = lib.types.package;
|
||
|
default = config.boot.kernelPackages.netatop;
|
||
|
defaultText = lib.literalExpression "config.boot.kernelPackages.netatop";
|
||
|
description = ''
|
||
|
Which package to use for netatop.
|
||
|
'';
|
||
|
};
|
||
|
};
|
||
|
|
||
|
atopgpu.enable = lib.mkOption {
|
||
|
type = lib.types.bool;
|
||
|
default = false;
|
||
|
description = ''
|
||
|
Whether to install and enable the atopgpud daemon to get information about
|
||
|
NVIDIA gpus.
|
||
|
'';
|
||
|
};
|
||
|
|
||
|
setuidWrapper.enable = lib.mkOption {
|
||
|
type = lib.types.bool;
|
||
|
default = false;
|
||
|
description = ''
|
||
|
Whether to install a setuid wrapper for Atop. This is required to use some of
|
||
|
the features as non-root user (e.g.: ipc information, netatop, atopgpu).
|
||
|
Atop tries to drop the root privileges shortly after starting.
|
||
|
'';
|
||
|
};
|
||
|
|
||
|
atopService.enable = lib.mkOption {
|
||
|
type = lib.types.bool;
|
||
|
default = true;
|
||
|
description = ''
|
||
|
Whether to enable the atop service responsible for storing statistics for
|
||
|
long-term analysis.
|
||
|
'';
|
||
|
};
|
||
|
atopRotateTimer.enable = lib.mkOption {
|
||
|
type = lib.types.bool;
|
||
|
default = true;
|
||
|
description = ''
|
||
|
Whether to enable the atop-rotate timer, which restarts the atop service
|
||
|
daily to make sure the data files are rotate.
|
||
|
'';
|
||
|
};
|
||
|
atopacctService.enable = lib.mkOption {
|
||
|
type = lib.types.bool;
|
||
|
default = true;
|
||
|
description = ''
|
||
|
Whether to enable the atopacct service which manages process accounting.
|
||
|
This allows Atop to gather data about processes that disappeared in between
|
||
|
two refresh intervals.
|
||
|
'';
|
||
|
};
|
||
|
settings = lib.mkOption {
|
||
|
type = lib.types.attrs;
|
||
|
default = { };
|
||
|
example = {
|
||
|
flags = "a1f";
|
||
|
interval = 5;
|
||
|
};
|
||
|
description = ''
|
||
|
Parameters to be written to {file}`/etc/atoprc`.
|
||
|
'';
|
||
|
};
|
||
|
};
|
||
|
};
|
||
|
|
||
|
config = lib.mkIf cfg.enable (
|
||
|
let
|
||
|
atop =
|
||
|
if cfg.atopgpu.enable then
|
||
|
(cfg.package.override { withAtopgpu = true; })
|
||
|
else
|
||
|
cfg.package;
|
||
|
in
|
||
|
{
|
||
|
environment.etc = lib.mkIf (cfg.settings != { }) {
|
||
|
atoprc.text = lib.concatStrings
|
||
|
(lib.mapAttrsToList
|
||
|
(n: v: ''
|
||
|
${n} ${builtins.toString v}
|
||
|
'')
|
||
|
cfg.settings);
|
||
|
};
|
||
|
environment.systemPackages = [ atop (lib.mkIf cfg.netatop.enable cfg.netatop.package) ];
|
||
|
boot.extraModulePackages = [ (lib.mkIf cfg.netatop.enable cfg.netatop.package) ];
|
||
|
systemd =
|
||
|
let
|
||
|
mkSystemd = type: name: restartTriggers: {
|
||
|
${name} = {
|
||
|
inherit restartTriggers;
|
||
|
wantedBy = [ (if type == "services" then "multi-user.target" else if type == "timers" then "timers.target" else null) ];
|
||
|
};
|
||
|
};
|
||
|
mkService = mkSystemd "services";
|
||
|
mkTimer = mkSystemd "timers";
|
||
|
in
|
||
|
{
|
||
|
packages = [ atop (lib.mkIf cfg.netatop.enable cfg.netatop.package) ];
|
||
|
services = lib.mkMerge [
|
||
|
(lib.mkIf cfg.atopService.enable (lib.recursiveUpdate
|
||
|
(mkService "atop" [ atop ])
|
||
|
{
|
||
|
# always convert logs to newer version first
|
||
|
# XXX might trigger TimeoutStart but restarting atop.service will
|
||
|
# convert remainings logs and start eventually
|
||
|
atop.preStart = ''
|
||
|
set -e -u
|
||
|
shopt -s nullglob
|
||
|
rm -f "$LOGPATH"/atop_*.new
|
||
|
for logfile in "$LOGPATH"/atop_*
|
||
|
do
|
||
|
${atop}/bin/atopconvert "$logfile" "$logfile".new
|
||
|
# only replace old file if version was upgraded to avoid
|
||
|
# false positives for atop-rotate.service
|
||
|
if ! ${pkgs.diffutils}/bin/cmp -s "$logfile" "$logfile".new
|
||
|
then
|
||
|
mv -v -f "$logfile".new "$logfile"
|
||
|
else
|
||
|
rm -f "$logfile".new
|
||
|
fi
|
||
|
done
|
||
|
'';
|
||
|
}))
|
||
|
(lib.mkIf cfg.atopacctService.enable (mkService "atopacct" [ atop ]))
|
||
|
(lib.mkIf cfg.netatop.enable (mkService "netatop" [ cfg.netatop.package ]))
|
||
|
(lib.mkIf cfg.atopgpu.enable (mkService "atopgpu" [ atop ]))
|
||
|
];
|
||
|
timers = lib.mkIf cfg.atopRotateTimer.enable (mkTimer "atop-rotate" [ atop ]);
|
||
|
};
|
||
|
|
||
|
security.wrappers = lib.mkIf cfg.setuidWrapper.enable {
|
||
|
atop = {
|
||
|
setuid = true;
|
||
|
owner = "root";
|
||
|
group = "root";
|
||
|
source = "${atop}/bin/atop";
|
||
|
};
|
||
|
};
|
||
|
}
|
||
|
);
|
||
|
}
|