Skip to content

Latest commit

 

History

History
161 lines (105 loc) · 5.03 KB

README.md

File metadata and controls

161 lines (105 loc) · 5.03 KB

Realtime Log Streaming with FastAPI and Server-Sent Events

Artigo original: https://amittallapragada.github.io/docker/fastapi/python/2020/12/23/server-side-events.html

Este projeto utiliza a https://developer.mozilla.org/pt-BR/docs/Web/API/EventSource

A interface EventSource é usada para receber eventos enviados pelo servidor (server-sent events). Ele se conecta via HTTP em um servidor e recebe eventos com o formato text/event-stream. A conexão permanece aberta até ser fechada chamando EventSource.close() (en-US).

Assim que a conexão estiver aberta, mensagens recebidas do servidor são entregues para o seu código na forma de eventos message.

Ao contrário dos WebSockets, server-sent events são unidirecionais; ou seja, mensagens são entregues em uma direção, do servidor para o cliente (por exemplo, um navegador web). Isso torna-os uma excelente escolha quando não há necessidade de enviar mensagens do cliente para o servidor.

Referências

https://sysid.github.io/sse/

Os eventos enviados pelo servidor fazem parte do padrão HTML, não HTTP 1 . Eles definem um protocolo que é invisível para a camada HTTP e não interrompe nenhuma das camadas inferiores.

Em sua essência, SSE é apenas um Content-Type cabeçalho que informa ao cliente que a resposta será entregue em partes. Ele também alerta o navegador que ele deve expor cada parte do código à medida que ele chega, e não esperar pela solicitação completa, como os quadros 2 do WebSocket .

No navegador, isso é implementado com a interface fácil de usar EventSource no código do lado do cliente:

var source = new EventSource('updates.cgi');
source.onmessage = function (event) {
  alert(event.data);
}
  1. https://www.starlette.io/responses/

StreamingResponse

Takes an async generator or a normal generator/iterator and streams the response body.

from starlette.responses import StreamingResponse
import asyncio


async def slow_numbers(minimum, maximum):
    yield('<html><body><ul>')
    for number in range(minimum, maximum + 1):
        yield '<li>%d</li>' % number
        await asyncio.sleep(0.5)
    yield('</ul></body></html>')


async def app(scope, receive, send):
    assert scope['type'] == 'http'
    generator = slow_numbers(1, 10)
    response = StreamingResponse(generator, media_type='text/html')
    await response(scope, receive, send)
  1. https://www.abrandao.com/2019/04/html5-eventsource-server-sent-events-tutorial/

  2. https://www.html5rocks.com/en/tutorials/eventsource/basics/

Controlling the Reconnection-timeout The browser attempts to reconnect to the source roughly 3 seconds after each connection is closed. You can change that timeout by including a line beginning with "retry:", followed by the number of milliseconds to wait before trying to reconnect. Exemplo:

      tempo_de_reconexao = 3000 # 3s
      yield {
          "event": "close",
          "retry": tempo_de_reconexao,
          "data": number,
      }

Demo

Sobre este projeto. Como executa-lo?

Basta ativar o ambiente com o comando:

poetry shell
poetry install

Rodando Exemplo 1:

Abra um terminal (não esqueça de ativar o ambiente virtual - poetry shell) para iniciar o gerador de Logs que ficará escrevendo de 0.3s uma linha no arquivo teste.log:

python gerador_logs.py

Ele ficará escrevendo as linhas em: app/logs/log_gerado.log

Em outro terminal (com o ambiente virtual ativado também) rode para subir o servidor FastAPI

uvicorn app.main:app --reload

E por sua vez, abra a página client/client_listen_logs.html no seu browser. Modifiquei o exemplo para você controlar quando o cliente deseja receber os eventos e quando não. Por isto existem dois botões.

Como funciona?

O servidor tem uma rota associada ao método logGenerator que por sua vez utiliza um gerador de eventos do pacote sh chamado tail cuja função é ficar lendo o arquivo de log e cada linha adicionada nele ele lê a linha e dispara um evento. A rota stream-logs então retorna o EventSource. E o HTML fica consumindo estas informações e apresentando na div logs.

async def logGenerator(request):
    status_stream_retry_timeout = 3000 # 3s
    count = 0
    for line in tail("-f", LOGFILE, _iter=True):

Exemplo 2 - Ouvindo o Gerador de Números

Se abrirmos client/client_listen_number.html teremos o mesmo princípio que o outro html só que agora a rota que conectaremos será: stream-numbers.

Para começarmos a receber as mensagens do servidor foi adicionado um botão Ler Números. Que abrirá a conexão Server Sent Events (SSE) e começará a receber as mensagens enviadas pelo servidor.

O botão parar simplesmente fecha a conexão pelo lado do cliente.

A rota stream-numbers

# Esta rota fica lendo os Números e envia para os Clientes
@app.get('/stream-numbers')
async def send_stream_numbers(request: Request):
    evento = gerador_numeros(1, 10, request)
    return EventSourceResponse(evento)

chama o gerador de números: