diff --git a/Dockerfile b/Dockerfile index c53616677df17d0321b1f84719917a86fa20f67e..4f0d43a4d3d3dc6f6ea2f1bf4fa241df3babf5d1 100644 --- a/Dockerfile +++ b/Dockerfile @@ -13,6 +13,11 @@ RUN apk add --update ca-certificates openssl COPY _output/bin/dex /usr/local/bin/dex +# Import frontend assets and set the correct CWD directory so the assets +# are in the default path. +COPY web /web +WORKDIR / + ENTRYPOINT ["dex"] CMD ["version"] diff --git a/Makefile b/Makefile index 62ab589529a283b8298ea3a536d0dd0de8e90761..5aa8144e0fd57e53b2be1ae4a69b26e7c41d0755 100644 --- a/Makefile +++ b/Makefile @@ -25,7 +25,7 @@ LD_FLAGS="-w -X $(REPO_PATH)/version.Version=$(VERSION)" build: bin/dex bin/example-app -bin/dex: FORCE generated +bin/dex: FORCE @go install -v -ldflags $(LD_FLAGS) $(REPO_PATH)/cmd/dex bin/example-app: FORCE @@ -35,9 +35,6 @@ bin/example-app: FORCE release-binary: @go build -o _output/bin/dex -v -ldflags $(LD_FLAGS) $(REPO_PATH)/cmd/dex -.PHONY: generated -generated: server/templates_default.go - test: @go test -v -i $(shell go list ./... | grep -v '/vendor/') @go test -v $(shell go list ./... | grep -v '/vendor/') @@ -57,9 +54,6 @@ lint: golint -set_exit_status $$package $$i || exit 1; \ done -server/templates_default.go: $(wildcard web/templates/**) - @go run server/templates_default_gen.go - _output/bin/dex: # Using rkt to build the dex binary. @./scripts/rkt-build diff --git a/cmd/dex/config.go b/cmd/dex/config.go index 2ee9e58ae2480e4516ea97295bf108b3eab636c7..dc3715b8f3ca3a2c4db48049845a404af201036a 100644 --- a/cmd/dex/config.go +++ b/cmd/dex/config.go @@ -30,7 +30,7 @@ type Config struct { GRPC GRPC `json:"grpc"` Expiry Expiry `json:"expiry"` - Templates server.TemplateConfig `json:"templates"` + Frontend server.WebConfig `json:"frontend"` // StaticClients cause the server to use this list of clients rather than // querying the storage. Write operations, like creating a client, will fail. diff --git a/cmd/dex/serve.go b/cmd/dex/serve.go index e218e4732c90b6ff8c801aa2b71a00818e73aed7..c9baa9d38ce6a0de4ad05812c3404c6eb103c3e1 100644 --- a/cmd/dex/serve.go +++ b/cmd/dex/serve.go @@ -151,7 +151,7 @@ func serve(cmd *cobra.Command, args []string) error { Issuer: c.Issuer, Connectors: connectors, Storage: s, - TemplateConfig: c.Templates, + Web: c.Frontend, EnablePasswordDB: c.EnablePasswordDB, } if c.Expiry.SigningKeys != "" { diff --git a/examples/config-dev.yaml b/examples/config-dev.yaml index 5f94e2023709753bbf3b98564d1b1b664e6d6390..1b5a48d3f15cf0eb498548b6ea32e0fd2561187b 100644 --- a/examples/config-dev.yaml +++ b/examples/config-dev.yaml @@ -14,7 +14,7 @@ storage: # Configuration for the HTTP endpoints. web: - http: 127.0.0.1:5556 + http: 0.0.0.0:5556 # Uncomment for HTTPS options. # https: 127.0.0.1:5554 # tlsCert: /etc/dex/tls.crt diff --git a/server/server.go b/server/server.go index b8d2c8d395d9295163bfb1d3a2616e79a19f9621..91d119e9fda8b06189eee939bbf3f3265bf9e911 100644 --- a/server/server.go +++ b/server/server.go @@ -56,7 +56,32 @@ type Config struct { EnablePasswordDB bool - TemplateConfig TemplateConfig + Web WebConfig +} + +// WebConfig holds the server's frontend templates and asset configuration. +// +// These are currently very custom to CoreOS and it's not recommended that +// outside users attempt to customize these. +type WebConfig struct { + // A filepath to web static. + // + // It is expected to contain the following directories: + // + // * static - Static static served at "( issuer URL )/static". + // * templates - HTML templates controlled by dex. + // * themes/(theme) - Static static served at "( issuer URL )/theme". + // + Dir string + + // Defaults to "( issuer URL )/theme/logo.png" + LogoURL string + + // Defaults to "dex" + Issuer string + + // Defaults to "coreos" + Theme string } func value(val, defaultValue time.Duration) time.Duration { @@ -130,9 +155,17 @@ func newServer(ctx context.Context, c Config, rotationStrategy rotationStrategy) supported[respType] = true } - tmpls, err := loadTemplates(c.TemplateConfig) + web := webConfig{ + dir: c.Web.Dir, + logoURL: c.Web.LogoURL, + issuerURL: c.Issuer, + issuer: c.Web.Issuer, + theme: c.Web.Theme, + } + + static, theme, tmpls, err := loadWebConfig(web) if err != nil { - return nil, fmt.Errorf("server: failed to load templates: %v", err) + return nil, fmt.Errorf("server: failed to load web static: %v", err) } now := c.Now @@ -159,6 +192,10 @@ func newServer(ctx context.Context, c Config, rotationStrategy rotationStrategy) handleFunc := func(p string, h http.HandlerFunc) { r.HandleFunc(path.Join(issuerURL.Path, p), h) } + handlePrefix := func(p string, h http.Handler) { + prefix := path.Join(issuerURL.Path, p) + r.PathPrefix(prefix).Handler(http.StripPrefix(prefix, h)) + } r.NotFoundHandler = http.HandlerFunc(s.notFound) discoveryHandler, err := s.discoveryHandler() @@ -175,6 +212,8 @@ func newServer(ctx context.Context, c Config, rotationStrategy rotationStrategy) handleFunc("/callback", s.handleConnectorCallback) handleFunc("/approval", s.handleApproval) handleFunc("/healthz", s.handleHealth) + handlePrefix("/static", static) + handlePrefix("/theme", theme) s.mux = r startKeyRotation(ctx, c.Storage, rotationStrategy, now) diff --git a/server/server_test.go b/server/server_test.go index a5865dfa33e1a7b3828222e7edc3b580d5cda65e..3ab41940826a8352990a9ee418959f31a02cc87f 100644 --- a/server/server_test.go +++ b/server/server_test.go @@ -11,6 +11,8 @@ import ( "net/http/httptest" "net/http/httputil" "net/url" + "os" + "path/filepath" "reflect" "sort" "strings" @@ -85,6 +87,9 @@ func newTestServer(ctx context.Context, t *testing.T, updateConfig func(c *Confi Connector: mock.NewCallbackConnector(), }, }, + Web: WebConfig{ + Dir: filepath.Join(os.Getenv("GOPATH"), "src/github.com/coreos/dex/web"), + }, } if updateConfig != nil { updateConfig(&config) diff --git a/server/templates.go b/server/templates.go index e8285fe31e26b579b076cc430289ec8fe1f2b10b..649d5b0837e3a4ab15da1f1f253078846bf6f2de 100644 --- a/server/templates.go +++ b/server/templates.go @@ -6,8 +6,10 @@ import ( "io/ioutil" "log" "net/http" + "os" "path/filepath" "sort" + "strings" "text/template" ) @@ -18,8 +20,6 @@ const ( tmplOOB = "oob.html" ) -const coreOSLogoURL = "https://coreos.com/assets/images/brand/coreos-wordmark-135x40px.png" - var requiredTmpls = []string{ tmplApproval, tmplLogin, @@ -27,65 +27,122 @@ var requiredTmpls = []string{ tmplOOB, } -// TemplateConfig describes. -type TemplateConfig struct { - // TODO(ericchiang): Asking for a directory with a set of templates doesn't indicate - // what the templates should look like and doesn't allow consumers of this package to - // provide their own templates in memory. In the future clean this up. - - // Directory of the templates. If empty, these will be loaded from memory. - Dir string `yaml:"dir"` +type templates struct { + loginTmpl *template.Template + approvalTmpl *template.Template + passwordTmpl *template.Template + oobTmpl *template.Template +} - // Defaults to the CoreOS logo and "dex". - LogoURL string `yaml:"logoURL"` - Issuer string `yaml:"issuerName"` +type webConfig struct { + dir string + logoURL string + issuer string + theme string + issuerURL string } -type globalData struct { - LogoURL string - Issuer string +func join(base, path string) string { + b := strings.HasSuffix(base, "/") + p := strings.HasPrefix(path, "/") + switch { + case b && p: + return base + path[1:] + case b || p: + return base + path + default: + return base + "/" + path + } } -func loadTemplates(config TemplateConfig) (*templates, error) { - var tmpls *template.Template - if config.Dir != "" { - files, err := ioutil.ReadDir(config.Dir) - if err != nil { - return nil, fmt.Errorf("read dir: %v", err) +func dirExists(dir string) error { + stat, err := os.Stat(dir) + if err != nil { + if os.IsNotExist(err) { + return fmt.Errorf("directory %q does not exist", dir) } - filenames := []string{} - for _, file := range files { - if file.IsDir() { - continue - } - filenames = append(filenames, filepath.Join(config.Dir, file.Name())) - } - if len(filenames) == 0 { - return nil, fmt.Errorf("no files in template dir %s", config.Dir) - } - if tmpls, err = template.ParseFiles(filenames...); err != nil { - return nil, fmt.Errorf("parse files: %v", err) + return fmt.Errorf("stat directory %q: %v", dir, err) + } + if !stat.IsDir() { + return fmt.Errorf("path %q is a file not a directory", dir) + } + return nil +} + +// loadWebConfig returns static assets, theme assets, and templates used by the frontend by +// reading the directory specified in the webConfig. +// +// The directory layout is expected to be: +// +// ( web directory ) +// |- static +// |- themes +// | |- (theme name) +// |- templates +// +func loadWebConfig(c webConfig) (static, theme http.Handler, templates *templates, err error) { + if c.theme == "" { + c.theme = "coreos" + } + if c.issuer == "" { + c.issuer = "dex" + } + if c.dir == "" { + c.dir = "./web" + } + if c.logoURL == "" { + c.logoURL = join(c.issuerURL, "theme/logo.png") + } + + if err := dirExists(c.dir); err != nil { + return nil, nil, nil, fmt.Errorf("load web dir: %v", err) + } + + staticDir := filepath.Join(c.dir, "static") + templatesDir := filepath.Join(c.dir, "templates") + themeDir := filepath.Join(c.dir, "themes", c.theme) + + for _, dir := range []string{staticDir, templatesDir, themeDir} { + if err := dirExists(dir); err != nil { + return nil, nil, nil, fmt.Errorf("load dir: %v", err) } - } else { - // Load templates from memory. This code is largely copied from the standard library's - // ParseFiles source code. - // See: https://goo.gl/6Wm4mN - for name, data := range defaultTemplates { - var t *template.Template - if tmpls == nil { - tmpls = template.New(name) - } - if name == tmpls.Name() { - t = tmpls - } else { - t = tmpls.New(name) - } - if _, err := t.Parse(data); err != nil { - return nil, fmt.Errorf("parsing %s: %v", name, err) - } + } + + static = http.FileServer(http.Dir(staticDir)) + theme = http.FileServer(http.Dir(themeDir)) + + templates, err = loadTemplates(c, templatesDir) + return +} + +// loadTemplates parses the expected templates from the provided directory. +func loadTemplates(c webConfig, templatesDir string) (*templates, error) { + files, err := ioutil.ReadDir(templatesDir) + if err != nil { + return nil, fmt.Errorf("read dir: %v", err) + } + + filenames := []string{} + for _, file := range files { + if file.IsDir() { + continue } + filenames = append(filenames, filepath.Join(templatesDir, file.Name())) + } + if len(filenames) == 0 { + return nil, fmt.Errorf("no files in template dir %q", templatesDir) } + funcs := map[string]interface{}{ + "issuer": func() string { return c.issuer }, + "logo": func() string { return c.logoURL }, + "url": func(s string) string { return join(c.issuerURL, s) }, + } + + tmpls, err := template.New("").Funcs(funcs).ParseFiles(filenames...) + if err != nil { + return nil, fmt.Errorf("parse files: %v", err) + } missingTmpls := []string{} for _, tmplName := range requiredTmpls { if tmpls.Lookup(tmplName) == nil { @@ -95,16 +152,7 @@ func loadTemplates(config TemplateConfig) (*templates, error) { if len(missingTmpls) > 0 { return nil, fmt.Errorf("missing template(s): %s", missingTmpls) } - - if config.LogoURL == "" { - config.LogoURL = coreOSLogoURL - } - if config.Issuer == "" { - config.Issuer = "dex" - } - return &templates{ - globalData: config, loginTmpl: tmpls.Lookup(tmplLogin), approvalTmpl: tmpls.Lookup(tmplApproval), passwordTmpl: tmpls.Lookup(tmplPassword), @@ -118,14 +166,6 @@ var scopeDescriptions = map[string]string{ "email": "View your email", } -type templates struct { - globalData TemplateConfig - loginTmpl *template.Template - approvalTmpl *template.Template - passwordTmpl *template.Template - oobTmpl *template.Template -} - type connectorInfo struct { ID string Name string @@ -142,21 +182,19 @@ func (t *templates) login(w http.ResponseWriter, connectors []connectorInfo, aut sort.Sort(byName(connectors)) data := struct { - TemplateConfig Connectors []connectorInfo AuthReqID string - }{t.globalData, connectors, authReqID} + }{connectors, authReqID} renderTemplate(w, t.loginTmpl, data) } func (t *templates) password(w http.ResponseWriter, authReqID, callback, lastUsername string, lastWasInvalid bool) { data := struct { - TemplateConfig AuthReqID string PostURL string Username string Invalid bool - }{t.globalData, authReqID, callback, lastUsername, lastWasInvalid} + }{authReqID, string(callback), lastUsername, lastWasInvalid} renderTemplate(w, t.passwordTmpl, data) } @@ -170,20 +208,18 @@ func (t *templates) approval(w http.ResponseWriter, authReqID, username, clientN } sort.Strings(accesses) data := struct { - TemplateConfig User string Client string AuthReqID string Scopes []string - }{t.globalData, username, clientName, authReqID, accesses} + }{username, clientName, authReqID, accesses} renderTemplate(w, t.approvalTmpl, data) } func (t *templates) oob(w http.ResponseWriter, code string) { data := struct { - TemplateConfig Code string - }{t.globalData, code} + }{code} renderTemplate(w, t.oobTmpl, data) } diff --git a/server/templates_default.go b/server/templates_default.go deleted file mode 100644 index 651c7411b8c9eaa9a7f26fae82aa59533ad31427..0000000000000000000000000000000000000000 --- a/server/templates_default.go +++ /dev/null @@ -1,362 +0,0 @@ -// This file was generated by the makefile. Do not edit. - -package server - -// defaultTemplates is a key for file name to file data of the files in web/templates. -var defaultTemplates = map[string]string{ - "approval.html": `{{ template "header.html" . }} - -<div class="panel"> - <h2 class="heading">Grant Access</h2> - - <hr> - <div class="list-with-title"> - <div class="subtle-text">{{ .Client }} would like to:</div> - {{ range $scope := .Scopes }} - <li class="bullet-point"> - <div class="subtle-text"> - {{ $scope }} - </div> - </li> - {{ end }} - </div> - <hr> - - <div> - <div class="form-row"> - <form method="post"> - <input type="hidden" name="req" value="{{ .AuthReqID }}"/> - <input type="hidden" name="approval" value="approve"> - <button type="submit" class="btn btn-success"> - <span class="btn-text">Grant Access</span> - </button> - </form> - </div> - <div class="form-row"> - <form method="post"> - <input type="hidden" name="req" value="{{ .AuthReqID }}"/> - <input type="hidden" name="approval" value="rejected"> - <button type="submit" class="btn btn-provider"> - <span class="btn-text">Cancel</span> - </button> - </form> - </div> - </div> - -</div> - -{{ template "footer.html" . }} -`, - "footer.html": ` </div> - </body> -</html> -`, - "header.html": `<!DOCTYPE html> -<html> - <head> - <meta charset="utf-8"> - <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> - <title>{{ .Issuer }}</title> - <meta name="viewport" content="width=device-width, initial-scale=1.0"> - <style> - * { - -webkit-box-sizing: border-box; - -moz-box-sizing: border-box; - box-sizing: border-box; - } - - html, - body { - margin: 0; - background-color: #efefef; - font-family: 'Source Sans Pro', Helvetica, sans-serif; - color: #333; - } - a { - color: #428BCA; - text-decoration: none; - } - a:active, a:hover, a:visited { - color: #2A6596; - text-decoration: underline; - } - #navbar { - background-color: #fff; - color: #333; - height: 46px; - box-shadow: 0 2px 2px rgba(0, 0, 0, 0.2); - font-size: 13px; - font-weight: 100; - overflow: hidden; - padding: 0 10px; - } - #navbar-logo-wrap { - width: 300px; - height: 100%; - display: inline-block; - overflow: hidden; - padding: 10px 15px; - } - #navbar-logo { - height: 100%; - max-height: 25px; - } - #container { - margin: 45px auto; - text-align: center; - max-width: 500px; - min-width: 320px; - } - .heading { - font-size: 20px; - font-weight: 500; - margin-top: 0; - margin-bottom: 10px; - } - .footer { - margin: 30px; - } - .input-label-right { - position: absolute; - right: 0; - bottom: 0; - } - .input-desc { - width: 250px; - margin: 4px auto; - text-align: left; - position: relative; - } - .subtle-text { - color: #999; - font-size: 12px; - } - .panel { - background-color: #fff; - padding: 30px; - box-shadow: 0px 5px 15px rgba(0, 0, 0, 0.5); - } - .explain { - font-size: 13px; - color: #666; - } - - .btn { - box-shadow: inset 0 1px 0px rgba(255, 255, 255, 0.2), 0 1px 2px rgba(0, 0, 0, 0.25), 0 0px 1px rgba(0, 0, 0, 0.25); - padding: 0; - font-size: 14px; - border-radius: 4px; - border: none; - cursor: pointer; - font-size: 16px; - } - .btn:focus { - outline: none; - } - .btn:active { - outline: none; - box-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125); - } - .btn-primary { - color: #fff; - background-color: #333; - padding: 6px 12px; - min-width: 200px; - border: none; - } - .btn-primary:hover { - background-color: #666; - color: #fff; - } - .btn-provider { - background-color: #fff; - color: #333; - width: 250px; - } - .btn-provider:hover { - color: #999; - } - .btn-success { - background-color: #2FC98E; - color: #fff; - width: 250px; - } - .btn-success:hover { - background-color: #49E3A8; - } - .btn-icon { - width: 36px; - height: 36px; - float: left; - margin-right: 5px; - background-repeat: no-repeat; - background-position: center; - background-size: 24px; - } - .btn-icon-google { - background-color: #DB4437; - background-image: url(); - } - .btn-icon-local { - background-color: #84B6EF; - background-image: url(); - } - .btn-icon-coreos { - /* B&W CoreOS SVG logo */ - background-image: url(); - } - .btn-icon-github { - background-color: #F5F5F5; - background-image: url(); - } - .btn-icon-bitbucket { - background-color: #205081; - background-image: url(); - } - .btn-text { - line-height: 36px; - padding: 6px 12px; - text-align: center; - font-weight: 600; - } - .form-row { - display: block; - margin: 20px auto; - } - label { - font-size: 13px; - font-weight: 600; - } - .input-box { - display: block; - height: 36px; - padding: 6px 12px; - font-size: 14px; - line-height: 1.42857143; - color: #666; - border: 1px solid #CCC; - border-radius: 4px; - box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075); - width: 250px; - margin: auto; - } - .input-box:focus, - .input-box:active { - outline: none; - border-color: #66AFE9; - } - .error-box-field, - .error-box { - background-color: #DD1327; - max-width: 320px; - color: #fff; - font-size: 14px; - font-weight: normal; - padding: 4px 0; - } - .error-box { - margin: 20px auto; - } - .error-box-field { - margin: 0 auto; - width: 250px; - } - .instruction-block { - font-size: 14px; - } - .detail-block { - color: #777; - font-size: 12px; - margin-top: 20px; - } - .bullet-point { - list-style: square; - } - .list-with-title { - text-align: left; - margin: 0 25%; - } - .hr { - color: #999; - } - </style> - </head> - - <body> - <div id="navbar"> - <div id="navbar-logo-wrap"> - <img id="navbar-logo" src="{{ .LogoURL }}"> - </div> - </div> - - <div id="container"> - -`, - "login.html": `{{ template "header.html" . }} - -<div class="panel"> - <h2 class="heading">Log in to {{ .Issuer }} </h2> - - <div> - {{ range $c := .Connectors }} - <div class="form-row"> - <a href="{{ $c.URL }}?req={{ $.AuthReqID }}" target="_self"> - <button class="btn btn-provider"> - <span class="btn-icon btn-icon-{{ $c.ID }}"></span> - <span class="btn-text">Log in with {{ $c.Name }}</span> - </button> - </a> - </div> - {{ end }} - </div> - -</div> - - -{{ template "footer.html" . }} -`, - "oob.html": `{{ template "header.html" . }} - -<div class="panel"> - <h2 class="heading">Login Successful</h2> - - Please copy this code, switch to your application and paste it there: - <br/> - <input type="text" value="{{ .Code }}" /> -</div> - -{{ template "footer.html" . }} -`, - "password.html": `{{ template "header.html" . }} - -<div class="panel"> - <h2 class="heading">Log in to Your Account</h2> - <form method="post" action="{{ .PostURL }}"> - <div class="form-row"> - <div class="input-desc"> - <label for="userid">Username</label> - </div> - <input tabindex="1" required id="login" name="login" type="text" class="input-box" placeholder="username" {{ if .Username }}value="{{ .Username }}" {{ else }} autofocus {{ end }}/> - </div> - <div class="form-row"> - <div class="input-desc"> - <label for="password">Password</label> - </div> - <input tabindex="2" required id="password" name="password" type="password" class="input-box" placeholder="password" {{ if .Invalid }} autofocus {{ end }}/> - </div> - <input type="hidden" name="req" value="{{ .AuthReqID }}"/> - - {{ if .Invalid }} - <div class="error-box"> - Invalid username and password. - </div> - {{ end }} - - <button tabindex="3" type="submit" class="btn btn-primary">Login</button> - - </form> -</div> - -{{ template "footer.html" . }} -`, -} diff --git a/server/templates_default_gen.go b/server/templates_default_gen.go deleted file mode 100644 index 0e46df780e9e1a3d8446cff1947f89f303913321..0000000000000000000000000000000000000000 --- a/server/templates_default_gen.go +++ /dev/null @@ -1,85 +0,0 @@ -// +build ignore - -package main - -import ( - "bytes" - "fmt" - "io/ioutil" - "log" - "os/exec" - "path/filepath" -) - -// ignoreFile uses "git check-ignore" to determine if we should ignore a file. -func ignoreFile(p string) (ok bool, err error) { - err = exec.Command("git", "check-ignore", p).Run() - if err == nil { - return true, nil - } - exitErr, ok := err.(*exec.ExitError) - if ok { - if sys := exitErr.Sys(); sys != nil { - e, ok := sys.(interface { - // Is the returned value something that returns an exit status? - ExitStatus() int - }) - if ok && e.ExitStatus() == 1 { - return false, nil - } - } - } - return false, err -} - -// Maps aren't deterministic, use a struct instead. - -type fileData struct { - name string - data string -} - -func main() { - // ReadDir guarentees result in sorted order. - dir, err := ioutil.ReadDir("web/templates") - if err != nil { - log.Fatal(err) - } - files := []fileData{} - for _, file := range dir { - p := filepath.Join("web/templates", file.Name()) - ignore, err := ignoreFile(p) - if err != nil { - log.Fatal(err) - } - if ignore { - continue - } - - data, err := ioutil.ReadFile(p) - if err != nil { - log.Fatal(err) - } - if bytes.Contains(data, []byte{'`'}) { - log.Fatalf("file %s contains escape character '`' and cannot be compiled into go source", p) - } - files = append(files, fileData{file.Name(), string(data)}) - } - - f := new(bytes.Buffer) - - fmt.Fprintln(f, "// This file was generated by the makefile. Do not edit.") - fmt.Fprintln(f) - fmt.Fprintln(f, "package server") - fmt.Fprintln(f) - fmt.Fprintln(f, "// defaultTemplates is a key for file name to file data of the files in web/templates.") - fmt.Fprintln(f, "var defaultTemplates = map[string]string{") - for _, file := range files { - fmt.Fprintf(f, "\t%q: `%s`,\n", file.name, file.data) - } - fmt.Fprintln(f, "}") - - if err := ioutil.WriteFile("server/templates_default.go", f.Bytes(), 0644); err != nil { - log.Fatal(err) - } -} diff --git a/server/templates_test.go b/server/templates_test.go index efbb29edf9ad03f12a894f8c3a6bb2f4138798d0..abb4e431abd516750a5a1e5e2b77073c236b8f9e 100644 --- a/server/templates_test.go +++ b/server/templates_test.go @@ -1,16 +1 @@ package server - -import "testing" - -func TestNewTemplates(t *testing.T) { - var config TemplateConfig - if _, err := loadTemplates(config); err != nil { - t.Fatal(err) - } -} - -func TestLoadTemplates(t *testing.T) { - var config TemplateConfig - - config.Dir = "../web/templates" -} diff --git a/web/static/main.css b/web/static/main.css new file mode 100644 index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/web/templates/header.html b/web/templates/header.html index cadb078d9750f0b1aab2b81a82eb3b85f60096d6..79438ec44e2a0927282ee56adfe8be002f7e20bf 100644 --- a/web/templates/header.html +++ b/web/templates/header.html @@ -3,8 +3,10 @@ <head> <meta charset="utf-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> - <title>{{ .Issuer }}</title> + <title>{{ issuer }}</title> <meta name="viewport" content="width=device-width, initial-scale=1.0"> + <link href="{{ url "static/main.css" }}" rel="stylesheet"> + <link href="{{ url "theme/style.css" }}" rel="stylesheet"> <style> * { -webkit-box-sizing: border-box; @@ -232,7 +234,7 @@ <body> <div id="navbar"> <div id="navbar-logo-wrap"> - <img id="navbar-logo" src="{{ .LogoURL }}"> + <img id="navbar-logo" src="{{ logo }}"> </div> </div> diff --git a/web/templates/login.html b/web/templates/login.html index ea43903a538e55b652babf21dc4f0ec722b6dd58..e52f775ac6979a12c945ddfb562a941e4e1584f0 100644 --- a/web/templates/login.html +++ b/web/templates/login.html @@ -1,7 +1,7 @@ {{ template "header.html" . }} <div class="panel"> - <h2 class="heading">Log in to {{ .Issuer }} </h2> + <h2 class="heading">Log in to {{ issuer }} </h2> <div> {{ range $c := .Connectors }} diff --git a/web/themes/coreos/logo.png b/web/themes/coreos/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..cf8caaaee53500d4efa77c587997788ac8d4fa6e Binary files /dev/null and b/web/themes/coreos/logo.png differ diff --git a/web/themes/coreos/style.css b/web/themes/coreos/style.css new file mode 100644 index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391