2020-06-23 04:11:56 +03:00
|
|
|
package core
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"io/ioutil"
|
|
|
|
"math"
|
|
|
|
"os"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
|
2020-07-14 09:49:22 +03:00
|
|
|
"github.com/gabek/owncast/config"
|
2020-06-23 04:11:56 +03:00
|
|
|
"github.com/gabek/owncast/models"
|
|
|
|
"github.com/gabek/owncast/utils"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
statsFilePath = "stats.json"
|
|
|
|
)
|
|
|
|
|
|
|
|
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() {
|
|
|
|
for clientID, lastConnectedtime := range _stats.Clients {
|
|
|
|
timeSinceLastActive := time.Since(lastConnectedtime).Minutes()
|
|
|
|
if timeSinceLastActive > 2 {
|
|
|
|
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-16 02:14:12 +03:00
|
|
|
// So account for that with an artificial buffer of four segments.
|
2020-06-23 04:11:56 +03:00
|
|
|
timeSinceLastConnected := time.Since(_stats.LastConnectTime).Seconds()
|
2020-07-16 02:14:12 +03:00
|
|
|
if timeSinceLastConnected < float64(config.Config.GetVideoSegmentSecondsLength()*4.0) {
|
2020-06-23 04:11:56 +03:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return _stats.StreamConnected
|
|
|
|
}
|
|
|
|
|
|
|
|
//SetClientActive sets a client as active and connected
|
|
|
|
func SetClientActive(clientID string) {
|
|
|
|
// if _, ok := s.clients[clientID]; !ok {
|
|
|
|
// fmt.Println("Marking client active:", clientID, s.GetViewerCount()+1, "clients connected.")
|
|
|
|
// }
|
|
|
|
|
|
|
|
_stats.Clients[clientID] = time.Now()
|
|
|
|
_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) {
|
2020-07-07 07:27:31 +03:00
|
|
|
log.Trace("Removing the client:", clientID)
|
2020-06-23 04:11:56 +03:00
|
|
|
|
|
|
|
delete(_stats.Clients, clientID)
|
|
|
|
}
|
|
|
|
|
|
|
|
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{
|
|
|
|
Clients: make(map[string]time.Time),
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|