You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
memos/server/router/frontend/frontend.go

190 lines
4.7 KiB
Go

package frontend
import (
"context"
"embed"
"encoding/xml"
"io/fs"
"net/http"
"path"
"strings"
"github.com/labstack/echo/v5"
"github.com/labstack/echo/v5/middleware"
"github.com/pkg/errors"
"github.com/usememos/memos/internal/profile"
"github.com/usememos/memos/store"
)
//go:embed dist/*
var embeddedFiles embed.FS
const (
frontendHTMLCacheControl = "no-cache, no-store, must-revalidate"
frontendStaticAssetCacheControl = "public, max-age=3600"
frontendHashedAssetCacheControl = "public, max-age=3600, immutable"
)
type FrontendService struct {
Profile *profile.Profile
Store *store.Store
}
func NewFrontendService(profile *profile.Profile, store *store.Store) *FrontendService {
return &FrontendService{
Profile: profile,
Store: store,
}
}
func (s *FrontendService) Serve(_ context.Context, e *echo.Echo) {
frontendFS := getFileSystem("dist")
skipper := func(c *echo.Context) bool {
requestPath := c.Request().URL.Path
if shouldSkipFrontendStatic(requestPath) {
return true
}
setFrontendCacheHeaders(c, requestPath)
return false
}
// Route to serve the frontend static assets.
e.Use(middleware.StaticWithConfig(middleware.StaticConfig{
Filesystem: frontendFS,
Skipper: skipper,
}))
e.Use(spaFallbackMiddleware(frontendFS))
s.registerRoutes(e)
}
func spaFallbackMiddleware(frontendFS fs.FS) echo.MiddlewareFunc {
return func(next echo.HandlerFunc) echo.HandlerFunc {
return func(c *echo.Context) error {
err := next(c)
if err == nil {
return nil
}
requestPath := c.Request().URL.Path
if shouldSkipFrontendStatic(requestPath) || !shouldServeFrontendHTML(requestPath) || echo.StatusCode(err) != http.StatusNotFound {
return err
}
setFrontendCacheHeaders(c, requestPath)
return c.FileFS("index.html", frontendFS)
}
}
}
func shouldSkipFrontendStatic(requestPath string) bool {
if requestPath == "/robots.txt" || requestPath == "/sitemap.xml" || strings.HasSuffix(requestPath, "/rss.xml") {
return true
}
return hasPathPrefix(requestPath, "/api") ||
hasPathPrefix(requestPath, "/file") ||
hasPathPrefix(requestPath, "/mcp") ||
requestPath == "/memos.api.v1" ||
strings.HasPrefix(requestPath, "/memos.api.v1.")
}
func setFrontendCacheHeaders(c *echo.Context, requestPath string) {
if shouldServeFrontendHTML(requestPath) {
c.Response().Header().Set(echo.HeaderCacheControl, frontendHTMLCacheControl)
c.Response().Header().Set("Pragma", "no-cache")
c.Response().Header().Set("Expires", "0")
return
}
cacheControl := frontendStaticAssetCacheControl
if strings.HasPrefix(requestPath, "/assets/") {
cacheControl = frontendHashedAssetCacheControl
}
c.Response().Header().Set(echo.HeaderCacheControl, cacheControl)
}
func shouldServeFrontendHTML(requestPath string) bool {
return requestPath == "/" || requestPath == "/index.html" || path.Ext(requestPath) == ""
}
func hasPathPrefix(requestPath, prefix string) bool {
return requestPath == prefix || strings.HasPrefix(requestPath, prefix+"/")
}
func getFileSystem(path string) fs.FS {
sub, err := fs.Sub(embeddedFiles, path)
if err != nil {
panic(err)
}
return sub
}
func (s *FrontendService) registerRoutes(e *echo.Echo) {
e.GET("/robots.txt", s.getRobotsTXT)
e.GET("/sitemap.xml", s.getSitemapXML)
}
func (s *FrontendService) getRobotsTXT(c *echo.Context) error {
instanceURL, err := normalizeInstanceURL(s.Profile.InstanceURL)
if err != nil {
return err
}
robotsTXT := strings.Join([]string{
"User-agent: *",
"Allow: /",
"Host: " + instanceURL,
"Sitemap: " + instanceURL + "/sitemap.xml",
}, "\n")
return c.String(http.StatusOK, robotsTXT)
}
func (s *FrontendService) getSitemapXML(c *echo.Context) error {
instanceURL, err := normalizeInstanceURL(s.Profile.InstanceURL)
if err != nil {
return err
}
memos, err := s.Store.ListMemos(c.Request().Context(), &store.FindMemo{
VisibilityList: []store.Visibility{store.Public},
})
if err != nil {
return errors.Wrap(err, "failed to list public memos for sitemap")
}
urls := make([]sitemapURL, 0, len(memos))
for _, memo := range memos {
urls = append(urls, sitemapURL{
Loc: instanceURL + "/memos/" + memo.UID,
})
}
return c.XML(http.StatusOK, sitemapURLSet{
XMLNS: sitemapXMLNamespace,
URLs: urls,
})
}
func normalizeInstanceURL(instanceURL string) (string, error) {
instanceURL = strings.TrimRight(instanceURL, "/")
if instanceURL == "" {
return "", echo.NewHTTPError(http.StatusNotFound, "instance URL is not configured")
}
return instanceURL, nil
}
type sitemapURLSet struct {
XMLName xml.Name `xml:"urlset"`
XMLNS string `xml:"xmlns,attr"`
URLs []sitemapURL `xml:"url"`
}
type sitemapURL struct {
Loc string `xml:"loc"`
}
//nolint:revive
const sitemapXMLNamespace = "http://www.sitemaps.org/schemas/sitemap/0.9"