# Changelog All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). ## [Unreleased] ## [0.9.14] - 2026-05-03 ### Fixed - **Live UDP loss counter no longer stalls under upload-mode saturation** (issue #70 final fix) — v0.9.13's `TCP_NODELAY` partially addressed the bug but brettowe's retest showed 8 subsequent intervals still bunched at one end-of-test client-side timestamp. Root cause was `tokio::time::interval` defaulting to `MissedTickBehavior::Burst` on the server's stats sampling timer: when `writer.write_all()` stalled under the back-pressure that the saturated UDP uplink induces on TCP control, missed ticks accumulated and fired as a burst when the writer unblocked, producing stale interval samples with fresh client-side arrival timestamps and misleading throughput numbers. `Skip` now drops the stale ticks; cumulative state in `StreamStats` atomics still surfaces correctly on the next live tick and at end-of-test. Applied unconditionally on both `run_test` interval-loop sites; benefits even pre-v0.9.14 clients pairing with a v0.9.14 server. - **`--omit` no longer folds hidden UDP loss into the first visible interval** — the new cumulative-loss tracker added below was advancing its cache on every progress arrival, but the printed-line baseline only advanced when a line printed. With `--omit 3`, the first visible interval would report all loss accumulated during seconds 0-3 as one jumbo delta, defeating the purpose of `--omit`. The baseline now advances during the omit window so visible lines reflect only loss observed during printed intervals. ### Added - **UDP receiver feedback (`udp_feedback_v1` capability)** (issue #70 final fix) — when both peers advertise the capability, the server now emits a 36-byte cumulative `(packets_received, packets_lost)` UDP packet back to the client at 2 Hz on the same data socket, sidestepping the TCP control channel for live UDP loss reporting. Wire format: `b"XFRF"` magic + version + kind + flags + `stream_id` + reserved + `elapsed_ms` + cumulative `packets_received` + cumulative `packets_lost`, all big-endian, fixed 36 bytes. Length-first demux at receive sites distinguishes feedback from data packets without inspecting sequence-number bits. Cumulative-not-delta semantics let the client recover from any dropped feedback packet without needing the lost intermediate state. Capability negotiation gates emission so older clients (which wouldn't know to listen) never see a packet they don't understand. - **Producer-side monotonic-denominator filter on the client** — both the TCP control `udp_progress` decode site and the UDP feedback aggregator funnel updates through `UdpProgressFilter::apply`, which admits only readings whose `(received + lost)` denominator is at-least-as-fresh as anything we've seen before. Atomic CAS via `fetch_update` so two producers cannot race a stale store after a fresh one. Applies in addition to TUI display: plain text, CSV, and JSON-stream output use the cached cumulative as the source of truth for the per-line `lost` field, so the freshest reading from either source flows through to scripted consumers, not just the TUI live counter. - **Live UDP loss in non-TUI output paths** — `--no-tui --json-stream` / `--csv` / plain interval output now reflects the freshest `udp_progress` from either TCP control or UDP feedback. Previously these consumers used per-stream `streams[].lost` from the most recent TCP `Interval` only, which under control-channel stalls could be several seconds stale. Falls back to the per-stream sum for sessions where `udp_progress` is never sent (paired with a pre-0.9.11 server, or non-UDP tests). - **Docker repro harness for issue #70** (`docker/Dockerfile.repro`, `docker/repro-issue-70.sh`, `docker/README.md`) — multi-stage build with the current branch and the released v0.9.13 baseline side-by-side. `docker run --rm --cap-add=NET_ADMIN xfr-repro` runs hard assertions on the new build (max bunch ≤ 2, time-to-first-loss < 5s, live mid-run loss observed); `--baseline` prints diagnostics for narrative comparison without gating on a threshold. Stays out of CI — the existing 2× oversubscription `control-channel-skew` job remains the regression floor; the harness is for human-driven A/B at brettowe's 10× recipe before publishing. ### Changed - **`TestProgress` schema (pre-1.0 break)** — adds `udp_feedback_only: bool` so consumers can distinguish a feedback-only update (only `udp_progress` carries truth; everything else is sentinel/None) from a full TCP `Interval` update. Three consumers handle the partial variant: `App::on_progress` early-returns after updating UDP loss state and preserves all other field values; `main.rs` print loop skips feedback-only entries entirely (the cumulative cache picks up the freshness for the next full interval); cross-version compat test path adopts the new field. - **Server bidir mode no longer emits UDP feedback** — feedback is upload-mode-only by design. Bidir's server-side recv half was passing `client_supports_udp_feedback` through to `receive_udp` even though the client's bidir recv has no consumer for those packets; emission was pure overhead on the return path. Bidir always passes `false` now. - **`receive_udp` skips feedback packets in `bytes_received` accounting** — the length-first demux previously rejected feedback before the data path but bumped `bytes_received` first. With server bidir gating that's a moot path post-fix, but defense-in-depth: feedback bytes never count toward `bytes_received`, which tracks test-data wire bandwidth. - **Capability list factored into a single `SUPPORTED_CAPABILITIES` const** — `client_hello`, `server_hello`, and `server_hello_with_auth` previously each had a duplicated `Vec` literal. Future capability additions now touch one line. New `capability_advertised(&capabilities, name)` helper centralizes the matcher used at both negotiation sites. ### Library API (pre-1.0 break) - `client::TestProgress` gains `udp_feedback_only: bool`. Constructors must supply it. - `client::UdpProgressFilter` and `client::UdpFeedbackAggregator` are new public types backing the producer-side filter and aggregator. - `udp::receive_udp` signature gains a trailing `feedback_enabled: bool` parameter. - `udp::receive_udp_feedback_only(socket, aggregator, stream_index, cancel)` is new; spawned per-stream on the client in upload mode. - `udp::UdpFeedbackPacket` and `UDP_FEEDBACK_SIZE` / `UDP_FEEDBACK_MAGIC` / `UDP_FEEDBACK_VERSION` / `UDP_FEEDBACK_KIND_RECEIVER_PROGRESS` constants exported. - `protocol::SUPPORTED_CAPABILITIES` and `protocol::capability_advertised` exported. - `stats::StreamStats::udp_progress_snapshot()` exported for callers that need a coherent `(received, lost)` pair. ### Maintenance - Bump `Cargo.toml` to `0.9.14`. ## [0.9.13] - 2026-05-03 ### Fixed - **Live UDP loss counter no longer stuck at 0% under saturated links** (issue #70 follow-up) — `TCP_NODELAY` was not being set on the control connection. With Nagle still active, the periodic `Interval` messages (~150-byte 1 Hz writes) coalesced waiting for an MSS-sized payload (which never arrives — they're tiny) or a delayed ACK from the peer. Under heavy parallel UDP data load on a saturated path (Wi-Fi, rate-limited links, anything where ACK turnaround stretches), the kernel held every queued segment for the duration of the test and flushed the entire backlog in a single burst when data traffic stopped. The TUI live counter appeared permanently stuck at 0% during the run, then jumped to the final value at quit. iperf3 sets `TCP_NODELAY` on its control channel for exactly this reason. New `tcp::configure_control_stream` helper applies it before splitting the stream into reader/writer halves; called at three sites (server's accepted control connection, client's connecting control connection, server's auth-handshake fallback path). Reproduced and verified with a `tc netem` 50 Mbps + 50ms-delay simulation: the pre-fix binary collapses 3+ interval lines to a single end-of-test timestamp; the post-fix binary spreads them across the run with at most a 2-line tail collision. ### Added - **CI regression test for the bunching pattern** (`test-control-channel-skew.sh`, runs as the `Control-channel skew (#70 regression)` job). Applies a 50 Mbps shaper + 50ms each-way delay to `lo`, runs an 8-second UDP test at 100 Mbps target (2× oversubscription), and asserts no 3-or-more interval lines share a client-side timestamp. Catches future regressions where the `TCP_NODELAY` plumbing is dropped from any of the three control-stream sites or a new code path forgets to call the helper. ### Maintenance - Rust dependency group bump (PR #76): `clap_complete` 4.6.2 → 4.6.3, `rustls` 0.23.39 → 0.23.40. Patch-version updates only, no source changes required. ## [0.9.12] - 2026-05-02 ### Added - **`-w`/`--window` now applies to UDP socket buffers** (#70 follow-up) — previously the flag only set TCP `SO_SNDBUF`/`SO_RCVBUF`. UDP sockets used the kernel default regardless. On high-rate UDP flows where the receiver's kernel UDP buffer can saturate (weak/loaded receivers, default `net.core.rmem_max` lower than line-rate × ~1s), the kernel tail-drops new arrivals and the loss only surfaces as a sequence-gap once traffic stops — rendering as live `Packet Loss: 0.0%` while the test is running and a high final loss percent on quit. `-w 16M` is the immediate workaround for that pattern; the value propagates from client to server in `TestStart` (already wired for TCP since v0.9.9) and now lands on UDP `SO_SNDBUF`/`SO_RCVBUF` on both ends. ### Changed - **`setsockopt` failures on `-w` now surface as warnings** — previously rejections (typically `net.core.rmem_max` exceeded without `CAP_NET_ADMIN`) were swallowed at debug level. Users running `-w 16M` for #70-style troubleshooting need to see the rejection rather than have it disappear silently. Both `SO_SNDBUF` and `SO_RCVBUF` are still attempted independently, and the warning identifies which buffer failed. ### Maintenance - Rust dependency group bump (PR #71): clap 4.5 → 4.6, clap_complete 4.5 → 4.6, hyper 1.8 → 1.9, mdns-sd 0.17 → 0.19, toml 0.9 → 1.1, rustls 0.23.37 → 0.23.39, hmac 0.12 → 0.13, sha2 0.10 → 0.11, rand 0.9 → 0.10, plus libc, uuid, once_cell, tracing-subscriber, tracing-appender, tempfile patch bumps. Adapted source to API moves in hmac (constructor moved from `Mac` to `KeyInit`) and rand (user-facing `Rng` methods moved to `RngExt`); no behavior change. ## [0.9.11] - 2026-04-30 ### Fixed - **Live UDP packet-loss counter during the run** (issue #70) — the Packet Loss line in the TUI was stuck at 0.0% for the entire test and only updated to the real value at completion. With `-t 0` (infinite mode) the real value was never visible. Server now ships a cumulative packet-counts snapshot (`UdpIntervalProgress { packets_received, packets_lost }`) on every periodic Interval message; client derives the loss percent locally and the TUI updates it live. Cumulative counts are snapshotted into `IntervalStats` at interval emission time so deltas between consecutive samples correspond to the same window. Reported by @brettowe. - **Final UDP loss accounting only counts valid xfr packets** — `UdpStats.packets_received` and `packets_sent` now exclude short, malformed, or foreign datagrams that can't be header-decoded. Previously such datagrams inflated `packets_received` and silently understated the final loss percent. `bytes_received` continues to count every byte the wire delivered. ### Added - **Throughput sparkline tints by per-interval loss severity** (issue #70) — lossy intervals are visually distinct from clean intervals at the same height: clean stays the graph color, light loss (<1% per-interval rate) tints warning, heavy loss (≥1%) tints error. Per-interval rate computed from `udp_progress` deltas, so a single-packet hiccup and a heavy drop burst no longer collapse to the same flat tint. Magnitude unknown (TCP run, pre-0.9.11 server, or first UDP sample) stays the graph color — honest "no signal" rather than a misleading tint. Sparkline widget gains a `.styles(&[Style])` per-sample override. - **Freshness signal for the Packet Loss line** — the line renders dimmed `--%` when paired against a pre-0.9.11 server, or before any UDP traffic has been observed. Without this, an old server would render a stale `0.0%` next to actual loss bursts in the sparkline. `App.udp_lost_percent` is now `Option` end-to-end. ### Changed - **Jitter rolling-window label** (issue #72) — the running display now reads `Jitter: 0.86 ms (10s avg: 0.03 ms)`. The previous `(10s: …)` form read like a stuck timer; @pythonwood opened an issue thinking the display was broken on a v0.9.10-client → v0.9.6-server pairing. ## [0.9.10] - 2026-04-22 ### Fixed - **TCP teardown no longer hangs under rate-limited paths** (issue #54) — the v0.9.8 SO_LINGER=0 fix didn't take effect when the send loop was parked in `stream.write().await` under heavy backpressure (tc rate limiting, slow peers, MPTCP subflows filling). The loop would block past the configured deadline, never reaching the drop/close that would trigger the abortive close. `send_data` and `send_data_half` now race the pending `write()` against cancel and deadline in a `biased tokio::select!`, so either signal breaks the loop and lets `SO_LINGER=0` do its job. Confirmed by @matttbe against his MPTCP + tc reproducer. - **TUI elapsed time stays live during data gaps** (issue #62) — `app.elapsed` was only updated when the server's `Interval` progress message arrived. On lossy paths that starved the control channel (e.g. brettowe's WiFi test with packet-drop bursts), the elapsed counter — and by extension the progress bar position — could freeze for several seconds until the next message landed, creating the impression of a "stall" even though the TUI was still redrawing at 20 Hz. The loop now refreshes `elapsed` from the wall clock on every iteration; `on_progress` no longer writes `elapsed` (doing so was a visual no-op, since the next tick immediately overwrote the server's value). Pause handling shifts `start_time` forward by the pause duration on resume, so the elapsed counter excludes paused time. Reported by @brettowe. - **Infinite-duration (`-t 0`) TUI now shows a live elapsed counter** — the `{}s/∞` display was relying on the same `elapsed` field that could go stale, so an infinite test on a flaky link looked frozen. Covered by the same fix. ### Added - **Client/server version in the Configuration panel** (issue #62) — the TUI now shows `xfr/` so cross-version test pairings are obvious at a glance. Server version is captured from the `Hello` handshake (already wire-supported; just wasn't surfaced). Requested by @brettowe. ### Changed - **TUI jitter line shows latest + smoothed together** (issue #48 follow-up) — the running display now reads `Jitter: 0.02 ms (10s: 0.03 ms)` so the latest per-interval aggregate and the 10-second rolling mean are visible side by side. Resolves the confusion where the rolling mean could stay above any single sample's value, making the live display look inconsistent with the server's authoritative final. Completed state still shows just the final value (no smoothing companion — the final is authoritative). Reported by @brettowe. ### Security - **Sanitize server-advertised version before rendering** — the `Hello.server` field crosses a network trust boundary and was being rendered verbatim into the Configuration panel. A hostile or compromised server could send ANSI escape sequences (clear screen, move cursor, OSC commands) or a very long string and have the user's terminal act on them. Control characters are now stripped, length capped at 32, and empty or all-control inputs fall back to `(unknown)`. - **Bump rustls-webpki 0.103.12 → 0.103.13** (RUSTSEC-2026-0104) — reachable panic in CRL parsing. Transitive upgrade via Cargo.lock; no manifest change. ## [0.9.9] - 2026-04-21 ### Added - **Max jitter and packet size in UDP summary** (issue #48 follow-up) — the final UDP summary now reports `Jitter Max` (peak of the RFC 3550 running estimate across the test) alongside the average, and `Packet Size` (UDP payload bytes). Surfaced in plain text and JSON. Requested by brettowe for NFS UDP packet-size tuning context. - **`-w` short alias for `--window`** (issue #60) — matches iperf3 muscle memory. ### Changed - **Bare-integer duration arguments mean seconds** (issue #61) — `-t 10`, `--max-duration 60`, `--rate-limit-window 30`, and discover `--timeout 5` now accept plain integers as seconds, matching iperf3 muscle memory. Unit-suffixed forms (`10s`, `1min`, `500ms`) continue to work unchanged. - Side effect: `--rate-limit-window` now rejects zero (`0`, `0s`, `0ms`) with `Duration must be greater than 0`. Previously `0s` was accepted and would later panic in the rate-limiter's cleanup task because `tokio::time::interval` requires a non-zero duration. Other duration flags (`-t`, `--max-duration`, `discover --timeout`) still accept `0` for their existing meanings (`-t 0` is infinite duration). - **Smoothed TUI jitter reading** (issue #48) — the UDP stats panel now shows jitter averaged over a 10-second rolling window rather than the raw per-second sample. The data pipeline is unchanged (samples still arrive every second from the server); only the aggregate display is smoothed. Per-stream jitter in the streams view continues to show the latest interval. While the test is running, the label shows `Jitter (10s):`; once completed, it reverts to `Jitter:` with the authoritative final value from the server's result. ### Fixed - **Duplicate receive-error log on the server** (issue #54) — `tcp::receive_data` and `tcp::receive_data_half` each warned at the read-error site, and the caller then warned again when it saw the returned `Err`. The duplicate inner `warn!` is removed so receive errors now log exactly once, matching the send path's pattern. Reported by @matttbe. - **Default to kernel TCP autotuning** (issue #60) — xfr no longer forces `SO_SNDBUF`/`SO_RCVBUF` to 4 MB on either side by default; both ends let the kernel autotune unless the user passes `-w`/`--window`. When set, the client's value propagates to the server over the control protocol so both sides apply the socket option symmetrically (matching iperf3). Reported by @matttbe. Caveats: - Loopback / intra-host benchmark numbers may decrease by roughly 10% — this is expected; the previous numbers were inflated by the oversized app-applied buffer. - On high-RTT paths, very short tests (e.g. `-t 1s` at high bitrate) may now show ramp-up-limited throughput in the final summary because kernel autotune takes a handful of RTTs to grow the window. Use a longer `-t`, or pass an explicit `-w` to skip autotune. Note that `-O`/`--omit` only hides the early intervals from output — the server-side final summary is still computed over the full test duration. - Explicit window sizes above `c_int::MAX` (≈2.1 GB on 64-bit) are now rejected with `InvalidInput` instead of silently wrapping before `setsockopt`. ### Removed - **Library API**: pre-1.0 break — `TcpConfig::high_speed()` and `TcpConfig::with_auto_detect()` are gone; construct `TcpConfig` directly with the fields you want set. The `HIGH_SPEED_BUFFER` and `HIGH_SPEED_WINDOW_THRESHOLD` constants (which were private) are also removed. Downstream code that constructs `ControlMessage::TestStart`, `protocol::UdpStats`, or `tui::app::App` by name now needs to supply the new fields (`window_size`, `jitter_max_ms`, `packet_size`, `jitter_history`); these are additive and have sensible None/default values. ## [0.9.8] - 2026-04-17 ### Added - **Separate send/recv reporting in bidir tests** (issue #56) — `--bidir` now reports per-direction bytes and throughput in the summary instead of just the combined total, which was useless on asymmetric links. Plain text shows `Send: X Recv: Y (Total: Z)`; JSON adds `bytes_sent`, `bytes_received`, `throughput_send_mbps`, `throughput_recv_mbps`; CSV gets four new columns; TUI shows `↑ X / ↓ Y` in the throughput panel. Unidirectional tests are unchanged (the existing `bytes_total`/`throughput_mbps` is already the single-direction number). ### Fixed - **Fast, accurate TCP teardown** (issue #54) — replaced the blocking `shutdown()` drain on the send path with `SO_LINGER=0` on Linux, so cancel and natural end-of-test no longer wait for bufferbloated send buffers to ACK through rate-limited paths. Fixes the "Timed out waiting 2s for N data streams to stop" warning matttbe reported with `-P 4 --mptcp -t 1sec`. - **Sender-side byte-count accuracy** — `stats.bytes_sent` is now clamped to `tcpi_bytes_acked` before abortive close, removing a quiet ~5-10% overcount where the send-buffer tail discarded by RST was being reported as transferred. Download and bidir tests are the primary beneficiaries. - **macOS preserves graceful shutdown** — non-Linux platforms lack `tcpi_bytes_acked`, so the Linux abortive-close path is cfg-gated; other platforms still use `shutdown()` for accurate accounting. ## [0.9.7] - 2026-04-16 ### Added - **Early exit summary** (issue #35) — Ctrl+C now displays a test summary with accumulated stats instead of silently exiting. Works in both plain text and TUI modes. Double Ctrl+C force-exits immediately. - **DSCP server-side propagation** — `--dscp` flag is now sent to the server and applied to server-side TCP/UDP sockets for download and bidirectional tests. Previously only client-side sockets were marked. - **Non-Unix `--dscp` warning** — platforms without socket TOS support now show a visible warning before the test starts, instead of silently no-oping. ### Fixed - **Cancel flow waits for server result** — client `Cancelled` handler now waits for the server's `Result` message instead of immediately erroring, providing accurate final stats after cancel. - **Server result ordering** — server sends `Result` before slow post-processing (push gateway, metrics hooks), preventing false cancel timeouts. - **Rust 1.95 clippy compatibility** — fixed `manual_checked_ops` and `collapsible_match_arms` lints. ### Changed - Bump `softprops/action-gh-release` from 2 to 3 in CI release workflow. ## [0.9.6] - 2026-03-18 ### Added - **`--dscp` flag** — set DSCP/TOS marking on TCP and UDP client sockets for QoS policy testing. Accepts numeric values (0-255) or standard DSCP names (EF, AF11-AF43, CS0-CS7). QUIC warns and ignores the flag; non-Unix platforms warn instead of applying socket marking. - **`omit_secs` config support** (issue #43) — `[client] omit_secs = N` in config file sets default `--omit` value. ## [0.9.5] - 2026-03-17 ### Added - **TCP `--cport` support** (issue #44) — `--cport` now pins client-side TCP data-stream source ports. Multi-stream TCP uses sequential ports (`cport`, `cport+1`, ...), matching UDP behavior. ### Changed - **TCP `--cport` semantics** — TCP control remains on an ephemeral source port while data streams use the requested source port or range. TCP data binds now match the remote address family the same way UDP/QUIC already do, so dual-stack clients can use `--cport` against IPv6 targets. ## [0.9.4] - 2026-03-11 ### Added - **`--no-mdns` flag** (issue #41) — `xfr serve --no-mdns` disables mDNS service registration for environments where multicast is unwanted or another service already uses mDNS. - **`server.no_mdns` config support** — mDNS registration can now also be disabled from `~/.config/xfr/config.toml` via `[server] no_mdns = true`. ### Changed - **Delta retransmits in interval reports** (issue #36) — plain text interval lines now show per-interval retransmit deltas instead of cumulative totals, making it easier to spot when retransmits actually occur. Hidden intervals from `--omit`, `--quiet`, or larger `--interval` settings no longer get folded into the next visible `rtx:` value. Final summary still shows cumulative totals. ## [0.9.3] - 2026-03-10 ### Added - **Server `--bind` flag** (issue #38) — `xfr serve --bind ` binds TCP, QUIC, and UDP data listeners to a specific address. Validates against `-4`/`-6` flags and rejects unspecified addresses (`::`, `0.0.0.0`). Requested by Windows users needing interface-specific binding. ### Changed - **Server sends random payloads** (issue #34) — server-side TCP and UDP send paths now use random bytes by default in reverse and bidirectional modes, matching the client's default-on behavior. `--zeros` only affects client-sent traffic; server payload mode is not negotiated over the wire (future enhancement). ### Fixed - **QUIC dual-stack on Windows** (issue #39) — QUIC server endpoint now creates its UDP socket via socket2 with explicit `IPV6_V6ONLY` handling instead of relying on Quinn's `Endpoint::server()`, which uses `std::net::UdpSocket::bind()` without dual-stack configuration. On Windows/macOS where `IPV6_V6ONLY` defaults to `true`, binding to `[::]` would only accept IPv6 connections. - **Server random payload on single-port TCP reverse** (issue #34) — the single-port TCP handler (DataHello path used by all modern clients) was missing `random_payload = true`, causing reverse-mode downloads to still send zeros. Legacy multi-port handler was correct but is dead code for current clients. ### Security - **quinn-proto DoS fix** — updated quinn-proto 0.11.13 → 0.11.14 (RUSTSEC-2026-0037, severity 8.7) ## [0.9.2] - 2026-03-06 ### Changed - **Random payloads by default** (issue #34) — TCP/UDP payloads now use random bytes by default to avoid silently inflated results on WAN-optimized or compressing paths. `--random` remains as an explicit no-op for clarity, and new `--zeros` forces zero-filled payloads for compression/dedup testing. ### Fixed - **Windows build regression** (issue #37) — `pacing_rate_bytes_per_sec()` used `libc::c_ulong` without a `#[cfg(target_os = "linux")]` guard, breaking compilation on Windows. The function is only called from the linux-gated `SO_MAX_PACING_RATE` path. - **MPTCP namespace test realism** (issue #32) — `test-mptcp-ns.sh` now combines `netem` shaping with `fq_codel` on the shaped transit links, matching common Linux defaults more closely and reducing false-positive high-stream failures caused by shallow unfair queues in the test harness. ## [0.9.1] - 2026-03-05 ### Added - **MPTCP support** (`--mptcp`) - Multi-Path TCP on Linux 5.6+ (issue #24). Uses `IPPROTO_MPTCP` at socket creation via socket2 — all TCP features (nodelay, congestion control, window size, bidir, multi-stream, single-port mode) work transparently. The server automatically creates MPTCP listeners when available (no flag needed) — MPTCP listeners accept both MPTCP and regular TCP clients transparently, with silent fallback to TCP if the kernel lacks MPTCP support. Client uses `--mptcp` to opt in. Clear error message on non-Linux clients or kernels without `CONFIG_MPTCP=y`. - **Kernel TCP pacing via `SO_MAX_PACING_RATE`** (issue #30) - On Linux, TCP bitrate pacing (`-b`) now uses the kernel's FQ scheduler with EDT (Earliest Departure Time) for precise per-packet timing, eliminating burst behavior from userspace sleep/wake cycles. Falls back to userspace pacing on non-Linux, MPTCP sockets (not yet supported in kernel, see [mptcp_net-next#578](https://github.com/multipath-tcp/mptcp_net-next/issues/578)), or if the setsockopt fails. Note: `-b` sets a global bitrate shared across all parallel streams (unlike iperf3 where `-b` is per-stream). Suggested by the kernel MPTCP maintainer. - **Random payload mode** (`--random`, issue #34) — client can fill TCP/UDP send buffers with random bytes (once at allocation) to reduce compression/dedup artifacts on shaped/WAN links. Current scope is client-sent payloads only: reverse mode sender remains server-side zeros until protocol negotiation is added. ### Changed - **Library API** — `create_tcp_listener()`, `connect_tcp()`, and `connect_tcp_with_bind()` now take a `mptcp: bool` parameter. Library consumers should pass `false` to preserve existing behavior. ### Fixed - **High stream-count TCP robustness** (issues #25, #32) — client now stops local data streams at local duration expiry instead of waiting for server `Result`, scales stream join timeout with stream count (`max(2s, streams*50ms)`), and TCP receivers drain briefly after cancel to reduce reset-on-close bursts. For single-port TCP setup, client also limits concurrent `connect + DataHello` handshakes (max 16 in flight) and server initial first-line read timeout is now adaptive to active stream counts (capped at 20s), reducing mid-test handshake-loss failures on constrained links. - **Best-effort send shutdown** — `send_data()` shutdown no longer propagates errors during normal teardown races, matching `send_data_half()` behavior. - **Kernel pacing rate width** — `SO_MAX_PACING_RATE` now uses native `c_ulong` instead of `u32`, removing an unintended ~34 Gbps ceiling on 64-bit Linux. - **JoinHandle panic with many parallel streams** (issue #24) — removed second `join_all` after aborting timed-out stream tasks, which polled already-completed handles - **Final summary showing 0 retransmits/RTT/cwnd** (issue #26) — each stream task now captures a final sender-side TCP_INFO snapshot before the socket closes; the Result handler overlays these saved snapshots deterministically instead of racing live fd polls - **Broken pipe / connection reset at teardown** (issue #25) — client now joins stream task handles with timeout before returning, preventing writes to already-closed sockets - **MPTCP label in server log** — server now displays "MPTCP" instead of "TCP" in the test info log when client uses `--mptcp`; adds backward-compatible `mptcp` field to TestStart control message ## [0.8.0] - 2026-02-12 ### Added - **Client source port pinning** (`--cport`) - pin the client's local port for firewall traversal (issue #16). Works with UDP and QUIC. Multi-stream UDP (`-P N`) assigns sequential ports starting from the specified port (e.g., `--cport 5300 -P 4` uses ports 5300-5303). QUIC multiplexes all streams on the single specified port. TCP rejects `--cport` since single-port mode already handles firewall traversal. Combines with `--bind` for full control (`--bind 10.0.0.1 --cport 5300`). Automatically matches the remote's address family so `--cport` works transparently with both IPv4 and IPv6 targets. ### Fixed - **`--bind` with IPv6 targets** — `--bind` with an unspecified IP (e.g., `0.0.0.0:0`) now auto-matches the remote's address family at socket creation time across TCP, UDP, and QUIC. Previously failed when connecting to IPv6 targets from dual-stack clients. - **UDP data_ports length validation** — server returning mismatched port count could panic on `stats.streams[i]`; now validates length before iterating, matching the existing TCP guard ## [0.7.1] - 2026-02-12 ### Fixed - **Server TUI `-0.0 Mbps` after test ends** (issue #20) - IEEE 754 negative zero now normalized via precision-aware `normalize_for_display()` helper across all throughput display paths - **TCP RTT/retransmits not updating live** (issue #13) - per-interval retransmits now computed from TCP_INFO deltas instead of a dead atomic counter; client stores socket fds for local TCP_INFO polling so sender-side metrics (upload/bidir) update live; download mode correctly uses server-reported metrics - **Plain-text zero retransmits dropped** - `rtx: 0` was omitted in plain/JSON/CSV interval output when all streams reported zero retransmits; now preserved - **`mbps_to_human()` unit-switch boundary** - `999.95 Mbps` displayed as `1000.0 Mbps` instead of `1.00 Gbps`; unit branch now uses rounded value ### Changed - **Consolidated throughput formatting** - server TUI now uses shared `mbps_to_human()` instead of inline formatting; Gbps display changes from 1 to 2 decimal places for consistency ## [0.7.0] - 2026-02-11 ### Added - **Real pause/resume** (`p` key) - pressing `p` now pauses actual data transfer, not just the TUI display. Uses `Pause`/`Resume` protocol messages and a dedicated `watch` channel to stop/resume data loops across TCP, UDP, and QUIC. Capability-gated via `pause_resume` in Hello messages: older servers without support fall back to display-only pause. TCP bitrate pacing resets its baseline on resume to prevent catch-up bursts. UDP receiver resets its inactivity timer during pause to prevent false timeouts. Resolves issue #19. ## [0.6.1] - 2026-02-10 ### Added - **TCP bitrate pacing** (`-b` for TCP) - `-b` flag now works for TCP, not just UDP. Uses byte-budget sleep pacing with interruptible sleeps for responsive cancellation. Buffer size auto-caps to prevent first-write burst at low bitrates. Resolves issue #14. ## [0.6.0] - 2026-02-06 ### Added - **Congestion control selection** (`--congestion`) - Select TCP congestion control algorithm (e.g. cubic, bbr, reno). Applied on both client and server sockets. Useful for BBR vs CUBIC comparison on WAN/cloud links. - **Live TCP_INFO polling** - RTT, cwnd now reported per interval during tests, not just in final result. Enables real-time TCP metric monitoring in TUI, plain text (`rtt: X.XXms`), JSON streaming, and CSV output. Essential for `-t 0` infinite tests where results are never finalized (issue #13). ### Fixed - **Congestion config errors surfaced** - Invalid `--congestion` algorithm is now validated before the test starts; client exits non-zero immediately and server sends an error message back to the client - **TCP_INFO stale fd cleared** - Stream handlers now clear the stored file descriptor on completion and on early-return error paths, preventing `poll_tcp_info()` from reading an unrelated socket if the OS reuses the fd - **Update banner double "v" prefix** - Version display now strips leading `v` from update-informer output to avoid showing `vv0.5.0` - **PSK unwrap panics** - Server no longer panics if PSK is misconfigured during auth; returns error instead - **UDP encode bounds check** - `UdpPacketHeader::encode()` now validates buffer length before writing - **Timestamp clock skew** - ISO8601/Unix timestamps now derived from monotonic elapsed time instead of calling `SystemTime::now()` ### Code Quality - **Named constants** - Replaced 12 hardcoded magic numbers in serve.rs with 6 named constants (STATS_INTERVAL, CANCEL_CHECK_TIMEOUT, RESULT_FLUSH_DELAY, STREAM_ACCEPT_TIMEOUT, STREAM_COLLECTION_TIMEOUT, DEFAULT_BITRATE_BPS) ## [0.5.0] - 2026-02-05 ### Changed - **Single-port TCP mode** (issue #16) - TCP tests now use only port 5201 for all connections, making them firewall-friendly. Data connections identify themselves via `DataHello` message instead of using ephemeral ports. - **Protocol version bump to 1.1** - Signals DataHello support; adds `single_port_tcp` capability for backward compatibility detection - **Client capabilities in Hello** - Client now advertises supported capabilities (tcp, udp, quic, multistream, single_port_tcp); server falls back to multi-port TCP for legacy clients without `single_port_tcp` - **Numeric version comparison** - `versions_compatible()` now parses major version as integer instead of string comparison ### Fixed - **QUIC IPv6 support** (issue #17) - QUIC clients can now connect to IPv6 addresses without requiring `-6` flag; endpoint now binds to matching address family - **mDNS discovery** (issue #15) - Server now advertises addresses via `enable_addr_auto()`; client uses non-blocking receive with proper timeout handling - **TCP RTT and retransmits display** (issue #13) - TUI now shows correct retransmit count from stream results (captured after transfer); TCP_INFO captured after transfer for accurate RTT/cwnd - **Data connections no longer consume rate-limit/semaphore slots** - Only control (Hello) connections acquire permits; DataHello connections route directly without resource consumption - **Cancel messages processed during TCP stream collection** - Interval loop now starts immediately; stream collection runs concurrently in background - **Client OOB panic on port mismatch** - Added bounds check when server returns fewer ports than requested streams - **DoS guard on oversized lines** - `read_first_line_unbuffered()` now returns error instead of truncating - **DataHello serialization panic** - Replaced `unwrap()` with proper error handling in spawned task - **One-off mode deadlock** - `--one-off` no longer blocks the accept loop waiting for test completion; uses shutdown channel to signal exit after test finishes - **QUIC one-off mode** - QUIC accept loop now responds to shutdown signal for proper `--one-off` exit - **cancel.changed() busy-loop** - Handle sender-dropped error in stream collection select! to prevent CPU spin - **IPv4-mapped IPv6 comparison** - DataHello IP validation now normalizes `::ffff:x.x.x.x` addresses for correct matching on dual-stack systems ### Security - **DataHello IP validation** - Server validates DataHello connections come from same IP as control connection to prevent connection hijacking - **Slow-loris protection** - Accept loop now spawns per-connection tasks with 5-second initial read timeout; slow clients can no longer block the listener - **DataHello flood protection** - Server validates test_id exists in active_tests before processing DataHello connections; unknown test_ids are dropped immediately - **Pre-handshake connection gate** - Limits concurrent unclassified connections (4x max_concurrent) to prevent connection-flood DoS before Hello/DataHello routing - **Multi-port TCP fallback IP validation** - Per-stream listeners validate connecting peer IP against control connection, preventing unauthorized data stream injection - **One-off mode hardened** - Failed handshakes and auth failures no longer trigger server shutdown in `--one-off` mode; only successful test completion exits ### Testing - Added regression test for QUIC IPv6 connectivity - Added `test_tcp_one_off_multi_stream` - verifies 4-stream TCP in `--one-off` mode with stream count assertion - Added `test_quic_one_off` - verifies 2-stream QUIC in `--one-off` mode with stream count assertion ### Code Quality - Log panics from `join_all` in QUIC, UDP, and TCP stream handlers instead of silently discarding JoinErrors - Multi-port fallback listener tasks cleaned up via cancel signal (no leaked tasks on partial connections) ## [0.4.4] - 2026-02-04 ### Changed - **Lower default max_concurrent** - reduced from 1000 to 100 for safer resource defaults ### Fixed - **Settings modal text truncation** - increased modal width to prevent help text from being cut off - **UDP session cleanup on client abort** (issue #12) - server now detects inactive UDP sessions after 30 seconds and cleans them up properly - **UTF-8 handling in protocol parser** - use lossy conversion to handle partial UTF-8 sequences at buffer boundaries - **Rate limiter cleanup on panic** - use RAII guard to ensure slot release even if task panics - **PSK length validation** - reject PSKs over 1024 bytes and empty PSKs to prevent abuse - **UDP per-stream bitrate underflow** - clamp to minimum 1 bps when total bitrate > 0 to prevent unlimited mode - **QUIC stream accept timeout** - add 30-second timeout and cancellation support to prevent infinite hangs - **active_tests cleanup order** - cleanup before result write to prevent stale entries on connection failure ### Testing - Added regression test for UDP bitrate underflow bug ### Code Quality - Added SAFETY comments to all 4 unsafe blocks (tcp_info.rs, tcp.rs, net.rs) ### Documentation - Added server memory footprint guide to README - Clarified Windows support is experimental (WSL2 recommended) - Emphasized PSK requirement for QUIC on untrusted networks - Fixed buffer size documentation (256KB → 128KB) - Fixed UDP bitrate default documentation (1 Gbps, not unlimited; use `-b 0` for unlimited) - Documented that data ports are unauthenticated by design - Updated KNOWN_ISSUES with QUIC cert verification, protocol versioning, Windows limitations - Added KNOWN_ISSUES entries for UDP data plane spoofing risk and IPv6 zone ID limitation - Added pre-1.0 roadmap items (structured errors, code refactoring, fuzz testing) - Expanded roadmap with security enhancements, testing, and optimization items ## [0.4.3] - 2026-02-04 ### Added - **In-app update notifications** - checks GitHub releases in background, shows banner with update command - Detects install method (Homebrew, Cargo, binary) and shows appropriate update command - Press `u` to dismiss the banner - Uses 1-hour cache to avoid GitHub API rate limits - **Android/Termux support** - pre-built `aarch64-linux-android` binary in releases - **Infinite duration mode** (`-t 0`) - run test indefinitely until manually stopped with `q` or Ctrl+C - **Local bind address** (`--bind`) - bind to specific local IP or IP:port for multi-homed hosts - **Theme hint in footer** - `[t] Theme` now shown in TUI status bar ### Fixed - **UDP cross-platform compatibility** (issue #10) - UDP mode now works between Linux server and macOS client - Client UDP socket now matches server's address family instead of using dual-stack - macOS handles dual-stack sockets differently than Linux; this fix ensures compatibility - **`--bind` with explicit port** - disallows explicit port for TCP (control + data connections would conflict) - **Non-blocking connect** - added EWOULDBLOCK to Unix error handling for edge cases ## [0.4.2] - 2026-02-03 ### Changed - Removed Intel Mac (x86_64-apple-darwin) pre-built binary - use `cargo install xfr` ### Fixed - UDP download/bidir race condition: client now retries hello packets ## [0.4.1] - 2026-02-03 ### Added - `-Q` short flag for `--quic` mode (uppercase to distinguish from `-q` quiet) - Security documentation section in README - 30-second timeout for control-plane handshakes (prevents DoS from idle connections) - KNOWN_ISSUES.md documenting edge cases and limitations - "quic" capability in server Hello message ### Changed - `-b/--bitrate` help text clarifies it only applies to UDP - Log messages now go to stderr instead of stdout (allows clean JSON/CSV piping) - Minimum test duration enforced at 1 second - Protocol documentation corrected (newline-delimited JSON, not length-prefixed) ### Fixed - UDP reverse mode (`-u -R`) now works - server learns client address before sending - UDP bidirectional mode on server now waits for client before sending - Division by zero guard in throughput calculations for zero-duration tests - Overflow protection in bitrate/size parsing (checked_mul instead of unchecked) - Conflicting CLI flags now produce clear errors (`--quic --udp`, `--bidir --reverse`, `--json --csv`) - Parallel streams validated to 1-128 range (prevents `-P 0` crash) - Documentation: corrected JSON field names in SCRIPTING.md (bytes_total, duration_ms) - Documentation: fixed Prometheus flag name in FEATURES.md (--prometheus not --prometheus-port) - QUIC bitrate warning now logged like TCP when -b flag is ignored ## [0.4.0] - 2026-02-02 ### Added - **Per-stream detail view** (`d` key) - toggle between History and Streams panels for multi-stream tests - Shows per-stream throughput bars with retransmit counts - Uses existing StreamBar widget for consistent visualization - **Pause overlay** - prominent "PAUSED" overlay when test is paused (not just footer text) - **History event logging** - automatically logs significant events: - Peak throughput moments (10%+ above previous peak) - Retransmit spikes (10+ in an interval) - UDP packet loss events (5+ packets lost) - **Settings modal** (`s` key) - adjust display and test settings on the fly: - Display tab: Theme, Timestamp format, Units (auto-persist) - Test tab: Streams, Protocol, Duration, Direction (session-only) - Vim-style navigation (j/k/h/l) and arrow keys supported - Tab key switches between categories - **QUIC transport** (`--quic`) via quinn crate - built-in TLS 1.3 encryption, stream multiplexing - **TUI visual overhaul** - cleaner design inspired by ttl: - Styled title bar and footer with status display - Completion results shown as centered modal overlay - Color-coded metrics (retransmits, loss, jitter, RTT) - Simplified layout with single outer border - **Documentation overhaul:** - `docs/COMPARISON.md` - Feature matrix vs iperf3/iperf2/rperf/nperf, migration guide - `docs/SCRIPTING.md` - CI/CD examples, Docker usage, Prometheus integration - `docs/FEATURES.md` - Comprehensive feature reference - `docs/ARCHITECTURE.md` - Module structure, data flow, threading model - Real-world use cases section in README - Enhanced module-level rustdoc for lib.rs, protocol.rs, quic.rs - `install.sh` - Cross-platform installer script (Linux/macOS) - `.github/FUNDING.yml` - Ko-fi sponsor link - QUIC supports upload, download, bidirectional, and multi-stream modes - QUIC works with PSK authentication ### Removed - TLS over TCP (`--tls`, `--tls-cert`, `--tls-key`, `--tls-ca`) - replaced by QUIC which provides built-in encryption ### Fixed - Use VecDeque for interval history to avoid O(n) removal on every interval - Report per-interval retransmit deltas instead of cumulative totals - Suppress console logging in TUI mode to prevent log messages corrupting display - Help modal now opens after test completion (was blocked by key handler) - Settings modal button cycling when Apply is hidden (only Close visible) - Rate limiter atomic underflow race condition (use fetch_update instead of fetch_sub) - Progress bar width now adaptive to terminal size (removed arbitrary 40-char max) ### Security - Add semaphore to limit concurrent handlers (defense against connection floods on public servers) ### Changed - Use constant for QUIC buffer size (consistency with TCP module) ### Dependencies - hyper-util 0.1.19 → 0.1.20 - slab 0.4.11 → 0.4.12 - unicode-width 0.2.0 → 0.2.2 - zmij 1.0.18 → 1.0.19 ### Testing - Add integration tests: UDP bidir, UDP multi-stream, QUIC bidir, ACL deny, IPv6 localhost - Add protocol parsing benchmarks (serialize/deserialize for Hello, TestStart, Interval, Result) ## [0.3.0] - 2026-01-31 ### Added - **Server TUI dashboard** (`xfr serve --tui`): - Real-time display of active tests, bandwidth, and client information - Shows blocked connections and authentication failures - Uptime counter and total test statistics - Help overlay with `?` key - Timestamp format options (`--timestamp-format`) - relative, iso8601, or unix epoch - Prometheus Push Gateway support (`--push-gateway`) for pushing metrics at test completion - File logging with daily rotation (`--log-file`, `--log-level`) - Integration tests for TCP, UDP, download, bidir, and multi-client modes - **Security & Enterprise features:** - Pre-shared key (PSK) authentication (`--psk`, `--psk-file`) - Per-IP rate limiting (`--rate-limit`, `--rate-limit-window`) - IP access control lists (`--allow`, `--deny`, `--acl-file`) - **TUI Improvements:** - 11 color themes: default, kawaii, cyber, dracula, monochrome, matrix, nord, gruvbox, catppuccin, tokyo_night, solarized - Theme selection via `--theme` flag or config file - Press `t` to cycle themes during TUI session - UDP-specific stats display (jitter, packet loss %) in TUI - Target bitrate shown in header for UDP mode - **Preferences persistence** (`~/.config/xfr/prefs.toml`) - Auto-saves last used theme - Remembers user preferences across sessions - **Full IPv6 support:** - `-4`/`--ipv4` and `-6`/`--ipv6` flags for address family selection - Dual-stack mode (default): server accepts both IPv4 and IPv6 clients - Proper `IPV6_V6ONLY` socket option handling via socket2 - ACL normalizes IPv4-mapped IPv6 addresses for consistent rule matching - CSV output format (`--csv`) - JSON streaming output (`--json-stream`) for real-time per-interval JSON - Quiet mode (`-q/--quiet`) to suppress interval output - Custom report interval (`-i/--interval`) - Omit option (`--omit`) to skip initial TCP ramp-up seconds - Server max duration (`--max-duration`) for server-side test limits - Environment variable support: `XFR_PORT` and `XFR_DURATION` - mDNS service registration on server start for discovery - send_data_half/receive_data_half for split socket operations ### Fixed - Stats now shared correctly between handlers and TestStats (real-time intervals) - Bidirectional mode properly splits sockets for concurrent send/receive - UDP receive uses recv_from for unconnected sockets - Server signals cancel when test duration elapses - Client control loop has 30s timeout to prevent hangs - Dynamic port allocation prevents multi-client port collisions - Hostname parsing provides proper error messages - Interval history bounded to 60 entries to prevent memory growth - Client cancel() method now functional and sends Cancel to server - Protocol version checking uses proper comparison function - Hostname resolution uses resolved IP from control connection for data streams - TUI stream throughput uses correct 1-second interval calculation - CSV header columns now match data format - Socket buffer tuning logs failures at debug level ### Security - Bounded control channel read to prevent memory DoS (max 8KB lines) - Validate stream count against MAX_STREAMS (128) - Enforce MAX_TEST_DURATION (1 hour) and server max_duration - Add 10s timeout on TCP data connection accept ### Changed - Replaced emoji indicators with ASCII [OK]/[WARN]/[FAIL] for terminal compatibility - `BackendType::from_str` and `AddressFamily::from_str` now implement `std::str::FromStr` trait ### Removed - Unused dependencies: thiserror, bytesize, rand ### Dependencies - ratatui 0.29 → 0.30 (fixes lru and paste warnings) - crossterm 0.28 → 0.29 - prometheus 0.13 → 0.14 (fixes protobuf vulnerability RUSTSEC-2024-0437) - mdns-sd 0.11 → 0.17 - toml 0.8 → 0.9 - dirs 5 → 6 - webpki-roots 0.26 → 1.0 - rand 0.8 → 0.9 - ipnetwork 0.20 → 0.21 - criterion 0.5 → 0.6 ## [0.2.0] - 2026-01-31 ### Added - Full Prometheus metrics with per-stream and TCP stats - Config file support (`~/.config/xfr/config.toml`) - Server presets for bandwidth limits and client restrictions - Grafana dashboard template (`examples/grafana-dashboard.json`) - Man page (`doc/xfr.1`) - CONTRIBUTING.md guide ### Changed - Prometheus metrics now properly registered and updated - CLI arguments can be overridden by config file defaults ## [0.1.2] - 2026-01-31 ### Added - Dual MIT/Apache-2.0 licensing - README documentation ### Fixed - License attribution ## [0.1.1] - 2026-01-31 [YANKED] ### Added - Initial README ## [0.1.0] - 2026-01-31 [YANKED] ### Added - Initial release - TCP bandwidth testing with configurable streams - UDP mode with bitrate limiting and jitter calculation - Live TUI with real-time throughput graphs - Multi-client server support - Reverse and bidirectional testing modes - JSON output format - Plain text output format - Prometheus metrics export (optional feature) - mDNS LAN discovery (optional feature) - `xfr diff` command to compare test results - TCP_INFO stats on Linux and macOS - Configurable TCP window size and nodelay [0.9.2]: https://github.com/lance0/xfr/compare/v0.9.1...v0.9.2 [0.9.1]: https://github.com/lance0/xfr/compare/v0.8.0...v0.9.1 [0.8.0]: https://github.com/lance0/xfr/compare/v0.7.1...v0.8.0 [0.7.1]: https://github.com/lance0/xfr/compare/v0.7.0...v0.7.1 [0.7.0]: https://github.com/lance0/xfr/compare/v0.6.1...v0.7.0 [0.6.1]: https://github.com/lance0/xfr/compare/v0.6.0...v0.6.1 [0.6.0]: https://github.com/lance0/xfr/compare/v0.5.0...v0.6.0 [0.5.0]: https://github.com/lance0/xfr/compare/v0.4.4...v0.5.0 [0.4.4]: https://github.com/lance0/xfr/compare/v0.4.3...v0.4.4 [0.4.3]: https://github.com/lance0/xfr/compare/v0.4.2...v0.4.3 [0.4.2]: https://github.com/lance0/xfr/compare/v0.4.1...v0.4.2 [0.4.1]: https://github.com/lance0/xfr/compare/v0.4.0...v0.4.1 [0.4.0]: https://github.com/lance0/xfr/compare/v0.3.0...v0.4.0 [0.3.0]: https://github.com/lance0/xfr/compare/v0.2.0...v0.3.0 [0.2.0]: https://github.com/lance0/xfr/compare/v0.1.2...v0.2.0 [0.1.2]: https://github.com/lance0/xfr/compare/v0.1.1...v0.1.2 [0.1.1]: https://github.com/lance0/xfr/compare/v0.1.0...v0.1.1 [0.1.0]: https://github.com/lance0/xfr/releases/tag/v0.1.0