Performance
From Nagle to io_uring: evolution of batching
In 1984, John Nagle watched a single telnet keystroke leave a host as a 41-byte packet: one byte of payload wrapped in a 40-byte TCP/IP header. On the congested ARPANET that was 97.5% waste, and thousands of those tinygrams were melting links. His fix was three lines of TCP logic. Forty years later you tune the exact same lever every time you set linger.ms in Kafka or batch SQEs into one io_uring_enter. The fixed cost changed names; the move never did.
The batching insight is timeless: when a fixed cost per operation dominates the variable cost of the payload — a TCP header, a network round-trip, a syscall context switch, a DB connection acquire — group operations so you pay the fixed cost once instead of N times. Everything in this lesson is that one idea, applied at a different layer of the stack across four decades. The only real design knob is the window: how long to wait, or how full to get, before you flush.
Nagle’s algorithm: the original batching tradeoff
RFC 896 (1984) introduced what we now call Nagle’s algorithm. The rule is small: while there is unacknowledged data already in flight, hold back any new small segment and coalesce it with later writes; flush immediately only when an ACK clears the in-flight data, or when you have a full MSS worth of bytes to send. The motivation was brutal arithmetic — a 1-byte telnet keystroke became a 41-byte packet, so 40 of every 41 bytes on the wire were header. Nagle’s rule turned a burst of keystrokes into one packet per round-trip instead of one packet per key.
The cost lands on interactive and request/response traffic. If your application does a small write() and then waits for a reply, Nagle may sit on that last small segment, hoping for more data that never comes — so it waits for the ACK instead, adding up to a full round-trip of dead time. The escape hatch is the TCP_NODELAY socket option, which disables the algorithm so every write goes out immediately. This is why HTTP/2, gRPC, Redis clients, and basically every modern RPC stack set TCP_NODELAY at connect time and do their own batching at the application layer, where they actually know message boundaries.
The Nagle + delayed-ACK deadlock (the postmortem beat)
The famous failure mode is not Nagle alone — it is Nagle interacting with TCP delayed ACK. Delayed ACK is the receiver-side mirror of Nagle: instead of ACKing every segment, the receiver waits (Linux default: up to ~40ms) hoping to piggyback the ACK onto a reply or batch it with the next one. Now compose the two. The sender writes a response slightly larger than one MSS: the first full segment goes out, but the small trailing segment is held by Nagle because the first is still unacknowledged. The receiver got the first segment but holds its ACK under delayed-ACK, waiting to piggyback. Neither side will move. The deadlock breaks only when the 40ms delayed-ACK timer fires.
The symptom in production is unmistakable and infuriating: a protocol that should do thousands of transactions per second mysteriously caps near 25/sec, with latency histograms spiking at a suspiciously round 40ms (or 200ms on some stacks). Marc Brooker’s line — “It’s always TCP_NODELAY. Every damn time.” — is folklore for a reason. The fix is one socket option; the diagnosis is the hard part, because the 40ms is paid by nobody’s CPU and shows up only as wall-clock stall.
The latency-throughput Pareto frontier
Strip away the layer-specific details and every batching system traces the same curve. On one axis: batch window (time or size). On the other two: throughput and per-item latency, which move in opposition.
| Operating point | Window | Per-item latency | Throughput |
|---|---|---|---|
| No batching | window = 0 | Minimum (send now) | Capped by fixed cost per op |
| SLO operating point | largest window with p99 < SLO | At the SLO ceiling | Near-max under that ceiling |
| Infinite batching | window = latency = ∞ | Unbounded (first item never flushes) | Max theoretical |
The senior workflow is not “pick a number” — it is: define the latency SLO ceiling first (say p99 < 50ms), then find the largest batch window that still fits under it, because that window gives you the most throughput you can buy without breaking the contract. Static systems tune this knob once and live with it. Adaptive systems track the curve at runtime: under light load they shrink the window toward zero (latency matters, there is nothing to batch anyway); under heavy load they let batches fill (throughput matters, and items are arriving fast enough that the wait is cheap). Kafka 4.0 quietly encoded this wisdom: the producer linger.ms default moved from 0 to 5ms, because the efficiency win from fuller batches usually pays for the 5ms wait — frequently yielding lower end-to-end latency, not higher, by reducing per-request overhead.
Batch coalescing and request deduplication
There is a sharper variant for cache/lookup workloads where many callers want the same result, not just any throughput. When concurrent requests miss the cache for the same key, you can collapse them into a single inflight load instead of N duplicate loads. Worker A misses key K and starts the DB query; worker B (and C, and D…) also miss K, see A’s request already pending, and attach to it rather than firing their own. One result fans out to all of them.
This is the cure for the cache stampede / thundering herd: a hot key expires, 100 requests arrive in the same millisecond, and without coalescing all 100 hammer the database at once — often enough to knock it over right when traffic is highest. With coalescing, those 100 misses become 1 query and 99 free-riders. The implementations are everywhere under different names: Go’s golang.org/x/sync/singleflight, Java’s Caffeine AsyncLoadingCache, request collapsing in Varnish and most CDNs. GraphQL’s DataLoader takes it one step further by combining coalescing with windowed batch loading: every distinct key requested within one tick is deduped and the unique keys are bundled into a single batched backend query, which is also how DataLoader kills the N+1 query problem.
Why this works
Coalescing and Nagle look different but share a spine. Nagle merges writes in time on one connection to amortize header cost. Singleflight merges reads across callers on one key to amortize a backend query. Both answer the same question — “several small things want the same expensive operation; can one trip serve them all?” — just along different dimensions (time vs. identity).
Why the lineage matters
The whole point of seeing Nagle, Kafka, and io_uring as one family is that the tuning method transfers. Each is just a different fixed cost wrapped in the same lever, so the diagnostic question is identical every time.
| Era | System | Fixed cost amortized | Window knob |
|---|---|---|---|
| 1984 | Nagle / RFC 896 | 40-byte TCP/IP header per segment | in-flight ACK or full MSS |
| 2011 | Kafka producer linger.ms | network round-trip + broker request | linger.ms + batch.size |
| 2019 | io_uring (Linux 5.1) | syscall + context switch into kernel | SQEs queued before one io_uring_enter |
io_uring is the cleanest modern echo: instead of one syscall per I/O, you fill a ring of submission queue entries (SQEs) in shared memory and submit a whole batch with a single io_uring_enter — amortizing the kernel-boundary crossing across many operations, exactly as Nagle amortized the header across many keystrokes. Same lever, new fixed cost. Once you see the pattern, the work is always the same three steps: measure the fixed cost vs. the variable cost, confirm the fixed cost actually dominates, then size the window to the largest value your latency SLO allows.
A request/response service over TCP mysteriously caps near 25 transactions/sec, with latencies clustered at exactly 40ms. What's the most likely cause?
A hot cache key expires and 100 concurrent requests miss it in the same millisecond. What does request coalescing (singleflight) do?
Order the senior workflow for tuning any batch window:
- 1 Measure the fixed cost per op vs. the variable cost of the payload
- 2 Confirm the fixed cost actually dominates (else batching buys little)
- 3 Define the latency SLO ceiling first (e.g. p99 under 50ms)
- 4 Find the largest batch window that still fits under that ceiling
- 5 Decide static vs. adaptive: fixed knob, or shrink/grow the window with load
A low-traffic internal RPC service does small request/response calls and is hitting a fixed ~40ms latency floor per call. Pick the fix a senior defends.
- 01What problem did Nagle's algorithm solve, what is the mechanism, and how does the classic deadlock arise?
- 02Explain the lineage Nagle → Kafka linger.ms → io_uring as one pattern, and the workflow for tuning the window.
Batching is one lever applied across four decades: when a fixed per-operation cost (TCP header, round-trip, syscall) dominates the variable cost of the payload, group operations to pay it once. Nagle’s algorithm (1984) coalesced tiny TCP writes until an ACK or a full MSS, and its famous deadlock with delayed ACK pins latency at 40ms until TCP_NODELAY turns it off. Kafka’s linger.ms and io_uring’s batched SQE submission are the same move at higher layers, which is why the tuning workflow never changes: measure fixed vs. variable cost, define the latency SLO ceiling, then take the largest window that fits under it — statically or adaptively. Request coalescing (singleflight, Caffeine, DataLoader) applies the idea across callers instead of over time, collapsing a thundering herd of identical cache misses into a single backend load.
appears again in260
- Why GraphQL gets N+1junior
- DataLoader mechanics: tick-boundary batchingmiddle
- Batch function contracts: ordering, shapes, errorsmiddle
- Federation and lookahead: batching beyond DataLoadermiddle
- Query complexity defences: depth, cost, persisted queriesmiddle
- Senior GraphQL API: scheduling contract, tenant isolation, observabilitysenior
- The journey of a request: seven stops from socket to responsejunior
- Accept and parse: from kernel queue to a typed requestmiddle
- Routing and middleware: choosing what runs, and in what ordermiddle
- Handler and response: from business logic to bytes on the wiremiddle
- Streaming and backpressure: when the client reads slower than you writesenior
- Timeouts and tail latency: budgets, deadlines, and the fan-out trapsenior
- Middleware and DI: the two patterns that shape every backendjunior
- Writing middleware: signatures, next(), and the three framework modelsmiddle
- Inversion of control: how dependencies reach a classmiddle
- DI scopes and lifecycles: singleton, request, transientmiddle
- DI as a testing seam: fakes, mocks, and the boundary that matterssenior
- DI containers in production: resolution graphs, circular deps, and when not tosenior
- Blocking vs non-blocking I/O: two ways to waitjunior
- The event loop: one thread, ordered phasesmiddle
- What blocks the loop: CPU work and sync callsmiddle
- Offloading CPU work: worker threads and the libuv poolmiddle
- Backpressure and bounded concurrencysenior
- Throughput under load: tail latency and saturationsenior
- Why pool: the cost of creating a connectionjunior
- Pool sizing: why bigger is not fastermiddle
- Acquisition and timeouts: the wait queue is the real latency dialmiddle
- Why idempotency: making retries safejunior
- Server-side state machine: four states of an idempotency keymiddle
- Retry strategies: backoff, jitter, and thundering herdmiddle
- Outbox and inbox: effectively-once across the dual-write boundarymiddle
- Concurrency and cache architecture for idempotency at scalesenior
- Observability, production failures, and global-scale designsenior
- The event loop: one thread, three queuesjunior
- Tasks, microtasks, and scheduler.yield()middle
- Timer accuracy, throttling, and idle workmiddle
- Microtask starvation, Long Tasks, and LoAFsenior
- Node.js event loop: phases, nextTick, and loop lagsenior
- React, Vue, and INP observability in productionsenior
- The render pipeline: six stages from bytes to pixelsjunior
- Stage costs and the renderer process modelmiddle
- Invalidation, dirty bits, and containmiddle
- Compositor layers: promotion, overlap, and GPU memorymiddle
- DevTools flame strip and the frame lifecyclemiddle
- Layout thrash: forced synchronous layoutsenior
- BeginMainFrame, compositor-driven animations, and GPU memorysenior
- Production observability: LoAF, INP, and the full attack surfacesenior
- What V8 is and why performance varies 100×junior
- V8''''s four-tier JIT pipeline and profile-guided tieringmiddle
- Hidden classes, transition trees, and memory layoutmiddle
- Inline caches, IC states, and deoptimizationmiddle
- Orinoco GC: parallel scavenger, concurrent marking, and write barriersmiddle
- TurboFan''''s speculative engine and the deopt-loop trapsenior
- V8 in production: isolates, pointer compression, and real failuressenior
- Service worker lifecycle and cache strategiesmiddle
- Service worker edge cases: version skew, durability, and navigation trapssenior
- What the reconciler does: render vs commitjunior
- The fiber object and the double-buffer treemiddle
- Render phase purity and commit phase sub-stepsmiddle
- Reconciliation: diffing heuristics and the key trapmiddle
- Priority lanes, time-slicing, and useTransitionmiddle
- Bailout, memoisation, and tearingsenior
- React Profiler, the Compiler, and production observabilitysenior
- Rendering strategies: SSG, SSR, ISR, streaming, and hydrationjunior
- SSG, SSR, ISR, streaming, and RSC — how each worksmiddle
- Hydration cost: selective, progressive, islands, resumabilitymiddle
- Hydration mismatch: causes, detection, and the determinism rulesenior
- RSC, per-route strategy, and production observabilitysenior
- Core Web Vitals: what LCP, INP, and CLS measurejunior
- LCP: four phases, one dominant costmiddle
- INP: input delay, processing, presentationmiddle
- CLS: why layout shifts happen and how to stop themmiddle
- Lab vs field: why the two disagree and how to use eachmiddle
- Metric tradeoffs, RUM attribution, and the CI+field loopsenior
- The full picture: URL to LCP to INP as a relay racejunior
- Eight layers traced: from the service worker to the second navigationmiddle
- Five canonical breaks: where production reliably diessenior
- The three-track method: reading traces and building a monitored systemsenior
- What is a cache stampede and why it makes things worsejunior
- Lock and single-flight: bounding concurrent rebuildsmiddle
- XFetch: coordination-free probabilistic early expirationmiddle
- Stale-while-revalidate and CDN request coalescingmiddle
- Detecting stampedes and designing TTL for productionmiddle
- Metastable failure, fencing tokens, and production postmortemssenior
- What a relation is: tables, rows, keys, and constraintsjunior
- Constraints, keys, and Postgres data typesmiddle
- Normal forms, denormalization, and why schemas stickmiddle
- JSONB, arrays, and when a side table winsmiddle
- Heap storage, TOAST, and column alignmentsenior
- Schema integrity: deferral, versioning, and production failure modessenior
- Relational vs document, wide-column, graph, and key-valuesenior
- What an index is and how it speeds up queriesjunior
- The leading-column rule and composite index designmiddle
- Partial, expression, and covering indexesmiddle
- Index types: GIN, GiST, BRIN, Hash, Bloom, and HOT updatesmiddle
- Index-only scans, the Visibility Map, and INCLUDEsenior
- Production failure modes and the index audit playbooksenior
- Index design exercise: full-text search strategysenior
- EXPLAIN and execution plans: what the planner decides and whyjunior
- Scan types: Seq, Index, Bitmap, Index-Onlymiddle
- Join algorithms and the row-estimate cascademiddle
- pg_statistic, ANALYZE, and production observabilitymiddle
- Extended statistics: fixing correlated-column estimate failuressenior
- Plan cache, cost-constant tuning, and planner internalssenior
- Production failure modes and plan stabilitysenior
- MVCC: why readers and writers never wait for each otherjunior
- Row versions and snapshots: the on-disk mechanicsmiddle
- HOT updates and isolation levels: what you gain and what you paymiddle
- Vacuum and bloat: keeping the storage tax boundedmiddle
- CLOG, XID wraparound, and MultiXact: deep visibility internalssenior
- SSI internals and production autovacuum tuningsenior
- Real-world MVCC failures, deployment patterns, and distributed snapshotssenior
- Connection pools: amortising the cost of a Postgres backendjunior
- PgBouncer session, transaction, and statement modesmiddle
- Pool sizing: the (cores × 2) + spindles formula and the two-layer stackmiddle
- Pool exhaustion and idle-in-transaction: the 3 AM failure modemiddle
- Migrating to transaction mode: rollout playbook and PgBouncer 1.21 prepared statementsmiddle
- The Postgres process model and why raising max_connections degrades throughputsenior
- Pooler landscape 2026, serverless connection storms, and the full failure-mode taxonomysenior
- What a schema migration is and why it replaces ad-hoc DDLjunior
- ADD COLUMN: instant in PG 11+ vs rewrite in older Postgresjunior
- The lock-queue failure mode: why instant DDL can freeze the databasemiddle
- Safe DDL patterns: NOT VALID, CONCURRENTLY, and unsafe-op fixesmiddle
- Expand-contract: zero-downtime for breaking schema changesmiddle
- Advisory locks, migration tools, and deploy coordinationsenior
- Migration failure taxonomy and production disciplinesenior
- Why sharding exists: the single-Postgres ceilingjunior
- Shard-key selection: hash, range, list, and directory strategiesmiddle
- Partitioning vs sharding: same word, two different thingsmiddle
- Co-location and Citus: the invariant that makes sharding usablemiddle
- The hot-shard failure mode: detection, isolation, and durable policymiddle
- Schema-based sharding and multi-tenancy alternativessenior
- Online resharding, 2PC, and the operational cost of shardingsenior
- The seven acts: from CREATE TABLE to Citusjunior
- Acts 1–3 in depth: schema, indexes, and planner statisticsmiddle
- Acts 4–6 in depth: MVCC bloat, connection pooling, and safe migrationsmiddle
- Act 7 in depth: sharding, co-location, and the seven-tier tradeoff cascademiddle
- Observability, anti-patterns, and production triagesenior
- Raft roles, terms, and why majority quorums prevent split brainjunior
- How Raft replicates a log entry and decides it is safe to commitmiddle
- Raft leader election: timeouts, voting rules, and the four safety propertiesmiddle
- Raft in the real world: partitions, slow disks, and client routingmiddle
- Raft extensions: pre-vote, learners, snapshots, and linearizable readssenior
- Raft in production: membership changes, Multi-Raft, and observabilitysenior
- Where data fetching happens — and why it decides LCPjunior
- Fetch waterfalls — diagnosis and the Promise.all curemiddle
- React Server Components and Suspense streamingmiddle
- Client-side cache: TanStack Query, SWR, and stale-while-revalidatemiddle
- LCP, prefetch, and race conditions in interactive fetchingmiddle
- Senior internals: RSC payload, caching layers, and production failure modessenior
- Bits on the wirejunior
- Latency mathmiddle
- Bufferbloat and congestionsenior
- The physical frontiersenior
- The three-way handshakejunior
- Sequence numbers and connection statemiddle
- Flow control and congestion controlmiddle
- BBR, production observability, and beyond TCPsenior
- DNS: what it does and why it existsjunior
- The resolver walk: referrals, record types, and gluemiddle
- TTL, caching, and DNS propagationmiddle
- The 1-RTT handshake: key shares and ECDHEmiddle
- Session resumption and 0-RTTmiddle
- CDN: putting content next doorjunior
- Anycast and GeoDNS: routing to the nearest edgemiddle
- Tiered cache and Cache-Controlmiddle
- Vary header and cache keysmiddle
- Stale-while-revalidate and cache stampedesenior
- Edge workers and edge-side compositionsenior
- CDN operations and observabilitysenior
- WebSocket: the HTTP upgrade handshakejunior
- WebSocket frame format: opcodes, masking, fragmentationmiddle
- WebSocket vs SSE vs long-polling: choosing the right transportmiddle
- WebSocket backpressure: when clients can''''t keep upmiddle
- Reconnection: jittered backoff, thundering herd, message resumptionsenior
- WebSocket at scale: HTTP/2 multiplexing, permessage-deflate, C10Msenior
- WebSocket in production: proxies, security, and distributed architecturesenior
- What reverse proxies dojunior
- Balancing algorithms: round-robin to power-of-two-choicesmiddle
- L4 vs L7 load balancing and client-IP preservationmiddle
- Health checks, connection draining, and slow startmiddle
- Session affinity, consistent hashing, and the right fixmiddle
- Retry storms, circuit breakers, and load sheddingsenior
- Resilient LB architecture: anycast, zone-aware routing, and observabilitysenior
- Why QUIC and not TCP+TLSjunior
- QUIC streams and head-of-line blockingjunior
- Integrated handshake and 1-RTTmiddle
- Connection IDs and network migrationmiddle
- Loss detection and congestion controlmiddle
- 0-RTT resumption and packet encryptionsenior
- Deployment tradeoffs and CPU costsenior
- DDoS: what it is and why it worksjunior
- Amplification attacks and state exhaustionmiddle
- Rate limiting: algorithms and architecturemiddle
- WAFs, firewalls, mTLS, and HSTSmiddle
- DNS cache poisoning and BGP hijackingsenior
- Defense-in-depth architecture and attack economicssenior
- The twelve layers: one URL, seven actorsjunior
- DNS, TCP, TLS in sequence: where the milliseconds gomiddle
- Critical render path and Core Web Vitalsmiddle
- Proxy intercepts and security gates: rate limiters, WAF, mTLSmiddle
- Alternate paths: QUIC 0-RTT, WebSocket upgrade, connection migrationmiddle
- Observability: distributed traces, USE/RED, and samplingsenior
- Resilience: cascading retries, circuit breakers, and error budgetssenior
- What the three signals are: logs, metrics, and tracesjunior
- Metrics and cardinality: the cost model of a time-series databasemiddle
- Logs and volume: the cost model of structured loggingmiddle
- Traces and sampling: the cost model of distributed tracingmiddle
- Join keys and exemplars: making the three signals composemiddle
- Observability 2.0: wide events and the cost shiftsenior
- Failure modes and engineering practice: cardinality budgets, PII, and samplingsenior
- Why structured logs exist: the diary vs the spreadsheetjunior
- The production log schema: fields every line must carrymiddle
- Log levels and alert routingmiddle
- Sampling strategies and log costmiddle
- PII redaction and log injectionsenior
- Trace context propagation in logssenior
- OTel Logs Data Model and audit logs as a subsystemsenior
- OTel signals, Semantic Conventions, and the OTLP wire formatmiddle
- Auto-instrumentation and manual spans: the 80/20 of OTelmiddle
- The OTel Collector: receivers, processors, exporters, and deployment patternsmiddle
- Sampling strategies: head, tail, and parent-basedmiddle
- Vendor neutrality, eBPF instrumentation, the Operator, and browser/serverless OTelsenior
- Operating the OTel Collector: reliability, version skew, failure modes, and governancesenior
- RED and USE: two checklists, one triage disciplinejunior
- Instrumenting RED in Prometheus: counters, histograms, and cardinality disciplinemiddle
- USE on Linux: CPU, memory, disk, network, and PSImiddle
- Golden signals, dashboard layout, and service mesh auto-REDmiddle
- Cardinality as a cost driver: labels, PII, exemplars, and samplingmiddle
- Native histograms, SLO tie-in, and production failure patternsmiddle
- SLI, SLO, and the error budget: reliability by the numbersjunior
- Choosing SLIs and SLO targets: ratios, not feelingsmiddle
- Multi-window multi-burn-rate alerting: why AND beats ORmiddle
- Error budget policy, latency SLOs, and composite journeysmiddle
- Iceberg SLIs, composite SLO math, and SLA vs SLOsenior
- Production SLO failures, self-observability, security, and the big picturesenior
- Flame graphs: reading the picture that shows where time goesjunior
- Sampling vs instrumentation profiling: why 99 Hz wins in productionmiddle
- Profile types: CPU, memory, off-CPU, mutex — which one to reach formiddle
- Continuous profiling: always-on flame graphs with eBPF and trace-id correlationmiddle
- How flame graphs are built from samples, and the production workflows that use themmiddle
- Linux perf, eBPF internals, PGO, and the limits of samplingsenior
- Profiling in production: security, war stories, OTel profiles, and the infrastructure designsenior
- The debugging funnel: SLO → RED → trace → profilejunior
- OTel architecture: one SDK, four signals, one wire formatmiddle
- Cost discipline: keeping observability under 5% of infra spendmiddle
- The incident loop: from pager to postmortem to preventionmiddle
- Scale, security, and the ROI of observable systemssenior
- At-most-once, at-least-once, exactly-once: the three delivery contractsjunior
- The three failure legs — where duplicates and losses actually happenmiddle
- Consumer-side dedup: the cheapest path to exactly-once processingmiddle
- Kafka exactly-once semantics: idempotent producer and transactionsmiddle
- SQS visibility timeout, DLQ, and the outbox patternmiddle
- Exactly-once in production: impossibility proof, hybrid patterns, and real incidentssenior
- What OAuth is and why passwords are not the answerjunior
- Authorization code flow with PKCEmiddle
- ID token validation and JWKS cache managementmiddle
- Refresh token rotation and scope-based least privilegemiddle
- Sender-constrained tokens: DPoP and mTLSsenior
- OAuth in production: audience attacks, observability, and real failuressenior