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 }