mattermost-community-enterp.../vendor/github.com/splitio/go-toolkit/v5/sse/sse.go
Claude ec1f89217a Merge: Complete Mattermost Server with Community Enterprise
Full Mattermost server source with integrated Community Enterprise features.
Includes vendor directory for offline/air-gapped builds.

Structure:
- enterprise-impl/: Enterprise feature implementations
- enterprise-community/: Init files that register implementations
- enterprise/: Bridge imports (community_imports.go)
- vendor/: All dependencies for offline builds

Build (online):
  go build ./cmd/mattermost

Build (offline/air-gapped):
  go build -mod=vendor ./cmd/mattermost

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2025-12-17 23:59:07 +09:00

185 lines
4.4 KiB
Go

package sse
import (
"bufio"
"context"
"errors"
"fmt"
"net/http"
"sync"
"time"
"github.com/splitio/go-toolkit/v5/logging"
"github.com/splitio/go-toolkit/v5/struct/traits/lifecycle"
)
const (
endOfLineChar = '\n'
endOfLineStr = "\n"
)
// Client struct
type Client struct {
lifecycle lifecycle.Manager
url string
client http.Client
timeout time.Duration
logger logging.LoggerInterface
}
// NewClient creates new SSEClient
func NewClient(url string, keepAlive int, dialTimeout int, logger logging.LoggerInterface) (*Client, error) {
if keepAlive < 1 {
return nil, errors.New("keepAlive timeout should be higher than 0")
}
if dialTimeout < 0 {
dialTimeout = 0
}
transport := http.DefaultTransport.(*http.Transport).Clone()
transport.Proxy = http.ProxyFromEnvironment
client := &Client{
url: url,
client: http.Client{Transport: transport},
timeout: time.Duration(keepAlive) * time.Second,
logger: logger,
}
client.lifecycle.Setup()
return client, nil
}
func (l *Client) readEvents(in *bufio.Reader, out chan<- RawEvent) {
eventBuilder := NewEventBuilder()
for {
line, err := in.ReadString(endOfLineChar)
l.logger.Debug("Incoming SSE line: ", line)
if err != nil {
if l.lifecycle.IsRunning() { // If it's supposed to be running, log an error
l.logger.Error(err)
}
close(out)
return
}
if line != endOfLineStr {
eventBuilder.AddLine(line)
continue
}
l.logger.Debug("Building SSE event")
if event := eventBuilder.Build(); event != nil {
out <- event
}
eventBuilder.Reset()
}
}
// Do starts streaming
func (l *Client) Do(params map[string]string, headers map[string]string, callback func(e RawEvent)) error {
if !l.lifecycle.BeginInitialization() {
return ErrNotIdle
}
activeGoroutines := sync.WaitGroup{}
ctx, cancel := context.WithCancel(context.Background())
defer func() {
l.logger.Info("SSE streaming exiting")
cancel()
activeGoroutines.Wait()
l.lifecycle.ShutdownComplete()
}()
req, err := l.buildCancellableRequest(ctx, params, headers)
if err != nil {
return &ErrConnectionFailed{wrapped: fmt.Errorf("error building request: %w", err)}
}
l.logger.Debug("[GET] ", req.URL.String())
l.logger.Debug(fmt.Sprintf("Headers: %v", req.Header))
resp, err := l.client.Do(req)
if err != nil {
l.logger.Error("Error performing get: ", req.URL.String(), err.Error())
return &ErrConnectionFailed{wrapped: fmt.Errorf("error issuing request: %w", err)}
}
if resp.StatusCode != 200 {
l.logger.Error(fmt.Sprintf("GET method: Status Code: %d - %s", resp.StatusCode, resp.Status))
return &ErrConnectionFailed{wrapped: fmt.Errorf("sse request status code: %d", resp.StatusCode)}
}
defer resp.Body.Close()
if !l.lifecycle.InitializationComplete() {
return nil
}
reader := bufio.NewReader(resp.Body)
eventChannel := make(chan RawEvent, 1000)
go l.readEvents(reader, eventChannel)
// Create timeout timer in case SSE dont receive notifications or keepalive messages
keepAliveTimer := time.NewTimer(l.timeout)
defer keepAliveTimer.Stop()
for {
select {
case <-l.lifecycle.ShutdownRequested():
l.logger.Info("Shutting down listener")
return nil
case event, ok := <-eventChannel:
keepAliveTimer.Reset(l.timeout)
if !ok {
if l.lifecycle.IsRunning() {
return ErrReadingStream
}
return nil
}
if event.IsEmpty() {
continue // don't forward empty/comment events
}
activeGoroutines.Add(1)
go func() {
defer activeGoroutines.Done()
callback(event)
}()
case <-keepAliveTimer.C: // Timeout
l.logger.Warning("SSE idle timeout.")
l.lifecycle.AbnormalShutdown()
return ErrTimeout
}
}
}
// Shutdown stops SSE
func (l *Client) Shutdown(blocking bool) {
if !l.lifecycle.BeginShutdown() {
l.logger.Info("SSE client stopped or shutdown in progress. Ignoring.")
return
}
if blocking {
l.lifecycle.AwaitShutdownComplete()
}
}
func (l *Client) buildCancellableRequest(ctx context.Context, params map[string]string, headers map[string]string) (*http.Request, error) {
req, err := http.NewRequest("GET", l.url, nil)
if err != nil {
return nil, fmt.Errorf("error instantiating request: %w", err)
}
req = req.WithContext(ctx)
query := req.URL.Query()
for key, value := range params {
query.Add(key, value)
}
for key, value := range headers {
req.Header.Set(key, value)
}
req.URL.RawQuery = query.Encode()
req.Header.Set("Accept", "text/event-stream")
return req, nil
}