2024-05-02 19:30:25 +00:00
|
|
|
package events
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
rumblelivestreamlib "github.com/tylertravisty/rumble-livestream-lib-go"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Chat struct {
|
2024-05-22 16:51:46 +00:00
|
|
|
Livestream string
|
|
|
|
Message rumblelivestreamlib.ChatView
|
|
|
|
Stop bool
|
|
|
|
Url string
|
2024-05-02 19:30:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type chatProducer struct {
|
2024-05-22 16:51:46 +00:00
|
|
|
cancel context.CancelFunc
|
|
|
|
cancelMu sync.Mutex
|
|
|
|
client *rumblelivestreamlib.Client
|
|
|
|
livestream string
|
|
|
|
url string
|
2024-05-02 19:30:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type chatProducerValFunc func(*chatProducer) error
|
|
|
|
|
|
|
|
func runChatProducerValFuncs(c *chatProducer, fns ...chatProducerValFunc) error {
|
|
|
|
if c == nil {
|
|
|
|
return fmt.Errorf("chat producer is nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, fn := range fns {
|
|
|
|
err := fn(c)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func chatProducerRequireClient(c *chatProducer) error {
|
|
|
|
if c.client == nil {
|
|
|
|
return fmt.Errorf("client is nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type ChatProducer struct {
|
|
|
|
Ch chan Chat
|
|
|
|
close bool
|
|
|
|
closeMu sync.Mutex
|
|
|
|
closeCh chan bool
|
|
|
|
logError *log.Logger
|
|
|
|
logInfo *log.Logger
|
|
|
|
producers map[string]*chatProducer
|
|
|
|
producersMu sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewChatProducer(logError *log.Logger, logInfo *log.Logger) *ChatProducer {
|
|
|
|
return &ChatProducer{
|
|
|
|
Ch: make(chan Chat, 10),
|
|
|
|
closeCh: make(chan bool),
|
|
|
|
logError: logError,
|
|
|
|
logInfo: logInfo,
|
|
|
|
producers: map[string]*chatProducer{},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// func (cp *ChatProducer) Active(url string) bool {
|
|
|
|
// cp.producersMu.Lock()
|
|
|
|
// defer cp.producersMu.Unlock()
|
|
|
|
// _, active := cp.producers[url]
|
|
|
|
|
|
|
|
// return active
|
|
|
|
// }
|
|
|
|
|
|
|
|
func (cp *ChatProducer) Start(liveStreamUrl string) (string, error) {
|
|
|
|
if liveStreamUrl == "" {
|
|
|
|
return "", pkgErr("", fmt.Errorf("url is empty"))
|
|
|
|
}
|
|
|
|
|
2024-05-22 16:51:46 +00:00
|
|
|
cp.producersMu.Lock()
|
|
|
|
defer cp.producersMu.Unlock()
|
|
|
|
if producer, active := cp.producers[liveStreamUrl]; active {
|
|
|
|
return producer.url, nil
|
|
|
|
}
|
|
|
|
|
2024-05-02 19:30:25 +00:00
|
|
|
client, err := rumblelivestreamlib.NewClient(rumblelivestreamlib.NewClientOptions{LiveStreamUrl: liveStreamUrl})
|
|
|
|
if err != nil {
|
|
|
|
return "", pkgErr("error creating new rumble client", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
chatInfo, err := client.ChatInfo(false)
|
|
|
|
if err != nil {
|
|
|
|
return "", pkgErr("error getting chat info", err)
|
|
|
|
}
|
|
|
|
chatStreamUrl := chatInfo.StreamUrl()
|
|
|
|
|
2024-05-22 16:51:46 +00:00
|
|
|
// cp.producersMu.Lock()
|
|
|
|
// defer cp.producersMu.Unlock()
|
|
|
|
// if _, active := cp.producers[chatStreamUrl]; active {
|
|
|
|
// return chatStreamUrl, nil
|
|
|
|
// }
|
2024-05-02 19:30:25 +00:00
|
|
|
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
producer := &chatProducer{
|
2024-05-22 16:51:46 +00:00
|
|
|
cancel: cancel,
|
|
|
|
client: client,
|
|
|
|
livestream: liveStreamUrl,
|
|
|
|
url: chatStreamUrl,
|
2024-05-02 19:30:25 +00:00
|
|
|
}
|
2024-05-22 16:51:46 +00:00
|
|
|
// cp.producers[chatStreamUrl] = producer
|
|
|
|
cp.producers[liveStreamUrl] = producer
|
2024-05-02 19:30:25 +00:00
|
|
|
go cp.run(ctx, producer)
|
|
|
|
|
|
|
|
return chatStreamUrl, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cp *ChatProducer) Stop(chatStreamUrl string) error {
|
|
|
|
cp.producersMu.Lock()
|
|
|
|
producer, exists := cp.producers[chatStreamUrl]
|
|
|
|
if !exists {
|
|
|
|
return pkgErr("", fmt.Errorf("producer does not exist for chat stream: %s", chatStreamUrl))
|
|
|
|
}
|
|
|
|
cp.producersMu.Unlock()
|
|
|
|
|
|
|
|
producer.cancelMu.Lock()
|
|
|
|
if producer.cancel != nil {
|
|
|
|
producer.cancel()
|
|
|
|
}
|
|
|
|
producer.cancelMu.Unlock()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cp *ChatProducer) run(ctx context.Context, producer *chatProducer) {
|
|
|
|
err := runChatProducerValFuncs(
|
|
|
|
producer,
|
|
|
|
chatProducerRequireClient,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
cp.logError.Println(pkgErr("invalid chat producer", err))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-05-22 16:51:46 +00:00
|
|
|
// TODO: handle the case when restarting stream with possibly missing messages
|
|
|
|
// Start new stream, make sure it's running, close old stream
|
2024-05-02 19:30:25 +00:00
|
|
|
for {
|
|
|
|
err = producer.client.StartChatStream(cp.handleChat(producer), cp.handleError(producer))
|
|
|
|
if err != nil {
|
|
|
|
cp.logError.Println(pkgErr("error starting chat stream", err))
|
|
|
|
cp.stop(producer)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
timer := time.NewTimer(90 * time.Minute)
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
timer.Stop()
|
|
|
|
producer.client.StopChatStream()
|
|
|
|
cp.stop(producer)
|
|
|
|
return
|
|
|
|
case <-timer.C:
|
2024-05-22 16:51:46 +00:00
|
|
|
producer.client.StopChatStream()
|
2024-05-02 19:30:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cp *ChatProducer) handleChat(p *chatProducer) func(cv rumblelivestreamlib.ChatView) {
|
|
|
|
return func(cv rumblelivestreamlib.ChatView) {
|
|
|
|
if p == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-05-22 16:51:46 +00:00
|
|
|
cp.Ch <- Chat{Livestream: p.livestream, Message: cv, Url: p.url}
|
2024-05-02 19:30:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cp *ChatProducer) handleError(p *chatProducer) func(err error) {
|
|
|
|
return func(err error) {
|
|
|
|
cp.logError.Println(pkgErr("chat stream returned error", err))
|
|
|
|
p.cancelMu.Lock()
|
|
|
|
if p.cancel != nil {
|
|
|
|
p.cancel()
|
|
|
|
}
|
|
|
|
p.cancelMu.Unlock()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cp *ChatProducer) stop(p *chatProducer) {
|
|
|
|
if p == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-05-22 16:51:46 +00:00
|
|
|
cp.Ch <- Chat{Livestream: p.livestream, Stop: true, Url: p.url}
|
2024-05-02 19:30:25 +00:00
|
|
|
|
|
|
|
cp.producersMu.Lock()
|
|
|
|
delete(cp.producers, p.url)
|
|
|
|
remaining := len(cp.producers)
|
|
|
|
cp.producersMu.Unlock()
|
|
|
|
|
|
|
|
cp.closeMu.Lock()
|
|
|
|
if remaining == 0 && cp.close {
|
|
|
|
select {
|
|
|
|
case cp.closeCh <- true:
|
|
|
|
default:
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cp.closeMu.Unlock()
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|