[feature] Support multiple subscriptions on single websocket connection (#1489)

- Allow Oauth authentication on websocket endpoint
- Make streamType query parameter optional
- Read websocket commands from client and update subscriptions
This commit is contained in:
darrinsmart 2023-03-11 02:10:58 -08:00 committed by GitHub
parent cb2f84e551
commit e323a930bf
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 74 additions and 26 deletions

View file

@ -20,14 +20,16 @@
import ( import (
"context" "context"
"errors"
"fmt"
"time" "time"
"codeberg.org/gruf/go-kv" "codeberg.org/gruf/go-kv"
apiutil "github.com/superseriousbusiness/gotosocial/internal/api/util" apiutil "github.com/superseriousbusiness/gotosocial/internal/api/util"
"github.com/superseriousbusiness/gotosocial/internal/gtserror" "github.com/superseriousbusiness/gotosocial/internal/gtserror"
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
"github.com/superseriousbusiness/gotosocial/internal/log" "github.com/superseriousbusiness/gotosocial/internal/log"
"github.com/superseriousbusiness/gotosocial/internal/oauth"
streampkg "github.com/superseriousbusiness/gotosocial/internal/stream"
"golang.org/x/exp/slices"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
"github.com/gorilla/websocket" "github.com/gorilla/websocket"
@ -134,32 +136,37 @@
// '400': // '400':
// description: bad request // description: bad request
func (m *Module) StreamGETHandler(c *gin.Context) { func (m *Module) StreamGETHandler(c *gin.Context) {
streamType := c.Query(StreamQueryKey)
if streamType == "" {
err := fmt.Errorf("no stream type provided under query key %s", StreamQueryKey)
apiutil.ErrorHandler(c, gtserror.NewErrorBadRequest(err, err.Error()), m.processor.InstanceGetV1)
return
}
var token string
// First we check for a query param provided access token // First we check for a query param provided access token
if token = c.Query(AccessTokenQueryKey); token == "" { token := c.Query(AccessTokenQueryKey)
if token == "" {
// Else we check the HTTP header provided token // Else we check the HTTP header provided token
if token = c.GetHeader(AccessTokenHeader); token == "" { token = c.GetHeader(AccessTokenHeader)
const errStr = "no access token provided" }
err := gtserror.NewErrorUnauthorized(errors.New(errStr), errStr)
var account *gtsmodel.Account
if token != "" {
// Check the explicit token
var errWithCode gtserror.WithCode
account, errWithCode = m.processor.Stream().Authorize(c.Request.Context(), token)
if errWithCode != nil {
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
return
}
} else {
// If no explicit token was provided, try regular oauth
auth, errStr := oauth.Authed(c, true, true, true, true)
if errStr != nil {
err := gtserror.NewErrorUnauthorized(errStr, errStr.Error())
apiutil.ErrorHandler(c, err, m.processor.InstanceGetV1) apiutil.ErrorHandler(c, err, m.processor.InstanceGetV1)
return return
} }
account = auth.Account
} }
account, errWithCode := m.processor.Stream().Authorize(c.Request.Context(), token) // Get the initial stream type, if there is one.
if errWithCode != nil { // streamType will be an empty string if one wasn't supplied. Open() will deal with this
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1) streamType := c.Query(StreamQueryKey)
return
}
stream, errWithCode := m.processor.Stream().Open(c.Request.Context(), account, streamType) stream, errWithCode := m.processor.Stream().Open(c.Request.Context(), account, streamType)
if errWithCode != nil { if errWithCode != nil {
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1) apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
@ -219,8 +226,9 @@ func (m *Module) StreamGETHandler(c *gin.Context) {
// We have to listen for received websocket messages in // We have to listen for received websocket messages in
// order to trigger the underlying wsConn.PingHandler(). // order to trigger the underlying wsConn.PingHandler().
// //
// So we wait on received messages but only act on errors. // Read JSON objects from the client and act on them
_, _, err := wsConn.ReadMessage() var msg map[string]string
err := wsConn.ReadJSON(&msg)
if err != nil { if err != nil {
if ctx.Err() == nil { if ctx.Err() == nil {
// Only log error if the connection was not closed // Only log error if the connection was not closed
@ -229,6 +237,33 @@ func (m *Module) StreamGETHandler(c *gin.Context) {
} }
return return
} }
l.Tracef("received message from websocket: %v", msg)
// If the message contains 'stream' and 'type' fields, we can
// update the set of timelines that are subscribed for events.
// everything else is ignored.
action := msg["type"]
streamType := msg["stream"]
// Ignore if the streamType is unknown (or missing), so a bad
// client can't cause extra memory allocations
if !slices.Contains(streampkg.AllStatusTimelines, streamType) {
l.Warnf("Unknown 'stream' field: %v", msg)
continue
}
switch action {
case "subscribe":
stream.Lock()
stream.Timelines[streamType] = true
stream.Unlock()
case "unsubscribe":
stream.Lock()
delete(stream.Timelines, streamType)
stream.Unlock()
default:
l.Warnf("Invalid 'type' field: %v", msg)
}
} }
}() }()

View file

@ -45,9 +45,17 @@ func (p *Processor) Open(ctx context.Context, account *gtsmodel.Account, streamT
return nil, gtserror.NewErrorInternalError(fmt.Errorf("error generating stream id: %s", err)) return nil, gtserror.NewErrorInternalError(fmt.Errorf("error generating stream id: %s", err))
} }
// Each stream can be subscibed to multiple timelines.
// Record them in a set, and include the initial one
// if it was given to us
timelines := map[string]bool{}
if streamTimeline != "" {
timelines[streamTimeline] = true
}
thisStream := &stream.Stream{ thisStream := &stream.Stream{
ID: streamID, ID: streamID,
Timeline: streamTimeline, Timelines: timelines,
Messages: make(chan *stream.Message, 100), Messages: make(chan *stream.Message, 100),
Hangup: make(chan interface{}, 1), Hangup: make(chan interface{}, 1),
Connected: true, Connected: true,

View file

@ -63,12 +63,15 @@ func (p *Processor) toAccount(payload string, event string, timelines []string,
} }
for _, t := range timelines { for _, t := range timelines {
if s.Timeline == string(t) { if _, found := s.Timelines[t]; found {
s.Messages <- &stream.Message{ s.Messages <- &stream.Message{
Stream: []string{string(t)}, Stream: []string{string(t)},
Event: string(event), Event: string(event),
Payload: payload, Payload: payload,
} }
// break out to the outer loop, to avoid sending duplicates
// of the same event to the same stream
break
} }
} }
} }

View file

@ -63,8 +63,10 @@ type StreamsForAccount struct {
type Stream struct { type Stream struct {
// ID of this stream, generated during creation. // ID of this stream, generated during creation.
ID string ID string
// Timeline of this stream: user/public/etc // A set of timelines of this stream: user/public/etc
Timeline string // a matching key means the timeline is subscribed. The value
// is ignored
Timelines map[string]bool
// Channel of messages for the client to read from // Channel of messages for the client to read from
Messages chan *Message Messages chan *Message
// Channel to close when the client drops away // Channel to close when the client drops away