2024-11-28 00:16:07 +00:00
|
|
|
package web
|
|
|
|
|
|
|
|
import (
|
|
|
|
"embed"
|
|
|
|
"errors"
|
|
|
|
"github.com/1f349/lavender/logger"
|
|
|
|
"github.com/1f349/lavender/utils"
|
|
|
|
"github.com/1f349/overlapfs"
|
|
|
|
"html/template"
|
|
|
|
"io"
|
|
|
|
"io/fs"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
2024-12-09 18:40:18 +00:00
|
|
|
"path"
|
2024-11-28 00:16:07 +00:00
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
//go:embed dist
|
|
|
|
webBuild embed.FS
|
|
|
|
|
|
|
|
webCombinedDir fs.FS
|
|
|
|
pageTemplates *template.Template
|
|
|
|
loadOnce utils.Once[error]
|
|
|
|
)
|
|
|
|
|
|
|
|
func LoadPages(wd string) error {
|
|
|
|
return loadOnce.Do(func() (err error) {
|
|
|
|
webCombinedDir, err = fs.Sub(webBuild, "dist")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if wd != "" {
|
|
|
|
webDir := filepath.Join(wd, "web")
|
|
|
|
|
|
|
|
err = os.Mkdir(webDir, os.ModePerm)
|
|
|
|
if err != nil && !errors.Is(err, fs.ErrExist) {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
wdFs := os.DirFS(webDir)
|
|
|
|
webCombinedDir = overlapfs.OverlapFS{A: webBuild, B: wdFs}
|
|
|
|
}
|
|
|
|
|
|
|
|
pageTemplates, err = template.New("web").Delims("[[", "]]").Funcs(template.FuncMap{
|
|
|
|
"emailHide": utils.EmailHide,
|
2024-12-09 18:40:18 +00:00
|
|
|
"renderTitle":
|
|
|
|
}).ParseFS(webCombinedDir, "*/index.html")
|
2024-11-28 00:16:07 +00:00
|
|
|
|
|
|
|
return err
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-12-09 18:40:18 +00:00
|
|
|
func renderTitle(title string, service string) string {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2024-11-28 00:16:07 +00:00
|
|
|
func RenderPageTemplate(wr io.Writer, name string, data any) {
|
2024-12-09 18:40:18 +00:00
|
|
|
p := path.Join(name, "index.html")
|
|
|
|
err := pageTemplates.ExecuteTemplate(wr, p, data)
|
2024-11-28 00:16:07 +00:00
|
|
|
if err != nil {
|
|
|
|
logger.Logger.Warn("Failed to render page", "name", name, "err", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-02 00:40:19 +00:00
|
|
|
func RenderWebAsset(rw http.ResponseWriter, req *http.Request, name string) {
|
2024-11-28 00:16:07 +00:00
|
|
|
// Disallow paths containing ".." - directory traversal is a security issue.
|
2024-12-02 00:40:19 +00:00
|
|
|
if containsDotDot(name) {
|
|
|
|
http.Error(rw, "400 Bad Request", http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
|
2024-11-28 00:16:07 +00:00
|
|
|
// Disallow paths ending in ".html" - these should only be processed by HTML
|
|
|
|
// template.
|
2024-12-02 00:40:19 +00:00
|
|
|
if strings.HasSuffix(name, ".html") {
|
2024-11-28 00:16:07 +00:00
|
|
|
http.Error(rw, "404 Not Found", http.StatusNotFound)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Enjoy the power of Go stdlib
|
|
|
|
http.ServeFileFS(rw, req, webCombinedDir, name)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Go stdlib net/http/fs.go (containsDotDot)
|
|
|
|
func containsDotDot(v string) bool {
|
|
|
|
if !strings.Contains(v, "..") {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
for _, ent := range strings.FieldsFunc(v, isSlashRune) {
|
|
|
|
if ent == ".." {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Go stdlib net/http/fs.go (isSlashRune)
|
|
|
|
func isSlashRune(r rune) bool { return r == '/' || r == '\\' }
|