ghostream/web/handler.go

109 lines
2.7 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-09-28 16:06:10 +00:00
"github.com/markbates/pkger"
"gitlab.crans.org/nounous/ghostream/internal/monitoring"
2020-09-29 16:17:55 +00:00
"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-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
2020-10-13 17:36:28 +00:00
}{Path: path, Cfg: cfg, WidgetURL: ""}
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) {
name := strings.SplitN(strings.Replace(r.URL.Path[7:], "/", "", -1), "@", 2)[0]
2020-10-17 10:38:18 +00:00
userCount := 0
// 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-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)
}
}