From b47ea89c06d9636537fd14dd88e7b047f77bc09a Mon Sep 17 00:00:00 2001 From: Gabe Kangas Date: Wed, 10 Jun 2020 23:52:55 -0700 Subject: [PATCH] Add basic persistent viewership stats --- main.go | 17 +++++--- stats.go | 106 +++++++++++++++++++++++++++++++++++++++++++++ status.go | 6 ++- webroot/index.html | 2 + webroot/js/app.js | 4 ++ 5 files changed, 128 insertions(+), 7 deletions(-) create mode 100644 stats.go diff --git a/main.go b/main.go index 6c4e816b4..c11d7c5d0 100644 --- a/main.go +++ b/main.go @@ -11,14 +11,16 @@ import ( var storage ChunkStorage var configuration = getConfig() var server *Server +var stats *Stats -var online = false var usingExternalStorage = false func main() { log.Println("Starting up. Please wait...") resetDirectories(configuration) checkConfig(configuration) + stats = getSavedStats() + stats.Setup() if configuration.IPFS.Enabled { storage = &IPFSStorage{} @@ -57,14 +59,17 @@ func startChatServer() { func getStatus(w http.ResponseWriter, r *http.Request) { status := Status{ - Online: online, - ViewerCount: server.ClientCount(), + Online: stats.IsStreamConnected(), + ViewerCount: stats.GetViewerCount(), + OverallMaxViewerCount: stats.GetOverallMaxViewerCount(), + SessionMaxViewerCount: stats.GetSessionMaxViewerCount(), } json.NewEncoder(w).Encode(status) } func streamConnected() { - online = true + stats.StreamConnected() + chunkPath := configuration.PublicHLSPath if usingExternalStorage { chunkPath = configuration.PrivateHLSPath @@ -73,11 +78,13 @@ func streamConnected() { } func streamDisconnected() { - online = false + stats.StreamDisconnected() } func viewerAdded() { + stats.SetViewerCount(server.ClientCount()) } func viewerRemoved() { + stats.SetViewerCount(server.ClientCount()) } diff --git a/stats.go b/stats.go new file mode 100644 index 000000000..a811e6228 --- /dev/null +++ b/stats.go @@ -0,0 +1,106 @@ +package main + +import ( + "encoding/json" + "io/ioutil" + "math" + "os" + "time" +) + +type Stats struct { + streamConnected bool `json:"-"` + ViewerCount int `json:"viewerCount"` + SessionMaxViewerCount int `json:"sessionMaxViewerCount"` + OverallMaxViewerCount int `json:"overallMaxViewerCount"` + LastDisconnectTime time.Time `json:"lastDisconnectTime"` +} + +func (s *Stats) Setup() { + ticker := time.NewTicker(2 * time.Minute) + quit := make(chan struct{}) + go func() { + for { + select { + case <-ticker.C: + s.save() + case <-quit: + ticker.Stop() + return + } + } + }() +} + +func (s *Stats) IsStreamConnected() bool { + return s.streamConnected +} + +func (s *Stats) SetViewerCount(count int) { + s.ViewerCount = count + s.SessionMaxViewerCount = int(math.Max(float64(s.ViewerCount), float64(s.SessionMaxViewerCount))) + s.OverallMaxViewerCount = int(math.Max(float64(s.SessionMaxViewerCount), float64(s.OverallMaxViewerCount))) +} + +func (s *Stats) GetViewerCount() int { + return s.ViewerCount +} + +func (s *Stats) GetSessionMaxViewerCount() int { + return s.SessionMaxViewerCount +} + +func (s *Stats) GetOverallMaxViewerCount() int { + return s.OverallMaxViewerCount +} + +func (s *Stats) ViewerConnected() { +} + +func (s *Stats) ViewerDisconnected() { +} + +func (s *Stats) StreamConnected() { + s.streamConnected = true + + timeSinceDisconnect := time.Since(s.LastDisconnectTime).Minutes() + if timeSinceDisconnect > 15 { + s.SessionMaxViewerCount = 0 + } +} + +func (s *Stats) StreamDisconnected() { + s.streamConnected = false + s.LastDisconnectTime = time.Now() +} + +func (s *Stats) save() { + jsonData, err := json.Marshal(&s) + verifyError(err) + + f, err := os.Create("config/stats.json") + defer f.Close() + + verifyError(err) + + _, err = f.Write(jsonData) + verifyError(err) +} + +func getSavedStats() *Stats { + filePath := "config/stats.json" + + if !fileExists(filePath) { + return &Stats{} + } + + jsonFile, err := ioutil.ReadFile(filePath) + + var stats Stats + err = json.Unmarshal(jsonFile, &stats) + if err != nil { + panic(err) + } + + return &stats +} diff --git a/status.go b/status.go index cea323b98..30b3b2d8c 100644 --- a/status.go +++ b/status.go @@ -1,6 +1,8 @@ package main type Status struct { - Online bool `json:"online"` - ViewerCount int `json:"viewerCount"` + Online bool `json:"online"` + ViewerCount int `json:"viewerCount"` + OverallMaxViewerCount int `json:"overallMaxViewerCount"` + SessionMaxViewerCount int `json:"sessionMaxViewerCount"` } diff --git a/webroot/index.html b/webroot/index.html index 4ff89b8d1..2895310c2 100644 --- a/webroot/index.html +++ b/webroot/index.html @@ -41,6 +41,8 @@
{{ streamStatus }} {{ viewerCount }} {{ 'viewer' | plural(viewerCount) }}. + Max {{ sessionMaxViewerCount }} {{ 'viewer' | plural(sessionMaxViewerCount) }}, + {{ overallMaxViewerCount }} overall.
diff --git a/webroot/js/app.js b/webroot/js/app.js index 083db7b61..a0b216225 100644 --- a/webroot/js/app.js +++ b/webroot/js/app.js @@ -12,6 +12,8 @@ function setupApp() { data: { streamStatus: "", viewerCount: 0, + sessionMaxViewerCount: 0, + overallMaxViewerCount: 0 }, }); @@ -56,6 +58,8 @@ async function getStatus() { : "Stream is offline." app.viewerCount = status.viewerCount + app.sessionMaxViewerCount = status.sessionMaxViewerCount + app.overallMaxViewerCount = status.overallMaxViewerCount } catch (e) { app.streamStatus = "Stream server is offline."