Support server-rendered index.html for all clients. Closes #1871
This commit is contained in:
parent
78c6189c02
commit
ff968616ba
@ -36,12 +36,6 @@ func IndexHandler(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
isIndexRequest := r.URL.Path == "/" || filepath.Base(r.URL.Path) == "index.html" || filepath.Base(r.URL.Path) == ""
|
isIndexRequest := r.URL.Path == "/" || filepath.Base(r.URL.Path) == "index.html" || filepath.Base(r.URL.Path) == ""
|
||||||
if isIndexRequest {
|
if isIndexRequest {
|
||||||
serveWeb(w, r)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// For search engine bots and social scrapers return a special
|
|
||||||
// server-rendered page.
|
|
||||||
if utils.IsUserAgentABot(r.UserAgent()) && isIndexRequest {
|
|
||||||
handleScraperMetadataPage(w, r)
|
handleScraperMetadataPage(w, r)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -78,7 +72,7 @@ func IndexHandler(w http.ResponseWriter, r *http.Request) {
|
|||||||
// Return a basic HTML page with server-rendered metadata from the config
|
// Return a basic HTML page with server-rendered metadata from the config
|
||||||
// to give to Opengraph clients and web scrapers (bots, web crawlers, etc).
|
// to give to Opengraph clients and web scrapers (bots, web crawlers, etc).
|
||||||
func handleScraperMetadataPage(w http.ResponseWriter, r *http.Request) {
|
func handleScraperMetadataPage(w http.ResponseWriter, r *http.Request) {
|
||||||
tmpl, err := static.GetBotMetadataTemplate()
|
tmpl, err := static.GetWebIndexTemplate()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Errorln(err)
|
log.Errorln(err)
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
16
static/static.go
vendored
16
static/static.go
vendored
@ -5,6 +5,8 @@ import (
|
|||||||
"html/template"
|
"html/template"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
)
|
)
|
||||||
|
|
||||||
//go:embed web/*
|
//go:embed web/*
|
||||||
@ -18,13 +20,15 @@ func GetWeb() embed.FS {
|
|||||||
return webFiles
|
return webFiles
|
||||||
}
|
}
|
||||||
|
|
||||||
//go:embed metadata.html.tmpl
|
// GetWebIndexTemplate will return the bot/scraper metadata template.
|
||||||
var botMetadataTemplate embed.FS
|
func GetWebIndexTemplate() (*template.Template, error) {
|
||||||
|
webFiles := GetWeb()
|
||||||
|
name := "web/index.html"
|
||||||
|
t, err := template.ParseFS(webFiles, name)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "unable to open index.html template")
|
||||||
|
}
|
||||||
|
|
||||||
// GetBotMetadataTemplate will return the bot/scraper metadata template.
|
|
||||||
func GetBotMetadataTemplate() (*template.Template, error) {
|
|
||||||
name := "metadata.html.tmpl"
|
|
||||||
t, err := template.ParseFS(botMetadataTemplate, name)
|
|
||||||
tmpl := template.Must(t, err)
|
tmpl := template.Must(t, err)
|
||||||
return tmpl, err
|
return tmpl, err
|
||||||
}
|
}
|
||||||
|
@ -15,7 +15,6 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/mssola/user_agent"
|
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
"github.com/yuin/goldmark"
|
"github.com/yuin/goldmark"
|
||||||
"github.com/yuin/goldmark/extension"
|
"github.com/yuin/goldmark/extension"
|
||||||
@ -98,34 +97,6 @@ func moveFallback(source, destination string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// IsUserAgentABot returns if a web client user-agent is seen as a bot.
|
|
||||||
func IsUserAgentABot(userAgent string) bool {
|
|
||||||
if userAgent == "" {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
botStrings := []string{
|
|
||||||
"mastodon",
|
|
||||||
"pleroma",
|
|
||||||
"applebot",
|
|
||||||
"whatsapp",
|
|
||||||
"matrix",
|
|
||||||
"synapse",
|
|
||||||
"element",
|
|
||||||
"rocket.chat",
|
|
||||||
"duckduckbot",
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, botString := range botStrings {
|
|
||||||
if strings.Contains(strings.ToLower(userAgent), botString) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ua := user_agent.New(userAgent)
|
|
||||||
return ua.Bot()
|
|
||||||
}
|
|
||||||
|
|
||||||
// IsUserAgentAPlayer returns if a web client user-agent is seen as a media player.
|
// IsUserAgentAPlayer returns if a web client user-agent is seen as a media player.
|
||||||
func IsUserAgentAPlayer(userAgent string) bool {
|
func IsUserAgentAPlayer(userAgent string) bool {
|
||||||
if userAgent == "" {
|
if userAgent == "" {
|
||||||
|
@ -4,21 +4,6 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestUserAgent(t *testing.T) {
|
|
||||||
testAgents := []string{
|
|
||||||
"Pleroma 1.0.0-1168-ge18c7866-pleroma-dot-site; https://pleroma.site info@pleroma.site",
|
|
||||||
"Mastodon 1.2.3 Bot",
|
|
||||||
"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_5) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/13.1.1 Safari/605.1.15 (Applebot/0.1; +http://www.apple.com/go/applebot)",
|
|
||||||
"WhatsApp",
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, agent := range testAgents {
|
|
||||||
if !IsUserAgentABot(agent) {
|
|
||||||
t.Error("Incorrect parsing of useragent", agent)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGetHashtagsFromText(t *testing.T) {
|
func TestGetHashtagsFromText(t *testing.T) {
|
||||||
text := `Some text with a #hashtag goes here.\n\n
|
text := `Some text with a #hashtag goes here.\n\n
|
||||||
Another #secondhashtag, goes here.\n\n
|
Another #secondhashtag, goes here.\n\n
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import { Layout } from 'antd';
|
import { Layout } from 'antd';
|
||||||
import { useRecoilValue } from 'recoil';
|
import { useRecoilValue } from 'recoil';
|
||||||
|
import Head from 'next/head';
|
||||||
import {
|
import {
|
||||||
ClientConfigStore,
|
ClientConfigStore,
|
||||||
isChatAvailableSelector,
|
isChatAvailableSelector,
|
||||||
@ -19,6 +20,63 @@ function Main() {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
<Head>
|
||||||
|
<meta name="description" content="{{.Summary}}" />
|
||||||
|
|
||||||
|
<meta property="og:title" content="{{.Name}}" />
|
||||||
|
<meta property="og:site_name" content="{{.Name}}" />
|
||||||
|
<meta property="og:url" content="{{.RequestedURL}}" />
|
||||||
|
<meta property="og:description" content="{{.Summary}}" />
|
||||||
|
<meta property="og:type" content="video.other" />
|
||||||
|
<meta property="video:tag" content="{{.TagsString}}" />
|
||||||
|
|
||||||
|
<meta property="og:image" content="{{.Thumbnail}}" />
|
||||||
|
<meta property="og:image:url" content="{{.Thumbnail}}" />
|
||||||
|
<meta property="og:image:alt" content="{{.Image}}" />
|
||||||
|
|
||||||
|
<meta property="og:video" content="{{.RequestedURL}}embed/video" />
|
||||||
|
<meta property="og:video:secure_url" content="{{.RequestedURL}}embed/video" />
|
||||||
|
<meta property="og:video:height" content="315" />
|
||||||
|
<meta property="og:video:width" content="560" />
|
||||||
|
<meta property="og:video:type" content="text/html" />
|
||||||
|
<meta property="og:video:actor" content="{{.Name}}" />
|
||||||
|
|
||||||
|
<meta property="twitter:title" content="{{.Name}}" />
|
||||||
|
<meta property="twitter:url" content="{{.RequestedURL}}" />
|
||||||
|
<meta property="twitter:description" content="{{.Summary}}" />
|
||||||
|
<meta property="twitter:image" content="{{.Image}}" />
|
||||||
|
<meta property="twitter:card" content="player" />
|
||||||
|
<meta property="twitter:player" content="{{.RequestedURL}}embed/video" />
|
||||||
|
<meta property="twitter:player:width" content="560" />
|
||||||
|
<meta property="twitter:player:height" content="315" />
|
||||||
|
|
||||||
|
<link rel="apple-touch-icon" sizes="57x57" href="/img/favicon/apple-icon-57x57.png" />
|
||||||
|
<link rel="apple-touch-icon" sizes="60x60" href="/img/favicon/apple-icon-60x60.png" />
|
||||||
|
<link rel="apple-touch-icon" sizes="72x72" href="/img/favicon/apple-icon-72x72.png" />
|
||||||
|
<link rel="apple-touch-icon" sizes="76x76" href="/img/favicon/apple-icon-76x76.png" />
|
||||||
|
<link rel="apple-touch-icon" sizes="114x114" href="/img/favicon/apple-icon-114x114.png" />
|
||||||
|
<link rel="apple-touch-icon" sizes="120x120" href="/img/favicon/apple-icon-120x120.png" />
|
||||||
|
<link rel="apple-touch-icon" sizes="144x144" href="/img/favicon/apple-icon-144x144.png" />
|
||||||
|
<link rel="apple-touch-icon" sizes="152x152" href="/img/favicon/apple-icon-152x152.png" />
|
||||||
|
<link rel="apple-touch-icon" sizes="180x180" href="/img/favicon/apple-icon-180x180.png" />
|
||||||
|
<link
|
||||||
|
rel="icon"
|
||||||
|
type="image/png"
|
||||||
|
sizes="192x192"
|
||||||
|
href="/img/favicon/android-icon-192x192.png"
|
||||||
|
/>
|
||||||
|
<link rel="icon" type="image/png" sizes="32x32" href="/img/favicon/favicon-32x32.png" />
|
||||||
|
<link rel="icon" type="image/png" sizes="96x96" href="/img/favicon/favicon-96x96.png" />
|
||||||
|
<link rel="icon" type="image/png" sizes="16x16" href="/img/favicon/favicon-16x16.png" />
|
||||||
|
<link rel="manifest" href="/manifest.json" />
|
||||||
|
|
||||||
|
<link href="/api/auth/provider/indieauth" />
|
||||||
|
|
||||||
|
<meta name="msapplication-TileColor" content="#ffffff" />
|
||||||
|
<meta name="msapplication-TileImage" content="/img/favicon/ms-icon-144x144.png" />
|
||||||
|
<meta name="theme-color" content="#ffffff" />
|
||||||
|
</Head>
|
||||||
|
|
||||||
<ClientConfigStore />
|
<ClientConfigStore />
|
||||||
<Layout>
|
<Layout>
|
||||||
<Header name={title || name} chatAvailable={isChatAvailable} />
|
<Header name={title || name} chatAvailable={isChatAvailable} />
|
||||||
|
Loading…
Reference in New Issue
Block a user