2020-03-25 07:58:58 +01:00
|
|
|
package console
|
|
|
|
|
|
|
|
import (
|
2020-05-13 14:41:43 +02:00
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"path"
|
2020-06-22 13:17:29 +02:00
|
|
|
"strings"
|
|
|
|
"time"
|
2020-05-13 14:41:43 +02:00
|
|
|
|
2020-06-22 13:17:29 +02:00
|
|
|
"github.com/caos/zitadel/internal/api/http/middleware"
|
2020-03-25 07:58:58 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type Config struct {
|
2021-07-27 14:34:56 +02:00
|
|
|
Port string
|
|
|
|
ConsoleOverwriteDir string
|
|
|
|
ShortCache middleware.CacheConfig
|
|
|
|
LongCache middleware.CacheConfig
|
|
|
|
CSPDomain string
|
2020-05-13 14:41:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type spaHandler struct {
|
|
|
|
fileSystem http.FileSystem
|
|
|
|
}
|
|
|
|
|
2020-06-09 07:38:44 +02:00
|
|
|
const (
|
2021-07-27 14:34:56 +02:00
|
|
|
envRequestPath = "/assets/environment.json"
|
|
|
|
consoleDefaultDir = "./console/"
|
|
|
|
handlerPrefix = "/console"
|
2020-06-24 14:26:27 +02:00
|
|
|
)
|
2020-06-22 13:17:29 +02:00
|
|
|
|
2020-06-24 14:26:27 +02:00
|
|
|
var (
|
2020-11-20 08:47:28 +01:00
|
|
|
shortCacheFiles = []string{
|
|
|
|
"/",
|
2020-06-24 14:26:27 +02:00
|
|
|
"/index.html",
|
|
|
|
"/manifest.webmanifest",
|
|
|
|
"/ngsw.json",
|
|
|
|
"/ngsw-worker.js",
|
|
|
|
"/safety-worker.js",
|
|
|
|
"/worker-basic.min.js",
|
|
|
|
}
|
2020-06-09 07:38:44 +02:00
|
|
|
)
|
|
|
|
|
2020-05-13 14:41:43 +02:00
|
|
|
func (i *spaHandler) Open(name string) (http.File, error) {
|
|
|
|
ret, err := i.fileSystem.Open(name)
|
|
|
|
if !os.IsNotExist(err) || path.Ext(name) != "" {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return i.fileSystem.Open("/index.html")
|
2020-03-25 07:58:58 +01:00
|
|
|
}
|
|
|
|
|
2020-08-31 08:49:35 +02:00
|
|
|
func Start(config Config) (http.Handler, string, error) {
|
2021-07-27 14:34:56 +02:00
|
|
|
consoleDir := consoleDefaultDir
|
|
|
|
if config.ConsoleOverwriteDir != "" {
|
|
|
|
consoleDir = config.ConsoleOverwriteDir
|
2020-06-09 07:38:44 +02:00
|
|
|
}
|
2021-07-27 14:34:56 +02:00
|
|
|
consoleHTTPDir := http.Dir(consoleDir)
|
2020-06-24 14:26:27 +02:00
|
|
|
cache := AssetsCacheInterceptorIgnoreManifest(
|
|
|
|
config.ShortCache.MaxAge.Duration,
|
|
|
|
config.ShortCache.SharedMaxAge.Duration,
|
|
|
|
config.LongCache.MaxAge.Duration,
|
|
|
|
config.LongCache.SharedMaxAge.Duration,
|
|
|
|
)
|
2020-06-22 13:17:29 +02:00
|
|
|
security := middleware.SecurityHeaders(csp(config.CSPDomain), nil)
|
2020-07-08 13:56:37 +02:00
|
|
|
handler := &http.ServeMux{}
|
2021-07-27 14:34:56 +02:00
|
|
|
handler.Handle("/", cache(security(http.FileServer(&spaHandler{consoleHTTPDir}))))
|
|
|
|
handler.Handle(envRequestPath, cache(security(http.StripPrefix("/assets", http.FileServer(consoleHTTPDir)))))
|
2020-08-31 08:49:35 +02:00
|
|
|
return handler, handlerPrefix, nil
|
2020-03-25 07:58:58 +01:00
|
|
|
}
|
2020-06-22 13:17:29 +02:00
|
|
|
|
|
|
|
func csp(zitadelDomain string) *middleware.CSP {
|
|
|
|
if !strings.HasPrefix(zitadelDomain, "*.") {
|
|
|
|
zitadelDomain = "*." + zitadelDomain
|
|
|
|
}
|
|
|
|
csp := middleware.DefaultSCP
|
2022-02-03 07:21:00 +01:00
|
|
|
csp.StyleSrc = csp.StyleSrc.AddInline()
|
2020-06-22 13:17:29 +02:00
|
|
|
csp.ScriptSrc = csp.ScriptSrc.AddEval()
|
2022-02-03 07:21:00 +01:00
|
|
|
csp.ConnectSrc = csp.ConnectSrc.AddHost(zitadelDomain)
|
2021-06-14 13:53:40 +02:00
|
|
|
csp.ImgSrc = csp.ImgSrc.AddHost(zitadelDomain).AddScheme("blob")
|
2020-06-22 13:17:29 +02:00
|
|
|
return &csp
|
|
|
|
}
|
|
|
|
|
2020-06-24 14:26:27 +02:00
|
|
|
func AssetsCacheInterceptorIgnoreManifest(shortMaxAge, shortSharedMaxAge, longMaxAge, longSharedMaxAge time.Duration) func(http.Handler) http.Handler {
|
2020-06-22 13:17:29 +02:00
|
|
|
return func(handler http.Handler) http.Handler {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
2020-11-20 08:47:28 +01:00
|
|
|
for _, file := range shortCacheFiles {
|
|
|
|
if r.URL.Path == file {
|
2020-06-24 14:26:27 +02:00
|
|
|
middleware.AssetsCacheInterceptor(shortMaxAge, shortSharedMaxAge, handler).ServeHTTP(w, r)
|
|
|
|
return
|
|
|
|
}
|
2020-06-22 13:17:29 +02:00
|
|
|
}
|
2020-11-20 08:47:28 +01:00
|
|
|
middleware.AssetsCacheInterceptor(longMaxAge, longSharedMaxAge, handler).ServeHTTP(w, r)
|
|
|
|
return
|
2020-06-22 13:17:29 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|