Skip to main content

ts_control/
config.rs

1use core::fmt::Debug;
2use std::net::SocketAddr;
3
4use url::Url;
5
6lazy_static::lazy_static! {
7    /// The default [`Url`] of the control plane server (aka "coordination server").
8    pub static ref DEFAULT_CONTROL_SERVER: Url = Url::parse("https://controlplane.tailscale.com/").unwrap();
9}
10
11/// Upstream-proxy wire protocol for [`ExitProxyConfig`]. Mirrors `ts_forwarder::ProxyScheme`;
12/// kept as a separate type here because `ts_control` must not depend on `ts_forwarder` (the
13/// runtime converts between them at the boundary).
14#[derive(Clone, Copy, Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize)]
15pub enum ExitProxyScheme {
16    /// SOCKS5 (RFC 1928), with optional username/password auth (RFC 1929).
17    Socks5,
18    /// HTTP `CONNECT` tunnelling, with optional `Proxy-Authorization: Basic` auth.
19    HttpConnect,
20}
21
22/// Transport-only description of an upstream proxy that exit-node egress is routed through, so a
23/// cloud exit node egresses via the proxy's (e.g. residential) IP rather than its own origin IP.
24///
25/// This is **not** read inside `ts_control`; like the other dataplane fields on [`Config`] it is
26/// carried for transport only and converted to a `ts_forwarder::ProxyConfig` by the runtime. It is
27/// only consulted when [`Config::forward_exit_egress`] is `true` (the anti-leak opt-in); on its own
28/// it changes nothing. See the proxy-egress docs in the repo's `AGENTS.md`/`CLAUDE.md`.
29#[derive(Clone, serde::Serialize, serde::Deserialize)]
30pub struct ExitProxyConfig {
31    /// Address of the upstream proxy to connect to.
32    pub addr: SocketAddr,
33    /// Wire protocol to speak to the proxy.
34    pub scheme: ExitProxyScheme,
35    /// Optional `(username, password)` credentials for proxy auth.
36    pub auth: Option<(String, String)>,
37}
38
39// Manual Debug that NEVER prints the proxy credentials, mirroring `ts_forwarder::ProxyConfig`. A
40// stray `tracing!(?cfg)` or `{:?}` must not leak the residential-proxy username/password.
41impl Debug for ExitProxyConfig {
42    fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
43        f.debug_struct("ExitProxyConfig")
44            .field("addr", &self.addr)
45            .field("scheme", &self.scheme)
46            .field("auth", &self.auth.as_ref().map(|_| "<redacted>"))
47            .finish()
48    }
49}
50
51/// How the node's **application** overlay data path is realized.
52///
53/// Defaults to [`Netstack`](TransportMode::Netstack), the userspace smoltcp netstack that needs no
54/// privileges and is the right choice for the fork's primary deployment (a privacy proxy / cloud
55/// exit node running unprivileged in a container). [`Tun`](TransportMode::Tun) instead hands the
56/// node's overlay packets to a real kernel TUN interface, for embedders that want the host OS
57/// networking stack (routes, sockets, DNS) to see the tailnet directly — closer to `tailscaled`'s
58/// model than to Go `tsnet`'s in-process netstack.
59///
60/// Like the other dataplane fields this is **not read inside `ts_control`**: it is carried for
61/// transport only and converted to a `ts_transport_tun` config by the runtime at the `ts_runtime`
62/// boundary (`ts_control` must not depend on `ts_transport_tun`). The mode governs only the
63/// application data path; it never changes the exit-node / forwarder egress path, which stays its
64/// own IPv4-only userspace netstack regardless.
65#[derive(Clone, Debug, Default, PartialEq, Eq, serde::Serialize, serde::Deserialize)]
66#[serde(rename_all = "snake_case")]
67pub enum TransportMode {
68    /// Userspace smoltcp netstack (default). No privileges required.
69    #[default]
70    Netstack,
71    /// Real kernel TUN interface. Requires privileges (root / `CAP_NET_ADMIN` on Linux) and a
72    /// platform that supports TUN (Linux `/dev/net/tun`, macOS `utun`).
73    Tun(TunConfig),
74}
75
76/// Transport-only parameters for [`TransportMode::Tun`].
77///
78/// The node's tailnet *prefix* is deliberately absent: it is assigned by control and only known at
79/// runtime, so the runtime supplies it when it builds the real `ts_transport_tun::Config`. Only the
80/// user-choosable knobs live here.
81#[derive(Clone, Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize)]
82pub struct TunConfig {
83    /// Desired interface name (e.g. `tailscale0`). `None` lets the OS pick (e.g. `utunN` on macOS).
84    #[serde(default)]
85    pub name: Option<String>,
86
87    /// Interface MTU. `None` uses the transport's default. Tailscale's overlay MTU is 1280.
88    #[serde(default)]
89    pub mtu: Option<u16>,
90}
91
92/// Default for [`Config::ephemeral`]: `true`, matching the historical behavior of this client.
93fn default_ephemeral() -> bool {
94    true
95}
96
97/// Default for [`Config::accept_dns`]: `true`, matching Go's `NewPrefs()` (`CorpDNS: true`).
98fn default_true() -> bool {
99    true
100}
101
102/// Default WireGuard persistent-keepalive interval: 25s.
103///
104/// Matches Tailscale, which sets `PersistentKeepalive = 25` on a peer when control marks it
105/// `KeepAlive=true`. 25s sits just under the ~30s lower bound for UDP NAT/firewall mapping
106/// timeouts, so the mapping (and any DERP relay path) is refreshed before it can expire.
107pub const DEFAULT_PERSISTENT_KEEPALIVE: std::time::Duration = std::time::Duration::from_secs(25);
108
109/// Default for [`Config::persistent_keepalive_interval`]: `Some(25s)`
110/// ([`DEFAULT_PERSISTENT_KEEPALIVE`]). On by default so a relayed, idle session keeps its path warm
111/// and doesn't wedge the next dial.
112fn default_persistent_keepalive() -> Option<std::time::Duration> {
113    Some(DEFAULT_PERSISTENT_KEEPALIVE)
114}
115
116/// Configuration for the control server.
117#[derive(Clone, serde::Serialize, serde::Deserialize)]
118pub struct Config {
119    /// The URL of the control server to connect to.
120    pub server_url: Url,
121
122    /// The hostname of the current node.
123    pub hostname: Option<String>,
124
125    /// A name for this type of client.
126    ///
127    /// This will be reported to the control server in the `HostInfo.App` field.
128    pub client_name: Option<String>,
129
130    /// Tags to request from the control server (`--advertise-tags` / `AdvertiseTags` in the Go
131    /// client).
132    ///
133    /// Sent as `HostInfo.RequestTags` on registration and on every map request, so a
134    /// tag-keyed control ACL (e.g. a self-hosted control plane's route auto-approver) can match this node. Each
135    /// entry is a full tag string including the `tag:` prefix (e.g. `tag:exit`). Defaults to
136    /// empty (claim no tags); an empty set omits the wire field entirely.
137    #[serde(default)]
138    pub tags: Vec<String>,
139
140    /// Whether this node registers as *ephemeral* (`--ephemeral` / `Ephemeral` in the Go client).
141    ///
142    /// An ephemeral node is garbage-collected by the control server shortly after it
143    /// disconnects. That is the right default for short-lived clients, but a persistent exit node
144    /// or subnet router must set this to `false` or it will be GC'd out of the tailnet while
145    /// briefly offline. Defaults to `true` to match the historical behavior of this client.
146    #[serde(default = "default_ephemeral")]
147    pub ephemeral: bool,
148
149    /// Whether to accept subnet routes advertised by peers (`--accept-routes` / `RouteAll` in the
150    /// Go client).
151    ///
152    /// When `false` (the default, matching the Go client on Linux/server platforms and our
153    /// fail-closed posture), only each peer's own tailnet addresses are routed; larger advertised
154    /// subnet routes are ignored. When `true`, traffic destined for an accepted subnet egresses
155    /// via the advertising peer.
156    ///
157    /// This is a client-side preference and is not read inside `ts_control`: control always sends
158    /// the full set of advertised routes, and the runtime trims them. It is carried here only to
159    /// be threaded through to the runtime's route filter.
160    #[serde(default)]
161    pub accept_routes: bool,
162
163    /// Whether to accept the tailnet's DNS configuration (MagicDNS + the pushed resolvers/search
164    /// domains) — `--accept-dns` / the `CorpDNS` pref in the Go client. **Defaults to `true`**, matching
165    /// Go's `NewPrefs()` (`CorpDNS: true`).
166    ///
167    /// When `true`, the MagicDNS responder serves the control-pushed [`DnsConfig`](crate::DnsConfig)
168    /// (overlay-name answers + split-DNS routes + recursive forwarding). When `false`, the node
169    /// **ignores the pushed DNS config** and the responder serves nothing (every query is `REFUSED`),
170    /// mirroring Go applying an essentially-empty `dns.Config` when `CorpDNS` is off — so a node can
171    /// join the tailnet for connectivity without taking over its DNS.
172    ///
173    /// Like [`accept_routes`](Config::accept_routes), this is a client-side preference not read inside
174    /// `ts_control` (control always pushes the full `DNSConfig`; the runtime decides whether to honor
175    /// it); it is carried here only to be threaded through to the runtime's MagicDNS responder, and is
176    /// runtime-settable via `Device::set_accept_dns` (the analog of `tailscale set --accept-dns`).
177    #[serde(default = "default_true")]
178    pub accept_dns: bool,
179
180    /// Which peer (if any) to use as an exit node (`--exit-node` / `ExitNodeID` in the Go client).
181    ///
182    /// The selector may name the peer by stable id, tailnet IP, or MagicDNS name (see
183    /// [`ExitNodeSelector`](crate::ExitNodeSelector)); it is resolved against the live peer set on
184    /// every route rebuild, so an IP/name selection follows the peer across netmap changes. When
185    /// set and resolvable, the selected peer's advertised default route (`0.0.0.0/0` / `::/0`) is
186    /// installed so internet-bound traffic egresses through it. When `None` (the default) or
187    /// unresolvable, no peer receives a default route and internet-bound traffic is dropped
188    /// (fail-closed).
189    ///
190    /// Like [`accept_routes`](Config::accept_routes), this is a client-side preference not read
191    /// inside `ts_control`; it is carried here only to be threaded through to the runtime's route
192    /// filter.
193    ///
194    /// **Full-tunnel exit vs. just reaching a peer's port — leave this `None` unless you mean
195    /// full-tunnel.** Set `exit_node` *only* to route **all** internet-bound traffic through a peer
196    /// that advertises a default route (`advertise_exit_node`). To merely **reach a specific peer's
197    /// service over the tailnet** — e.g. `Device::tcp_connect` to its `100.x.y.z:1080` — you do
198    /// **not** set `exit_node` at all; direct peer dials need no exit node. Setting `exit_node` to a
199    /// peer that is only a selective CONNECT proxy (advertises no `0.0.0.0/0`) leaves egress
200    /// fail-closed and logs a warning that internet-bound traffic is dropped — which looks like a
201    /// failure but is just "that peer isn't a full-tunnel exit." If you saw that warning while only
202    /// trying to dial a peer's port, the fix is to unset `exit_node`.
203    #[serde(default)]
204    pub exit_node: Option<crate::ExitNodeSelector>,
205
206    /// Subnet routes to advertise to the control server (`--advertise-routes` / `RoutableIPs` in
207    /// the Go client).
208    ///
209    /// Unlike [`accept_routes`](Config::accept_routes)/[`exit_node`](Config::exit_node), this field
210    /// *is* read inside `ts_control`: it populates `HostInfo.RoutableIPs` on every map request so
211    /// the control server can grant this node as a subnet router. Defaults to empty (advertise
212    /// nothing — fail-closed). Only IPv4 prefixes are advertised; IPv6 prefixes are dropped to
213    /// uphold the IPv6-off posture (advertising a route we won't forward would be a black hole).
214    #[serde(default)]
215    pub advertise_routes: Vec<ipnet::IpNet>,
216
217    /// Whether to advertise this node as an exit node (`--advertise-exit-node` in the Go client).
218    ///
219    /// When `true`, the default route `0.0.0.0/0` is added to the advertised
220    /// [`routable_ips`](Config::advertise_routes) so the control server can grant this node as an
221    /// exit node, after which other peers may egress internet-bound traffic through our real IP.
222    /// Defaults to `false` (fail-closed): being an exit node means *other* peers' traffic leaves
223    /// via our real origin IP, so it must be explicit opt-in. IPv6 (`::/0`) is never advertised,
224    /// per the IPv6-off posture.
225    #[serde(default)]
226    pub advertise_exit_node: bool,
227
228    /// TCP ports the inbound forwarder accepts and splices to real OS sockets for every advertised
229    /// route (`advertise_routes` / `advertise_exit_node`).
230    ///
231    /// smoltcp has no all-port accept mode (see the `ts_forwarder` crate docs), so the forwarder
232    /// forwards a configured set of ports rather than the full 1–65535 range. Defaults to empty: a
233    /// node that advertises routes but configures no forward ports accepts inbound flows into its
234    /// dedicated forwarder netstack but forwards none of them (fail-closed — nothing is dialed).
235    #[serde(default)]
236    pub forward_tcp_ports: Vec<u16>,
237
238    /// UDP ports the inbound forwarder accepts and splices to real OS sockets for every advertised
239    /// route. See [`forward_tcp_ports`](Config::forward_tcp_ports); defaults to empty.
240    #[serde(default)]
241    pub forward_udp_ports: Vec<u16>,
242
243    /// Forward **all** TCP/UDP ports (1–65535) on every advertised route, like a Go subnet router
244    /// (`tailscale up --advertise-routes` forwards all ports), instead of the explicit
245    /// [`forward_tcp_ports`](Config::forward_tcp_ports) /
246    /// [`forward_udp_ports`](Config::forward_udp_ports) sets.
247    ///
248    /// smoltcp cannot wildcard-port-accept, so all-port mode is implemented with an on-demand
249    /// per-port listener manager driven by a raw-socket port observer on the dedicated forwarder
250    /// netstack (see the `ts_forwarder` crate docs). When `true`, the explicit port sets are
251    /// ignored. Anti-leak is unchanged: every flow still routes through the same
252    /// `RouteTable`→dialer chokepoint, so [`forward_exit_egress`](Config::forward_exit_egress) still
253    /// governs exit-node egress. Defaults to `false`.
254    #[serde(default)]
255    pub forward_all_ports: bool,
256
257    /// Whether exit-node (`0.0.0.0/0`) inbound flows are actually egressed via **this host's real
258    /// origin IP**.
259    ///
260    /// This is the anti-leak opt-in, kept separate from
261    /// [`advertise_exit_node`](Config::advertise_exit_node): advertising the default route only
262    /// makes control *offer* this node as an exit; it does not by itself egress a peer's traffic.
263    /// When `false` (the default, fail-closed), the forwarder uses a dialer that **structurally
264    /// refuses** exit-node egress — a `0.0.0.0/0` flow is dropped at dial time, never leaked out our
265    /// real IP. Set to `true` only on a node whose real IP *is* the intended egress (e.g. a
266    /// residential exit), never on a node whose host IP must stay hidden (e.g. a cloud VPS). Subnet
267    /// routes are dialed identically regardless of this flag.
268    #[serde(default)]
269    pub forward_exit_egress: bool,
270
271    /// Shields-up (Go `ipn` prefs `ShieldsUp`): when `true`, refuse all **inbound** connections from
272    /// peers that terminate on this node — the packet filter drops inbound packets aimed at this
273    /// node's own addresses. Replies to connections this node itself initiated, and forwarded
274    /// subnet/exit transit, are unaffected (the deny is scoped to self-destined packets; see
275    /// `ts_packetfilter::ShieldsUpFilter`). Transport-only client preference — `ts_control` never
276    /// reads it; the runtime's packet-filter updater consumes it. Defaults to `false`.
277    #[serde(default)]
278    pub block_incoming: bool,
279
280    /// Optional upstream proxy that exit-node egress is routed through, so the node egresses via
281    /// the proxy's IP rather than its own origin IP.
282    ///
283    /// Only consulted when [`forward_exit_egress`](Config::forward_exit_egress) is `true`. When
284    /// set, the runtime wires the forwarder with a proxy dialer (SOCKS5 / HTTP `CONNECT`) that
285    /// **fails closed** — any proxy connect or handshake failure drops the flow rather than falling
286    /// back to a direct host-IP dial, so the real origin IP never leaks. When `None` (the default)
287    /// and exit egress is enabled, egress uses this host's real IP (`HostExitDialer`).
288    ///
289    /// Like the other dataplane fields, this is a client-side preference not read inside
290    /// `ts_control`; it is carried here only to be threaded through to the runtime's dialer
291    /// selection. This is a product capability (residential-proxy egress) beyond strict tsnet
292    /// parity — see the repo's `AGENTS.md`/`CLAUDE.md`.
293    #[serde(default)]
294    pub exit_proxy: Option<ExitProxyConfig>,
295
296    /// The IPv4 peerAPI port this node binds to serve exit-node DoH (DNS-over-HTTPS) proxying for
297    /// peers that select it as their exit node (`peerapi4` + `peerapi-dns-proxy` services).
298    ///
299    /// When `Some(port)`, the runtime binds a peerAPI DoH server on this host's overlay IPv4
300    /// address at `port`, and registration / map requests advertise both the `peerapi4` service
301    /// (at `port`) and the `peerapi-dns-proxy` service (Go quirk: its advertised port is always
302    /// `1`) so peers know they can delegate DNS to us. When `None` (the default, fail-closed), no
303    /// peerAPI is run and no services are advertised — this node never offers DNS proxying.
304    ///
305    /// The DoH server always answers authoritative/overlay records (MagicDNS peer names,
306    /// `ExtraRecords`, PTR); *recursive* resolution to real upstream resolvers is gated separately
307    /// behind [`forward_exit_egress`](Config::forward_exit_egress), so a cloud exit node can serve
308    /// overlay DNS without ever exposing its real origin IP via a recursive lookup.
309    #[serde(default)]
310    pub peerapi_port: Option<u16>,
311
312    /// Filesystem directory that received Taildrop files land in, or `None` to disable Taildrop
313    /// (the default, fail-closed).
314    ///
315    /// When `Some(dir)` **and** [`peerapi_port`](Config::peerapi_port) is also set, the runtime
316    /// serves the Taildrop peerAPI route `PUT /v0/put/<name>` on the shared peerAPI listener, and
317    /// incoming files are written under `dir` (created if absent). When `None`, no Taildrop server
318    /// is run — a peer's `PUT` is refused. This is a pure on-disk destination: like the other
319    /// dataplane fields it is not read inside `ts_control`; it is carried here only to be threaded
320    /// into the runtime, which constructs the file store from it.
321    ///
322    /// Independently of the network server, the embedder consumes received files via the
323    /// `Device::taildrop_*` methods (Go exposes these over LocalAPI; this fork exposes them on the
324    /// device). With no `peerapi_port`, the store still exists for those read APIs but no peer can
325    /// deliver to it.
326    #[serde(default)]
327    pub taildrop_dir: Option<std::path::PathBuf>,
328
329    /// Per-direction TCP send/receive buffer size (bytes) for the userspace netstack, or `None` to
330    /// use the netstack default (256 KiB per direction, ~512 KiB per socket).
331    ///
332    /// smoltcp has no window auto-tuning, so this is the hard cap on a single flow's
333    /// bandwidth-delay product; raising it helps large model-API responses on high-RTT links, at
334    /// the cost of more memory per concurrent socket (each socket allocates this size for both rx
335    /// and tx). Like the other dataplane fields, this is a client-side preference not read inside
336    /// `ts_control`; it is carried here only to be threaded into the runtime's netstack
337    /// configuration.
338    #[serde(default)]
339    pub tcp_buffer_size: Option<usize>,
340
341    /// Whether IPv6 is enabled on the tailnet overlay. Defaults to `false` (IPv4-only).
342    ///
343    /// Like the other dataplane fields, this is a client-side preference not read inside
344    /// `ts_control`; it is carried here only to be threaded into the runtime's underlay socket,
345    /// disco candidate filter, netstack address assignment, and MagicDNS AAAA handling. It governs
346    /// only the overlay and never the exit-node / forwarder egress path, which stays IPv4-only
347    /// regardless to uphold the real-origin-IP isolation invariant.
348    #[serde(default)]
349    pub enable_ipv6: bool,
350
351    /// Whether the runtime runs an internal OS network-link monitor that auto-re-binds + re-probes
352    /// connectivity on a link change (Wi-Fi switch, sleep/wake, default-route change). Defaults to
353    /// `false` (no monitor — the embedder drives `Device::rebind` itself).
354    ///
355    /// Like the other dataplane fields, this is a client-side preference not read inside
356    /// `ts_control`; it is carried here only to be threaded into the runtime, which (when set, and
357    /// when built with the `network-monitor` feature) spawns a `NetmonSupervisor`. It is off by
358    /// default to preserve the fork's pure-engine posture (it is an engine, not a daemon): with it
359    /// off, the runtime starts zero monitor threads/sockets and behaves byte-for-byte as before.
360    #[serde(default)]
361    pub network_monitor: bool,
362
363    /// The fixed UDP port magicsock binds for WireGuard + disco, or `None` for an OS-chosen
364    /// ephemeral port (Go `tailscaled --port` / `ListenPort`). Defaults to `None`.
365    ///
366    /// Like the other dataplane fields, this is a client-side preference not read inside
367    /// `ts_control`; it is carried here only to be threaded into the runtime's *initial* underlay
368    /// socket bind. `None` binds `0.0.0.0:0` (ephemeral, today's behavior); `Some(p)` pins port `p`
369    /// with an ephemeral fallback if it is already taken (a port collision never fails bring-up).
370    /// Governs only the bound port, never the bind family — the IPv4-only-by-default, fail-closed
371    /// underlay posture is unchanged.
372    #[serde(default)]
373    pub wireguard_listen_port: Option<u16>,
374
375    /// WireGuard persistent-keepalive interval applied to every peer, or `None` to disable persistent
376    /// keepalives (`PersistentKeepalive`; Tailscale uses 25s).
377    ///
378    /// When `Some(interval)`, each peer emits an empty authenticated keepalive every `interval` of
379    /// outbound silence, holding the (typically DERP-relayed) path/NAT mapping warm so an idle
380    /// session doesn't age past expiry and wedge the next dial — the failure this fork's primary
381    /// userspace-netstack deployment hits, where the relay is the only path to a peer. Unlike the
382    /// reactive WireGuard §6.5 keepalive (armed only by inbound traffic), this re-arms unconditionally
383    /// and fires on a fully idle tunnel; the empty packet does not advance the session's
384    /// rotation/expiry timers, so a genuinely dead peer is still detected. Defaults to `Some(25s)`
385    /// ([`DEFAULT_PERSISTENT_KEEPALIVE`]). Like the other dataplane fields it is not read inside
386    /// `ts_control`; it is carried here only to be threaded into the runtime's dataplane actor.
387    #[serde(default = "default_persistent_keepalive")]
388    pub persistent_keepalive_interval: Option<std::time::Duration>,
389
390    /// How the application overlay data path is realized: userspace netstack (default) or a real
391    /// kernel TUN interface. See [`TransportMode`].
392    ///
393    /// Like the other dataplane fields, this is a client-side preference not read inside
394    /// `ts_control`; it is carried here only to be threaded into the runtime, which builds either a
395    /// netstack actor or a TUN transport from it. `ts_control` must not depend on `ts_transport_tun`.
396    #[serde(default)]
397    pub transport_mode: TransportMode,
398
399    /// Whether to ask control to wire this node up server-side for Tailscale Funnel
400    /// (`HostInfo.WireIngress`, the capver-113 client→control Funnel signal), even when no Funnel
401    /// endpoint is currently active.
402    ///
403    /// Unlike the dataplane fields above, this one *is* read inside `ts_control`: it sets
404    /// `HostInfo.WireIngress` on registration and the streaming map request, asking control to
405    /// provision the DNS / ingress records a Funnel node needs so a later `serve`/funnel session
406    /// works immediately. It mirrors Go `tsnet`'s "would like to be wired up for Funnel" signal.
407    ///
408    /// This fork cannot yet *terminate* public Funnel ingress — [`crate::listen_funnel`] is
409    /// fail-closed (no client-side ACME engine, and a self-hosted control plane provides no public
410    /// ingress relay). So `HostInfo.IngressEnabled` (Funnel endpoints actually live) is never set;
411    /// only `WireIngress` is, and only when this flag is `true`. Defaults to `false` (fail-closed):
412    /// a node requests Funnel wiring only when explicitly opted in.
413    #[serde(default)]
414    pub wire_ingress: bool,
415
416    /// Live signal that this node currently has an active Funnel ingress listener
417    /// (`Device::listen_funnel` was called and its listener is up), driving `HostInfo.IngressEnabled`
418    /// on the streaming map request.
419    ///
420    /// Unlike [`wire_ingress`](Self::wire_ingress) (a static "please provision Funnel records" hint),
421    /// this is a *dynamic* flag: the runtime flips it `true` when a funnel listener starts serving and
422    /// back to `false` when it stops, so the next map request advertises `IngressEnabled` accordingly
423    /// (Go sets `HostInfo.IngressEnabled` only while Funnel endpoints are actually live, and
424    /// `IngressEnabled` implies `WireIngress`). Shared (`Arc`) with the runtime so the device can flip
425    /// it without rebuilding the config. Defaults to a fresh `false` (fail-closed: no live endpoint).
426    /// Not serialized — it is process-local runtime state, not persisted configuration.
427    #[serde(skip, default)]
428    pub ingress_active: std::sync::Arc<std::sync::atomic::AtomicBool>,
429
430    /// VIP services this node advertises that it **hosts** (`svc:<dns-label>` names), the
431    /// advertise side of Tailscale VIP services (Go `tsnet`'s `Hostinfo.ServicesHash` +
432    /// c2n `GET /vip-services`).
433    ///
434    /// Each entry is a full `svc:`-prefixed service name. This field *is* read inside `ts_control`:
435    /// the valid names ([`validate_service_name`](crate::validate_service_name) is applied
436    /// fail-closed; malformed names are dropped and logged) are hashed into `HostInfo.ServicesHash`
437    /// on every map request, and answered when control fetches the list via the c2n
438    /// `/vip-services` endpoint. Defaults to empty: with no entries the hash is `""` and behavior is
439    /// byte-for-byte the historical non-advertising path. Hosting a service additionally requires
440    /// control to assign it a VIP and the node to be tagged (the *consume* side, unchanged here).
441    #[serde(default)]
442    pub advertise_services: Vec<String>,
443
444    /// Whether to advertise this node as an **app connector** (Go `Prefs.AppConnector.Advertise` /
445    /// `tailscale set --advertise-connector`). When `true`, this *is* read inside `ts_control`: it
446    /// sets `HostInfo.AppConnector = Some(true)` on registration and every map request, mirroring Go's
447    /// `applyPrefsToHostinfoLocked` (`hi.AppConnector.Set(prefs.AppConnector().Advertise)`).
448    ///
449    /// Advertising the bool is the **faithful engine minimum** — exactly the boundary Go draws. The
450    /// actual app-connector *data path* (control pushing the connector's domain routes, the 4via6
451    /// domain→route mapping, the per-domain DNS observation that learns target IPs) is a separate
452    /// subsystem this fork does not implement; advertising the capability without that data path is
453    /// identical in effect to Go advertising it before control has assigned any domains. Defaults to
454    /// `false` (fail-closed): a node offers itself as an app connector only when explicitly opted in.
455    #[serde(default)]
456    pub advertise_app_connector: bool,
457
458    /// Whether this node opts in to control-console-triggered auto-updates (Go
459    /// `Prefs.AutoUpdate.Apply` / `tailscale set --auto-update`). When `Some(true)`, this *is* read
460    /// inside `ts_control`: it sets `HostInfo.AllowsUpdate = true` on registration and every map
461    /// request, mirroring Go's `applyPrefsToHostinfoLocked`
462    /// (`hi.AllowsUpdate = … || prefs.AutoUpdate().Apply.EqualBool(true)`), so the admin console knows
463    /// the node accepts remote update triggers.
464    ///
465    /// Advertising the bool is the faithful engine minimum: this fork runs **no updater** (it is an
466    /// embeddable engine, not a packaged daemon), so it never *applies* an update — the actual
467    /// self-update machinery is a daemon/OS-package concern. `Some(false)` and `None` both leave
468    /// `AllowsUpdate` at its default `false` (the node advertises it does not accept remote updates);
469    /// the tri-state mirrors Go's `opt.Bool` (unset vs explicitly-off vs on). Defaults to `None`.
470    #[serde(default)]
471    pub auto_update_apply: Option<bool>,
472
473    /// Whether this node's (hypothetical) background updater should *check* for available updates
474    /// (Go `Prefs.AutoUpdate.Check`). **Carried pref only — not read inside `ts_control` and never
475    /// sent to control.** In Go this gates a purely local background update-check loop in the daemon;
476    /// it is not part of `Hostinfo` and never crosses the control wire, so storing it is the faithful
477    /// mirror of tsnet state. This fork has no updater (engine, not daemon), so the pref is carried
478    /// for a downstream daemon to consult and has no effect inside the engine. Defaults to `false`.
479    #[serde(default)]
480    pub auto_update_check: bool,
481
482    /// The OS username permitted to operate this node over the local API (Go `Prefs.OperatorUser` /
483    /// `tailscale set --operator`). **Carried pref only — not read inside `ts_control` and never sent
484    /// to control.** In Go this is purely a daemon-side LocalAPI authorization check (which Unix uid
485    /// may drive the daemon without root); it never touches the control protocol. Storing it is the
486    /// faithful mirror of tsnet state — a downstream daemon that exposes a local API consults it; the
487    /// engine itself has no local API to gate. Defaults to `None` (no operator delegated).
488    #[serde(default)]
489    pub operator_user: Option<String>,
490
491    /// A local display label for this node's profile (Go `Prefs.ProfileName`, set by
492    /// `tailscale switch`/profile management). **Carried pref only — not read inside `ts_control` and
493    /// never sent to control.** In Go this is a client-local cosmetic name for the login profile; it
494    /// is never advertised in `Hostinfo` (distinct from the `Hostinfo.Hostname` the node requests).
495    /// Storing it faithfully mirrors tsnet state for a downstream daemon's profile UI; the engine
496    /// makes no use of it. Defaults to `None`.
497    #[serde(default)]
498    pub node_nickname: Option<String>,
499
500    /// Whether device posture identity collection is enabled (Go `Prefs.PostureChecking` /
501    /// `tailscale set --posture-checking`). **Carried pref only — not read inside `ts_control` and
502    /// never sent to control.**
503    ///
504    /// There is deliberately **no `Hostinfo.PostureChecking` field to wire it to**: posture is a
505    /// control-to-node (c2n) *pull* mechanism — control requests posture attributes (serial numbers,
506    /// etc.) from the node on demand — which this fork does not implement. Storing the pref is
507    /// therefore the faithful mirror: with no c2n posture responder, control simply never pulls
508    /// posture identity, which is byte-for-byte identical to the posture-disabled case. A downstream
509    /// daemon that implements the c2n posture endpoint consults this pref to decide whether to answer.
510    /// Defaults to `false` (fail-closed: no posture identity collected).
511    #[serde(default)]
512    pub posture_checking: bool,
513
514    /// Whether this node runs a local web client (Go `Prefs.RunWebClient` /
515    /// `tailscale set --webclient`). **Carried pref only — not read inside `ts_control` and never
516    /// sent to control.** In Go this gates a daemon-hosted local web-client HTTP server (the
517    /// device-management web UI on `100.x:5252`); it is a separate subsystem, not advertised in
518    /// `Hostinfo`. This fork has no web-client server, so storing the pref faithfully mirrors tsnet
519    /// state for a downstream daemon that does; the engine never acts on it. Defaults to `false`.
520    #[serde(default)]
521    pub run_web_client: bool,
522
523    /// Whether a peer using this node as an exit node may also reach this node's **local LAN**
524    /// (Go `Prefs.ExitNodeAllowLANAccess` / `tailscale set --exit-node-allow-lan-access`).
525    /// **Carried pref only for now — not read inside `ts_control` and never sent to control.**
526    ///
527    /// In Go this is an **OS-router route-shaping** flag: when acting as an exit node it controls
528    /// whether the host router excludes the local LAN ranges from the routes pulled through the
529    /// tunnel. On a platform with no host router it has "no effect" — and this fork's default data
530    /// path is the userspace netstack with no host-route layer, so there is nothing to shape today.
531    /// The pref is stored so a downstream daemon (or a future host-route layer in this engine) can
532    /// consume it; until such a layer exists it is inert. It is never advertised to control. Defaults
533    /// to `false`.
534    #[serde(default)]
535    pub exit_node_allow_lan_access: bool,
536
537    /// Whether to automatically re-authenticate (rotate the node key + re-register with the stored
538    /// auth key, Go `doLogin`) when control reports this node's node key has expired, instead of
539    /// going terminally offline.
540    ///
541    /// Defaults to `true`: an auth-key-registered node whose key expires recovers itself without
542    /// human intervention — the common reusable-auth-key case (a persistent exit node / subnet
543    /// router) self-heals. Set to `false` for the most conservative posture (the historical behavior:
544    /// an expired key surfaces the terminal "expired" state and the node stays offline until
545    /// re-paired). Auto-reauth is additionally gated at runtime on a usable auth key being retained
546    /// and Tailnet Lock NOT being enforced (a rotation on a locked tailnet would install an unsigned
547    /// key); see the runtime's `expiry_action`. A one-shot auth key (already consumed by the first
548    /// registration) cannot re-register and degrades to the terminal state regardless of this flag.
549    ///
550    /// Like the client-preference fields, this is **not read inside `ts_control`**: it is carried for
551    /// transport only and consulted by the runtime's self-node expiry handler.
552    #[serde(default = "default_true")]
553    pub reauth_on_expiry: bool,
554
555    /// Allow fetching the control server's machine public key (`GET /key`) over plain **http** when
556    /// the [`server_url`](Config::server_url) is itself `http://`.
557    ///
558    /// By default (`false`) the `/key` fetch is always upgraded to `https`, even when the control
559    /// URL is `http://` — matching Tailscale's posture that the unauthenticated key bootstrap must
560    /// be TLS-protected. That upgrade makes registration **fail** against a control plane that only
561    /// serves plain http (e.g. a self-hosted Headscale exposed over a `http://host:port` LAN
562    /// endpoint / NodePort with no TLS), even though the rest of the control connection already
563    /// honors the `http` scheme. Set this to `true` for such a deployment to fetch `/key` over the
564    /// same `http` scheme as the control URL.
565    ///
566    /// Security: only enable this when you control both ends and the control plane is reachable
567    /// over a trusted network path — an on-path attacker could otherwise substitute the control
568    /// key. It has no effect when `server_url` is `https://` (the fetch stays https regardless).
569    /// Fail-closed default is `false`.
570    #[serde(default)]
571    pub allow_http_key_fetch: bool,
572}
573
574impl Config {
575    /// Get the full client name as a string.
576    ///
577    /// This takes the form `tailscale-rs ({client_name})`, where the parenthetical is only
578    /// provided if self.client_name is set.
579    pub fn format_client_name(&self) -> String {
580        let mut full_name = "tailscale-rs".to_owned();
581        if let Some(client_name) = &self.client_name {
582            full_name.push_str(&format!(" ({client_name})"));
583        }
584
585        full_name
586    }
587
588    /// Compute the set of IP prefixes to advertise in `HostInfo.RoutableIPs`, combining
589    /// [`advertise_routes`](Config::advertise_routes) with the exit-node default route when
590    /// [`advertise_exit_node`](Config::advertise_exit_node) is set.
591    ///
592    /// IPv6 prefixes are filtered out (IPv6-off posture): we never forward IPv6, so advertising an
593    /// IPv6 route would create a black hole. The exit-node default route is therefore `0.0.0.0/0`
594    /// only, never `::/0`. The result is deduplicated and order-preserving; an empty result means
595    /// "advertise nothing", and callers omit the wire field entirely.
596    pub fn advertised_routes(&self) -> Vec<ipnet::IpNet> {
597        let mut routes: Vec<ipnet::IpNet> = Vec::new();
598        let mut push_unique = |net: ipnet::IpNet| {
599            if !routes.contains(&net) {
600                routes.push(net);
601            }
602        };
603
604        for net in &self.advertise_routes {
605            // IPv6-off: drop v6 prefixes so we never advertise a route we won't forward.
606            if matches!(net, ipnet::IpNet::V4(_)) {
607                push_unique(*net);
608            } else {
609                tracing::warn!(prefix = %net, "dropping IPv6 advertise_routes prefix (IPv6-off posture)");
610            }
611        }
612
613        if self.advertise_exit_node {
614            let default_v4 = ipnet::IpNet::V4(
615                ipnet::Ipv4Net::new(core::net::Ipv4Addr::UNSPECIFIED, 0)
616                    .expect("0.0.0.0/0 is a valid prefix"),
617            );
618            push_unique(default_v4);
619        }
620
621        routes
622    }
623
624    /// The services to advertise in `HostInfo.Services`, derived from
625    /// [`peerapi_port`](Config::peerapi_port).
626    ///
627    /// When a peerAPI port is configured, we advertise the `peerapi4` service at that port plus the
628    /// `peerapi-dns-proxy` service (whose advertised port is always `1`, matching the Go client's
629    /// quirk) so peers learn they can delegate exit-node DNS to us. When `None`, the result is empty
630    /// and callers omit the `HostInfo.Services` wire field entirely (advertise no services). IPv6
631    /// peerAPI (`peerapi6`) is never advertised, per the IPv6-off posture.
632    pub fn advertised_services(&self) -> Vec<ts_control_serde::Service<'static>> {
633        use ts_control_serde::{Service, ServiceProto};
634
635        let Some(port) = self.peerapi_port else {
636            return Vec::new();
637        };
638
639        vec![
640            Service {
641                proto: ServiceProto::PeerApi4,
642                port,
643                description: "tailscale-rs".into(),
644            },
645            Service {
646                // Go quirk: the peerapi-dns-proxy service always advertises port 1.
647                proto: ServiceProto::PeerApiDnsProxy,
648                port: 1,
649                description: "tailscale-rs".into(),
650            },
651        ]
652    }
653
654    /// The validated set of VIP services this node advertises that it hosts, derived from
655    /// [`advertise_services`](Config::advertise_services).
656    ///
657    /// Each configured name is validated with
658    /// [`validate_service_name`](crate::validate_service_name) (fail-closed: a name that is not a
659    /// well-formed `svc:<dns-label>` is dropped with a warning, never advertised). Each surviving
660    /// service is advertised on **all ports** (a single `0/0..=65535`
661    /// [`ProtoPortRange`](ts_control_serde::ProtoPortRange), matching
662    /// Go's default `ServicePortRange()` when no explicit ports are configured) and marked active.
663    /// The result is the canonical input to both [`services_hash`] and the c2n `/vip-services`
664    /// response. An empty config yields an empty `Vec` (advertise nothing — the hash is `""`).
665    pub fn advertised_vip_services(&self) -> Vec<ts_control_serde::VipServiceOwned> {
666        use ts_control_serde::{ProtoPortRange, VipServiceOwned};
667
668        self.advertise_services
669            .iter()
670            .filter_map(|name| {
671                if crate::validate_service_name(name).is_none() {
672                    tracing::warn!(
673                        service = %name,
674                        "dropping invalid advertise_services name (expected svc:<dns-label>)"
675                    );
676                    return None;
677                }
678                Some(VipServiceOwned {
679                    name: name.clone(),
680                    // All ports: proto 0 (all protocols), full 0..=65535 span — Go's default
681                    // ServicePortRange() for a service with no explicit port restriction.
682                    ports: vec![ProtoPortRange {
683                        proto: 0,
684                        first: 0,
685                        last: 65535,
686                    }],
687                    active: true,
688                })
689            })
690            .collect()
691    }
692}
693
694/// Compute the `HostInfo.ServicesHash` for a node's advertised VIP services, mirroring Go's
695/// `vipServiceHash`.
696///
697/// The services are sorted by name, serialized to canonical (whitespace-free) JSON as a
698/// [`ts_control_serde::VipServiceOwned`] list, SHA-256'd, and hex-encoded. An empty list hashes to
699/// the empty string `""` (the "no services advertised" sentinel, which omits/clears the wire
700/// field). The hash is byte-stable and order-independent: the same set in any input order yields the
701/// same value, so control reliably refetches only on a genuine change.
702///
703/// Uses `ring`'s SHA-256 (the same crypto backend the rest of the stack links — no aws-lc-rs /
704/// openssl is introduced).
705pub fn services_hash(services: &[ts_control_serde::VipServiceOwned]) -> String {
706    if services.is_empty() {
707        return String::new();
708    }
709
710    let mut sorted = services.to_vec();
711    sorted.sort_by(|a, b| a.name.cmp(&b.name));
712
713    // Canonical, whitespace-free JSON so the digest is byte-stable across builds.
714    let json = serde_json::to_vec(&sorted).expect("VipServiceOwned list always serializes");
715    let digest = ring::digest::digest(&ring::digest::SHA256, &json);
716
717    let mut hex = String::with_capacity(digest.as_ref().len() * 2);
718    for byte in digest.as_ref() {
719        hex.push_str(&format!("{byte:02x}"));
720    }
721    hex
722}
723
724impl Debug for Config {
725    fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
726        f.debug_struct("Config")
727            .field("hostname", &self.hostname)
728            .field("server_url", &self.server_url.as_str())
729            .field("client_name", &self.client_name)
730            .finish()
731    }
732}
733
734impl Default for Config {
735    fn default() -> Self {
736        Self {
737            server_url: DEFAULT_CONTROL_SERVER.clone(),
738            hostname: gethostname::gethostname().into_string().ok(),
739            client_name: None,
740            tags: Default::default(),
741            ephemeral: default_ephemeral(),
742            accept_routes: false,
743            accept_dns: default_true(),
744            exit_node: None,
745            advertise_routes: Vec::new(),
746            advertise_exit_node: false,
747            forward_tcp_ports: Vec::new(),
748            forward_udp_ports: Vec::new(),
749            forward_all_ports: false,
750            forward_exit_egress: false,
751            block_incoming: false,
752            exit_proxy: None,
753            peerapi_port: None,
754            taildrop_dir: None,
755            tcp_buffer_size: None,
756            enable_ipv6: false,
757            network_monitor: false,
758            wireguard_listen_port: None,
759            persistent_keepalive_interval: default_persistent_keepalive(),
760            transport_mode: TransportMode::default(),
761            wire_ingress: false,
762            ingress_active: std::sync::Arc::new(std::sync::atomic::AtomicBool::new(false)),
763            advertise_services: Vec::new(),
764            advertise_app_connector: false,
765            auto_update_apply: None,
766            auto_update_check: false,
767            operator_user: None,
768            node_nickname: None,
769            posture_checking: false,
770            run_web_client: false,
771            exit_node_allow_lan_access: false,
772            reauth_on_expiry: default_true(),
773            allow_http_key_fetch: false,
774        }
775    }
776}
777
778#[cfg(test)]
779mod tests {
780    use super::*;
781
782    fn v4(s: &str) -> ipnet::IpNet {
783        ipnet::IpNet::V4(s.parse().unwrap())
784    }
785
786    fn v6(s: &str) -> ipnet::IpNet {
787        ipnet::IpNet::V6(s.parse().unwrap())
788    }
789
790    #[test]
791    fn default_advertises_nothing() {
792        let cfg = Config::default();
793        assert!(cfg.advertised_routes().is_empty());
794    }
795
796    #[test]
797    fn advertises_v4_subnet_routes() {
798        let cfg = Config {
799            advertise_routes: vec![v4("10.0.0.0/24"), v4("192.168.1.0/24")],
800            ..Default::default()
801        };
802        assert_eq!(
803            cfg.advertised_routes(),
804            vec![v4("10.0.0.0/24"), v4("192.168.1.0/24")]
805        );
806    }
807
808    #[test]
809    fn exit_node_adds_default_v4_route() {
810        let cfg = Config {
811            advertise_exit_node: true,
812            ..Default::default()
813        };
814        assert_eq!(cfg.advertised_routes(), vec![v4("0.0.0.0/0")]);
815    }
816
817    #[test]
818    fn v6_prefixes_are_dropped() {
819        let cfg = Config {
820            advertise_routes: vec![v4("10.0.0.0/24"), v6("fd00::/64")],
821            ..Default::default()
822        };
823        // IPv6-off: only the v4 prefix survives.
824        assert_eq!(cfg.advertised_routes(), vec![v4("10.0.0.0/24")]);
825    }
826
827    #[test]
828    fn exit_node_never_advertises_v6_default() {
829        let cfg = Config {
830            advertise_routes: vec![v6("::/0")],
831            advertise_exit_node: true,
832            ..Default::default()
833        };
834        // ::/0 is dropped; only the v4 default route is advertised.
835        assert_eq!(cfg.advertised_routes(), vec![v4("0.0.0.0/0")]);
836    }
837
838    #[test]
839    fn default_is_ephemeral() {
840        // Preserves the historical hardcoded behavior; persistent nodes must opt out explicitly.
841        assert!(Config::default().ephemeral);
842    }
843
844    #[test]
845    fn ephemeral_deserializes_default_true_when_absent() {
846        // A config that predates the field still registers ephemeral.
847        let cfg: Config = serde_json::from_str(r#"{"server_url":"https://example.com/"}"#).unwrap();
848        assert!(cfg.ephemeral);
849    }
850
851    #[test]
852    fn ephemeral_can_be_disabled_for_persistent_nodes() {
853        let cfg: Config =
854            serde_json::from_str(r#"{"server_url":"https://example.com/","ephemeral":false}"#)
855                .unwrap();
856        assert!(!cfg.ephemeral);
857    }
858
859    #[test]
860    fn tags_default_empty_and_deserialize() {
861        let cfg: Config =
862            serde_json::from_str(r#"{"server_url":"https://example.com/","tags":["tag:exit"]}"#)
863                .unwrap();
864        assert_eq!(cfg.tags, vec!["tag:exit".to_owned()]);
865        assert!(Config::default().tags.is_empty());
866    }
867
868    #[test]
869    fn advertises_no_services_without_peerapi_port() {
870        // Fail-closed default: no peerAPI port means no services advertised.
871        assert!(Config::default().advertised_services().is_empty());
872    }
873
874    #[test]
875    fn advertises_peerapi4_and_dns_proxy_when_port_set() {
876        use ts_control_serde::ServiceProto;
877
878        let cfg = Config {
879            peerapi_port: Some(8080),
880            ..Default::default()
881        };
882        let services = cfg.advertised_services();
883        assert_eq!(services.len(), 2);
884
885        // peerapi4 carries the real bind port.
886        assert_eq!(services[0].proto, ServiceProto::PeerApi4);
887        assert_eq!(services[0].port, 8080);
888
889        // peerapi-dns-proxy always advertises port 1 (Go quirk).
890        assert_eq!(services[1].proto, ServiceProto::PeerApiDnsProxy);
891        assert_eq!(services[1].port, 1);
892    }
893
894    #[test]
895    fn peerapi_port_deserializes_default_none() {
896        let cfg: Config = serde_json::from_str(r#"{"server_url":"https://example.com/"}"#).unwrap();
897        assert_eq!(cfg.peerapi_port, None);
898    }
899
900    #[test]
901    fn advertise_services_default_empty() {
902        assert!(Config::default().advertise_services.is_empty());
903        assert!(Config::default().advertised_vip_services().is_empty());
904    }
905
906    #[test]
907    fn advertise_services_deserializes() {
908        let cfg: Config = serde_json::from_str(
909            r#"{"server_url":"https://example.com/","advertise_services":["svc:samba"]}"#,
910        )
911        .unwrap();
912        assert_eq!(cfg.advertise_services, vec!["svc:samba".to_owned()]);
913    }
914
915    #[test]
916    fn advertised_vip_services_validates_and_drops_bad_names() {
917        let cfg = Config {
918            advertise_services: vec![
919                "svc:good".to_owned(),
920                "bad-no-prefix".to_owned(),
921                "svc:-bad-label".to_owned(),
922            ],
923            ..Default::default()
924        };
925        let svcs = cfg.advertised_vip_services();
926        assert_eq!(svcs.len(), 1);
927        assert_eq!(svcs[0].name, "svc:good");
928        // All-ports default range, active.
929        assert_eq!(svcs[0].ports.len(), 1);
930        assert_eq!(svcs[0].ports[0].first, 0);
931        assert_eq!(svcs[0].ports[0].last, 65535);
932        assert!(svcs[0].active);
933    }
934
935    #[test]
936    fn services_hash_empty_is_empty_string() {
937        assert_eq!(services_hash(&[]), "");
938    }
939
940    #[test]
941    fn services_hash_is_order_independent() {
942        let a = Config {
943            advertise_services: vec!["svc:a".to_owned(), "svc:b".to_owned()],
944            ..Default::default()
945        };
946        let b = Config {
947            advertise_services: vec!["svc:b".to_owned(), "svc:a".to_owned()],
948            ..Default::default()
949        };
950        let ha = services_hash(&a.advertised_vip_services());
951        let hb = services_hash(&b.advertised_vip_services());
952        assert_eq!(ha, hb);
953        assert!(!ha.is_empty());
954    }
955
956    #[test]
957    fn services_hash_changes_with_set() {
958        let one = Config {
959            advertise_services: vec!["svc:a".to_owned()],
960            ..Default::default()
961        };
962        let two = Config {
963            advertise_services: vec!["svc:a".to_owned(), "svc:b".to_owned()],
964            ..Default::default()
965        };
966        assert_ne!(
967            services_hash(&one.advertised_vip_services()),
968            services_hash(&two.advertised_vip_services())
969        );
970    }
971
972    #[test]
973    fn services_hash_known_answer() {
974        // KAT: pin the hash of a single all-ports `svc:samba` so a future serialization change
975        // (field order, whitespace) that would silently break the node's own change-detection fails
976        // this test. The hash is a SELF-CONSISTENCY TOKEN: this node computes it, sends it in
977        // `HostInfo.ServicesHash`, and echoes the same value in `C2NVIPServicesResponse.ServicesHash`;
978        // control treats it as opaque and only refetches when it CHANGES — control never recomputes
979        // it, so the node only needs to be internally consistent (it is — one `services_hash`).
980        //
981        // It is NOT byte-equal to Go `vipServiceHash` and is not meant to be: Go does
982        // `json.NewEncoder(sha256).Encode(services)` which (a) appends a trailing `\n` that
983        // `serde_json::to_vec` here does not, and (b) Go's advertise path (`vipServicesFromPrefsLocked`)
984        // leaves `Ports` nil → `"Ports":null`, whereas this fork injects an explicit all-ports
985        // `ProtoPortRange` → `"Ports":["*"]`. (The element form IS now Go-correct — `ProtoPortRange`
986        // serializes as the TextMarshaler string `"*"`, not a `{Proto,First,Last}` object — which is
987        // what moved this value from the old object-form hash.) Full Go-faithful ServicesHash is
988        // tracked separately; benign because the token is opaque to control.
989        let cfg = Config {
990            advertise_services: vec!["svc:samba".to_owned()],
991            ..Default::default()
992        };
993        let hash = services_hash(&cfg.advertised_vip_services());
994        // 64 hex chars = SHA-256.
995        assert_eq!(hash.len(), 64);
996        assert!(hash.bytes().all(|b| b.is_ascii_hexdigit()));
997        assert_eq!(
998            hash,
999            "9593a969d3df19c81e5c47a5caeca701ab60b732b99004f15aa00384d922c40c"
1000        );
1001    }
1002
1003    /// All eight up/set pref fields default off/None on a fresh `ts_control::Config`: the two
1004    /// advertise-side ones (`advertise_app_connector`, `auto_update_apply`) and the six store-only
1005    /// carried prefs. Fail-closed: a default node advertises no app-connector / auto-update and
1006    /// carries no operator/nickname/posture/webclient/LAN-access preference.
1007    #[test]
1008    fn up_set_pref_fields_default_off() {
1009        let cfg = Config::default();
1010        // Advertise-side.
1011        assert!(!cfg.advertise_app_connector);
1012        assert_eq!(cfg.auto_update_apply, None);
1013        // Store-only carried prefs.
1014        assert!(!cfg.auto_update_check);
1015        assert_eq!(cfg.operator_user, None);
1016        assert_eq!(cfg.node_nickname, None);
1017        assert!(!cfg.posture_checking);
1018        assert!(!cfg.run_web_client);
1019        assert!(!cfg.exit_node_allow_lan_access);
1020    }
1021
1022    /// End-to-end: a `Config` with `advertise_app_connector` / `auto_update_apply` set drives the
1023    /// `HostInfo.AppConnector` / `HostInfo.AllowsUpdate` wire fields through the SAME expressions the
1024    /// streaming map request (`client.rs`) and registration (`register.rs`) use. Guards that the
1025    /// advertise fields reach the wire, and that the default config omits both keys.
1026    #[test]
1027    fn advertise_prefs_drive_host_info_wire_fields() {
1028        use crate::map_request_builder::MapRequestBuilder;
1029
1030        let node_state = ts_keys::NodeState::generate();
1031
1032        // Advertising config: mirrors `.app_connector(config.advertise_app_connector)` and
1033        // `.allows_update(config.auto_update_apply == Some(true))` from client.rs.
1034        let cfg = Config {
1035            advertise_app_connector: true,
1036            auto_update_apply: Some(true),
1037            ..Default::default()
1038        };
1039        let req = MapRequestBuilder::new(&node_state)
1040            .app_connector(cfg.advertise_app_connector)
1041            .allows_update(cfg.auto_update_apply == Some(true))
1042            .build();
1043        let hi = req.host_info.unwrap();
1044        assert_eq!(hi.app_connector, Some(true));
1045        assert!(hi.allows_update);
1046        let v = serde_json::to_value(&hi).unwrap();
1047        assert_eq!(
1048            v.get("AppConnector").and_then(serde_json::Value::as_bool),
1049            Some(true)
1050        );
1051        assert_eq!(
1052            v.get("AllowsUpdate").and_then(serde_json::Value::as_bool),
1053            Some(true)
1054        );
1055
1056        // Default config (advertise off): `AppConnector` is sent as `false` (Go calls
1057        // `hi.AppConnector.Set(advertise)` unconditionally, and `.Set(false)` marshals to `false`, not
1058        // omitted), while `AllowsUpdate` (a plain `omitzero` bool) IS omitted when false. This
1059        // asymmetry matches Go's wire bytes exactly: a default node sends `AppConnector:false` but no
1060        // `AllowsUpdate` key.
1061        let cfg = Config::default();
1062        let req = MapRequestBuilder::new(&node_state)
1063            .app_connector(cfg.advertise_app_connector)
1064            .allows_update(cfg.auto_update_apply == Some(true))
1065            .build();
1066        let hi = req.host_info.unwrap();
1067        assert_eq!(hi.app_connector, Some(false));
1068        assert!(!hi.allows_update);
1069        let v = serde_json::to_value(&hi).unwrap();
1070        assert_eq!(
1071            v.get("AppConnector").and_then(serde_json::Value::as_bool),
1072            Some(false),
1073            "default node sends AppConnector:false (Go .Set(false)), not an omitted key"
1074        );
1075        assert!(
1076            v.get("AllowsUpdate").is_none(),
1077            "AllowsUpdate is an omitzero bool, omitted when false"
1078        );
1079
1080        // `auto_update_apply == Some(false)` advertises NO update (AllowsUpdate stays unset),
1081        // matching the `== Some(true)` gate.
1082        let cfg = Config {
1083            auto_update_apply: Some(false),
1084            ..Default::default()
1085        };
1086        let req = MapRequestBuilder::new(&node_state)
1087            .allows_update(cfg.auto_update_apply == Some(true))
1088            .build();
1089        assert!(!req.host_info.unwrap().allows_update);
1090    }
1091
1092    /// The pref fields deserialize from their snake_case keys (a daemon persists the config as JSON)
1093    /// and a config that predates the fields still loads with them defaulted off (the `#[serde(default)]`
1094    /// on each).
1095    #[test]
1096    fn up_set_pref_fields_deserialize_and_default_when_absent() {
1097        // Absent: defaults apply.
1098        let cfg: Config = serde_json::from_str(r#"{"server_url":"https://example.com/"}"#).unwrap();
1099        assert!(!cfg.advertise_app_connector);
1100        assert_eq!(cfg.auto_update_apply, None);
1101        assert!(!cfg.posture_checking);
1102        assert_eq!(cfg.operator_user, None);
1103
1104        // Present: parsed.
1105        let cfg: Config = serde_json::from_str(
1106            r#"{"server_url":"https://example.com/","advertise_app_connector":true,"auto_update_apply":true,"auto_update_check":true,"operator_user":"alice","node_nickname":"laptop","posture_checking":true,"run_web_client":true,"exit_node_allow_lan_access":true}"#,
1107        )
1108        .unwrap();
1109        assert!(cfg.advertise_app_connector);
1110        assert_eq!(cfg.auto_update_apply, Some(true));
1111        assert!(cfg.auto_update_check);
1112        assert_eq!(cfg.operator_user.as_deref(), Some("alice"));
1113        assert_eq!(cfg.node_nickname.as_deref(), Some("laptop"));
1114        assert!(cfg.posture_checking);
1115        assert!(cfg.run_web_client);
1116        assert!(cfg.exit_node_allow_lan_access);
1117    }
1118
1119    #[test]
1120    fn deduplicates_routes() {
1121        let cfg = Config {
1122            advertise_routes: vec![v4("0.0.0.0/0"), v4("10.0.0.0/24")],
1123            advertise_exit_node: true,
1124            ..Default::default()
1125        };
1126        // Explicit 0.0.0.0/0 plus the exit-node default route collapse to one entry.
1127        assert_eq!(
1128            cfg.advertised_routes(),
1129            vec![v4("0.0.0.0/0"), v4("10.0.0.0/24")]
1130        );
1131    }
1132}