2022-06-16 17:23:12 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
let
|
|
|
|
cfg = config.services.bird-lg;
|
2023-04-29 16:46:19 +00:00
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
stringOrConcat = sep: v: if builtins.isString v then v else lib.concatStringsSep sep v;
|
2023-04-29 16:46:19 +00:00
|
|
|
|
|
|
|
frontend_args = let
|
|
|
|
fe = cfg.frontend;
|
|
|
|
in {
|
2024-09-19 14:19:46 +00:00
|
|
|
"--servers" = lib.concatStringsSep "," fe.servers;
|
2023-04-29 16:46:19 +00:00
|
|
|
"--domain" = fe.domain;
|
|
|
|
"--listen" = fe.listenAddress;
|
|
|
|
"--proxy-port" = fe.proxyPort;
|
|
|
|
"--whois" = fe.whois;
|
|
|
|
"--dns-interface" = fe.dnsInterface;
|
2024-09-19 14:19:46 +00:00
|
|
|
"--bgpmap-info" = lib.concatStringsSep "," cfg.frontend.bgpMapInfo;
|
2023-04-29 16:46:19 +00:00
|
|
|
"--title-brand" = fe.titleBrand;
|
|
|
|
"--navbar-brand" = fe.navbar.brand;
|
|
|
|
"--navbar-brand-url" = fe.navbar.brandURL;
|
|
|
|
"--navbar-all-servers" = fe.navbar.allServers;
|
|
|
|
"--navbar-all-url" = fe.navbar.allServersURL;
|
|
|
|
"--net-specific-mode" = fe.netSpecificMode;
|
2024-09-19 14:19:46 +00:00
|
|
|
"--protocol-filter" = lib.concatStringsSep "," cfg.frontend.protocolFilter;
|
2023-04-29 16:46:19 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
proxy_args = let
|
|
|
|
px = cfg.proxy;
|
|
|
|
in {
|
2024-09-19 14:19:46 +00:00
|
|
|
"--allowed" = lib.concatStringsSep "," px.allowedIPs;
|
2023-04-29 16:46:19 +00:00
|
|
|
"--bird" = px.birdSocket;
|
|
|
|
"--listen" = px.listenAddress;
|
|
|
|
"--traceroute_bin" = px.traceroute.binary;
|
2024-09-19 14:19:46 +00:00
|
|
|
"--traceroute_flags" = lib.concatStringsSep " " px.traceroute.flags;
|
2023-04-29 16:46:19 +00:00
|
|
|
"--traceroute_raw" = px.traceroute.rawOutput;
|
|
|
|
};
|
|
|
|
|
|
|
|
mkArgValue = value:
|
2024-09-19 14:19:46 +00:00
|
|
|
if lib.isString value
|
|
|
|
then lib.escapeShellArg value
|
|
|
|
else if lib.isBool value
|
|
|
|
then lib.boolToString value
|
2023-04-29 16:46:19 +00:00
|
|
|
else toString value;
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
filterNull = lib.filterAttrs (_: v: v != "" && v != null && v != []);
|
2023-04-29 16:46:19 +00:00
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
argsAttrToList = args: lib.mapAttrsToList (name: value: "${name} " + mkArgValue value ) (filterNull args);
|
2022-06-16 17:23:12 +00:00
|
|
|
in
|
|
|
|
{
|
|
|
|
options = {
|
|
|
|
services.bird-lg = {
|
2024-09-19 14:19:46 +00:00
|
|
|
package = lib.mkPackageOption pkgs "bird-lg" { };
|
2022-06-16 17:23:12 +00:00
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
user = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "bird-lg";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "User to run the service.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
group = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "bird-lg";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Group to run the service.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
frontend = {
|
2024-09-19 14:19:46 +00:00
|
|
|
enable = lib.mkEnableOption "Bird Looking Glass Frontend Webserver";
|
2022-06-16 17:23:12 +00:00
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
listenAddress = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "127.0.0.1:5000";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Address to listen on.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
proxyPort = lib.mkOption {
|
|
|
|
type = lib.types.port;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = 8000;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Port bird-lg-proxy is running on.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
domain = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
example = "dn42.lantian.pub";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Server name domain suffixes.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
servers = lib.mkOption {
|
|
|
|
type = lib.types.listOf lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
example = [ "gigsgigscloud" "hostdare" ];
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Server name prefixes.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
whois = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "whois.verisign-grs.com";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Whois server for queries.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
dnsInterface = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "asn.cymru.com";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "DNS zone to query ASN information.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
bgpMapInfo = lib.mkOption {
|
|
|
|
type = lib.types.listOf lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = [ "asn" "as-name" "ASName" "descr" ];
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Information displayed in bgpmap.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
titleBrand = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "Bird-lg Go";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Prefix of page titles in browser tabs.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
netSpecificMode = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "";
|
|
|
|
example = "dn42";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Apply network-specific changes for some networks.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
protocolFilter = lib.mkOption {
|
|
|
|
type = lib.types.listOf lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = [ ];
|
|
|
|
example = [ "ospf" ];
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Information displayed in bgpmap.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
nameFilter = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "";
|
|
|
|
example = "^ospf";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Protocol names to hide in summary tables (RE2 syntax),";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
timeout = lib.mkOption {
|
|
|
|
type = lib.types.int;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = 120;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Time before request timed out, in seconds.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
navbar = {
|
2024-09-19 14:19:46 +00:00
|
|
|
brand = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "Bird-lg Go";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Brand to show in the navigation bar .";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
brandURL = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "/";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "URL of the brand to show in the navigation bar.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
allServers = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "ALL Servers";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Text of 'All server' button in the navigation bar.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
allServersURL = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "all";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "URL of 'All servers' button.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
extraArgs = lib.mkOption {
|
|
|
|
type = with lib.types; either lines (listOf str);
|
2023-04-29 16:46:19 +00:00
|
|
|
default = [ ];
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2022-08-12 12:06:08 +00:00
|
|
|
Extra parameters documented [here](https://github.com/xddxdd/bird-lg-go#frontend).
|
2023-04-29 16:46:19 +00:00
|
|
|
|
|
|
|
:::{.note}
|
|
|
|
Passing lines (plain strings) is deprecated in favour of passing lists of strings.
|
|
|
|
:::
|
2022-08-12 12:06:08 +00:00
|
|
|
'';
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
proxy = {
|
2024-09-19 14:19:46 +00:00
|
|
|
enable = lib.mkEnableOption "Bird Looking Glass Proxy";
|
2022-06-16 17:23:12 +00:00
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
listenAddress = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "127.0.0.1:8000";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Address to listen on.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
allowedIPs = lib.mkOption {
|
|
|
|
type = lib.types.listOf lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = [ ];
|
2024-04-21 15:54:59 +00:00
|
|
|
example = [ "192.168.25.52" "192.168.25.53" "192.168.0.0/24" ];
|
|
|
|
description = "List of IPs or networks to allow (default all allowed).";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
birdSocket = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2023-04-29 16:46:19 +00:00
|
|
|
default = "/var/run/bird/bird.ctl";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Bird control socket path.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
traceroute = {
|
2024-09-19 14:19:46 +00:00
|
|
|
binary = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = "${pkgs.traceroute}/bin/traceroute";
|
2024-09-19 14:19:46 +00:00
|
|
|
defaultText = lib.literalExpression ''"''${pkgs.traceroute}/bin/traceroute"'';
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Traceroute's binary path.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
flags = lib.mkOption {
|
|
|
|
type = with lib.types; listOf str;
|
2023-04-29 16:46:19 +00:00
|
|
|
default = [ ];
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Flags for traceroute process";
|
2023-04-29 16:46:19 +00:00
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
rawOutput = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2022-06-16 17:23:12 +00:00
|
|
|
default = false;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Display traceroute output in raw format.";
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
extraArgs = lib.mkOption {
|
|
|
|
type = with lib.types; either lines (listOf str);
|
2023-04-29 16:46:19 +00:00
|
|
|
default = [ ];
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2022-08-12 12:06:08 +00:00
|
|
|
Extra parameters documented [here](https://github.com/xddxdd/bird-lg-go#proxy).
|
2023-04-29 16:46:19 +00:00
|
|
|
|
|
|
|
:::{.note}
|
|
|
|
Passing lines (plain strings) is deprecated in favour of passing lists of strings.
|
|
|
|
:::
|
2022-08-12 12:06:08 +00:00
|
|
|
'';
|
2022-06-16 17:23:12 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
###### implementation
|
|
|
|
|
|
|
|
config = {
|
2023-04-29 16:46:19 +00:00
|
|
|
|
|
|
|
warnings =
|
|
|
|
lib.optional (cfg.frontend.enable && builtins.isString cfg.frontend.extraArgs) ''
|
|
|
|
Passing strings to `services.bird-lg.frontend.extraOptions' is deprecated. Please pass a list of strings instead.
|
|
|
|
''
|
|
|
|
++ lib.optional (cfg.proxy.enable && builtins.isString cfg.proxy.extraArgs) ''
|
|
|
|
Passing strings to `services.bird-lg.proxy.extraOptions' is deprecated. Please pass a list of strings instead.
|
|
|
|
''
|
|
|
|
;
|
|
|
|
|
2022-06-16 17:23:12 +00:00
|
|
|
systemd.services = {
|
2024-09-19 14:19:46 +00:00
|
|
|
bird-lg-frontend = lib.mkIf cfg.frontend.enable {
|
2022-06-16 17:23:12 +00:00
|
|
|
enable = true;
|
|
|
|
after = [ "network.target" ];
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
description = "Bird Looking Glass Frontend Webserver";
|
|
|
|
serviceConfig = {
|
|
|
|
Type = "simple";
|
|
|
|
Restart = "on-failure";
|
|
|
|
ProtectSystem = "full";
|
|
|
|
ProtectHome = "yes";
|
|
|
|
MemoryDenyWriteExecute = "yes";
|
|
|
|
User = cfg.user;
|
|
|
|
Group = cfg.group;
|
|
|
|
};
|
|
|
|
script = ''
|
|
|
|
${cfg.package}/bin/frontend \
|
2024-09-19 14:19:46 +00:00
|
|
|
${lib.concatStringsSep " \\\n " (argsAttrToList frontend_args)} \
|
2023-04-29 16:46:19 +00:00
|
|
|
${stringOrConcat " " cfg.frontend.extraArgs}
|
2022-06-16 17:23:12 +00:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
bird-lg-proxy = lib.mkIf cfg.proxy.enable {
|
2022-06-16 17:23:12 +00:00
|
|
|
enable = true;
|
|
|
|
after = [ "network.target" ];
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
description = "Bird Looking Glass Proxy";
|
|
|
|
serviceConfig = {
|
|
|
|
Type = "simple";
|
|
|
|
Restart = "on-failure";
|
|
|
|
ProtectSystem = "full";
|
|
|
|
ProtectHome = "yes";
|
|
|
|
MemoryDenyWriteExecute = "yes";
|
|
|
|
User = cfg.user;
|
|
|
|
Group = cfg.group;
|
|
|
|
};
|
|
|
|
script = ''
|
|
|
|
${cfg.package}/bin/proxy \
|
2024-09-19 14:19:46 +00:00
|
|
|
${lib.concatStringsSep " \\\n " (argsAttrToList proxy_args)} \
|
2023-04-29 16:46:19 +00:00
|
|
|
${stringOrConcat " " cfg.proxy.extraArgs}
|
2022-06-16 17:23:12 +00:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
users = lib.mkIf (cfg.frontend.enable || cfg.proxy.enable) {
|
|
|
|
groups."bird-lg" = lib.mkIf (cfg.group == "bird-lg") { };
|
|
|
|
users."bird-lg" = lib.mkIf (cfg.user == "bird-lg") {
|
2022-06-16 17:23:12 +00:00
|
|
|
description = "Bird Looking Glass user";
|
|
|
|
extraGroups = lib.optionals (config.services.bird2.enable) [ "bird2" ];
|
|
|
|
group = cfg.group;
|
|
|
|
isSystemUser = true;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
2023-04-29 16:46:19 +00:00
|
|
|
|
|
|
|
meta.maintainers = with lib.maintainers; [
|
|
|
|
e1mo
|
|
|
|
tchekda
|
|
|
|
];
|
2022-06-16 17:23:12 +00:00
|
|
|
}
|