ghostream/web/web.go

122 lines
3.2 KiB
Go
Raw Normal View History

2020-09-21 15:47:31 +00:00
package web
import (
2020-09-23 11:52:12 +00:00
"encoding/json"
2020-09-21 15:47:31 +00:00
"html/template"
"log"
"net/http"
"os"
2020-09-21 17:59:41 +00:00
2020-09-23 11:52:12 +00:00
"github.com/pion/webrtc/v3"
2020-09-21 19:38:11 +00:00
"gitlab.crans.org/nounous/ghostream/internal/monitoring"
2020-09-21 15:47:31 +00:00
)
2020-09-22 09:42:57 +00:00
// Options holds web package configuration
type Options struct {
ListenAddress string
Name string
Hostname string
Favicon string
WidgetURL string
}
2020-09-24 09:24:13 +00:00
var (
cfg *Options
// WebRTC session description channels
remoteSdpChan chan webrtc.SessionDescription
localSdpChan chan webrtc.SessionDescription
// Preload templates
templates = template.Must(template.ParseGlob("web/template/*.html"))
)
2020-09-21 15:47:31 +00:00
2020-09-23 11:52:12 +00:00
// Handle WebRTC session description exchange via POST
2020-09-24 09:24:13 +00:00
func viewerPostHandler(w http.ResponseWriter, r *http.Request) {
2020-09-23 11:52:12 +00:00
// Limit response body to 128KB
r.Body = http.MaxBytesReader(w, r.Body, 131072)
// 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
}
2020-09-24 09:24:13 +00:00
// Exchange session descriptions with WebRTC stream server
remoteSdpChan <- remoteDescription
localDescription := <-localSdpChan
2020-09-23 11:52:12 +00:00
// Send server description as JSON
jsonDesc, err := json.Marshal(localDescription)
if err != nil {
http.Error(w, "An error occured while formating response", http.StatusInternalServerError)
log.Println("An error occured while sending session description", err)
return
}
w.Header().Set("Content-Type", "application/json")
w.Write(jsonDesc)
2020-09-24 09:24:13 +00:00
// Increment monitoring
monitoring.WebSessions.Inc()
}
func viewerGetHandler(w http.ResponseWriter, r *http.Request) {
// Render template
data := struct {
Path string
Cfg *Options
}{Path: r.URL.Path[1:], Cfg: cfg}
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()
2020-09-23 11:52:12 +00:00
}
2020-09-21 15:47:31 +00:00
// Handle site index and viewer pages
2020-09-23 11:52:12 +00:00
// POST requests are used to exchange WebRTC session descriptions
2020-09-24 09:24:13 +00:00
func viewerHandler(w http.ResponseWriter, r *http.Request) {
2020-09-21 18:38:21 +00:00
// FIXME validation on path: https://golang.org/doc/articles/wiki/#tmp_11
2020-09-21 15:47:31 +00:00
2020-09-24 09:24:13 +00:00
// Route depending on HTTP method
2020-09-23 11:52:12 +00:00
switch r.Method {
2020-09-24 09:24:13 +00:00
case http.MethodGet:
viewerGetHandler(w, r)
case http.MethodPost:
viewerPostHandler(w, r)
2020-09-23 11:52:12 +00:00
default:
http.Error(w, "Sorry, only GET and POST methods are supported.", http.StatusBadRequest)
2020-09-21 15:47:31 +00:00
}
}
// Handle static files
// We do not use http.FileServer as we do not want directory listing
2020-09-24 09:24:13 +00:00
func staticHandler(w http.ResponseWriter, r *http.Request) {
2020-09-21 15:47:31 +00:00
path := "./web/" + r.URL.Path
if f, err := os.Stat(path); err == nil && !f.IsDir() {
http.ServeFile(w, r, path)
} else {
http.NotFound(w, r)
}
}
2020-09-24 09:24:13 +00:00
// Serve HTTP server
func Serve(rSdpChan chan webrtc.SessionDescription, lSdpChan chan webrtc.SessionDescription, c *Options) {
remoteSdpChan = rSdpChan
localSdpChan = lSdpChan
cfg = c
2020-09-21 18:38:21 +00:00
2020-09-21 15:47:31 +00:00
// Set up HTTP router and server
2020-09-21 19:33:32 +00:00
mux := http.NewServeMux()
2020-09-24 09:24:13 +00:00
mux.HandleFunc("/", viewerHandler)
mux.HandleFunc("/static/", staticHandler)
2020-09-22 09:42:57 +00:00
log.Printf("HTTP server listening on %s", cfg.ListenAddress)
log.Fatal(http.ListenAndServe(cfg.ListenAddress, mux))
2020-09-21 15:47:31 +00:00
}