-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathconfig.go
173 lines (140 loc) · 3.63 KB
/
config.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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
package goPinotAPI
import (
"fmt"
"log/slog"
"net/http"
"net/url"
"os"
"strings"
)
const controllerUrl = "controllerUrl"
const authToken = "authToken"
const authType = "authType"
const logger = "logger"
type Opt interface {
apply(*cfg)
Type() string
}
type cfg struct {
controllerUrl string
authToken string
authType string
httpAuthWriter httpAuthWriter
logger *slog.Logger
}
type clientOpt struct{ fn func(*cfg) }
// controllerUrlOpt is an option to set the controller url for the client
type controllerUrlOpt struct {
controllerUrl string
}
func (o *controllerUrlOpt) apply(c *cfg) {
c.controllerUrl = o.controllerUrl
}
func (o *controllerUrlOpt) Type() string {
return controllerUrl
}
// authTokenOpt is an option to set the auth token for the client
type authTokenOpt struct {
authToken string
}
func (o *authTokenOpt) apply(c *cfg) {
c.authToken = o.authToken
}
func (o *authTokenOpt) Type() string {
return authToken
}
// authTypeOpt is an option to set the auth type for the client
type authTypeOpt struct {
authType string
}
func (o *authTypeOpt) apply(c *cfg) {
c.authType = o.authType
}
func (o *authTypeOpt) Type() string {
return authType
}
// loggerOpt is an option to set the logger for the client
type loggerOpt struct {
logger *slog.Logger
}
func (o *loggerOpt) apply(c *cfg) {
c.logger = o.logger
}
func (o *loggerOpt) Type() string {
return logger
}
func (opt clientOpt) apply(cfg *cfg) { opt.fn(cfg) }
func ControllerUrl(pinotControllerUrl string) Opt {
return &controllerUrlOpt{controllerUrl: pinotControllerUrl}
}
func AuthToken(token string) Opt {
return &authTokenOpt{authToken: token}
}
func Logger(logger *slog.Logger) Opt {
return &loggerOpt{logger: logger}
}
func AuthType(authType string) Opt {
return &authTypeOpt{authType: authType}
}
func validateOpts(opts ...Opt) (*cfg, *url.URL, error) {
// with default auth writer that does nothing
optCfg := defaultCfg()
optCounts := make(map[string]int)
for _, opt := range opts {
switch opt.(type) {
case *authTypeOpt:
optCounts[authType]++
case *authTokenOpt:
optCounts[authToken]++
case *controllerUrlOpt:
optCounts[controllerUrl]++
case *loggerOpt:
optCounts[logger]++
default:
optCounts[opt.Type()]++
}
opt.apply(optCfg)
if optCounts[authType] > 1 {
return nil, nil, fmt.Errorf("multiple auth types provided")
}
}
// validate controller url
pinotControllerUrl, err := url.Parse(optCfg.controllerUrl)
if err != nil {
return nil, nil, fmt.Errorf("controller url is invalid: %w", err)
}
// TODO: remove the redundant check
// Currently this is designed to avoid a breaking change
if optCfg.authType != "" && optCfg.authToken == "" {
return nil, nil, fmt.Errorf("auth token is required when auth type is set")
}
// if auth token passed, handle authenticated requests
if optCfg.authToken != "" {
switch strings.ToLower(optCfg.authType) {
case "bearer":
optCfg.httpAuthWriter = func(req *http.Request) {
req.Header.Set("Authorization", fmt.Sprintf("Bearer %s", optCfg.authToken))
}
default:
optCfg.httpAuthWriter = func(req *http.Request) {
req.Header.Set("Authorization", fmt.Sprintf("Basic %s", optCfg.authToken))
}
fmt.Println("auth type not supported, defaulting to basic auth")
}
}
return optCfg, pinotControllerUrl, nil
}
func defaultCfg() *cfg {
return &cfg{
httpAuthWriter: defaultAuthWriter(),
logger: defaultLogger(),
}
}
func defaultAuthWriter() func(*http.Request) {
return func(req *http.Request) {
// do nothing
}
}
func defaultLogger() *slog.Logger {
return slog.New(slog.NewTextHandler(os.Stdout, &slog.HandlerOptions{Level: slog.LevelInfo}))
}