{ config, lib, pkgs, ... }: let inherit (lib) mkIf mkOption mkEnableOption types optional optionals; inherit (lib.types) nullOr bool listOf str; cfg = config.services.i2pd; homeDir = "/var/lib/i2pd"; strOpt = k: v: k + " = " + v; boolOpt = k: v: k + " = " + lib.boolToString v; intOpt = k: v: k + " = " + toString v; lstOpt = k: xs: k + " = " + lib.concatStringsSep "," xs; optionalNullString = o: s: optional (s != null) (strOpt o s); optionalNullBool = o: b: optional (b != null) (boolOpt o b); optionalNullInt = o: i: optional (i != null) (intOpt o i); optionalEmptyList = o: l: optional ([] != l) (lstOpt o l); mkEnableTrueOption = name: mkEnableOption name // { default = true; }; mkEndpointOpt = name: addr: port: { enable = mkEnableOption name; name = mkOption { type = types.str; default = name; description = "The endpoint name."; }; address = mkOption { type = types.str; default = addr; description = "Bind address for ${name} endpoint."; }; port = mkOption { type = types.port; default = port; description = "Bind port for ${name} endpoint."; }; }; i2cpOpts = name: { length = mkOption { type = types.int; description = "Guaranteed minimum hops for ${name} tunnels."; default = 3; }; quantity = mkOption { type = types.int; description = "Number of simultaneous ${name} tunnels."; default = 5; }; }; mkKeyedEndpointOpt = name: addr: port: keyloc: (mkEndpointOpt name addr port) // { keys = mkOption { type = nullOr str; default = keyloc; description = '' File to persist ${lib.toUpper name} keys. ''; }; inbound = i2cpOpts name; outbound = i2cpOpts name; latency.min = mkOption { type = with types; nullOr int; description = "Min latency for tunnels."; default = null; }; latency.max = mkOption { type = with types; nullOr int; description = "Max latency for tunnels."; default = null; }; }; commonTunOpts = name: { outbound = i2cpOpts name; inbound = i2cpOpts name; crypto.tagsToSend = mkOption { type = types.int; description = "Number of ElGamal/AES tags to send."; default = 40; }; destination = mkOption { type = types.str; description = "Remote endpoint, I2P hostname or b32.i2p address."; }; keys = mkOption { type = types.str; default = name + "-keys.dat"; description = "Keyset used for tunnel identity."; }; } // mkEndpointOpt name "127.0.0.1" 0; sec = name: "\n[" + name + "]"; notice = "# DO NOT EDIT -- this file has been generated automatically."; i2pdConf = let opts = [ notice (strOpt "loglevel" cfg.logLevel) (boolOpt "logclftime" cfg.logCLFTime) (boolOpt "ipv4" cfg.enableIPv4) (boolOpt "ipv6" cfg.enableIPv6) (boolOpt "notransit" cfg.notransit) (boolOpt "floodfill" cfg.floodfill) (intOpt "netid" cfg.netid) ] ++ (optionalNullInt "bandwidth" cfg.bandwidth) ++ (optionalNullInt "port" cfg.port) ++ (optionalNullString "family" cfg.family) ++ (optionalNullString "datadir" cfg.dataDir) ++ (optionalNullInt "share" cfg.share) ++ (optionalNullBool "ssu" cfg.ssu) ++ (optionalNullBool "ntcp" cfg.ntcp) ++ (optionalNullString "ntcpproxy" cfg.ntcpProxy) ++ (optionalNullString "ifname" cfg.ifname) ++ (optionalNullString "ifname4" cfg.ifname4) ++ (optionalNullString "ifname6" cfg.ifname6) ++ [ (sec "limits") (intOpt "transittunnels" cfg.limits.transittunnels) (intOpt "coresize" cfg.limits.coreSize) (intOpt "openfiles" cfg.limits.openFiles) (intOpt "ntcphard" cfg.limits.ntcpHard) (intOpt "ntcpsoft" cfg.limits.ntcpSoft) (intOpt "ntcpthreads" cfg.limits.ntcpThreads) (sec "upnp") (boolOpt "enabled" cfg.upnp.enable) (sec "precomputation") (boolOpt "elgamal" cfg.precomputation.elgamal) (sec "reseed") (boolOpt "verify" cfg.reseed.verify) ] ++ (optionalNullString "file" cfg.reseed.file) ++ (optionalEmptyList "urls" cfg.reseed.urls) ++ (optionalNullString "floodfill" cfg.reseed.floodfill) ++ (optionalNullString "zipfile" cfg.reseed.zipfile) ++ (optionalNullString "proxy" cfg.reseed.proxy) ++ [ (sec "trust") (boolOpt "enabled" cfg.trust.enable) (boolOpt "hidden" cfg.trust.hidden) ] ++ (optionalEmptyList "routers" cfg.trust.routers) ++ (optionalNullString "family" cfg.trust.family) ++ [ (sec "websockets") (boolOpt "enabled" cfg.websocket.enable) (strOpt "address" cfg.websocket.address) (intOpt "port" cfg.websocket.port) (sec "exploratory") (intOpt "inbound.length" cfg.exploratory.inbound.length) (intOpt "inbound.quantity" cfg.exploratory.inbound.quantity) (intOpt "outbound.length" cfg.exploratory.outbound.length) (intOpt "outbound.quantity" cfg.exploratory.outbound.quantity) (sec "ntcp2") (boolOpt "enabled" cfg.ntcp2.enable) (boolOpt "published" cfg.ntcp2.published) (intOpt "port" cfg.ntcp2.port) (sec "addressbook") (strOpt "defaulturl" cfg.addressbook.defaulturl) ] ++ (optionalEmptyList "subscriptions" cfg.addressbook.subscriptions) ++ [ (sec "meshnets") (boolOpt "yggdrasil" cfg.yggdrasil.enable) ] ++ (optionalNullString "yggaddress" cfg.yggdrasil.address) ++ (lib.flip map (lib.collect (proto: proto ? port && proto ? address) cfg.proto) (proto: let protoOpts = [ (sec proto.name) (boolOpt "enabled" proto.enable) (strOpt "address" proto.address) (intOpt "port" proto.port) ] ++ (optionals (proto ? keys) (optionalNullString "keys" proto.keys)) ++ (optionals (proto ? auth) (optionalNullBool "auth" proto.auth)) ++ (optionals (proto ? user) (optionalNullString "user" proto.user)) ++ (optionals (proto ? pass) (optionalNullString "pass" proto.pass)) ++ (optionals (proto ? strictHeaders) (optionalNullBool "strictheaders" proto.strictHeaders)) ++ (optionals (proto ? hostname) (optionalNullString "hostname" proto.hostname)) ++ (optionals (proto ? outproxy) (optionalNullString "outproxy" proto.outproxy)) ++ (optionals (proto ? outproxyPort) (optionalNullInt "outproxyport" proto.outproxyPort)) ++ (optionals (proto ? outproxyEnable) (optionalNullBool "outproxy.enabled" proto.outproxyEnable)); in (lib.concatStringsSep "\n" protoOpts) )); in pkgs.writeText "i2pd.conf" (lib.concatStringsSep "\n" opts); tunnelConf = let mkOutTunnel = tun: let outTunOpts = [ (sec tun.name) "type = client" (intOpt "port" tun.port) (strOpt "destination" tun.destination) ] ++ (optionals (tun ? destinationPort) (optionalNullInt "destinationport" tun.destinationPort)) ++ (optionals (tun ? keys) (optionalNullString "keys" tun.keys)) ++ (optionals (tun ? address) (optionalNullString "address" tun.address)) ++ (optionals (tun ? inbound.length) (optionalNullInt "inbound.length" tun.inbound.length)) ++ (optionals (tun ? inbound.quantity) (optionalNullInt "inbound.quantity" tun.inbound.quantity)) ++ (optionals (tun ? outbound.length) (optionalNullInt "outbound.length" tun.outbound.length)) ++ (optionals (tun ? outbound.quantity) (optionalNullInt "outbound.quantity" tun.outbound.quantity)) ++ (optionals (tun ? crypto.tagsToSend) (optionalNullInt "crypto.tagstosend" tun.crypto.tagsToSend)); in lib.concatStringsSep "\n" outTunOpts; mkInTunnel = tun: let inTunOpts = [ (sec tun.name) "type = server" (intOpt "port" tun.port) (strOpt "host" tun.address) ] ++ (optionals (tun ? destination) (optionalNullString "destination" tun.destination)) ++ (optionals (tun ? keys) (optionalNullString "keys" tun.keys)) ++ (optionals (tun ? inPort) (optionalNullInt "inport" tun.inPort)) ++ (optionals (tun ? accessList) (optionalEmptyList "accesslist" tun.accessList)); in lib.concatStringsSep "\n" inTunOpts; allOutTunnels = lib.collect (tun: tun ? port && tun ? destination) cfg.outTunnels; allInTunnels = lib.collect (tun: tun ? port && tun ? address) cfg.inTunnels; opts = [ notice ] ++ (map mkOutTunnel allOutTunnels) ++ (map mkInTunnel allInTunnels); in pkgs.writeText "i2pd-tunnels.conf" (lib.concatStringsSep "\n" opts); i2pdFlags = lib.concatStringsSep " " ( optional (cfg.address != null) ("--host=" + cfg.address) ++ [ "--service" ("--conf=" + i2pdConf) ("--tunconf=" + tunnelConf) ]); in { imports = [ (lib.mkRenamedOptionModule [ "services" "i2pd" "extIp" ] [ "services" "i2pd" "address" ]) ]; ###### interface options = { services.i2pd = { enable = mkEnableOption "I2Pd daemon" // { description = '' Enables I2Pd as a running service upon activation. Please read for further configuration help. ''; }; package = lib.mkPackageOption pkgs "i2pd" { }; logLevel = mkOption { type = types.enum ["debug" "info" "warn" "error"]; default = "error"; description = '' The log level. {command}`i2pd` defaults to "info" but that generates copious amounts of log messages. We default to "error" which is similar to the default log level of {command}`tor`. ''; }; logCLFTime = mkEnableOption "full CLF-formatted date and time to log"; address = mkOption { type = nullOr str; default = null; description = '' Your external IP or hostname. ''; }; family = mkOption { type = nullOr str; default = null; description = '' Specify a family the router belongs to. ''; }; dataDir = mkOption { type = nullOr str; default = null; description = '' Alternative path to storage of i2pd data (RI, keys, peer profiles, ...) ''; }; share = mkOption { type = types.int; default = 100; description = '' Limit of transit traffic from max bandwidth in percents. ''; }; ifname = mkOption { type = nullOr str; default = null; description = '' Network interface to bind to. ''; }; ifname4 = mkOption { type = nullOr str; default = null; description = '' IPv4 interface to bind to. ''; }; ifname6 = mkOption { type = nullOr str; default = null; description = '' IPv6 interface to bind to. ''; }; ntcpProxy = mkOption { type = nullOr str; default = null; description = '' Proxy URL for NTCP transport. ''; }; ntcp = mkEnableTrueOption "ntcp"; ssu = mkEnableTrueOption "ssu"; notransit = mkEnableOption "notransit" // { description = '' Tells the router to not accept transit tunnels during startup. ''; }; floodfill = mkEnableOption "floodfill" // { description = '' If the router is declared to be unreachable and needs introduction nodes. ''; }; netid = mkOption { type = types.int; default = 2; description = '' I2P overlay netid. ''; }; bandwidth = mkOption { type = with types; nullOr int; default = null; description = '' Set a router bandwidth limit integer in KBps. If not set, {command}`i2pd` defaults to 32KBps. ''; }; port = mkOption { type = with types; nullOr int; default = null; description = '' I2P listen port. If no one is given the router will pick between 9111 and 30777. ''; }; enableIPv4 = mkEnableTrueOption "IPv4 connectivity"; enableIPv6 = mkEnableOption "IPv6 connectivity"; nat = mkEnableTrueOption "NAT bypass"; upnp.enable = mkEnableOption "UPnP service discovery"; upnp.name = mkOption { type = types.str; default = "I2Pd"; description = '' Name i2pd appears in UPnP forwardings list. ''; }; precomputation.elgamal = mkEnableTrueOption "Precomputed ElGamal tables" // { description = '' Whenever to use precomputated tables for ElGamal. {command}`i2pd` defaults to `false` to save 64M of memory (and looses some performance). We default to `true` as that is what most users want anyway. ''; }; reseed.verify = mkEnableOption "SU3 signature verification"; reseed.file = mkOption { type = nullOr str; default = null; description = '' Full path to SU3 file to reseed from. ''; }; reseed.urls = mkOption { type = listOf str; default = []; description = '' Reseed URLs. ''; }; reseed.floodfill = mkOption { type = nullOr str; default = null; description = '' Path to router info of floodfill to reseed from. ''; }; reseed.zipfile = mkOption { type = nullOr str; default = null; description = '' Path to local .zip file to reseed from. ''; }; reseed.proxy = mkOption { type = nullOr str; default = null; description = '' URL for reseed proxy, supports http/socks. ''; }; addressbook.defaulturl = mkOption { type = types.str; default = "http://joajgazyztfssty4w2on5oaqksz6tqoxbduy553y34mf4byv6gpq.b32.i2p/export/alive-hosts.txt"; description = '' AddressBook subscription URL for initial setup ''; }; addressbook.subscriptions = mkOption { type = listOf str; default = [ "http://inr.i2p/export/alive-hosts.txt" "http://i2p-projekt.i2p/hosts.txt" "http://stats.i2p/cgi-bin/newhosts.txt" ]; description = '' AddressBook subscription URLs ''; }; trust.enable = mkEnableOption "explicit trust options"; trust.family = mkOption { type = nullOr str; default = null; description = '' Router Family to trust for first hops. ''; }; trust.routers = mkOption { type = listOf str; default = []; description = '' Only connect to the listed routers. ''; }; trust.hidden = mkEnableOption "router concealment"; websocket = mkEndpointOpt "websockets" "127.0.0.1" 7666; exploratory.inbound = i2cpOpts "exploratory"; exploratory.outbound = i2cpOpts "exploratory"; ntcp2.enable = mkEnableTrueOption "NTCP2"; ntcp2.published = mkEnableOption "NTCP2 publication"; ntcp2.port = mkOption { type = types.port; default = 0; description = '' Port to listen for incoming NTCP2 connections (0=auto). ''; }; limits.transittunnels = mkOption { type = types.int; default = 2500; description = '' Maximum number of active transit sessions. ''; }; limits.coreSize = mkOption { type = types.int; default = 0; description = '' Maximum size of corefile in Kb (0 - use system limit). ''; }; limits.openFiles = mkOption { type = types.int; default = 0; description = '' Maximum number of open files (0 - use system default). ''; }; limits.ntcpHard = mkOption { type = types.int; default = 0; description = '' Maximum number of active transit sessions. ''; }; limits.ntcpSoft = mkOption { type = types.int; default = 0; description = '' Threshold to start probabalistic backoff with ntcp sessions (default: use system limit). ''; }; limits.ntcpThreads = mkOption { type = types.int; default = 1; description = '' Maximum number of threads used by NTCP DH worker. ''; }; yggdrasil.enable = mkEnableOption "Yggdrasil"; yggdrasil.address = mkOption { type = nullOr str; default = null; description = '' Your local yggdrasil address. Specify it if you want to bind your router to a particular address. ''; }; proto.http = (mkEndpointOpt "http" "127.0.0.1" 7070) // { auth = mkEnableOption "webconsole authentication"; user = mkOption { type = types.str; default = "i2pd"; description = '' Username for webconsole access ''; }; pass = mkOption { type = types.str; default = "i2pd"; description = '' Password for webconsole access. ''; }; strictHeaders = mkOption { type = nullOr bool; default = null; description = '' Enable strict host checking on WebUI. ''; }; hostname = mkOption { type = nullOr str; default = null; description = '' Expected hostname for WebUI. ''; }; }; proto.httpProxy = (mkKeyedEndpointOpt "httpproxy" "127.0.0.1" 4444 "httpproxy-keys.dat") // { outproxy = mkOption { type = nullOr str; default = null; description = "Upstream outproxy bind address."; }; }; proto.socksProxy = (mkKeyedEndpointOpt "socksproxy" "127.0.0.1" 4447 "socksproxy-keys.dat") // { outproxyEnable = mkEnableOption "SOCKS outproxy"; outproxy = mkOption { type = types.str; default = "127.0.0.1"; description = "Upstream outproxy bind address."; }; outproxyPort = mkOption { type = types.int; default = 4444; description = "Upstream outproxy bind port."; }; }; proto.sam = mkEndpointOpt "sam" "127.0.0.1" 7656; proto.bob = mkEndpointOpt "bob" "127.0.0.1" 2827; proto.i2cp = mkEndpointOpt "i2cp" "127.0.0.1" 7654; proto.i2pControl = mkEndpointOpt "i2pcontrol" "127.0.0.1" 7650; outTunnels = mkOption { default = {}; type = with types; attrsOf (submodule ( { name, ... }: { options = { destinationPort = mkOption { type = with types; nullOr int; default = null; description = "Connect to particular port at destination."; }; } // commonTunOpts name; config = { name = mkDefault name; }; } )); description = '' Connect to someone as a client and establish a local accept endpoint ''; }; inTunnels = mkOption { default = {}; type = with types; attrsOf (submodule ( { name, ... }: { options = { inPort = mkOption { type = types.int; default = 0; description = "Service port. Default to the tunnel's listen port."; }; accessList = mkOption { type = listOf str; default = []; description = "I2P nodes that are allowed to connect to this service."; }; } // commonTunOpts name; config = { name = mkDefault name; }; } )); description = '' Serve something on I2P network at port and delegate requests to address inPort. ''; }; }; }; ###### implementation config = mkIf cfg.enable { users.users.i2pd = { group = "i2pd"; description = "I2Pd User"; home = homeDir; createHome = true; uid = config.ids.uids.i2pd; }; users.groups.i2pd.gid = config.ids.gids.i2pd; systemd.services.i2pd = { description = "Minimal I2P router"; after = [ "network.target" ]; wantedBy = [ "multi-user.target" ]; serviceConfig = { User = "i2pd"; WorkingDirectory = homeDir; Restart = "on-abort"; ExecStart = "${cfg.package}/bin/i2pd ${i2pdFlags}"; }; }; }; }