2022-07-29 06:29:21 +02:00
|
|
|
// Copyright (C) 2022 NHR@FAU, University Erlangen-Nuremberg.
|
|
|
|
// All rights reserved.
|
|
|
|
// Use of this source code is governed by a MIT-style
|
|
|
|
// license that can be found in the LICENSE file.
|
2022-07-06 14:55:39 +02:00
|
|
|
package web
|
|
|
|
|
|
|
|
import (
|
|
|
|
"embed"
|
|
|
|
"html/template"
|
|
|
|
"io/fs"
|
|
|
|
"net/http"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/ClusterCockpit/cc-backend/internal/config"
|
|
|
|
"github.com/ClusterCockpit/cc-backend/pkg/log"
|
|
|
|
)
|
|
|
|
|
|
|
|
/// Go's embed is only allowed to embed files in a subdirectory of the embedding package ([see here](https://github.com/golang/go/issues/46056)).
|
|
|
|
|
|
|
|
//go:embed frontend/public/*
|
|
|
|
var frontendFiles embed.FS
|
|
|
|
|
|
|
|
func ServeFiles() http.Handler {
|
|
|
|
publicFiles, err := fs.Sub(frontendFiles, "frontend/public")
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return http.FileServer(http.FS(publicFiles))
|
|
|
|
}
|
|
|
|
|
|
|
|
//go:embed templates/*
|
|
|
|
var templateFiles embed.FS
|
|
|
|
|
|
|
|
var templates map[string]*template.Template = map[string]*template.Template{}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
base := template.Must(template.ParseFS(templateFiles, "templates/base.tmpl"))
|
|
|
|
if err := fs.WalkDir(templateFiles, "templates", func(path string, d fs.DirEntry, err error) error {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if d.IsDir() || path == "templates/base.tmpl" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
templates[strings.TrimPrefix(path, "templates/")] = template.Must(template.Must(base.Clone()).ParseFS(templateFiles, path))
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
_ = base
|
|
|
|
}
|
|
|
|
|
|
|
|
type User struct {
|
|
|
|
Username string // Username of the currently logged in user
|
|
|
|
IsAdmin bool
|
|
|
|
}
|
|
|
|
|
|
|
|
type Page struct {
|
|
|
|
Title string // Page title
|
|
|
|
Error string // For generic use (e.g. the exact error message on /login)
|
|
|
|
Info string // For generic use (e.g. "Logout successfull" on /login)
|
|
|
|
User User // Information about the currently logged in user
|
|
|
|
Clusters []string // List of all clusters for use in the Header
|
|
|
|
FilterPresets map[string]interface{} // For pages with the Filter component, this can be used to set initial filters.
|
|
|
|
Infos map[string]interface{} // For generic use (e.g. username for /monitoring/user/<id>, job id for /monitoring/job/<id>)
|
|
|
|
Config map[string]interface{} // UI settings for the currently logged in user (e.g. line width, ...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func RenderTemplate(rw http.ResponseWriter, r *http.Request, file string, page *Page) {
|
|
|
|
t, ok := templates[file]
|
|
|
|
if !ok {
|
|
|
|
panic("template not found")
|
|
|
|
}
|
|
|
|
|
|
|
|
if page.Clusters == nil {
|
|
|
|
for _, c := range config.Clusters {
|
|
|
|
page.Clusters = append(page.Clusters, c.Name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := t.Execute(rw, page); err != nil {
|
|
|
|
log.Errorf("template error: %s", err.Error())
|
|
|
|
}
|
|
|
|
}
|