惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

L
LangChain Blog
Security Latest
Security Latest
P
Proofpoint News Feed
GbyAI
GbyAI
PCI Perspectives
PCI Perspectives
博客园 - Franky
N
Netflix TechBlog - Medium
博客园_首页
WordPress大学
WordPress大学
K
Kaspersky official blog
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
Vercel News
Vercel News
T
Threatpost
The Hacker News
The Hacker News
H
Help Net Security
S
Securelist
Recent Announcements
Recent Announcements
腾讯CDC
T
Tailwind CSS Blog
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
Engineering at Meta
Engineering at Meta
C
Cisco Blogs
V
V2EX
C
Check Point Blog
S
Schneier on Security
Cyberwarzone
Cyberwarzone
C
Cybersecurity and Infrastructure Security Agency CISA
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
B
Blog RSS Feed
H
Hackread – Cybersecurity News, Data Breaches, AI and More
Jina AI
Jina AI
M
MIT News - Artificial intelligence
T
Threat Research - Cisco Blogs
博客园 - 叶小钗
A
Arctic Wolf
AWS News Blog
AWS News Blog
Latest news
Latest news
Martin Fowler
Martin Fowler
Recorded Future
Recorded Future
Last Week in AI
Last Week in AI
The GitHub Blog
The GitHub Blog
小众软件
小众软件
B
Blog
aimingoo的专栏
aimingoo的专栏
C
Cyber Attacks, Cyber Crime and Cyber Security
V
Visual Studio Blog
P
Palo Alto Networks Blog
Spread Privacy
Spread Privacy

DEV Community

Authentication Security Deep Dive: From Brute Force to Salted Hashing (With Java Examples) Why AI Systems Don’t Fail — They Drift Spilling beans for how i learn for exam😁"Reinforcement Learning Cheat Sheet" I Replaced Chrome with Safari for AI Browser Automation. Here's What Broke (and What Finally Worked) How Python Borrows Other People's Work The $40 Architecture: Processing 1 Billion API Requests with 99.99% Uptime Vibe Coding: A Workflow Guide (From Zero to SaaS) Most webhook security guides protect the wrong side. The scary part is delivery. Headless CMS for TanStack Start: Build a Blog with Cosmic EU Age Verification App "Hacked in 2 Minutes" — What Actually Happened Comfy Cloud’s delete function does not actually remove files Running AI Models on GPU Cloud Servers: A Beginner Guide Event-driven media intelligence with AWS Step Functions and Bedrock I scored 500 AI prompts across 8 quality dimensions — here's what broke How to Call Google Gemini API from Next.js (Free Tier, No Backend Needed) The Portal Protocol: Reclaiming Human Connection in the Age of AI How to Fix Your Team's Scattered Knowledge Problem With a Self-Hosted Forum Intro to tc Cloud Functors: A Graph-First Mental Model for the Modern Cloud Designing Multi-Tenant Backends With Both Ownership and Team Access I Built a Neumorphic CSS Library with 77+ Components — Here's What I Learned PostgreSQL Performance Optimization: Why Connection Pooling Is Critical at Scale Cómo construí un SaaS multi-rubro para gestionar expensas en Argentina con FastAPI + Vue 3 🚀 I Built an Ethical Hacking Scanner Tool – Open Source Project I Replaced /usage and /context in Claude Code With a Single Statusline A Pythonic Way to Handle Emails (IMAP/SMTP) with Auto-Discovery and AI-Ready Design I Collected 8.9 Million Polymarket Price Points — Here's What I Found About How Markets Really Move EcoTrack AI — Carbon Footprint Tracker & Dashboard Everyone's Using AI. No One Agrees How. 5 self-hosted ebook managers worth trying in 2026 Building Your First AI Agent with LangChain: From Chatbot to Autonomous Assistant Common SOC 2 Failures (Real World) Stop Vibe-Checking Your AI App: A Practical Guide to Evals How to Use SonarQube and SonarScanner Locally to Level Up Your Code Quality Your Next To-Do App Is Dead — I Replaced Mine with an OpenClaw AI Sign a Nostr event in 60 lines of Python using coincurve — no nostr-sdk, no nbxplorer, no rust toolchain ITGC Audit Explained Like You’re in Big 4 Patch Tuesday abril 2026: Microsoft parcha 163 vulnerabilidades y un zero-day en SharePoint Stop scraping everything: a better way to track competitor price changes Listing on MCPize + the Official MCP Registry while routing payments OUTSIDE the marketplace — how I kept 100% of my x402 revenue Building an AI-Powered Risk Intelligence System Using Serverless Architecture Why We Ripped Function Overloading Out of Our AI Toolchain Testing AI-Generated Code: How to Actually Know If It Works SaaS Churn Is Killing Your Business. Here Is What to Do About It (Without a Support Team) The Speed of AI Is No Longer Linear - And Self-Improving Models Are Why How to Implement RBAC for MCP Tools: A Practical Guide for Engineering Teams From Standard Quote to Persuasive Proposal: AI Automation for Arborists I built a CLI that scaffolds complete multi-tenant SaaS apps Axios CVE-2025–62718: The Silent SSRF Bug That Could Be Hiding in Your Node.js App Right Now The dashboard that ended our friendship Data Pipelines Explained Simply (and How to Build Them with Python) The Hidden Cost of AI Systems Nobody Talks About. undefined vs undeclared, and how typeof behaves Switching from file-based jobs to NATS/Kafka in Rust without changing code io_uring Adventures: Rust Servers That Love Syscalls Why Agentic AI is Killing the Traditional Database The POUR principles of web accessibility for developers and designers Quantum Neural Network 3D — A Deep Dive into Interactive WebGL Visualization How To Install Caveman In Codex On macOS And Windows Automation Pipeline Reliability: Why Your Workflow Breaks When Nobody Is Watching I Built an 'Open World' AI Coding Agent — It Works From ANY Folder From Freelancing to Product: A Tech Service Company's SaaS Transformation China's AI Giants: Adding Tencent Hunyuan & ByteDance Doubao to AI University (74 Providers) On the Vibe Coders and Their Lies clerk: Auto-Summarize Your Claude Code Sessions AI Weekly — 2026/04/10–04/17 | The Model Lockdown Is Here, but the Toolchain Is the Real Battleground AI 週報 — 2026/04/10–2026/04/17 模型封鎖潮來了,但工具鏈才是真戰場 Maybe this is how Open-Source apps are born... 🚀 Fine-Tune LLMs with LoRA and QLoRA: 2026 Guide tRPC v11 + Next.js App Router: End-to-End Type Safety Without the Boilerplate ShadCN UI in 2026: Why I Stopped Installing Component Libraries and Started Owning My Components SaaS Billing in React Server Components: Stripe + Supabase Without a Single `useEffect` Join our DEV Weekend Challenge — $1,000 in Prizes Across TEN winners! Submissions Due April 20 at 6:59 AM UTC. Implementing FSRS Spaced Repetition in Flutter + Supabase — Adding Memory Science to an AI Learning App "I Texted My Localhost From the Train — Claude Code Fixed the Bug Before I Got Home" I Built a Sales Prep AI and It Went Deeper Than Expected Design to Code #2: One JSON, Eleven Outputs Solving the 100M-Row Problem: A Summary Table Pattern for High-Volume Push Notification Logs Flutter Web With Wasm: What Actually Changes For Developers I Built 50 Royalty-Free Soundtracks for My Side Project in a Weekend Using AI Music Generation The Vibe Coding Security Checklist: 7 Things to Check Before You Ship Stop Letting Googlebot Guess Fix Your React App's SEO Right Desconstruindo o Streaming do LinkedIn: Como Criar um Engine de Extração de Vídeo de Alta Performance com HLS e FFmpeg (EDA Part-1) EDA (Exploratory Data Analysis) Explained With Real Life — Why Looking at Your Data Is the Most Important Step in Machine Learning Brand Relationship Management at Scale: Our 4-Touch Outreach System for 200+ Brands Why String.fromEnvironment() Might Return an Empty String in Dart JGuardrails 1.0.0 — Hardening Java LLM Apps Against Jailbreaks, Toxicity, and Prompt Injection Plan and Schedule a Full Week of Threads Content From One Claude Conversation Coding Cat Oran Ep3, Five Tables Changed Everything BFF模式详解:构建前后端协同的中间层 I'm done watching freelancers get buried by 200 proposals. So I'm building the alternative. This is my first post BFS Algorithm in Java Step by Step Tutorial with Examples Tracking LLM Pricing Monthly: An Open Dataset for 22 AI Models How We Measure Content ROI on a Comparison Site: Revenue Attribution Without Perfect Data Introducing Nova AI Ops: The AI-Native Operating System for SRE Teams I built a free desktop video downloader for Windows — Grabbit How Talkie OCR Helps Vision-Impaired & Dyslexic Users Read the World Around Them VRCFaceTracking安装和iPhone面捕配置教程,有bug Even CrowdStrike Can't See Your Agents The Automation Gold Rush: What n8n Workflows and Claude Are Opening Up for Developers Right Now
What Are Server-Sent Events (SSE)? A Developer's Guide for 2026
Rahul J · 2026-05-24 · via DEV Community

TL;DR

Server-Sent Events (SSE) is a one-way streaming protocol from server → browser built on plain HTTP. The browser opens a connection with new EventSource(url), and the server keeps it open and pushes data: lines whenever it wants. That's it. No WebSocket handshake, no ws://, no upgrade. It auto-reconnects, it works through every proxy that speaks HTTP, and it's been in every browser since 2012.

If you need server → client streaming and you don't need bidirectional messaging, SSE beats WebSockets in almost every way that matters in 2026: less code, simpler infra, auto-reconnect, automatic event IDs for resumability. The only reasons to reach for WebSockets are bidirectional chat-style traffic or binary frames.

Want to see one streaming right now? Open the free SSE tester, paste any SSE endpoint, watch events arrive live.

The 30-Second Mental Model

[ Browser ]  ──── GET /events ───►  [ Server ]
              ◄── HTTP 200, keep alive
              ◄── data: hello\n\n
              ◄── data: world\n\n
              ◄── data: ...

It's just a long-lived HTTP GET that never closes, with a specific text format on the response body. The MIME type is text/event-stream. Every two newlines flush an event to onmessage. The browser handles framing, parsing, and reconnection — you write eventSource.onmessage = ... and you're done.

The Smallest Possible Example

Server (Node.js — no library needed):

import http from 'node:http';

http.createServer((req, res) => {
  if (req.url !== '/events') {
    res.writeHead(404).end();
    return;
  }

  res.writeHead(200, {
    'Content-Type': 'text/event-stream',
    'Cache-Control': 'no-cache',
    'Connection': 'keep-alive',
  });

  let n = 0;
  const interval = setInterval(() => {
    res.write(`data: tick ${++n}\n\n`);
  }, 1000);

  req.on('close', () => clearInterval(interval));
}).listen(3000);

Client (anywhere — including a <script> tag):

const es = new EventSource('/events');
es.onmessage = (e) => console.log('got:', e.data);
es.onerror = () => console.warn('disconnected, browser will retry');

Run the server, paste the client into the browser console, you have streaming in under 20 lines of code. No build step. No library. No upgrade negotiation. That's the entire pitch.

Why SSE Is Underrated in 2026

I see teams reach for WebSockets reflexively whenever they hear "real-time," then spend a week debugging proxy timeouts, sticky sessions, and ALB upgrade headers. Half the time the use case was server → client only — a feed, a notifications drawer, a build log, a live counter. That's SSE territory.

Here's what you actually get for free with SSE that you'd have to build yourself with WebSockets:

1. Auto-reconnect. When the connection drops, the browser waits ~3 seconds and reconnects. You don't write any code for this. With WebSockets you write the same backoff loop in every project for the rest of your career.

2. Event IDs and resume. If the server sends id: 42\n, the browser sends Last-Event-ID: 42 as a header on the next reconnect. The server can resume from where it left off. This is the feature that makes SSE great for build logs, AI streaming, and audit feeds. With WebSockets, this is a custom protocol you design yourself.

3. Standard HTTP everywhere. SSE is a GET with Accept: text/event-stream. Every proxy, every CDN, every WAF, every reverse proxy understands it (with one caveat — see below). Cookies, Authorization, CORS, compression — all work normally. WebSockets need the upgrade dance and proxies often mishandle it.

4. No framing. Text only, line-delimited. You can curl an SSE endpoint and read the stream in your terminal. Try that with a WebSocket.

curl -N -H "Accept: text/event-stream" https://api.example.com/events

The Anatomy of an SSE Frame

The format is dead simple but trips people up. Each event is one or more lines, terminated by two newlines (a blank line):

event: user-joined
id: 423
retry: 5000
data: {"userId":42,"name":"Ada"}

data: simple message
data: continuation of the same event

  • data: — the payload. Multiple data: lines in the same event get joined with \n.
  • event: — sets the event name. Listen with es.addEventListener('user-joined', ...). If omitted, fires onmessage.
  • id: — what the browser sends back as Last-Event-ID on reconnect.
  • retry: — milliseconds to wait before reconnecting.

The blank line is mandatory. Forgetting the second \n is the #1 SSE bug. Your events buffer in the proxy and the client sees nothing.

When to Use SSE vs WebSockets vs Long Polling

This is the only decision tree you need:

Use SSE when:

  • Traffic is server → client only (notifications, feeds, dashboards, logs, AI token streaming)
  • You want auto-reconnect without writing it
  • You're streaming text (JSON, markdown chunks, log lines)
  • You want to debug with curl

Use WebSockets when:

  • Traffic is bidirectional and high-frequency (chat, collaborative editing, multiplayer games)
  • You need binary frames (audio/video, custom protocols)
  • You need sub-100ms round trips for client → server messages

Use long polling when:

  • You're stuck on infra that breaks both of the above (rare in 2026 but happens behind aggressive corporate proxies)

A full breakdown with code on each side is in the SSE vs WebSockets vs Long Polling deep dive.

The Three Production Gotchas

These are the bugs that hit every team the first time they ship SSE.

1. Proxy buffering kills you

Nginx, by default, buffers responses. Your res.write lands in the buffer; the client sees nothing for minutes. Two fixes, do both:

Server response header:

X-Accel-Buffering: no

Nginx config:

proxy_buffering off;
proxy_cache off;
proxy_read_timeout 24h;

Cloudflare buffers too — you need to put SSE endpoints on a non-cached route, or use a Cloudflare-aware streaming setup.

2. The browser's 6-connection-per-origin limit

Browsers limit you to ~6 open HTTP/1.1 connections per origin. If your app opens an EventSource and the user opens 5 more tabs, the 7th tab hangs because there's no connection slot. Two fixes:

  • Use HTTP/2 or HTTP/3 — connection multiplexing means hundreds of streams over one TCP connection. This is the modern fix.
  • Coordinate across tabs via BroadcastChannel or a SharedWorker — one tab holds the EventSource, broadcasts events to siblings.

3. Heartbeats and idle timeouts

Same trap as WebSockets: load balancers close idle connections. AWS ALB defaults to 50 seconds. Send a comment line every 15-30 seconds:

setInterval(() => res.write(':keepalive\n\n'), 15000);

Lines starting with : are comments per the spec — they keep the connection warm without firing onmessage on the client.

Auth and SSE

EventSource has one annoying limitation: you cannot set custom headers when constructing it. No Authorization: Bearer .... Your options:

1. Cookies (recommended for browser apps) — the browser sends them automatically. Use withCredentials: true and CORS-allow your origin.

const es = new EventSource('/events', { withCredentials: true });

2. Query string token — works but tokens end up in server logs.

3. The fetch + ReadableStream pattern — drops EventSource for fetch, which does accept custom headers. You lose auto-reconnect and have to parse the format yourself, but it's the right call for modern apps with token auth:

const res = await fetch('/events', {
  headers: { Authorization: `Bearer ${token}` },
});
const reader = res.body.getReader();
// ... parse text/event-stream chunks manually

A library like @microsoft/fetch-event-source does this for you and is what most modern AI streaming clients use under the hood.

SSE Is What's Powering AI Streaming

If you've used ChatGPT, Claude, or any AI chat UI in the last two years, you've watched SSE in action. The server streams tokens as they're generated:

data: {"choices":[{"delta":{"content":"Hello"}}]}

data: {"choices":[{"delta":{"content":" world"}}]}

data: [DONE]

Anthropic's, OpenAI's, and Google's streaming APIs all use SSE (or a near-identical custom event-stream format). Knowing SSE means you can debug AI streaming endpoints with the same tools you'd use for any other web protocol.

How to Test an SSE Endpoint

Three ways, in order of when to use each:

1. curl -N — the no-buffer flag. Fastest sanity check.

curl -N -H "Accept: text/event-stream" https://api.example.com/events

2. An in-browser SSE tester — paste the URL, see events live, see headers, see reconnect behavior. Free one here, no install.

3. Browser DevTools → Network → EventStream tab — once your app is open, you can see the parsed event stream live. Chromium-based browsers have a dedicated tab for it.

Common Misconceptions

  • "SSE is dead, WebSockets replaced it." No. WebSockets have grown but SSE never went anywhere — it powers AI streaming, GitHub Actions logs, Vercel deploy logs, Stripe webhooks (well, not technically SSE but the same idea), and most "notifications" features you use daily.
  • "SSE doesn't work with HTTP/2." It works better with HTTP/2 — no 6-connection limit, all streams multiplexed.
  • "SSE is text-only so you can't send binary." True, but base64-encoding small binary payloads in a JSON field is fine, and for real binary streams you should be using WebSockets or just HTTP downloads anyway.
  • "SSE doesn't have ping/pong." Send a comment line (:heartbeat\n\n) every 15s. Done.

FAQ

Is SSE the same as long polling?
No. Long polling = client makes a request, server holds it open until there's data, server responds, client makes another request. SSE = one request stays open forever, server pushes when it has data. Long polling reopens the connection every time. SSE doesn't.

Does SSE work on mobile browsers?
Yes, full support in iOS Safari, Chrome Android, Firefox Mobile. The auto-reconnect handles cellular handoffs fine.

Can I use SSE with React / Vue / Svelte / Solid?
Yes, exactly like you'd use any subscription. Spin up an EventSource in a useEffect (or equivalent), close it in the cleanup, set state on each message.

What's the maximum number of concurrent SSE connections a server can handle?
With Node.js and a sensible setup, tens of thousands per process — connections are mostly idle and consume a few KB of RAM each. Compare to ~32k-65k open file descriptors per process as the real upper bound.

Should I use EventSource or fetch + ReadableStream?
Use EventSource if cookie auth works for you (simpler, auto-reconnect for free). Use fetch if you need Authorization headers or custom request behavior, and pull in a library to handle reconnect.


Originally published on AllDevToolsHub. For 250+ free, privacy-first browser-based developer tools — SSE Tester, WebSocket Tester, JWT Decoder, and more — see alldevtoolshub.com.