forked from openshift/origin
-
Notifications
You must be signed in to change notification settings - Fork 1
/
selectprovider.go
114 lines (94 loc) · 3.27 KB
/
selectprovider.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
package selectprovider
import (
"bytes"
"errors"
"fmt"
"html/template"
"net/http"
"github.com/openshift/origin/pkg/auth/oauth/handlers"
utilruntime "k8s.io/kubernetes/pkg/util/runtime"
)
type SelectProviderRenderer interface {
Render(redirectors []handlers.ProviderInfo, w http.ResponseWriter, req *http.Request)
}
type SelectProvider struct {
render SelectProviderRenderer
forceInterstitial bool
}
var _ = handlers.AuthenticationSelectionHandler(&SelectProvider{})
func NewSelectProvider(render SelectProviderRenderer, forceInterstitial bool) *SelectProvider {
return &SelectProvider{
render: render,
forceInterstitial: forceInterstitial,
}
}
type ProviderData struct {
Providers []handlers.ProviderInfo
}
// NewSelectProviderRenderer creates a select provider renderer that takes in an optional custom template to
// allow branding of the page. Uses the default if customSelectProviderTemplateFile is not set.
func NewSelectProviderRenderer(customSelectProviderTemplateFile string) (*selectProviderTemplateRenderer, error) {
r := &selectProviderTemplateRenderer{}
if len(customSelectProviderTemplateFile) > 0 {
customTemplate, err := template.ParseFiles(customSelectProviderTemplateFile)
if err != nil {
return nil, err
}
r.selectProviderTemplate = customTemplate
} else {
r.selectProviderTemplate = defaultSelectProviderTemplate
}
return r, nil
}
func (s *SelectProvider) SelectAuthentication(providers []handlers.ProviderInfo, w http.ResponseWriter, req *http.Request) (*handlers.ProviderInfo, bool, error) {
if len(providers) == 0 {
return nil, false, nil
}
if len(providers) == 1 && !s.forceInterstitial {
return &providers[0], false, nil
}
s.render.Render(providers, w, req)
return nil, true, nil
}
func ValidateSelectProviderTemplate(templateContent []byte) []error {
var allErrs []error
template, err := template.New("selectProviderTemplateTest").Parse(string(templateContent))
if err != nil {
return append(allErrs, err)
}
// Execute the template with dummy values and check if they're there.
providerData := ProviderData{
Providers: []handlers.ProviderInfo{
{
Name: "provider_1",
URL: "http://example.com/redirect_1/",
},
{
Name: "provider_2",
URL: "http://example.com/redirect_2/",
},
},
}
var buffer bytes.Buffer
err = template.Execute(&buffer, providerData)
if err != nil {
return append(allErrs, err)
}
output := buffer.Bytes()
// We only care that they are using the URLs we provide, and that they are iterating over all providers
// for when multiple providers are allowed
if !bytes.Contains(output, []byte(providerData.Providers[1].URL)) {
allErrs = append(allErrs, errors.New("template must iterate over all {{.Providers}} and use the {{ .URL }} for each one"))
}
return allErrs
}
type selectProviderTemplateRenderer struct {
selectProviderTemplate *template.Template
}
func (r selectProviderTemplateRenderer) Render(providers []handlers.ProviderInfo, w http.ResponseWriter, req *http.Request) {
w.Header().Add("Content-Type", "text/html; charset=UTF-8")
w.WriteHeader(http.StatusOK)
if err := r.selectProviderTemplate.Execute(w, ProviderData{Providers: providers}); err != nil {
utilruntime.HandleError(fmt.Errorf("unable to render select provider template: %v", err))
}
}