mirror of
https://github.com/owncast/owncast.git
synced 2024-11-21 20:28:15 +03:00
7399bee276
* feat: add general purpose key/val caching layer * feat: cache bot/metadata response page for 10 seconds
206 lines
5.4 KiB
Go
206 lines
5.4 KiB
Go
package controllers
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net/http"
|
|
"net/url"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/owncast/owncast/config"
|
|
"github.com/owncast/owncast/core"
|
|
"github.com/owncast/owncast/core/cache"
|
|
"github.com/owncast/owncast/core/data"
|
|
"github.com/owncast/owncast/models"
|
|
"github.com/owncast/owncast/router/middleware"
|
|
"github.com/owncast/owncast/static"
|
|
"github.com/owncast/owncast/utils"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
var gc = cache.GetGlobalCache()
|
|
|
|
// IndexHandler handles the default index route.
|
|
func IndexHandler(w http.ResponseWriter, r *http.Request) {
|
|
middleware.EnableCors(w)
|
|
|
|
isIndexRequest := r.URL.Path == "/" || filepath.Base(r.URL.Path) == "index.html" || filepath.Base(r.URL.Path) == ""
|
|
|
|
if utils.IsUserAgentAPlayer(r.UserAgent()) && isIndexRequest {
|
|
http.Redirect(w, r, "/hls/stream.m3u8", http.StatusTemporaryRedirect)
|
|
return
|
|
}
|
|
|
|
// For search engine bots and social scrapers return a special
|
|
// server-rendered page.
|
|
if utils.IsUserAgentABot(r.UserAgent()) && isIndexRequest {
|
|
handleScraperMetadataPage(w, r)
|
|
return
|
|
}
|
|
|
|
// Set a cache control max-age header
|
|
middleware.SetCachingHeaders(w, r)
|
|
|
|
nonceRandom, _ := utils.GenerateRandomString(5)
|
|
|
|
// Set our global HTTP headers
|
|
middleware.SetHeaders(w, fmt.Sprintf("nonce-%s", nonceRandom))
|
|
|
|
if isIndexRequest {
|
|
renderIndexHtml(w, nonceRandom)
|
|
return
|
|
}
|
|
|
|
serveWeb(w, r)
|
|
}
|
|
|
|
func renderIndexHtml(w http.ResponseWriter, nonce string) {
|
|
type serverSideContent struct {
|
|
Name string
|
|
Summary string
|
|
RequestedURL string
|
|
TagsString string
|
|
ThumbnailURL string
|
|
Thumbnail string
|
|
Image string
|
|
StatusJSON string
|
|
ServerConfigJSON string
|
|
EmbedVideo string
|
|
Nonce string
|
|
}
|
|
|
|
status := getStatusResponse()
|
|
sb, err := json.Marshal(status)
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
config := getConfigResponse()
|
|
cb, err := json.Marshal(config)
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
content := serverSideContent{
|
|
Name: data.GetServerName(),
|
|
Summary: data.GetServerSummary(),
|
|
RequestedURL: fmt.Sprintf("%s%s", data.GetServerURL(), "/"),
|
|
TagsString: strings.Join(data.GetServerMetadataTags(), ","),
|
|
ThumbnailURL: "thumbnail.jpg",
|
|
Thumbnail: "thumbnail.jpg",
|
|
Image: "logo/external",
|
|
StatusJSON: string(sb),
|
|
ServerConfigJSON: string(cb),
|
|
EmbedVideo: "embed/video",
|
|
Nonce: nonce,
|
|
}
|
|
|
|
index, err := static.GetWebIndexTemplate()
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
if err := index.Execute(w, content); err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
}
|
|
}
|
|
|
|
// MetadataPage represents a server-rendered web page for bots and web scrapers.
|
|
type MetadataPage struct {
|
|
RequestedURL string
|
|
Image string
|
|
Thumbnail string
|
|
TagsString string
|
|
Summary string
|
|
Name string
|
|
Tags []string
|
|
SocialHandles []models.SocialHandle
|
|
}
|
|
|
|
// Return a basic HTML page with server-rendered metadata from the config
|
|
// to give to Opengraph clients and web scrapers (bots, web crawlers, etc).
|
|
func handleScraperMetadataPage(w http.ResponseWriter, r *http.Request) {
|
|
cacheKey := "bot-scraper-html"
|
|
cacheHtmlExpiration := time.Duration(10) * time.Second
|
|
c := gc.GetOrCreateCache(cacheKey, cacheHtmlExpiration)
|
|
|
|
cachedHtml := c.GetValueForKey(cacheKey)
|
|
if cachedHtml != nil {
|
|
w.Header().Set("Content-Type", "text/html")
|
|
_, _ = w.Write(cachedHtml)
|
|
return
|
|
}
|
|
|
|
tmpl, err := static.GetBotMetadataTemplate()
|
|
if err != nil {
|
|
log.Errorln(err)
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
scheme := "http"
|
|
|
|
if siteURL := data.GetServerURL(); siteURL != "" {
|
|
if parsed, err := url.Parse(siteURL); err == nil && parsed.Scheme != "" {
|
|
scheme = parsed.Scheme
|
|
}
|
|
}
|
|
|
|
fullURL, err := url.Parse(fmt.Sprintf("%s://%s%s", scheme, r.Host, r.URL.Path))
|
|
if err != nil {
|
|
log.Errorln(err)
|
|
}
|
|
imageURL, err := url.Parse(fmt.Sprintf("%s://%s%s", scheme, r.Host, "/logo/external"))
|
|
if err != nil {
|
|
log.Errorln(err)
|
|
}
|
|
|
|
status := core.GetStatus()
|
|
|
|
// If the thumbnail does not exist or we're offline then just use the logo image
|
|
var thumbnailURL string
|
|
if status.Online && utils.DoesFileExists(filepath.Join(config.DataDirectory, "tmp", "thumbnail.jpg")) {
|
|
thumbnail, err := url.Parse(fmt.Sprintf("%s://%s%s", scheme, r.Host, "/thumbnail.jpg"))
|
|
if err != nil {
|
|
log.Errorln(err)
|
|
thumbnailURL = imageURL.String()
|
|
} else {
|
|
thumbnailURL = thumbnail.String()
|
|
}
|
|
} else {
|
|
thumbnailURL = imageURL.String()
|
|
}
|
|
|
|
tagsString := strings.Join(data.GetServerMetadataTags(), ",")
|
|
metadata := MetadataPage{
|
|
Name: data.GetServerName(),
|
|
RequestedURL: fullURL.String(),
|
|
Image: imageURL.String(),
|
|
Summary: data.GetServerSummary(),
|
|
Thumbnail: thumbnailURL,
|
|
TagsString: tagsString,
|
|
Tags: data.GetServerMetadataTags(),
|
|
SocialHandles: data.GetSocialHandles(),
|
|
}
|
|
|
|
// Cache the rendered HTML
|
|
var b bytes.Buffer
|
|
if err := tmpl.Execute(&b, metadata); err != nil {
|
|
log.Errorln(err)
|
|
}
|
|
c.Set(cacheKey, b.Bytes())
|
|
|
|
// Set a cache header
|
|
middleware.SetCachingHeaders(w, r)
|
|
|
|
w.Header().Set("Content-Type", "text/html")
|
|
if _, err = w.Write(b.Bytes()); err != nil {
|
|
log.Errorln(err)
|
|
}
|
|
}
|