You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
151 lines
3.5 KiB
151 lines
3.5 KiB
package core |
|
|
|
import ( |
|
"encoding/json" |
|
"io/ioutil" |
|
"math" |
|
"os" |
|
"sync" |
|
"time" |
|
|
|
log "github.com/sirupsen/logrus" |
|
|
|
"github.com/owncast/owncast/config" |
|
"github.com/owncast/owncast/core/chat" |
|
"github.com/owncast/owncast/geoip" |
|
"github.com/owncast/owncast/models" |
|
"github.com/owncast/owncast/utils" |
|
) |
|
|
|
var l = sync.Mutex{} |
|
|
|
func setupStats() error { |
|
s, err := getSavedStats() |
|
if err != nil { |
|
return err |
|
} |
|
|
|
_stats = &s |
|
|
|
statsSaveTimer := time.NewTicker(1 * time.Minute) |
|
go func() { |
|
for range statsSaveTimer.C { |
|
if err := saveStatsToFile(); err != nil { |
|
panic(err) |
|
} |
|
} |
|
}() |
|
|
|
return nil |
|
} |
|
|
|
// 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. |
|
// So account for that with an artificial buffer of four segments. |
|
timeSinceLastConnected := time.Since(_stats.LastConnectTime.Time).Seconds() |
|
if timeSinceLastConnected < float64(config.Config.GetVideoSegmentSecondsLength())*2.3 { |
|
return false |
|
} |
|
|
|
return _stats.StreamConnected |
|
} |
|
|
|
// SetClientActive sets a client as active and connected. |
|
func SetClientActive(client models.Client) { |
|
l.Lock() |
|
// 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 |
|
} |
|
l.Unlock() |
|
|
|
// 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))) |
|
} |
|
} |
|
|
|
// RemoveClient removes a client from the active clients record. |
|
func RemoveClient(clientID string) { |
|
log.Trace("Removing the client:", clientID) |
|
|
|
l.Lock() |
|
delete(_stats.Clients, clientID) |
|
l.Unlock() |
|
} |
|
|
|
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 |
|
} |
|
|
|
func saveStatsToFile() error { |
|
jsonData, err := json.Marshal(_stats) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
f, err := os.Create(config.StatsFile) |
|
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{ |
|
Clients: make(map[string]models.Client), |
|
} |
|
|
|
if !utils.DoesFileExists(config.StatsFile) { |
|
return result, nil |
|
} |
|
|
|
jsonFile, err := ioutil.ReadFile(config.StatsFile) |
|
if err != nil { |
|
return result, err |
|
} |
|
|
|
if err := json.Unmarshal(jsonFile, &result); err != nil { |
|
return result, err |
|
} |
|
|
|
// If the stats were saved > 5min ago then ignore the |
|
// peak session count value, since the session is over. |
|
if !result.LastDisconnectTime.Valid || time.Since(result.LastDisconnectTime.Time).Minutes() > 5 { |
|
result.SessionMaxViewerCount = 0 |
|
} |
|
|
|
return result, err |
|
}
|
|
|