2020-09-28 16:06:10 +00:00
|
|
|
package web
|
|
|
|
|
|
|
|
import (
|
2020-10-02 19:13:50 +00:00
|
|
|
"bytes"
|
2020-09-28 16:06:10 +00:00
|
|
|
"encoding/json"
|
2020-10-05 20:41:19 +00:00
|
|
|
"gitlab.crans.org/nounous/ghostream/stream/srt"
|
2020-10-02 19:13:50 +00:00
|
|
|
"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-04 16:10:11 +00:00
|
|
|
"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
|
|
|
)
|
|
|
|
|
|
|
|
// Handle WebRTC session description exchange via POST
|
|
|
|
func viewerPostHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// Limit response body to 128KB
|
|
|
|
r.Body = http.MaxBytesReader(w, r.Body, 131072)
|
|
|
|
|
2020-10-05 20:00:08 +00:00
|
|
|
// Get stream ID from URL, or from domain name
|
|
|
|
path := r.URL.Path[1:]
|
|
|
|
if cfg.OneStreamPerDomain {
|
|
|
|
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
|
|
|
|
}
|
|
|
|
host = realHost
|
|
|
|
}
|
|
|
|
path = host
|
|
|
|
}
|
|
|
|
|
2020-09-28 16:06:10 +00:00
|
|
|
// Decode client description
|
|
|
|
dec := json.NewDecoder(r.Body)
|
|
|
|
dec.DisallowUnknownFields()
|
|
|
|
remoteDescription := webrtc.SessionDescription{}
|
|
|
|
if err := dec.Decode(&remoteDescription); err != nil {
|
|
|
|
http.Error(w, "The JSON WebRTC offer is malformed", http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exchange session descriptions with WebRTC stream server
|
2020-10-05 20:00:08 +00:00
|
|
|
remoteSdpChan <- struct {
|
|
|
|
StreamID string
|
|
|
|
RemoteDescription webrtc.SessionDescription
|
|
|
|
}{StreamID: path, RemoteDescription: remoteDescription}
|
2020-09-28 16:06:10 +00:00
|
|
|
localDescription := <-localSdpChan
|
|
|
|
|
|
|
|
// Send server description as JSON
|
|
|
|
jsonDesc, err := json.Marshal(localDescription)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, "An error occurred while formating response", http.StatusInternalServerError)
|
|
|
|
log.Println("An error occurred while sending session description", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
2020-10-05 20:00:08 +00:00
|
|
|
_, err = w.Write(jsonDesc)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("An error occurred while sending session description", err)
|
|
|
|
}
|
2020-09-28 16:06:10 +00:00
|
|
|
|
|
|
|
// Increment monitoring
|
|
|
|
monitoring.WebSessions.Inc()
|
|
|
|
}
|
|
|
|
|
|
|
|
func viewerGetHandler(w http.ResponseWriter, r *http.Request) {
|
2020-10-04 15:56:03 +00:00
|
|
|
// Get stream ID from URL, or from domain name
|
|
|
|
path := r.URL.Path[1:]
|
|
|
|
if cfg.OneStreamPerDomain {
|
2020-10-04 16:12:35 +00:00
|
|
|
host := r.Host
|
2020-10-04 16:10:11 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
host = realHost
|
2020-10-04 15:56:03 +00:00
|
|
|
}
|
|
|
|
path = host
|
|
|
|
}
|
|
|
|
|
2020-09-28 16:06:10 +00:00
|
|
|
// Render template
|
|
|
|
data := struct {
|
2020-10-02 19:13:50 +00:00
|
|
|
Cfg *Options
|
|
|
|
Path string
|
|
|
|
WidgetURL string
|
2020-10-04 15:56:03 +00:00
|
|
|
}{Path: path, Cfg: cfg}
|
|
|
|
|
|
|
|
// Compute the WidgetURL with the stream path
|
2020-10-02 19:13:50 +00:00
|
|
|
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()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle site index and viewer pages
|
|
|
|
// POST requests are used to exchange WebRTC session descriptions
|
|
|
|
func viewerHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// Validation on path
|
|
|
|
if validPath.FindStringSubmatch(r.URL.Path) == nil {
|
|
|
|
http.NotFound(w, r)
|
|
|
|
log.Print(r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Route depending on HTTP method
|
|
|
|
switch r.Method {
|
|
|
|
case http.MethodGet:
|
|
|
|
viewerGetHandler(w, r)
|
|
|
|
case http.MethodPost:
|
|
|
|
viewerPostHandler(w, r)
|
|
|
|
default:
|
|
|
|
http.Error(w, "Sorry, only GET and POST methods are supported.", http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func staticHandler() http.Handler {
|
|
|
|
// Set up static files server
|
|
|
|
staticFs := http.FileServer(pkger.Dir("/web/static"))
|
|
|
|
return http.StripPrefix("/static/", staticFs)
|
|
|
|
}
|
2020-09-29 16:03:28 +00:00
|
|
|
|
|
|
|
func statisticsHandler(w http.ResponseWriter, r *http.Request) {
|
2020-10-05 20:41:19 +00:00
|
|
|
// Display connected users stats, from WebRTC or streaming directly from a video player
|
|
|
|
streamID := strings.Replace(r.URL.Path[7:], "/", "", -1)
|
2020-09-29 16:03:28 +00:00
|
|
|
enc := json.NewEncoder(w)
|
2020-09-29 16:20:24 +00:00
|
|
|
err := enc.Encode(struct {
|
2020-09-29 16:03:28 +00:00
|
|
|
ConnectedViewers int
|
2020-10-05 20:41:19 +00:00
|
|
|
}{webrtc.GetNumberConnectedSessions(streamID) + srt.GetNumberConnectedSessions(streamID)})
|
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)
|
|
|
|
}
|
2020-09-29 16:03:28 +00:00
|
|
|
}
|