schnutibox/pkg/sselog/sselog.go

74 lines
1.5 KiB
Go

// package sselog is work in progress to implement a writer that sends its logs
// to a http server side event.
// nolint:gochecknoglobals,godox
package sselog
import (
"fmt"
"net/http"
"go.xsfx.dev/logginghandler"
)
// Log is the global sse logger struct.
var Log *SSELog
type SSELog struct {
Receivers map[chan []byte]struct{}
}
func NewSSELog() *SSELog {
return &SSELog{
Receivers: make(map[chan []byte]struct{}),
}
}
func (l SSELog) Write(p []byte) (n int, err error) {
// Send log message to all receiver channels.
for r := range l.Receivers {
r <- p
}
return len(p), nil
}
func LogHandler(w http.ResponseWriter, r *http.Request) {
logger := logginghandler.Logger(r)
logger.Info().Msg("registering a new sse logger")
flusher, ok := w.(http.Flusher)
if !ok {
logger.Error().Msg("streaming unsupported")
http.Error(w, "streaming unsupported", http.StatusInternalServerError)
return
}
w.Header().Set("Content-Type", "text/event-stream")
w.Header().Set("Cache-Control", "no-cache")
w.Header().Set("Connection", "keep-alive")
// TODO: has to be something else!
w.Header().Set("Access-Control-Allow-Origin", "*")
cChan := make(chan []byte)
Log.Receivers[cChan] = struct{}{}
for {
select {
case e := <-cChan:
// Send event to client.
fmt.Fprintf(w, "data: %s\n\n", e)
// Send it right now and not buffering it.
flusher.Flush()
case <-r.Context().Done():
close(cChan)
delete(Log.Receivers, cChan)
return
}
}
}