This repository has been archived by the owner on Oct 2, 2020. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathconfig.go
224 lines (194 loc) · 6.11 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
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
// Copyright (c) 2016 Uber Technologies, Inc.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
package gwr
import (
"errors"
"net"
"os"
"sync/atomic"
"github.com/uber-common/stacked"
)
var (
// ErrAlreadyConfigured is returned by gwr.Configure when called more than
// once.
ErrAlreadyConfigured = errors.New("gwr already configured")
// ErrAlreadyStarted is returned by ConfiguredServer.Start if the server is
// already listening.
ErrAlreadyStarted = errors.New("gwr server already started")
)
// Config defines configuration for GWR. For now this only defines server
// configuration; however once we have reporting support we'll add something
// ReportingCofiguration here.
type Config struct {
// Enabled controls whether GWR is enabled or not, it defaults true.
// Currently this only controls whether ConfiguredServer starting.
Enabled *bool `yaml:"enabled"`
// ListenAddr controls what address ConfiguredServer will listen on. It is
// superceded by the $GWR_LISTEN environment variable.
//
// If no listen address is set, then GWR does not start its own listening
// server; however GWR can still be accessed under "/gwr/..." from any
// default http servers.
ListenAddr string `yaml:"listen"`
}
var theServer *ConfiguredServer
// Configure sets up the gwr library and starts any resources (like a listening
// server) if enabled.
// - if nil config is passed, it's a convenience for &gwr.Config{}
// - if called more than once, ErrAlreadyConfigured is returned
// - otherwise any ConfiguredServer.Start error is returned.
func Configure(config *Config) error {
if theServer != nil {
return ErrAlreadyConfigured
}
if config == nil {
config = &Config{}
}
theServer = NewConfiguredServer(*config)
return theServer.Start()
}
// Enabled returns true if the gwr library is configured and enabled.
func Enabled() bool {
if theServer == nil {
return false
}
return theServer.Enabled()
}
// DefaultServer returns the configured gwr server, or nil if Configure
// hasn't been called yet.
func DefaultServer() *ConfiguredServer {
return theServer
}
type serverConfig struct {
enabled bool
listenAddr string
}
var defaultServerConfig = serverConfig{
enabled: true,
listenAddr: "",
}
// ConfiguredServer manages the lifecycle of a configured GWR server, as
// created by gwr.NewServer.
type ConfiguredServer struct {
config serverConfig
stacked stacked.Server
ln net.Listener
stopping uint32
done chan error
}
// NewConfiguredServer creates a new ConfiguredServer for a given config.
func NewConfiguredServer(cfg Config) *ConfiguredServer {
srv := &ConfiguredServer{
config: defaultServerConfig,
stacked: NewServer(DefaultDataSources),
}
if cfg.Enabled != nil {
srv.config.enabled = *cfg.Enabled
}
if envListen := os.Getenv("GWR_LISTEN"); envListen != "" {
srv.config.listenAddr = envListen
} else if cfg.ListenAddr != "" {
srv.config.listenAddr = cfg.ListenAddr
}
return srv
}
// Enabled return true if the server is enabled.
func (srv *ConfiguredServer) Enabled() bool {
return srv.config.enabled
}
// ListenAddr returns the configured listen address string.
func (srv *ConfiguredServer) ListenAddr() string {
return srv.config.listenAddr
}
// Addr returns the current listening address, if any.
func (srv *ConfiguredServer) Addr() net.Addr {
if srv.ln == nil {
return nil
}
return srv.ln.Addr()
}
// Start starts the server by creating the listener and a server goroutine to
// accept connections.
// - if not enabled, or if no listen address is configured, noops and returns
// nil
// - if already listening, returns ErrAlreadyStarted
// - otherwise any net.Listen error is returned.
func (srv *ConfiguredServer) Start() error {
if !srv.config.enabled {
return nil
}
if srv.config.listenAddr == "" {
return nil
}
if srv.ln != nil {
return ErrAlreadyStarted
}
ln, err := net.Listen("tcp", srv.config.listenAddr)
if err != nil {
return err
}
srv.ln = ln
srv.done = make(chan error, 1)
go func(ln net.Listener, done chan<- error) {
err := srv.stacked.Serve(ln)
if atomic.LoadUint32(&srv.stopping) == 0 {
done <- err
} else {
done <- nil
}
}(srv.ln, srv.done)
return nil
}
// StartOn starts the server on a given listening address. If the start
// succeeds, it also updates the configured listening address for later
// reference. It has all the same error cases as ConfiguredServer.Start.
func (srv *ConfiguredServer) StartOn(laddr string) error {
if !srv.config.enabled {
return nil
}
if srv.ln != nil {
return ErrAlreadyStarted
}
oldLaddr := srv.config.listenAddr
srv.config.listenAddr = laddr
err := srv.Start()
if err != nil {
srv.config.listenAddr = oldLaddr
}
return err
}
// Stop closes the current listener and shuts down the server goroutine started
// by Start (if any).
func (srv *ConfiguredServer) Stop() error {
if srv.ln == nil {
return nil
}
if !atomic.CompareAndSwapUint32(&srv.stopping, 0, 1) {
return nil
}
ln, done := srv.ln, srv.done
srv.ln, srv.done = nil, nil
err := ln.Close()
if serveErr := <-done; err == nil && serveErr != nil {
err = serveErr
}
atomic.CompareAndSwapUint32(&srv.stopping, 1, 0)
return err
}