Loading nixos/doc/manual/release-notes/rl-2511.section.md +2 −0 Original line number Diff line number Diff line Loading @@ -182,6 +182,8 @@ - `services.monero` now includes the `environmentFile` option for adding secrets to the Monero daemon config. - `services.netbird.server` now uses dedicated packages split out due to relicensing of server components to AGPLv3 with version `0.53.0`, - The new option [networking.ipips](#opt-networking.ipips) has been added to create IP within IP kind of tunnels (including 4in6, ip6ip6 and ipip). With the existing [networking.sits](#opt-networking.sits) option (6in4), it is now possible to create all combinations of IPv4 and IPv6 encapsulation. Loading nixos/modules/services/networking/netbird/management.nix +1 −1 Original line number Diff line number Diff line Loading @@ -139,7 +139,7 @@ in options.services.netbird.server.management = { enable = mkEnableOption "Netbird Management Service"; package = mkPackageOption pkgs "netbird" { }; package = mkPackageOption pkgs "netbird-management" { }; domain = mkOption { type = str; Loading nixos/modules/services/networking/netbird/signal.nix +1 −1 Original line number Diff line number Diff line Loading @@ -31,7 +31,7 @@ in options.services.netbird.server.signal = { enable = mkEnableOption "Netbird's Signal Service"; package = mkPackageOption pkgs "netbird" { }; package = mkPackageOption pkgs "netbird-signal" { }; enableNginx = mkEnableOption "Nginx reverse-proxy for the netbird signal service"; Loading nixos/tests/netbird.nix +72 −28 Original line number Diff line number Diff line Loading @@ -7,7 +7,7 @@ ]; nodes = { clients = node = { ... }: { services.netbird.enable = true; Loading @@ -15,34 +15,16 @@ }; }; # TODO: confirm the whole solution is working end-to-end when netbird server is implemented testScript = '' start_all() def did_start(node, name, interval=0.5, timeout=10): node.wait_for_unit(f"{name}.service") node.wait_for_file(f"/var/run/{name}/sock") # `netbird status` returns a full "Disconnected" status during initialization # only after a while passes it starts returning "NeedsLogin" help message start = time.time() output = node.succeed(f"{name} status") while "Disconnected" in output and (time.time() - start) < timeout: time.sleep(interval) output = node.succeed(f"{name} status") assert "NeedsLogin" in output did_start(clients, "netbird") did_start(clients, "netbird-custom") ''; /* `netbird status` used to print `Daemon status: NeedsLogin` https://github.com/netbirdio/netbird/blob/23a14737974e3849fa86408d136cc46db8a885d0/client/cmd/status.go#L154-L164 as the first line, but now it is just: Historically waiting for the NetBird client daemon initialization helped catch number of bugs with the service, so we keep try to keep it here in as much details as it makes sense. Daemon version: 0.26.3 CLI version: 0.26.3 Management: Disconnected Initially `netbird status` returns a "Disconnected" messages: OS: linux/amd64 Daemon version: 0.54.0 CLI version: 0.54.0 Profile: default Management: Disconnected, reason: rpc error: code = FailedPrecondition desc = failed connecting to Management Service : context deadline exceeded Signal: Disconnected Relays: 0/0 Available Nameservers: 0/0 Available Loading @@ -50,7 +32,69 @@ NetBird IP: N/A Interface type: N/A Quantum resistance: false Routes: - Lazy connection: false Networks: - Forwarding rules: 0 Peers count: 0/0 Connected After a while passes it should start returning "NeedsLogin" help message. As of ~0.53.0+ in ~30 second intervals the `netbird status` instead of "NeedsLogin" it briefly (for under 2 seconds) crashes with: Error: status failed: failed connecting to Management Service : context deadline exceeded This might be related to the following log line: 2025-08-11T15:03:25Z ERRO shared/management/client/grpc.go:65: failed creating connection to Management Service: context deadline exceeded */ # TODO: confirm the whole solution is working end-to-end when netbird server is implemented testScript = '' import textwrap import time start_all() def run_with_debug(node, cmd, check=True, display=True, **kwargs): cmd = f"{cmd} 2>&1" start = time.time() ret, output = node.execute(cmd, **kwargs) duration = time.time() - start txt = f">>> {cmd=} {ret=} {duration=:.2f}:\n{textwrap.indent(output, '... ')}" if check: assert ret == 0, txt if display: print(txt) return ret, output def wait_until_rcode(node, cmd, rcode=0, retries=30, **kwargs): def check_success(_last_try): nonlocal output ret, output = run_with_debug(node, cmd, **kwargs) return ret == rcode kwargs.setdefault('check', False) output = None with node.nested(f"waiting for {cmd=} to exit with {rcode=}"): retry(check_success, retries) return output instances = ["netbird", "netbird-custom"] for name in instances: node.wait_for_unit(f"{name}.service") node.wait_for_file(f"/var/run/{name}/sock") for name in instances: wait_until_rcode(node, f"{name} status |& grep -C20 Disconnected", 0, retries=5) '' # The status used to turn into `NeedsLogin`, but recently started crashing instead. # leaving the snippets in here, in case some update goes back to the old behavior and can be tested again + lib.optionalString false '' for name in instances: #wait_until_rcode(node, f"{name} status |& grep -C20 NeedsLogin", 0, retries=20) output = wait_until_rcode(node, f"{name} status", 1, retries=61) msg = "Error: status failed: failed connecting to Management Service : context deadline exceeded" assert output.strip() == msg, f"expected {msg=}, got {output=} instead" wait_until_rcode(node, f"{name} status |& grep -C20 Disconnected", 0, retries=10) ''; } pkgs/by-name/ne/netbird-management/package.nix 0 → 100644 +5 −0 Original line number Diff line number Diff line { netbird }: netbird.override { componentName = "management"; } Loading
nixos/doc/manual/release-notes/rl-2511.section.md +2 −0 Original line number Diff line number Diff line Loading @@ -182,6 +182,8 @@ - `services.monero` now includes the `environmentFile` option for adding secrets to the Monero daemon config. - `services.netbird.server` now uses dedicated packages split out due to relicensing of server components to AGPLv3 with version `0.53.0`, - The new option [networking.ipips](#opt-networking.ipips) has been added to create IP within IP kind of tunnels (including 4in6, ip6ip6 and ipip). With the existing [networking.sits](#opt-networking.sits) option (6in4), it is now possible to create all combinations of IPv4 and IPv6 encapsulation. Loading
nixos/modules/services/networking/netbird/management.nix +1 −1 Original line number Diff line number Diff line Loading @@ -139,7 +139,7 @@ in options.services.netbird.server.management = { enable = mkEnableOption "Netbird Management Service"; package = mkPackageOption pkgs "netbird" { }; package = mkPackageOption pkgs "netbird-management" { }; domain = mkOption { type = str; Loading
nixos/modules/services/networking/netbird/signal.nix +1 −1 Original line number Diff line number Diff line Loading @@ -31,7 +31,7 @@ in options.services.netbird.server.signal = { enable = mkEnableOption "Netbird's Signal Service"; package = mkPackageOption pkgs "netbird" { }; package = mkPackageOption pkgs "netbird-signal" { }; enableNginx = mkEnableOption "Nginx reverse-proxy for the netbird signal service"; Loading
nixos/tests/netbird.nix +72 −28 Original line number Diff line number Diff line Loading @@ -7,7 +7,7 @@ ]; nodes = { clients = node = { ... }: { services.netbird.enable = true; Loading @@ -15,34 +15,16 @@ }; }; # TODO: confirm the whole solution is working end-to-end when netbird server is implemented testScript = '' start_all() def did_start(node, name, interval=0.5, timeout=10): node.wait_for_unit(f"{name}.service") node.wait_for_file(f"/var/run/{name}/sock") # `netbird status` returns a full "Disconnected" status during initialization # only after a while passes it starts returning "NeedsLogin" help message start = time.time() output = node.succeed(f"{name} status") while "Disconnected" in output and (time.time() - start) < timeout: time.sleep(interval) output = node.succeed(f"{name} status") assert "NeedsLogin" in output did_start(clients, "netbird") did_start(clients, "netbird-custom") ''; /* `netbird status` used to print `Daemon status: NeedsLogin` https://github.com/netbirdio/netbird/blob/23a14737974e3849fa86408d136cc46db8a885d0/client/cmd/status.go#L154-L164 as the first line, but now it is just: Historically waiting for the NetBird client daemon initialization helped catch number of bugs with the service, so we keep try to keep it here in as much details as it makes sense. Daemon version: 0.26.3 CLI version: 0.26.3 Management: Disconnected Initially `netbird status` returns a "Disconnected" messages: OS: linux/amd64 Daemon version: 0.54.0 CLI version: 0.54.0 Profile: default Management: Disconnected, reason: rpc error: code = FailedPrecondition desc = failed connecting to Management Service : context deadline exceeded Signal: Disconnected Relays: 0/0 Available Nameservers: 0/0 Available Loading @@ -50,7 +32,69 @@ NetBird IP: N/A Interface type: N/A Quantum resistance: false Routes: - Lazy connection: false Networks: - Forwarding rules: 0 Peers count: 0/0 Connected After a while passes it should start returning "NeedsLogin" help message. As of ~0.53.0+ in ~30 second intervals the `netbird status` instead of "NeedsLogin" it briefly (for under 2 seconds) crashes with: Error: status failed: failed connecting to Management Service : context deadline exceeded This might be related to the following log line: 2025-08-11T15:03:25Z ERRO shared/management/client/grpc.go:65: failed creating connection to Management Service: context deadline exceeded */ # TODO: confirm the whole solution is working end-to-end when netbird server is implemented testScript = '' import textwrap import time start_all() def run_with_debug(node, cmd, check=True, display=True, **kwargs): cmd = f"{cmd} 2>&1" start = time.time() ret, output = node.execute(cmd, **kwargs) duration = time.time() - start txt = f">>> {cmd=} {ret=} {duration=:.2f}:\n{textwrap.indent(output, '... ')}" if check: assert ret == 0, txt if display: print(txt) return ret, output def wait_until_rcode(node, cmd, rcode=0, retries=30, **kwargs): def check_success(_last_try): nonlocal output ret, output = run_with_debug(node, cmd, **kwargs) return ret == rcode kwargs.setdefault('check', False) output = None with node.nested(f"waiting for {cmd=} to exit with {rcode=}"): retry(check_success, retries) return output instances = ["netbird", "netbird-custom"] for name in instances: node.wait_for_unit(f"{name}.service") node.wait_for_file(f"/var/run/{name}/sock") for name in instances: wait_until_rcode(node, f"{name} status |& grep -C20 Disconnected", 0, retries=5) '' # The status used to turn into `NeedsLogin`, but recently started crashing instead. # leaving the snippets in here, in case some update goes back to the old behavior and can be tested again + lib.optionalString false '' for name in instances: #wait_until_rcode(node, f"{name} status |& grep -C20 NeedsLogin", 0, retries=20) output = wait_until_rcode(node, f"{name} status", 1, retries=61) msg = "Error: status failed: failed connecting to Management Service : context deadline exceeded" assert output.strip() == msg, f"expected {msg=}, got {output=} instead" wait_until_rcode(node, f"{name} status |& grep -C20 Disconnected", 0, retries=10) ''; }
pkgs/by-name/ne/netbird-management/package.nix 0 → 100644 +5 −0 Original line number Diff line number Diff line { netbird }: netbird.override { componentName = "management"; }