package server import ( "context" "time" "github.com/bluesky-social/indigo/events" "github.com/bluesky-social/indigo/lex/util" "github.com/btcsuite/websocket" "github.com/labstack/echo/v4" ) func (s *Server) handleSyncSubscribeRepos(e echo.Context) error { ctx := e.Request().Context() logger := s.logger.With("component", "subscribe-repos-websocket") conn, err := websocket.Upgrade(e.Response().Writer, e.Request(), e.Response().Header(), 1<<10, 1<<10) if err != nil { logger.Error("unable to establish websocket with relay", "err", err) return err } ident := e.RealIP() + "-" + e.Request().UserAgent() logger = logger.With("ident", ident) logger.Info("new connection established") evts, cancel, err := s.evtman.Subscribe(ctx, ident, func(evt *events.XRPCStreamEvent) bool { return true }, nil) if err != nil { return err } defer cancel() header := events.EventHeader{Op: events.EvtKindMessage} for evt := range evts { wc, err := conn.NextWriter(websocket.BinaryMessage) if err != nil { logger.Error("error writing message to relay", "err", err) break } if ctx.Err() != nil { logger.Error("context error", "err", err) break } var obj util.CBOR switch { case evt.Error != nil: header.Op = events.EvtKindErrorFrame obj = evt.Error case evt.RepoCommit != nil: header.MsgType = "#commit" obj = evt.RepoCommit case evt.RepoIdentity != nil: header.MsgType = "#identity" obj = evt.RepoIdentity case evt.RepoAccount != nil: header.MsgType = "#account" obj = evt.RepoAccount case evt.RepoInfo != nil: header.MsgType = "#info" obj = evt.RepoInfo default: logger.Warn("unrecognized event kind") return nil } if err := header.MarshalCBOR(wc); err != nil { logger.Error("failed to write header to relay", "err", err) break } if err := obj.MarshalCBOR(wc); err != nil { logger.Error("failed to write event to relay", "err", err) break } if err := wc.Close(); err != nil { logger.Error("failed to flush-close our event write", "err", err) break } } // we should tell the relay to request a new crawl at this point if we got disconnected // use a new context since the old one might be cancelled at this point ctx, cancel = context.WithTimeout(context.Background(), 10*time.Second) defer cancel() if err := s.requestCrawl(ctx); err != nil { logger.Error("error requesting crawls", "err", err) } return nil }