ghostream/web/handler.go

145 lines
3.8 KiB
Go
Raw Permalink Normal View History

2020-10-14 19:37:05 +00:00
// Package web serves the JavaScript player and WebRTC negotiation
2020-09-28 16:06:10 +00:00
package web
import (
"bytes"
2020-09-28 16:06:10 +00:00
"encoding/json"
"html/template"
2020-09-28 16:06:10 +00:00
"log"
2020-10-04 15:56:03 +00:00
"net"
2020-09-28 16:06:10 +00:00
"net/http"
2020-10-16 19:23:13 +00:00
"regexp"
"strings"
2020-11-09 17:11:42 +00:00
"sync"
2020-11-09 16:57:55 +00:00
"time"
2020-11-09 17:11:42 +00:00
"github.com/markbates/pkger"
"gitlab.crans.org/nounous/ghostream/internal/monitoring"
"gitlab.crans.org/nounous/ghostream/stream/ovenmediaengine"
"gitlab.crans.org/nounous/ghostream/stream/webrtc"
2020-09-28 16:06:10 +00:00
)
2020-10-16 19:23:13 +00:00
var (
// Precompile regex
2020-10-17 10:38:18 +00:00
validPath = regexp.MustCompile("^/[a-z0-9@_-]*$")
2020-11-09 16:57:55 +00:00
2020-11-09 17:11:42 +00:00
counterMutex = new(sync.Mutex)
2020-11-09 16:57:55 +00:00
connectedClients = make(map[string]map[string]int64)
2020-10-16 19:23:13 +00:00
)
2020-10-20 17:12:15 +00:00
// Handle site index and viewer pages
func viewerHandler(w http.ResponseWriter, r *http.Request) {
// Validation on path
if validPath.FindStringSubmatch(r.URL.Path) == nil {
http.NotFound(w, r)
log.Printf("Replied not found on %s", r.URL.Path)
2020-10-19 19:45:23 +00:00
return
}
2020-10-20 17:12:15 +00:00
// Check method
if r.Method != http.MethodGet {
http.Error(w, "Method not allowed.", http.StatusMethodNotAllowed)
}
2020-09-28 16:06:10 +00:00
2020-10-04 15:56:03 +00:00
// Get stream ID from URL, or from domain name
path := r.URL.Path[1:]
host := r.Host
if strings.Contains(host, ":") {
realHost, _, err := net.SplitHostPort(r.Host)
if err != nil {
log.Printf("Failed to split host and port from %s", r.Host)
return
2020-10-04 15:56:03 +00:00
}
host = realHost
}
host = strings.Replace(host, ".", "-", -1)
if streamID, ok := cfg.MapDomainToStream[host]; ok {
2020-10-13 17:36:28 +00:00
// Move home page to /about
if path == "about" {
path = ""
} else {
path = streamID
}
2020-10-04 15:56:03 +00:00
}
2020-09-28 16:06:10 +00:00
// Render template
data := struct {
Cfg *Options
Path string
WidgetURL string
OMECfg *ovenmediaengine.Options
}{Path: path, Cfg: cfg, WidgetURL: "", OMECfg: omeCfg}
2020-10-04 15:56:03 +00:00
2020-10-13 17:36:28 +00:00
// Load widget is user does not disable it with ?nowidget
if _, ok := r.URL.Query()["nowidget"]; !ok {
// Compute the WidgetURL with the stream path
b := &bytes.Buffer{}
_ = template.Must(template.New("").Parse(cfg.WidgetURL)).Execute(b, data)
data.WidgetURL = b.String()
}
2020-09-28 16:06:10 +00:00
if err := templates.ExecuteTemplate(w, "base", data); err != nil {
log.Println(err.Error())
http.Error(w, "Internal Server Error", http.StatusInternalServerError)
return
}
// Increment monitoring
monitoring.WebViewerServed.Inc()
}
func staticHandler() http.Handler {
// Set up static files server
staticFs := http.FileServer(pkger.Dir("/web/static"))
return http.StripPrefix("/static/", staticFs)
}
func statisticsHandler(w http.ResponseWriter, r *http.Request) {
2020-11-09 16:57:55 +00:00
// Retrieve stream name from URL
name := strings.SplitN(strings.Replace(r.URL.Path[7:], "/", "", -1), "@", 2)[0]
2020-10-17 10:38:18 +00:00
userCount := 0
2020-11-09 16:57:55 +00:00
// Clients have a unique generated identifier per session, that expires in 40 seconds.
// Each time the client connects to this page, the identifier is renewed.
// Yeah, that's not a good way to have stats, but it works...
if connectedClients[name] == nil {
2020-11-09 17:11:42 +00:00
counterMutex.Lock()
2020-11-09 16:57:55 +00:00
connectedClients[name] = make(map[string]int64)
2020-11-09 17:11:42 +00:00
counterMutex.Unlock()
2020-11-09 16:57:55 +00:00
}
currentTime := time.Now().Unix()
if _, ok := r.URL.Query()["uid"]; ok {
uid := r.URL.Query()["uid"][0]
2020-11-09 17:11:42 +00:00
counterMutex.Lock()
2020-11-09 16:57:55 +00:00
connectedClients[name][uid] = currentTime
2020-11-09 17:11:42 +00:00
counterMutex.Unlock()
2020-11-09 16:57:55 +00:00
}
2020-11-09 17:03:15 +00:00
toDelete := make([]string, 0)
2020-11-09 17:11:42 +00:00
counterMutex.Lock()
2020-11-09 16:57:55 +00:00
for uid, oldTime := range connectedClients[name] {
if currentTime-oldTime > 40 {
2020-11-09 17:03:15 +00:00
toDelete = append(toDelete, uid)
2020-11-09 16:57:55 +00:00
}
}
2020-11-09 17:03:15 +00:00
for _, uid := range toDelete {
delete(connectedClients[name], uid)
}
2020-11-09 17:11:42 +00:00
counterMutex.Unlock()
2020-11-09 16:57:55 +00:00
// Get requested stream
stream, err := streams.Get(name)
if err == nil {
userCount = stream.ClientCount()
2020-10-19 18:05:20 +00:00
userCount += webrtc.GetNumberConnectedSessions(name)
2020-11-09 16:57:55 +00:00
userCount += len(connectedClients[name])
2020-10-17 10:38:18 +00:00
}
// Display connected users statistics
enc := json.NewEncoder(w)
err = enc.Encode(struct{ ConnectedViewers int }{userCount})
2020-09-29 16:20:24 +00:00
if err != nil {
http.Error(w, "Failed to generate JSON.", http.StatusInternalServerError)
log.Printf("Failed to generate JSON: %s", err)
}
}