6 Commits

Author SHA1 Message Date
d3513ae4fa Merge pull request '[🚀] Ticker' (#13) from dev into main
Reviewed-on: #13
2024-09-07 15:44:39 +08:00
99264c6715 [🚀] Ticker 2024-09-07 15:44:13 +08:00
3c70ad2377 Merge pull request '[🚀] SSE FUN' (#12) from dev into main
Reviewed-on: #12
2024-09-07 15:13:47 +08:00
af451c08b8 [🚀] SSE FUN 2024-09-07 15:13:38 +08:00
75fd09620c Merge pull request '[🚀] sse' (#11) from dev into main
Reviewed-on: #11
2024-09-07 13:41:59 +08:00
8fa8153ba6 [🚀] sse 2024-09-07 13:41:57 +08:00

View File

@ -1,20 +1,23 @@
package sse package sse
import ( import (
"fmt"
"gitea.bvbej.com/bvbej/base-golang/pkg/mux" "gitea.bvbej.com/bvbej/base-golang/pkg/mux"
"gitea.bvbej.com/bvbej/base-golang/pkg/ticker"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
"io" "io"
"net/http" "net/http"
"sync" "sync"
"sync/atomic" "sync/atomic"
"time"
) )
var _ Server = (*event)(nil) var _ Server = (*event)(nil)
type Server interface { type Server interface {
GinHandle(ctx *gin.Context, user any)
HandlerFunc() mux.HandlerFunc HandlerFunc() mux.HandlerFunc
Push(user, name, msg string) bool GinHandlerFunc(auth func(c *gin.Context) (string, error)) gin.HandlerFunc
Push(user any, name, msg string) bool
Broadcast(name, msg string) Broadcast(name, msg string)
} }
@ -31,9 +34,9 @@ type msgChan struct {
type event struct { type event struct {
SessionList sync.Map SessionList sync.Map
Count atomic.Int32 Count atomic.Int32
Register chan clientChan Register chan clientChan
Unregister chan any Unregister chan any
Ticker ticker.Ticker
} }
func NewServer() Server { func NewServer() Server {
@ -42,8 +45,12 @@ func NewServer() Server {
Count: atomic.Int32{}, Count: atomic.Int32{},
Register: make(chan clientChan), Register: make(chan clientChan),
Unregister: make(chan any), Unregister: make(chan any),
Ticker: ticker.New(time.Second * 3),
} }
go e.Ticker.Process(func() {
e.Broadcast("ping", fmt.Sprintf("%d", time.Now().Unix()))
})
go e.listen() go e.listen()
return e return e
@ -67,11 +74,47 @@ func (stream *event) listen() {
} }
} }
func (stream *event) GinHandle(ctx *gin.Context, user any) { func (stream *event) HandlerFunc() mux.HandlerFunc {
if user == nil { return func(c mux.Context) {
ctx.AbortWithStatus(http.StatusUnauthorized) auth := c.Auth()
if auth == nil {
c.Context().AbortWithStatus(http.StatusBadRequest)
return return
} }
e := make(chan msgChan)
client := clientChan{
User: auth,
Chan: e,
}
stream.Register <- client
defer func() {
stream.Unregister <- auth
}()
c.Context().Writer.Header().Set("Content-Type", "text/event-stream")
c.Context().Writer.Header().Set("Cache-Control", "no-cache")
c.Context().Writer.Header().Set("Connection", "keep-alive")
c.Context().Writer.Header().Set("Transfer-Encoding", "chunked")
c.Context().Stream(func(w io.Writer) bool {
if msg, ok := <-e; ok {
c.Context().SSEvent(msg.Name, msg.Message)
return true
}
return false
})
}
}
func (stream *event) GinHandlerFunc(auth func(c *gin.Context) (string, error)) gin.HandlerFunc {
return func(c *gin.Context) {
user, err := auth(c)
if err != nil {
c.AbortWithStatus(http.StatusBadRequest)
return
}
e := make(chan msgChan) e := make(chan msgChan)
client := clientChan{ client := clientChan{
User: user, User: user,
@ -82,29 +125,22 @@ func (stream *event) GinHandle(ctx *gin.Context, user any) {
stream.Unregister <- user stream.Unregister <- user
}() }()
ctx.Writer.Header().Set("Content-Type", "text/event-stream") c.Writer.Header().Set("Content-Type", "text/event-stream")
ctx.Writer.Header().Set("Cache-Control", "no-cache") c.Writer.Header().Set("Cache-Control", "no-cache")
ctx.Writer.Header().Set("Connection", "keep-alive") c.Writer.Header().Set("Connection", "keep-alive")
ctx.Writer.Header().Set("Transfer-Encoding", "chunked") c.Writer.Header().Set("Transfer-Encoding", "chunked")
ctx.Stream(func(w io.Writer) bool { c.Stream(func(w io.Writer) bool {
if msg, ok := <-e; ok { if msg, ok := <-e; ok {
ctx.SSEvent(msg.Name, msg.Message) c.SSEvent(msg.Name, msg.Message)
return true return true
} }
return false return false
}) })
ctx.Next()
}
func (stream *event) HandlerFunc() mux.HandlerFunc {
return func(c mux.Context) {
stream.GinHandle(c.Context(), c.Auth())
} }
} }
func (stream *event) Push(user, name, msg string) bool { func (stream *event) Push(user any, name, msg string) bool {
value, ok := stream.SessionList.Load(user) value, ok := stream.SessionList.Load(user)
if ok { if ok {
val := value.(chan msgChan) val := value.(chan msgChan)