Koneksi berjalan lama
Server-Sent Events (SSE) adalah koneksi HTTP yang sudah berjalan lama **, jadi untuk permulaan kita membutuhkan ini:
proxy_http_version 1.1;
proxy_set_header Connection "";
CATATAN: Koneksi TCP dalam HTTP / 1.1 persisten secara default, jadi mengatur header Koneksi untuk mengosongkan melakukan hal yang benar dan merupakan saran Nginx.
Transfer-Pengkodean Terpotong
Di samping itu; Respons SSE tidak menetapkan header Panjang Konten karena mereka tidak tahu berapa banyak data akan dikirim, sebaliknya mereka perlu menggunakan header Transfer-Pengkodean [0] [1], yang memungkinkan koneksi streaming. Perhatikan juga: jika Anda tidak menambahkan Panjang Konten, sebagian besar server HTTP akan mengaturnya Transfer-Encoding: chunked;
untuk Anda. Anehnya, HTTP chunking memperingatkan dan menyebabkan kebingungan.
Kebingungan berasal dari peringatan yang agak kabur di bagian Catatan dari deskripsi W3 EventSource:
Penulis juga diperingatkan bahwa chunking HTTP dapat memiliki efek negatif yang tidak terduga pada keandalan protokol ini. Jika memungkinkan, chunking harus dinonaktifkan untuk melayani aliran acara kecuali jika laju pesannya cukup tinggi sehingga hal ini tidak menjadi masalah.
Yang akan membuat orang percaya Transfer-Encoding: chunked;
adalah hal buruk bagi SSE. Namun: ini belum tentu demikian, itu hanya masalah ketika server web Anda melakukan chunking untuk Anda (tidak mengetahui informasi tentang data Anda). Jadi, sementara sebagian besar posting akan menyarankan menambahkan chunked_transfer_encoding off;
ini tidak perlu dalam kasus khas [3].
Buffering (masalah sebenarnya)
Di mana sebagian besar masalah berasal adalah memiliki segala jenis buffering antara server aplikasi dan klien. Secara default [4], Nginx menggunakan
proxy_buffering on
(juga melihat uwsgi_buffering
dan fastcgi_buffering
tergantung pada aplikasi Anda) dan dapat memilih untuk buffer potongan yang ingin Anda sampaikan kepada klien Anda. Ini adalah hal yang buruk karena sifat SSE yang realtime pecah.
Namun, alih-alih beralih proxy_buffering off
untuk semuanya, lebih baik (jika Anda bisa) menambahkan X-Accel-Buffering: no
sebagai header respons dalam kode server aplikasi Anda untuk hanya mematikan buffering untuk respons berbasis SSE dan tidak untuk semua respons yang datang dari aplikasi Anda server. Bonus: ini juga akan bekerja untuk uwsgi
dan fastcgi
.
Larutan
Dan pengaturan yang sangat penting sebenarnya adalah header respons aplikasi-server:
Content-Type: text/event-stream;
Cache-Control: no-cache;
X-Accel-Buffering: no;
Dan berpotensi penerapan beberapa mekanisme ping sehingga koneksi tidak tinggal diam terlalu lama. Bahaya dari ini adalah bahwa Nginx akan menutup koneksi idle sebagaimana diatur menggunakan keepalive
pengaturan.
[0] https://tools.ietf.org/html/rfc2616#section-3.6
[1] https://en.wikipedia.org/wiki/Chunked_transfer_encoding
[2] https://www.w3.org/TR / 2009 / WD-eventsource-20091029 / # text-event-stream
[3] https://github.com/whatwg/html/issues/515
[4] http://nginx.org/en/docs/http/ ngx_http_proxy_module.html # proxy_buffering
[5] https://tools.ietf.org/html/rfc7230#section-6.3
[6] https://gist.github.com/CMCDragonkai/6bfade6431e9ffb7fe88
X-Accel-Buffering: no
header adalah kunci bagi saya, tetapi yang penting, saya harus melakukan seperti yang ditulis oleh @ c4urself: "tambahkan X-Accel-Buffering: tidak sebagai header respons dalam kode server aplikasi Anda ". Menambahkan header ini ke bagian lokasi di konfigurasi nginx saya tidak berfungsi - seluruh aliran acara menunggu untuk dikirim sampai setelah aplikasi selesai / dihentikan.proxy_http_version 1.1;
perlu? Saya mencoba menjalankan lebih dari 6 SSE stream dari browser dan karenanya saya perlu HTTP2.