Networking & Protocols
Session affinity, consistent hashing, and the right fix
A user logs in. Their session lives on backend B2. The next request routes to B3. Session not found. They are logged out mid-flow. This is what happens when your application stores state on a single backend and you rely on session affinity to hide it — until affinity breaks.
Why sticky sessions exist
Some applications store session state on the backend — in memory, or in a local file. If the next request routes to a different backend, the state is missing: the user is logged out, the shopping cart is empty, the upload context is gone.
Sticky sessions (session affinity) make the LB route all requests from one client to the same backend. The client’s requests are “pinned” to one server.
Cookie-based affinity
The LB sets a cookie on the first response:
Set-Cookie: LB_ROUTE=backend_2; Path=/On every subsequent request, the browser sends this cookie. The LB reads it and routes to backend_2.
Advantages:
- Survives client-IP changes (mobile handover, NAT failover).
- Allows graceful failover: if
backend_2is unhealthy, the LB can pick a new backend and update the cookie.
Disadvantage: Cookie-based affinity requires HTTP. Non-HTTP protocols cannot use it.
IP-hash affinity
Route by Hash(client_IP) % num_backends. No cookie needed — works for any protocol.
Disadvantages:
- Breaks when the client IP changes (mobile handover, VPN reconnect).
- Clusters badly: 50 clients behind one corporate NAT all hash to the same backend — that backend gets 50× the load of others.
- Load imbalance with sticky sessions
- 1.5–2.5x worse
- Session loss on pinned backend crash
- 100% for that session
- Cookie-based: survives IP change
- yes
- IP-hash: survives IP change
- no
- IP-hash with 50 clients behind NAT
- all 50 hit one backend
- Right fix: session TTL in Redis
- any backend can resume
The right fix: externalized session state
Store session data in a distributed cache (Redis, Memcached, DynamoDB) keyed by session ID. Every backend reads from and writes to this store.
Now the LB can route requests freely using power-of-two-choices. No affinity needed. No pinning. Session data survives backend failures because it is in Redis, not in memory on one backend.
This is the pattern used at Netflix, Airbnb, Shopify, and any service that scales horizontally.
Consistent hashing: for cache locality, not session state
Session affinity is about user state. Consistent hashing is about cache locality: routing a cache key to the same backend so you get a cache hit instead of a miss.
Problem with round-robin for caching:
- Request
user:42→ B1 (cached there). - Next request
user:42→ B2 (cache miss, fetches from DB). - No locality, no benefit from caching.
Consistent hashing maps each key to a point on a hash ring. Backends are also mapped to points (with many virtual nodes per backend to spread them evenly, ~150–300). A key routes to the nearest backend clockwise on the ring.
Key property: When a backend joins or leaves, only ~1/N keys remap. All other keys stay on the same backend. This minimizes cache disruption on topology changes.
Lookup cost: O(log N) with a sorted tree (binary search over virtual node positions).
Use consistent hashing for:
- Distributed caches (Memcached shards, Redis cluster).
- Database sharding (route by shard key).
- Sticky request routing for cache locality (media encoding jobs, analytics aggregations).
Do not use consistent hashing for general request balancing — power-of-two-choices adapts to real-time load, consistent hashing does not.
Rendezvous hashing (highest-random-weight)
Alternative to ring-based consistent hashing:
- For each backend, compute
Hash(key, backend_id). - Route to the backend with the highest hash value.
No ring, no virtual nodes — simpler to implement. O(N) per lookup but for small N (<100 backends) the difference is negligible. Hash distribution is often more uniform than ring-based consistent hashing in practice. Some CDNs and Facebook’s TAO use rendezvous hashing for sharding.
Why this works
Why virtual nodes matter. Without virtual nodes each backend occupies one arc on the ring. With 4 backends (A, B, C, D) placed at 0, 90, 180, 270 degrees, the arcs are perfectly even — but that is the ideal case. In practice, hash(backend_id) clusters backends unevenly. Virtual nodes map each backend to 150–300 positions on the ring, spreading it into 150–300 small arcs. This averages out the distribution so no single backend claims a disproportionately large arc.
Order the steps of session affinity failover (cookie-based) showing why it fails without Redis:
- 1 Client sends a request with cookie LB_ROUTE=backend_2.
- 2 LB reads the cookie and routes to backend_2.
- 3 Backend_2 crashes or becomes unhealthy.
- 4 LB stops routing new requests to backend_2 but cannot migrate the existing session.
- 5 Client's session is lost because only backend_2 held it in memory.
- 6 Client retries; LB picks a new backend, but the session data is gone.
- 7 Right fix: store session in Redis, accessible from any backend, so failover is transparent.
What is the main drawback of IP-hash session affinity compared to cookie-based affinity?
When a backend is removed from a consistent-hash ring, what fraction of keys must remap to a new backend?
- 01Why is session affinity considered an anti-pattern, and what is the correct fix?
- 02What problem does consistent hashing solve that round-robin cannot?
- 03What are virtual nodes in consistent hashing and why are they necessary?
Session affinity routes each client to one backend using a cookie (LB_ROUTE=backend_2) or IP hash. Cookie-based affinity survives IP changes; IP-hash breaks on IP changes and clusters badly behind NAT. Both cause 1.5–2.5× worse load imbalance and lose sessions on backend failure — they are workarounds, not solutions. The correct architecture: externalize session state to Redis so any backend can resume any session and the LB can route freely. Consistent hashing is a different tool for cache locality: it maps a key to the same backend with ~1/N remapping on membership change, using virtual nodes (150–300 per backend) to even out the ring distribution. Use consistent hashing for caches and sharding; use power-of-two-choices for live request balancing.
appears again in152
- 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
- Why idempotency: making retries safejunior
- Server-side state machine: four states of an idempotency keymiddle
- 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
- 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
- CLS: why layout shifts happen and how to stop themmiddle
- 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
- Index-only scans, the Visibility Map, and INCLUDEsenior
- Production failure modes and the index audit playbooksenior
- pg_statistic, ANALYZE, and production observabilitymiddle
- 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
- What the three signals are: logs, metrics, and tracesjunior
- Why structured logs exist: the diary vs the spreadsheetjunior
- The production log schema: fields every line must carrymiddle
- PII redaction and log injectionsenior
- OTel Logs Data Model and audit logs as a subsystemsenior
- SLI, SLO, and the error budget: reliability by the numbersjunior
- Error budget policy, latency SLOs, and composite journeysmiddle
- Production SLO failures, self-observability, security, and the big picturesenior
- The incident loop: from pager to postmortem to preventionmiddle
- Cache lines, struct layout, and false sharingmiddle
- SIMD, SoA vs AoS, and memory bandwidthmiddle
- Cache-oblivious algorithms, PGO, and production failuressenior
- GC in production: observability, security, edge cases, and fleet governancesenior
- Batching: amortize fixed cost per operationjunior
- The batching window: size and wait timemiddle
- Batching in Kafka and Postgresmiddle
- io_uring and observability of batchingmiddle
- From Nagle to io_uring: evolution of batchingmiddle
- Backpressure, failure isolation, and batch security in productionsenior
- CI enforcement and RUM: making budgets stickmiddle
- V8 JIT pipeline, HTTP priorities, and bundle securitysenior
- The performance loop: discipline, not a projectjunior
- Classify and fix: matching bottleneck families to remediesmiddle
- Observability stack and CI gates: catching regressions before they shipmiddle
- Incident to enforcement: SLO burn to verified fix in 35 minutesmiddle
- Culture, economics, and org-scale performancesenior
- 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