diff --git a/Makefile b/Makefile index 20b7ce104c9b55e0b096e930c3f9766ed6575eaa..e22ac57d5c0a96fa83fd560342aa6882e235fbe5 100644 --- a/Makefile +++ b/Makefile @@ -19,7 +19,7 @@ GOARCH=$(shell go env GOARCH) build: bin/dex bin/example-app -bin/dex: FORCE +bin/dex: FORCE server/templates_default.go @go install -ldflags $(LD_FLAGS) $(REPO_PATH)/cmd/dex bin/example-app: FORCE @@ -42,6 +42,9 @@ lint: golint $$package; \ done +server/templates_default.go: $(wildcard web/templates/**) + @go run server/templates_default_gen.go + .PHONY: docker-build docker-build: bin/dex @docker build -t $(DOCKER_IMAGE) . diff --git a/cmd/dex/config.go b/cmd/dex/config.go index ea7641566e020e2728f414d486754e3533ddd942..2b08885d1cc9238438f82792ebb94c0a172d0df0 100644 --- a/cmd/dex/config.go +++ b/cmd/dex/config.go @@ -8,6 +8,7 @@ import ( "github.com/coreos/dex/connector/ldap" "github.com/coreos/dex/connector/mock" "github.com/coreos/dex/connector/oidc" + "github.com/coreos/dex/server" "github.com/coreos/dex/storage" "github.com/coreos/dex/storage/kubernetes" "github.com/coreos/dex/storage/memory" @@ -21,6 +22,8 @@ type Config struct { Web Web `yaml:"web"` OAuth2 OAuth2 `yaml:"oauth2"` + Templates server.TemplateConfig `yaml:"templates"` + StaticClients []storage.Client `yaml:"staticClients"` } @@ -111,9 +114,15 @@ func (c *Connector) UnmarshalYAML(unmarshal func(interface{}) error) error { var err error switch c.Type { - case "mock": + case "mockCallback": + var config struct { + Config mock.CallbackConfig `yaml:"config"` + } + err = unmarshal(&config) + c.Config = &config.Config + case "mockPassword": var config struct { - Config mock.Config `yaml:"config"` + Config mock.PasswordConfig `yaml:"config"` } err = unmarshal(&config) c.Config = &config.Config diff --git a/cmd/dex/serve.go b/cmd/dex/serve.go index be27f12b65a44d05ddb3c24a2c357f6a61dca221..24beb4b6569064d3ac051b6bbbf87e710c6bc59f 100644 --- a/cmd/dex/serve.go +++ b/cmd/dex/serve.go @@ -89,11 +89,11 @@ func serve(cmd *cobra.Command, args []string) error { } serverConfig := server.Config{ - Issuer: c.Issuer, - Connectors: connectors, - Storage: s, - SupportedResponseTypes: c.OAuth2.ResponseTypes, + Issuer: c.Issuer, + Connectors: connectors, + Storage: s, + TemplateConfig: c.Templates, } serv, err := server.New(serverConfig) diff --git a/connector/mock/connectortest.go b/connector/mock/connectortest.go index 71c97d9275c42651c4ecce5e1426c24dc52632bf..0d4b87baf333ad4ae2eb00f84bbcac3312caf26f 100644 --- a/connector/mock/connectortest.go +++ b/connector/mock/connectortest.go @@ -1,4 +1,4 @@ -// Package mock implements a mock connector which requires no user interaction. +// Package mock implements connectors which help test various server components. package mock import ( @@ -11,22 +11,24 @@ import ( "github.com/coreos/dex/connector" ) -// New returns a mock connector which requires no user interaction. It always returns +// NewCallbackConnector returns a mock connector which requires no user interaction. It always returns // the same (fake) identity. -func New() connector.Connector { - return mockConnector{} +func NewCallbackConnector() connector.Connector { + return callbackConnector{} } var ( - _ connector.CallbackConnector = mockConnector{} - _ connector.GroupsConnector = mockConnector{} + _ connector.CallbackConnector = callbackConnector{} + _ connector.GroupsConnector = callbackConnector{} + + _ connector.PasswordConnector = passwordConnector{} ) -type mockConnector struct{} +type callbackConnector struct{} -func (m mockConnector) Close() error { return nil } +func (m callbackConnector) Close() error { return nil } -func (m mockConnector) LoginURL(callbackURL, state string) (string, error) { +func (m callbackConnector) LoginURL(callbackURL, state string) (string, error) { u, err := url.Parse(callbackURL) if err != nil { return "", fmt.Errorf("failed to parse callbackURL %q: %v", callbackURL, err) @@ -39,7 +41,7 @@ func (m mockConnector) LoginURL(callbackURL, state string) (string, error) { var connectorData = []byte("foobar") -func (m mockConnector) HandleCallback(r *http.Request) (connector.Identity, string, error) { +func (m callbackConnector) HandleCallback(r *http.Request) (connector.Identity, string, error) { return connector.Identity{ UserID: "0-385-28089-0", Username: "Kilgore Trout", @@ -49,17 +51,54 @@ func (m mockConnector) HandleCallback(r *http.Request) (connector.Identity, stri }, r.URL.Query().Get("state"), nil } -func (m mockConnector) Groups(identity connector.Identity) ([]string, error) { +func (m callbackConnector) Groups(identity connector.Identity) ([]string, error) { if !bytes.Equal(identity.ConnectorData, connectorData) { return nil, errors.New("connector data mismatch") } return []string{"authors"}, nil } -// Config holds the configuration parameters for the mock connector. -type Config struct{} +// CallbackConfig holds the configuration parameters for a connector which requires no interaction. +type CallbackConfig struct{} // Open returns an authentication strategy which requires no user interaction. -func (c *Config) Open() (connector.Connector, error) { - return New(), nil +func (c *CallbackConfig) Open() (connector.Connector, error) { + return NewCallbackConnector(), nil +} + +// PasswordConfig holds the configuration for a mock connector which prompts for the supplied +// username and password. +type PasswordConfig struct { + Username string `yaml:"username"` + Password string `yaml:"password"` +} + +// Open returns an authentication strategy which prompts for a predefined username and password. +func (c *PasswordConfig) Open() (connector.Connector, error) { + if c.Username == "" { + return nil, errors.New("no username supplied") + } + if c.Password == "" { + return nil, errors.New("no password supplied") + } + return &passwordConnector{c.Username, c.Password}, nil +} + +type passwordConnector struct { + username string + password string +} + +func (p passwordConnector) Close() error { return nil } + +func (p passwordConnector) Login(username, password string) (identity connector.Identity, validPassword bool, err error) { + if username == p.username && password == p.password { + return connector.Identity{ + UserID: "0-385-28089-0", + Username: "Kilgore Trout", + Email: "kilgore@kilgore.trout", + EmailVerified: true, + }, true, nil + } + return identity, false, nil } diff --git a/examples/config-dev.yaml b/examples/config-dev.yaml index 65267d31e814ef1b115e7746a38d43e18ad41624..fa61cc5ead2ad9342ed7b41915adc235b0a61736 100644 --- a/examples/config-dev.yaml +++ b/examples/config-dev.yaml @@ -7,25 +7,15 @@ web: http: 127.0.0.1:5556 connectors: -- type: mock - id: mock +- type: mockCallback + id: mock-callback name: Mock -- type: github - id: github - name: GitHub +- type: mockPassword + id: mock-password + name: Password config: - clientID: "$GITHUB_CLIENT_ID" - clientSecret: "$GITHUB_CLIENT_SECRET" - redirectURI: http://127.0.0.1:5556/callback/github - org: kubernetes -- type: oidc - id: google - name: Google Account - config: - issuer: https://accounts.google.com - clientID: "$GOOGLE_OAUTH2_CLIENT_ID" - clientSecret: "$GOOGLE_OAUTH2_CLIENT_SECRET" - redirectURI: http://127.0.0.1:5556/callback/google + username: "admin" + password: "PASSWORD" staticClients: - id: example-app diff --git a/server/handlers.go b/server/handlers.go index 3aa0b368bcdb8850ce092da45b932b9f227b4439..ebbc87408347c34d777db48d7878cf9138778061 100644 --- a/server/handlers.go +++ b/server/handlers.go @@ -129,15 +129,16 @@ func (s *Server) handleAuthorization(w http.ResponseWriter, r *http.Request) { connectorInfos := make([]connectorInfo, len(s.connectors)) i := 0 - for id := range s.connectors { + for id, conn := range s.connectors { connectorInfos[i] = connectorInfo{ - DisplayName: id, - URL: s.absPath("/auth", id), + ID: id, + Name: conn.DisplayName, + URL: s.absPath("/auth", id), } i++ } - renderLoginOptions(w, connectorInfos, state) + s.templates.login(w, connectorInfos, state) } func (s *Server) handleConnectorLogin(w http.ResponseWriter, r *http.Request) { @@ -163,7 +164,7 @@ func (s *Server) handleConnectorLogin(w http.ResponseWriter, r *http.Request) { } http.Redirect(w, r, callbackURL, http.StatusFound) case connector.PasswordConnector: - renderPasswordTmpl(w, state, r.URL.String(), "") + s.templates.password(w, state, r.URL.String(), "", false) default: s.notFound(w, r) } @@ -174,7 +175,7 @@ func (s *Server) handleConnectorLogin(w http.ResponseWriter, r *http.Request) { return } - username := r.FormValue("username") + username := r.FormValue("login") password := r.FormValue("password") identity, ok, err := passwordConnector.Login(username, password) @@ -184,7 +185,7 @@ func (s *Server) handleConnectorLogin(w http.ResponseWriter, r *http.Request) { return } if !ok { - renderPasswordTmpl(w, state, r.URL.String(), "Invalid credentials") + s.templates.password(w, state, r.URL.String(), username, true) return } redirectURL, err := s.finalizeLogin(identity, state, connID, conn.Connector) @@ -299,7 +300,7 @@ func (s *Server) handleApproval(w http.ResponseWriter, r *http.Request) { s.renderError(w, http.StatusInternalServerError, errServerError, "") return } - renderApprovalTmpl(w, authReq.ID, *authReq.Claims, client, authReq.Scopes) + s.templates.approval(w, authReq.ID, authReq.Claims.Username, client.Name, authReq.Scopes) case "POST": if r.FormValue("approval") != "approve" { s.renderError(w, http.StatusInternalServerError, "approval rejected", "") diff --git a/server/server.go b/server/server.go index ef2697dbe2d781a27d6200bb75d0aecf52f9e0a2..2e3c00af3eee2d94121eaf1c701c83af24029549 100644 --- a/server/server.go +++ b/server/server.go @@ -43,6 +43,8 @@ type Config struct { // If specified, the server will use this function for determining time. Now func() time.Time + + TemplateConfig TemplateConfig } func value(val, defaultValue time.Duration) time.Duration { @@ -63,6 +65,8 @@ type Server struct { mux http.Handler + templates *templates + // If enabled, don't prompt user for approval after logging in through connector. // No package level API to set this, only used in tests. skipApproval bool @@ -107,6 +111,11 @@ func newServer(c Config, rotationStrategy rotationStrategy) (*Server, error) { supported[respType] = true } + tmpls, err := loadTemplates(c.TemplateConfig) + if err != nil { + return nil, fmt.Errorf("server: failed to load templates: %v", err) + } + now := c.Now if now == nil { now = time.Now @@ -124,6 +133,7 @@ func newServer(c Config, rotationStrategy rotationStrategy) (*Server, error) { supportedResponseTypes: supported, idTokensValidFor: value(c.IDTokensValidFor, 24*time.Hour), now: now, + templates: tmpls, } for _, conn := range c.Connectors { diff --git a/server/server_test.go b/server/server_test.go index 4865ab1f45391c4a12cdc9de35279447e2ace22e..296f22ccca78a2d1f3b18d6f4e5ade78754eaf9d 100644 --- a/server/server_test.go +++ b/server/server_test.go @@ -64,7 +64,7 @@ FDWV28nTP9sqbtsmU8Tem2jzMvZ7C/Q0AuDoKELFUpux8shm8wfIhyaPnXUGZoAZ Np4vUwMSYV5mopESLWOg3loBxKyLGFtgGKVCjGiQvy6zISQ4fQo= -----END RSA PRIVATE KEY-----`) -func newTestServer(updateConfig func(c *Config)) (*httptest.Server, *Server) { +func newTestServer(t *testing.T, updateConfig func(c *Config)) (*httptest.Server, *Server) { var server *Server s := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { server.ServeHTTP(w, r) @@ -76,7 +76,7 @@ func newTestServer(updateConfig func(c *Config)) (*httptest.Server, *Server) { { ID: "mock", DisplayName: "Mock", - Connector: mock.New(), + Connector: mock.NewCallbackConnector(), }, }, } @@ -87,21 +87,21 @@ func newTestServer(updateConfig func(c *Config)) (*httptest.Server, *Server) { var err error if server, err = newServer(config, staticRotationStrategy(testKey)); err != nil { - panic(err) + t.Fatal(err) } server.skipApproval = true // Don't prompt for approval, just immediately redirect with code. return s, server } func TestNewTestServer(t *testing.T) { - newTestServer(nil) + newTestServer(t, nil) } func TestDiscovery(t *testing.T) { ctx, cancel := context.WithCancel(context.Background()) defer cancel() - httpServer, _ := newTestServer(func(c *Config) { + httpServer, _ := newTestServer(t, func(c *Config) { c.Issuer = c.Issuer + "/non-root-path" }) defer httpServer.Close() @@ -129,7 +129,7 @@ func TestOAuth2CodeFlow(t *testing.T) { ctx, cancel := context.WithCancel(context.Background()) defer cancel() - httpServer, s := newTestServer(func(c *Config) { + httpServer, s := newTestServer(t, func(c *Config) { c.Issuer = c.Issuer + "/non-root-path" }) defer httpServer.Close() @@ -255,7 +255,7 @@ func TestOAuth2ImplicitFlow(t *testing.T) { ctx, cancel := context.WithCancel(context.Background()) defer cancel() - httpServer, s := newTestServer(func(c *Config) { + httpServer, s := newTestServer(t, func(c *Config) { // Enable support for the implicit flow. c.SupportedResponseTypes = []string{"code", "token"} }) diff --git a/server/templates.go b/server/templates.go index 7bc64144095cc6cbcb9d7617d33c400f1bfa2c71..a0d1ce17d5289468541d67ef2d5b7bf92b133933 100644 --- a/server/templates.go +++ b/server/templates.go @@ -1,101 +1,196 @@ package server import ( + "fmt" + "io" + "io/ioutil" "log" "net/http" + "path/filepath" + "sort" "text/template" +) - "github.com/coreos/dex/storage" +const ( + tmplApproval = "approval.html" + tmplLogin = "login.html" + tmplPassword = "password.html" ) +const coreOSLogoURL = "https://coreos.com/assets/images/brand/coreos-wordmark-135x40px.png" + +var requiredTmpls = []string{ + tmplApproval, + tmplLogin, + tmplPassword, +} + +// TemplateConfig describes. +type TemplateConfig struct { + // Directory of the templates. If empty, these will be loaded from memory. + Dir string `yaml:"dir"` + + // Defaults to the CoreOS logo and "dex". + LogoURL string `yaml:"logoURL"` + Issuer string `yaml:"issuerName"` +} + +type globalData struct { + LogoURL string + Issuer string +} + +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) + } + 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) + } + } 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) + } + } + } + + missingTmpls := []string{} + for _, tmplName := range requiredTmpls { + if tmpls.Lookup(tmplName) == nil { + missingTmpls = append(missingTmpls, tmplName) + } + } + 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), + }, nil +} + +var scopeDescriptions = map[string]string{ + "offline_access": "Have offline access", + "profile": "View basic profile information", + "email": "View your email", +} + +type templates struct { + globalData TemplateConfig + loginTmpl *template.Template + approvalTmpl *template.Template + passwordTmpl *template.Template +} + type connectorInfo struct { - DisplayName string - URL string + ID string + Name string + URL string } -var loginTmpl = template.Must(template.New("login-template").Parse(`<html> -<head></head> -<body> -<p>Login options</p> -{{ range $i, $connector := .Connectors }} -<a href="{{ $connector.URL }}?state={{ $.State }}">{{ $connector.DisplayName }}</a> -{{ end }} -</body> -</html>`)) - -func renderLoginOptions(w http.ResponseWriter, connectors []connectorInfo, state string) { +type byName []connectorInfo + +func (n byName) Len() int { return len(n) } +func (n byName) Less(i, j int) bool { return n[i].Name < n[j].Name } +func (n byName) Swap(i, j int) { n[i], n[j] = n[j], n[i] } + +func (t *templates) login(w http.ResponseWriter, connectors []connectorInfo, state string) { + sort.Sort(byName(connectors)) + data := struct { + TemplateConfig Connectors []connectorInfo State string - }{connectors, state} - renderTemplate(w, loginTmpl, data) + }{t.globalData, connectors, state} + renderTemplate(w, t.loginTmpl, data) } -var passwordTmpl = template.Must(template.New("password-template").Parse(`<html> -<body> -<p>Login</p> -<form action="{{ .Callback }}" method="POST"> -Login: <input type="text" name="login"/><br/> -Password: <input type="password" name="password"/><br/> -<input type="hidden" name="state" value="{{ .State }}"/> -<input type="submit"/> -{{ if .Message }} -<p>Error: {{ .Message }}</p> -{{ end }} -</form> -</body> -</html>`)) - -func renderPasswordTmpl(w http.ResponseWriter, state, callback, message string) { +func (t *templates) password(w http.ResponseWriter, state, callback, lastUsername string, lastWasInvalid bool) { data := struct { + TemplateConfig State string - Callback string - Message string - }{state, callback, message} - renderTemplate(w, passwordTmpl, data) + PostURL string + Username string + Invalid bool + }{t.globalData, state, callback, lastUsername, lastWasInvalid} + renderTemplate(w, t.passwordTmpl, data) } -var approvalTmpl = template.Must(template.New("approval-template").Parse(`<html> -<body> -<p>User: {{ .User }}</p> -<p>Client: {{ .ClientName }}</p> -<form method="post"> -<input type="hidden" name="state" value="{{ .State }}"/> -<input type="hidden" name="approval" value="approve"> -<button type="submit">Approve</button> -</form> -<form method="post"> -<input type="hidden" name="state" value="{{ .State }}"/> -<input type="hidden" name="approval" value="reject"> -<button type="submit">Reject</button> -</form> -</body> -</html>`)) - -func renderApprovalTmpl(w http.ResponseWriter, state string, identity storage.Claims, client storage.Client, scopes []string) { +func (t *templates) approval(w http.ResponseWriter, state, username, clientName string, scopes []string) { + accesses := []string{} + for _, scope := range scopes { + access, ok := scopeDescriptions[scope] + if ok { + accesses = append(accesses, access) + } + } + sort.Strings(accesses) data := struct { - User string - ClientName string - State string - }{identity.Email, client.Name, state} - renderTemplate(w, approvalTmpl, data) + TemplateConfig + User string + Client string + State string + Scopes []string + }{t.globalData, username, clientName, state, accesses} + renderTemplate(w, t.approvalTmpl, data) } -func renderTemplate(w http.ResponseWriter, tmpl *template.Template, data interface{}) { - err := tmpl.Execute(w, data) - if err == nil { - return - } +// small io.Writer utilitiy to determine if executing the template wrote to the underlying response writer. +type writeRecorder struct { + wrote bool + w io.Writer +} + +func (w *writeRecorder) Write(p []byte) (n int, err error) { + w.wrote = true + return w.w.Write(p) +} - switch err := err.(type) { - case template.ExecError: - // An ExecError guarentees that Execute has not written to the underlying reader. +func renderTemplate(w http.ResponseWriter, tmpl *template.Template, data interface{}) { + wr := &writeRecorder{w: w} + if err := tmpl.Execute(wr, data); err != nil { log.Printf("Error rendering template %s: %s", tmpl.Name(), err) - // TODO(ericchiang): replace with better internal server error. - http.Error(w, "Internal server error", http.StatusInternalServerError) - default: - // An error with the underlying write, such as the connection being - // dropped. Ignore for now. + if !wr.wrote { + // TODO(ericchiang): replace with better internal server error. + http.Error(w, "Internal server error", http.StatusInternalServerError) + } } + return } diff --git a/server/templates_default.go b/server/templates_default.go new file mode 100644 index 0000000000000000000000000000000000000000..fc272675039ea0253c49f39afd11653d5fd4b924 --- /dev/null +++ b/server/templates_default.go @@ -0,0 +1,12 @@ +// 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\" . }}\n\n<div class=\"panel\">\n <h2 class=\"heading\">Grant Access</h2>\n\n <hr>\n <div class=\"list-with-title\">\n <div class=\"subtle-text\">{{ .Client }} would like to:</div>\n {{ range $scope := .Scopes }}\n <li class=\"bullet-point\">\n <div class=\"subtle-text\">\n {{ $scope }}\n </div>\n </li>\n {{ end }}\n </div>\n <hr>\n\n <div>\n <div class=\"form-row\">\n <form method=\"post\">\n <input type=\"hidden\" name=\"state\" value=\"{{ .State }}\"/>\n <input type=\"hidden\" name=\"approval\" value=\"approve\">\n <button type=\"submit\" class=\"btn btn-success\">\n <span class=\"btn-text\">Grant Access</span>\n </button>\n </form>\n </div>\n <div class=\"form-row\">\n <form method=\"post\">\n <input type=\"hidden\" name=\"state\" value=\"{{ .State }}\"/>\n <input type=\"hidden\" name=\"approval\" value=\"rejected\">\n <button type=\"submit\" class=\"btn btn-provider\">\n <span class=\"btn-text\">Cancel</span>\n </button>\n </form>\n </div>\n </div>\n\n</div>\n\n{{ template \"footer.html\" . }}\n", + "footer.html": " </div>\n </body>\n</html>\n", + "header.html": "<!DOCTYPE html>\n<html>\n <head>\n <meta charset=\"utf-8\">\n <meta http-equiv=\"X-UA-Compatible\" content=\"IE=edge,chrome=1\">\n <title>{{ .Issuer }}</title>\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\">\n <style>\n * {\n -webkit-box-sizing: border-box;\n -moz-box-sizing: border-box;\n box-sizing: border-box;\n }\n\n html,\n body {\n margin: 0;\n background-color: #efefef;\n font-family: 'Source Sans Pro', Helvetica, sans-serif;\n color: #333;\n }\n a {\n color: #428BCA;\n text-decoration: none;\n }\n a:active, a:hover, a:visited {\n color: #2A6596;\n text-decoration: underline;\n }\n #navbar {\n background-color: #fff;\n color: #333;\n height: 46px;\n box-shadow: 0 2px 2px rgba(0, 0, 0, 0.2);\n font-size: 13px;\n font-weight: 100;\n overflow: hidden;\n padding: 0 10px;\n }\n #navbar-logo-wrap {\n width: 300px;\n height: 100%;\n display: inline-block;\n overflow: hidden;\n padding: 10px 15px;\n }\n #navbar-logo {\n height: 100%;\n max-height: 25px;\n }\n #container {\n margin: 45px auto;\n text-align: center;\n max-width: 500px;\n min-width: 320px;\n }\n .heading {\n font-size: 20px;\n font-weight: 500;\n margin-top: 0;\n margin-bottom: 10px;\n }\n .footer {\n margin: 30px;\n }\n .input-label-right {\n position: absolute;\n right: 0;\n bottom: 0;\n }\n .input-desc {\n width: 250px;\n margin: 4px auto;\n text-align: left;\n position: relative;\n }\n .subtle-text {\n color: #999;\n font-size: 12px;\n }\n .panel {\n background-color: #fff;\n padding: 30px;\n box-shadow: 0px 5px 15px rgba(0, 0, 0, 0.5);\n }\n .explain {\n font-size: 13px;\n color: #666;\n }\n\n .btn {\n 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);\n padding: 0;\n font-size: 14px;\n border-radius: 4px;\n border: none;\n cursor: pointer;\n font-size: 16px;\n }\n .btn:focus {\n outline: none;\n }\n .btn:active {\n outline: none;\n box-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125);\n }\n .btn-primary {\n color: #fff;\n background-color: #333;\n padding: 6px 12px;\n min-width: 200px;\n border: none;\n }\n .btn-primary:hover {\n background-color: #666;\n color: #fff;\n }\n .btn-provider {\n background-color: #fff;\n color: #333;\n width: 250px;\n }\n .btn-provider:hover {\n color: #999;\n }\n .btn-success {\n background-color: #2FC98E;\n color: #fff;\n width: 250px;\n }\n .btn-success:hover {\n background-color: #49E3A8;\n }\n .btn-icon {\n width: 36px;\n height: 36px;\n float: left;\n margin-right: 5px;\n background-repeat: no-repeat;\n background-position: center;\n background-size: 24px;\n }\n .btn-icon-google {\n background-color: #DB4437;\n background-image: url();\n }\n .btn-icon-local {\n background-color: #84B6EF;\n background-image: url();\n }\n .btn-icon-coreos {\n /* B&W CoreOS SVG logo */\n background-image: url();\n }\n .btn-icon-github {\n background-color: #F5F5F5;\n background-image: url();\n }\n .btn-icon-bitbucket {\n background-color: #205081;\n background-image: url();\n }\n .btn-text {\n line-height: 36px;\n padding: 6px 12px;\n text-align: center;\n font-weight: 600;\n }\n .form-row {\n display: block;\n margin: 20px auto;\n }\n label {\n font-size: 13px;\n font-weight: 600;\n }\n .input-box {\n display: block;\n height: 36px;\n padding: 6px 12px;\n font-size: 14px;\n line-height: 1.42857143;\n color: #666;\n border: 1px solid #CCC;\n border-radius: 4px;\n box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075);\n width: 250px;\n margin: auto;\n }\n .input-box:focus,\n .input-box:active {\n outline: none;\n border-color: #66AFE9;\n }\n .error-box-field,\n .error-box {\n background-color: #DD1327;\n max-width: 320px;\n color: #fff;\n font-size: 14px;\n font-weight: normal;\n padding: 4px 0;\n }\n .error-box {\n margin: 20px auto;\n }\n .error-box-field {\n margin: 0 auto;\n width: 250px;\n }\n .instruction-block {\n font-size: 14px;\n }\n .detail-block {\n color: #777;\n font-size: 12px;\n margin-top: 20px;\n }\n .bullet-point {\n list-style: square;\n }\n .list-with-title {\n text-align: left;\n margin: 0 25%;\n }\n .hr {\n color: #999;\n }\n </style>\n </head>\n\n <body>\n <div id=\"navbar\">\n <div id=\"navbar-logo-wrap\">\n <img id=\"navbar-logo\" src=\"{{ .LogoURL }}\">\n </div>\n </div>\n\n <div id=\"container\">\n\n", + "login.html": "{{ template \"header.html\" . }}\n\n<div class=\"panel\">\n <h2 class=\"heading\">Log in to {{ .Issuer }} </h2>\n\n <div>\n {{ range $c := .Connectors }}\n <div class=\"form-row\">\n <a href=\"{{ $c.URL }}?state={{ $.State }}\" target=\"_self\">\n <button class=\"btn btn-provider\">\n <span class=\"btn-icon btn-icon-{{ $c.ID }}\"></span>\n <span class=\"btn-text\">Log in with {{ $c.Name }}</span>\n </button>\n </a>\n </div>\n {{ end }}\n </div>\n\n</div>\n\n\n{{ template \"footer.html\" . }}\n", + "password.html": "{{ template \"header.html\" . }}\n\n<div class=\"panel\">\n <h2 class=\"heading\">Log in to Your Account</h2>\n <form method=\"post\" action=\"{{ .PostURL }}\">\n <div class=\"form-row\">\n <div class=\"input-desc\">\n <label for=\"userid\">Username</label>\n </div>\n\t <input tabindex=\"1\" required id=\"login\" name=\"login\" type=\"text\" class=\"input-box\" placeholder=\"username\" {{ if .Username }}value=\"{{ .Username }}\" {{ else }} autofocus {{ end }}/>\n </div>\n <div class=\"form-row\">\n <div class=\"input-desc\">\n <label for=\"password\">Password</label>\n </div>\n\t <input tabindex=\"2\" required id=\"password\" name=\"password\" type=\"password\" class=\"input-box\" placeholder=\"password\" {{ if .Invalid }} autofocus {{ end }}/>\n </div>\n <input type=\"hidden\" name=\"state\" value=\"{{ .State }}\"/>\n\n {{ if .Invalid }}\n <div class=\"error-box\">\n Invalid username and password.\n </div>\n {{ end }}\n\n <button tabindex=\"3\" type=\"submit\" class=\"btn btn-primary\">Login</button>\n\n </form>\n</div>\n\n{{ template \"footer.html\" . }}\n", +} diff --git a/server/templates_default_gen.go b/server/templates_default_gen.go new file mode 100644 index 0000000000000000000000000000000000000000..0a5ab78aae6eee1bc1ace32cb21f72e6a1734645 --- /dev/null +++ b/server/templates_default_gen.go @@ -0,0 +1,79 @@ +// +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 +} + +type fileData struct { + name string + data string +} + +func main() { + 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) + } + 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: %q,\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 abb4e431abd516750a5a1e5e2b77073c236b8f9e..efbb29edf9ad03f12a894f8c3a6bb2f4138798d0 100644 --- a/server/templates_test.go +++ b/server/templates_test.go @@ -1 +1,16 @@ 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/templates/approval.html b/web/templates/approval.html new file mode 100644 index 0000000000000000000000000000000000000000..c73a522e0d6cfde30e9a4aa7fbde26e7065cf122 --- /dev/null +++ b/web/templates/approval.html @@ -0,0 +1,42 @@ +{{ 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="state" value="{{ .State }}"/> + <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="state" value="{{ .State }}"/> + <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" . }} diff --git a/web/templates/footer.html b/web/templates/footer.html new file mode 100644 index 0000000000000000000000000000000000000000..5b6e2d6530625f5f9c574ed684e04d82ffae6c71 --- /dev/null +++ b/web/templates/footer.html @@ -0,0 +1,3 @@ + </div> + </body> +</html> diff --git a/web/templates/header.html b/web/templates/header.html new file mode 100644 index 0000000000000000000000000000000000000000..cadb078d9750f0b1aab2b81a82eb3b85f60096d6 --- /dev/null +++ b/web/templates/header.html @@ -0,0 +1,240 @@ +<!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"> + diff --git a/web/templates/login.html b/web/templates/login.html new file mode 100644 index 0000000000000000000000000000000000000000..d43b5142773425057774ead2a8b09ecdbbc1ddff --- /dev/null +++ b/web/templates/login.html @@ -0,0 +1,22 @@ +{{ 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 }}?state={{ $.State }}" 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" . }} diff --git a/web/templates/password.html b/web/templates/password.html new file mode 100644 index 0000000000000000000000000000000000000000..89f833fcdcd2f5a840ab8584877061669324aeb8 --- /dev/null +++ b/web/templates/password.html @@ -0,0 +1,31 @@ +{{ 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="state" value="{{ .State }}"/> + + {{ 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" . }}