2020-06-22 20:11:56 -05:00
|
|
|
package core
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"io/ioutil"
|
|
|
|
"math"
|
|
|
|
"os"
|
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/config"
|
2020-10-06 23:14:33 -07:00
|
|
|
"github.com/owncast/owncast/core/chat"
|
|
|
|
"github.com/owncast/owncast/geoip"
|
2020-10-06 01:07:09 +08:00
|
|
|
"github.com/owncast/owncast/models"
|
|
|
|
"github.com/owncast/owncast/utils"
|
2020-06-22 20:11:56 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
statsFilePath = "stats.json"
|
|
|
|
)
|
|
|
|
|
2020-07-22 23:09:11 -07:00
|
|
|
var l = sync.Mutex{}
|
|
|
|
|
2020-06-22 20:11:56 -05:00
|
|
|
func setupStats() error {
|
|
|
|
s, err := getSavedStats()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
_stats = &s
|
|
|
|
|
|
|
|
statsSaveTimer := time.NewTicker(1 * time.Minute)
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-statsSaveTimer.C:
|
|
|
|
if err := saveStatsToFile(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
staleViewerPurgeTimer := time.NewTicker(3 * time.Second)
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-staleViewerPurgeTimer.C:
|
|
|
|
purgeStaleViewers()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func purgeStaleViewers() {
|
2020-10-06 23:14:33 -07:00
|
|
|
for clientID, client := range _stats.Clients {
|
|
|
|
if client.LastSeen.IsZero() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
timeSinceLastActive := time.Since(client.LastSeen).Minutes()
|
|
|
|
if timeSinceLastActive > 1 {
|
2020-06-22 20:11:56 -05:00
|
|
|
RemoveClient(clientID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//IsStreamConnected checks if the stream is connected or not
|
|
|
|
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()
|
2020-07-15 16:14:12 -07:00
|
|
|
if timeSinceLastConnected < float64(config.Config.GetVideoSegmentSecondsLength()*4.0) {
|
2020-06-22 20:11:56 -05:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return _stats.StreamConnected
|
|
|
|
}
|
|
|
|
|
|
|
|
//SetClientActive sets a client as active and connected
|
2020-10-06 23:14:33 -07:00
|
|
|
func SetClientActive(client models.Client) {
|
2020-07-22 23:09:11 -07:00
|
|
|
l.Lock()
|
2020-10-06 23:14:33 -07:00
|
|
|
// If this clientID already exists then update it.
|
|
|
|
// Otherwise set a new one.
|
|
|
|
if existingClient, ok := _stats.Clients[client.ClientID]; ok {
|
|
|
|
existingClient.LastSeen = time.Now()
|
|
|
|
existingClient.Username = client.Username
|
|
|
|
existingClient.MessageCount = client.MessageCount
|
|
|
|
existingClient.Geo = geoip.GetGeoFromIP(existingClient.IPAddress)
|
|
|
|
_stats.Clients[client.ClientID] = existingClient
|
|
|
|
} else {
|
|
|
|
if client.Geo == nil {
|
|
|
|
geoip.FetchGeoForIP(client.IPAddress)
|
|
|
|
}
|
|
|
|
_stats.Clients[client.ClientID] = client
|
|
|
|
}
|
2020-07-22 23:09:11 -07:00
|
|
|
l.Unlock()
|
2020-09-24 20:27:47 -07:00
|
|
|
|
|
|
|
// Don't update viewer counts if a live stream session is not active.
|
|
|
|
if _stats.StreamConnected {
|
|
|
|
_stats.SessionMaxViewerCount = int(math.Max(float64(len(_stats.Clients)), float64(_stats.SessionMaxViewerCount)))
|
|
|
|
_stats.OverallMaxViewerCount = int(math.Max(float64(_stats.SessionMaxViewerCount), float64(_stats.OverallMaxViewerCount)))
|
|
|
|
}
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
//RemoveClient removes a client from the active clients record
|
|
|
|
func RemoveClient(clientID string) {
|
2020-07-06 21:27:31 -07:00
|
|
|
log.Trace("Removing the client:", clientID)
|
2020-06-22 20:11:56 -05:00
|
|
|
|
|
|
|
delete(_stats.Clients, clientID)
|
|
|
|
}
|
|
|
|
|
2020-10-06 23:14:33 -07:00
|
|
|
func GetClients() []models.Client {
|
|
|
|
clients := make([]models.Client, 0)
|
|
|
|
for _, client := range _stats.Clients {
|
|
|
|
chatClient := chat.GetClient(client.ClientID)
|
|
|
|
if chatClient != nil {
|
|
|
|
clients = append(clients, chatClient.GetViewerClientFromChatClient())
|
|
|
|
} else {
|
|
|
|
clients = append(clients, client)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return clients
|
|
|
|
}
|
|
|
|
|
2020-06-22 20:11:56 -05:00
|
|
|
func saveStatsToFile() error {
|
|
|
|
jsonData, err := json.Marshal(_stats)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
f, err := os.Create(statsFilePath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
if _, err := f.Write(jsonData); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getSavedStats() (models.Stats, error) {
|
|
|
|
result := models.Stats{
|
2020-10-06 23:14:33 -07:00
|
|
|
Clients: make(map[string]models.Client),
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if !utils.DoesFileExists(statsFilePath) {
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
jsonFile, err := ioutil.ReadFile(statsFilePath)
|
|
|
|
if err != nil {
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(jsonFile, &result); err != nil {
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|