nixos/opensnitch: Add option to configure rules

This commit is contained in:
Jonas Heinrich 2022-11-19 16:08:21 +01:00 committed by Yt
parent 11fdcb163e
commit 3aff916914
3 changed files with 74 additions and 16 deletions

View File

@ -83,7 +83,10 @@
<itemizedlist spacing="compact"> <itemizedlist spacing="compact">
<listitem> <listitem>
<para> <para>
Create the first release note entry in this section! The module for the application firewall
<literal>opensnitch</literal> got the ability to configure
rules. Available as
<link linkend="opt-services.opensnitch.rules">services.opensnitch.rules</link>
</para> </para>
</listitem> </listitem>
</itemizedlist> </itemizedlist>

View File

@ -33,4 +33,4 @@ In addition to numerous new and upgraded packages, this release has the followin
<!-- To avoid merge conflicts, consider adding your item at an arbitrary place in the list instead. --> <!-- To avoid merge conflicts, consider adding your item at an arbitrary place in the list instead. -->
- Create the first release note entry in this section! - The module for the application firewall `opensnitch` got the ability to configure rules. Available as [services.opensnitch.rules](#opt-services.opensnitch.rules)

View File

@ -5,10 +5,47 @@ with lib;
let let
cfg = config.services.opensnitch; cfg = config.services.opensnitch;
format = pkgs.formats.json {}; format = pkgs.formats.json {};
predefinedRules = flip mapAttrs cfg.rules (name: cfg: {
file = pkgs.writeText "rule" (builtins.toJSON cfg);
});
in { in {
options = { options = {
services.opensnitch = { services.opensnitch = {
enable = mkEnableOption (lib.mdDoc "Opensnitch application firewall"); enable = mkEnableOption (mdDoc "Opensnitch application firewall");
rules = mkOption {
default = {};
example = literalExpression ''
{
"tor" = {
"name" = "tor";
"enabled" = true;
"action" = "allow";
"duration" = "always";
"operator" = {
"type" ="simple";
"sensitive" = false;
"operand" = "process.path";
"data" = "''${lib.getBin pkgs.tor}/bin/tor";
};
};
};
'';
description = mdDoc ''
Declarative configuration of firewall rules.
All rules will be stored in `/var/lib/opensnitch/rules`.
See [upstream documentation](https://github.com/evilsocket/opensnitch/wiki/Rules)
for available options.
'';
type = types.submodule {
freeformType = format.type;
};
};
settings = mkOption { settings = mkOption {
type = types.submodule { type = types.submodule {
freeformType = format.type; freeformType = format.type;
@ -18,7 +55,7 @@ in {
Address = mkOption { Address = mkOption {
type = types.str; type = types.str;
description = lib.mdDoc '' description = mdDoc ''
Unix socket path (unix:///tmp/osui.sock, the "unix:///" part is Unix socket path (unix:///tmp/osui.sock, the "unix:///" part is
mandatory) or TCP socket (192.168.1.100:50051). mandatory) or TCP socket (192.168.1.100:50051).
''; '';
@ -26,7 +63,7 @@ in {
LogFile = mkOption { LogFile = mkOption {
type = types.path; type = types.path;
description = lib.mdDoc '' description = mdDoc ''
File to write logs to (use /dev/stdout to write logs to standard File to write logs to (use /dev/stdout to write logs to standard
output). output).
''; '';
@ -36,7 +73,7 @@ in {
DefaultAction = mkOption { DefaultAction = mkOption {
type = types.enum [ "allow" "deny" ]; type = types.enum [ "allow" "deny" ];
description = lib.mdDoc '' description = mdDoc ''
Default action whether to block or allow application internet Default action whether to block or allow application internet
access. access.
''; '';
@ -46,28 +83,28 @@ in {
type = types.enum [ type = types.enum [
"once" "always" "until restart" "30s" "5m" "15m" "30m" "1h" "once" "always" "until restart" "30s" "5m" "15m" "30m" "1h"
]; ];
description = lib.mdDoc '' description = mdDoc ''
Default duration of firewall rule. Default duration of firewall rule.
''; '';
}; };
InterceptUnknown = mkOption { InterceptUnknown = mkOption {
type = types.bool; type = types.bool;
description = lib.mdDoc '' description = mdDoc ''
Wheter to intercept spare connections. Wheter to intercept spare connections.
''; '';
}; };
ProcMonitorMethod = mkOption { ProcMonitorMethod = mkOption {
type = types.enum [ "ebpf" "proc" "ftrace" "audit" ]; type = types.enum [ "ebpf" "proc" "ftrace" "audit" ];
description = lib.mdDoc '' description = mdDoc ''
Which process monitoring method to use. Which process monitoring method to use.
''; '';
}; };
LogLevel = mkOption { LogLevel = mkOption {
type = types.enum [ 0 1 2 3 4 ]; type = types.enum [ 0 1 2 3 4 ];
description = lib.mdDoc '' description = mdDoc ''
Default log level from 0 to 4 (debug, info, important, warning, Default log level from 0 to 4 (debug, info, important, warning,
error). error).
''; '';
@ -75,7 +112,7 @@ in {
Firewall = mkOption { Firewall = mkOption {
type = types.enum [ "iptables" "nftables" ]; type = types.enum [ "iptables" "nftables" ];
description = lib.mdDoc '' description = mdDoc ''
Which firewall backend to use. Which firewall backend to use.
''; '';
}; };
@ -84,14 +121,14 @@ in {
MaxEvents = mkOption { MaxEvents = mkOption {
type = types.int; type = types.int;
description = lib.mdDoc '' description = mdDoc ''
Max events to send to the GUI. Max events to send to the GUI.
''; '';
}; };
MaxStats = mkOption { MaxStats = mkOption {
type = types.int; type = types.int;
description = lib.mdDoc '' description = mdDoc ''
Max stats per item to keep in backlog. Max stats per item to keep in backlog.
''; '';
}; };
@ -99,9 +136,8 @@ in {
}; };
}; };
}; };
description = lib.mdDoc '' description = mdDoc ''
opensnitchd configuration. Refer to opensnitchd configuration. Refer to [upstream documentation](https://github.com/evilsocket/opensnitch/wiki/Configurations)
<https://github.com/evilsocket/opensnitch/wiki/Configurations>
for details on supported values. for details on supported values.
''; '';
}; };
@ -118,6 +154,25 @@ in {
services.opensnitchd.wantedBy = [ "multi-user.target" ]; services.opensnitchd.wantedBy = [ "multi-user.target" ];
}; };
systemd.services.opensnitchd.preStart = mkIf (cfg.rules != {}) (let
rules = flip mapAttrsToList predefinedRules (file: content: {
inherit (content) file;
local = "/var/lib/opensnitch/rules/${file}.json";
});
in ''
# Remove all firewall rules from `/var/lib/opensnitch/rules` that are symlinks to a store-path,
# but aren't declared in `cfg.rules` (i.e. all networks that were "removed" from
# `cfg.rules`).
find /var/lib/opensnitch/rules -type l -lname '${builtins.storeDir}/*' ${optionalString (rules != {}) ''
-not \( ${concatMapStringsSep " -o " ({ local, ... }:
"-name '${baseNameOf local}*'")
rules} \) \
''} -delete
${concatMapStrings ({ file, local }: ''
ln -sf '${file}' "${local}"
'') rules}
'');
environment.etc."opensnitchd/default-config.json".source = format.generate "default-config.json" cfg.settings; environment.etc."opensnitchd/default-config.json".source = format.generate "default-config.json" cfg.settings;
}; };