2020-06-22 20:11:56 -05:00
|
|
|
package core
|
|
|
|
|
|
|
|
import (
|
|
|
|
"math"
|
2020-07-22 23:09:11 -07:00
|
|
|
"sync"
|
2020-06-22 20:11:56 -05:00
|
|
|
"time"
|
|
|
|
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
|
2020-10-06 01:07:09 +08:00
|
|
|
"github.com/owncast/owncast/models"
|
2024-11-15 19:20:58 -08:00
|
|
|
"github.com/owncast/owncast/persistence/configrepository"
|
2024-07-02 21:26:33 -07:00
|
|
|
"github.com/owncast/owncast/services/geoip"
|
2020-06-22 20:11:56 -05:00
|
|
|
)
|
|
|
|
|
2022-01-12 21:03:50 -08:00
|
|
|
var (
|
|
|
|
l = &sync.RWMutex{}
|
|
|
|
_activeViewerPurgeTimeout = time.Second * 15
|
2022-03-06 17:31:47 -08:00
|
|
|
_geoIPClient = geoip.NewClient()
|
2022-01-12 21:03:50 -08:00
|
|
|
)
|
2020-07-22 23:09:11 -07:00
|
|
|
|
2020-06-22 20:11:56 -05:00
|
|
|
func setupStats() error {
|
2021-02-18 23:05:52 -08:00
|
|
|
s := getSavedStats()
|
2020-06-22 20:11:56 -05:00
|
|
|
_stats = &s
|
|
|
|
|
|
|
|
statsSaveTimer := time.NewTicker(1 * time.Minute)
|
|
|
|
go func() {
|
2020-11-14 18:39:53 -08:00
|
|
|
for range statsSaveTimer.C {
|
2021-07-19 23:37:06 -07:00
|
|
|
saveStats()
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2021-05-20 20:29:01 -07:00
|
|
|
viewerCountPruneTimer := time.NewTicker(5 * time.Second)
|
|
|
|
go func() {
|
|
|
|
for range viewerCountPruneTimer.C {
|
|
|
|
pruneViewerCount()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2020-06-22 20:11:56 -05:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-11-13 00:14:59 +01:00
|
|
|
// IsStreamConnected checks if the stream is connected or not.
|
2020-06-22 20:11:56 -05:00
|
|
|
func IsStreamConnected() bool {
|
|
|
|
if !_stats.StreamConnected {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Kind of a hack. It takes a handful of seconds between a RTMP connection and when HLS data is available.
|
2020-07-15 16:14:12 -07:00
|
|
|
// So account for that with an artificial buffer of four segments.
|
2020-07-18 15:06:54 -07:00
|
|
|
timeSinceLastConnected := time.Since(_stats.LastConnectTime.Time).Seconds()
|
2024-11-15 19:20:58 -08:00
|
|
|
configRepository := configrepository.Get()
|
|
|
|
waitTime := math.Max(float64(configRepository.GetStreamLatencyLevel().SecondsPerSegment)*3.0, 7)
|
2021-02-18 23:05:52 -08:00
|
|
|
if timeSinceLastConnected < waitTime {
|
2020-06-22 20:11:56 -05:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return _stats.StreamConnected
|
|
|
|
}
|
|
|
|
|
2021-05-20 20:29:01 -07:00
|
|
|
// RemoveChatClient removes a client from the active clients record.
|
|
|
|
func RemoveChatClient(clientID string) {
|
2020-07-06 21:27:31 -07:00
|
|
|
log.Trace("Removing the client:", clientID)
|
2020-06-22 20:11:56 -05:00
|
|
|
|
2020-10-14 14:07:38 -07:00
|
|
|
l.Lock()
|
2021-05-20 20:29:01 -07:00
|
|
|
delete(_stats.ChatClients, clientID)
|
2020-10-14 14:07:38 -07:00
|
|
|
l.Unlock()
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
|
|
|
|
2022-03-06 17:31:47 -08:00
|
|
|
// SetViewerActive sets a client as active and connected.
|
|
|
|
func SetViewerActive(viewer *models.Viewer) {
|
|
|
|
// Don't update viewer counts if a live stream session is not active.
|
|
|
|
if !_stats.StreamConnected {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-05-20 20:29:01 -07:00
|
|
|
l.Lock()
|
|
|
|
defer l.Unlock()
|
|
|
|
|
2024-11-15 19:20:58 -08:00
|
|
|
// Asynchronously, optionally, fetch GeoIP configRepository.
|
2022-03-06 17:31:47 -08:00
|
|
|
go func(viewer *models.Viewer) {
|
|
|
|
viewer.Geo = _geoIPClient.GetGeoFromIP(viewer.IPAddress)
|
|
|
|
}(viewer)
|
2021-05-20 20:29:01 -07:00
|
|
|
|
2022-03-06 17:31:47 -08:00
|
|
|
if _, exists := _stats.Viewers[viewer.ClientID]; exists {
|
|
|
|
_stats.Viewers[viewer.ClientID].LastSeen = time.Now()
|
|
|
|
} else {
|
|
|
|
_stats.Viewers[viewer.ClientID] = viewer
|
2021-05-20 20:29:01 -07:00
|
|
|
}
|
2022-03-06 17:31:47 -08:00
|
|
|
_stats.SessionMaxViewerCount = int(math.Max(float64(len(_stats.Viewers)), float64(_stats.SessionMaxViewerCount)))
|
|
|
|
_stats.OverallMaxViewerCount = int(math.Max(float64(_stats.SessionMaxViewerCount), float64(_stats.OverallMaxViewerCount)))
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetActiveViewers will return the active viewers.
|
|
|
|
func GetActiveViewers() map[string]*models.Viewer {
|
|
|
|
return _stats.Viewers
|
2021-05-20 20:29:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func pruneViewerCount() {
|
2022-03-06 17:31:47 -08:00
|
|
|
viewers := make(map[string]*models.Viewer)
|
2021-05-20 20:29:01 -07:00
|
|
|
|
2021-07-22 15:26:19 -07:00
|
|
|
l.Lock()
|
|
|
|
defer l.Unlock()
|
2021-08-03 17:47:16 -07:00
|
|
|
|
2022-03-06 17:31:47 -08:00
|
|
|
for viewerID, viewer := range _stats.Viewers {
|
|
|
|
viewerLastSeenTime := _stats.Viewers[viewerID].LastSeen
|
2021-05-20 20:29:01 -07:00
|
|
|
if time.Since(viewerLastSeenTime) < _activeViewerPurgeTimeout {
|
2022-03-06 17:31:47 -08:00
|
|
|
viewers[viewerID] = viewer
|
2021-05-20 20:29:01 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_stats.Viewers = viewers
|
|
|
|
}
|
|
|
|
|
2021-07-19 23:37:06 -07:00
|
|
|
func saveStats() {
|
2024-11-15 19:20:58 -08:00
|
|
|
configRepository := configrepository.Get()
|
|
|
|
if err := configRepository.SetPeakOverallViewerCount(_stats.OverallMaxViewerCount); err != nil {
|
2021-02-18 23:05:52 -08:00
|
|
|
log.Errorln("error saving viewer count", err)
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
2024-11-15 19:20:58 -08:00
|
|
|
if err := configRepository.SetPeakSessionViewerCount(_stats.SessionMaxViewerCount); err != nil {
|
2021-02-18 23:05:52 -08:00
|
|
|
log.Errorln("error saving viewer count", err)
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
2021-08-03 17:47:16 -07:00
|
|
|
if _stats.LastDisconnectTime != nil && _stats.LastDisconnectTime.Valid {
|
2024-11-15 19:20:58 -08:00
|
|
|
if err := configRepository.SetLastDisconnectTime(_stats.LastDisconnectTime.Time); err != nil {
|
2021-06-29 11:38:13 -07:00
|
|
|
log.Errorln("error saving disconnect time", err)
|
|
|
|
}
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-18 23:05:52 -08:00
|
|
|
func getSavedStats() models.Stats {
|
2024-11-15 19:20:58 -08:00
|
|
|
configRepository := configrepository.Get()
|
|
|
|
savedLastDisconnectTime, _ := configRepository.GetLastDisconnectTime()
|
2021-06-20 11:30:29 -07:00
|
|
|
|
2020-06-22 20:11:56 -05:00
|
|
|
result := models.Stats{
|
2021-05-20 20:29:01 -07:00
|
|
|
ChatClients: make(map[string]models.Client),
|
2022-03-06 17:31:47 -08:00
|
|
|
Viewers: make(map[string]*models.Viewer),
|
2024-11-15 19:20:58 -08:00
|
|
|
SessionMaxViewerCount: configRepository.GetPeakSessionViewerCount(),
|
|
|
|
OverallMaxViewerCount: configRepository.GetPeakOverallViewerCount(),
|
2021-06-28 13:59:23 -07:00
|
|
|
LastDisconnectTime: savedLastDisconnectTime,
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
|
|
|
|
2020-12-05 15:09:57 -08:00
|
|
|
// If the stats were saved > 5min ago then ignore the
|
|
|
|
// peak session count value, since the session is over.
|
2021-08-03 17:47:16 -07:00
|
|
|
if result.LastDisconnectTime == nil || !result.LastDisconnectTime.Valid || time.Since(result.LastDisconnectTime.Time).Minutes() > 5 {
|
2020-12-05 15:09:57 -08:00
|
|
|
result.SessionMaxViewerCount = 0
|
|
|
|
}
|
|
|
|
|
2021-02-18 23:05:52 -08:00
|
|
|
return result
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|