zaply/server/sse.go

93 lines
1.8 KiB
Go
Raw Normal View History

2024-12-27 02:54:09 +00:00
package server
import (
"bytes"
"fmt"
"io"
"log"
"time"
2024-12-27 17:02:34 +00:00
"github.com/a-h/templ"
"github.com/ekzyis/zaply/components"
2024-12-27 02:54:09 +00:00
"github.com/ekzyis/zaply/lightning"
"github.com/labstack/echo/v4"
)
type Event struct {
2024-12-27 17:02:34 +00:00
Id []byte
2024-12-27 02:54:09 +00:00
Event []byte
Data []byte
}
func (ev *Event) MarshalTo(w io.Writer) error {
if _, err := fmt.Fprintf(w, "event: %s\n", ev.Event); err != nil {
return err
}
for _, line := range bytes.Split(ev.Data, []byte("\n")) {
if _, err := fmt.Fprintf(w, "data: %s\n", line); err != nil {
return err
}
}
2024-12-27 17:02:34 +00:00
if _, err := fmt.Fprintf(w, "id: %s\n", ev.Id); err != nil {
return err
}
2024-12-27 02:54:09 +00:00
if _, err := fmt.Fprint(w, "\n"); err != nil {
return err
}
return nil
}
func sseHandler(invSrc chan *lightning.Invoice) echo.HandlerFunc {
return func(c echo.Context) error {
w := c.Response()
w.Header().Set("Content-Type", "text/event-stream")
w.Header().Set("Cache-Control", "no-cache")
w.Header().Set("Connection", "keep-alive")
// disable nginx buffering
w.Header().Set("X-Accel-Buffering", "no")
ticker := time.NewTicker(1 * time.Second)
defer ticker.Stop()
for {
select {
case <-c.Request().Context().Done():
return nil
case <-ticker.C:
event := Event{
Event: []byte("message"),
Data: []byte("keepalive"),
}
if err := event.MarshalTo(w); err != nil {
return err
}
case inv := <-invSrc:
2024-12-27 17:02:34 +00:00
buf := templ.GetBuffer()
defer templ.ReleaseBuffer(buf)
if err := components.Zap(inv).Render(c.Request().Context(), buf); err != nil {
2024-12-27 02:54:09 +00:00
return err
}
event := Event{
2024-12-27 17:02:34 +00:00
Id: []byte(inv.PaymentHash),
2024-12-27 02:54:09 +00:00
Event: []byte("zap"),
2024-12-27 17:02:34 +00:00
Data: buf.Bytes(),
2024-12-27 02:54:09 +00:00
}
log.Printf("sending zap event: %s", inv.PaymentHash)
if err := event.MarshalTo(w); err != nil {
return err
}
}
w.Flush()
}
}
}