-
-
Notifications
You must be signed in to change notification settings - Fork 141
/
Copy pathserve.go
296 lines (268 loc) · 9.68 KB
/
serve.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
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
package engine
import (
"errors"
"net/http"
"os"
"strings"
"sync"
"sync/atomic"
"time"
"github.com/caddyserver/certmagic"
"github.com/sirupsen/logrus"
"github.com/tylerb/graceful"
"github.com/xyproto/env/v2"
"golang.org/x/net/http2"
)
// List of functions to run at shutdown
var (
shutdownFunctions []func()
serverServingMutex sync.Mutex
completed bool
)
// AtShutdown adds a function to the list of functions that will be ran at shutdown
func AtShutdown(shutdownFunction func()) {
serverServingMutex.Lock()
defer serverServingMutex.Unlock()
shutdownFunctions = append(shutdownFunctions, shutdownFunction)
}
// NewGracefulServer creates a new graceful server configuration
func (ac *Config) NewGracefulServer(handler http.Handler, http2support bool, addr string) *graceful.Server {
// Server configuration
s := &http.Server{
Addr: addr,
Handler: handler, // Use the provided http.Handler (e.g. httprouter)
// The timeout values are also the maximum time it can take
// for a complete page of Server-Sent Events (SSE).
ReadTimeout: 10 * time.Second,
WriteTimeout: time.Duration(ac.writeTimeout) * time.Second,
MaxHeaderBytes: 1 << 20,
}
if http2support {
// Enable HTTP/2 support
http2.ConfigureServer(s, nil)
}
gracefulServer := &graceful.Server{
Server: s,
Timeout: ac.shutdownTimeout,
}
// Handle ctrl-c: run the shutdown functions
gracefulServer.ShutdownInitiated = ac.GenerateShutdownFunction(gracefulServer)
return gracefulServer
}
// GenerateShutdownFunction generates a function that will run the postponed
// shutdown functions. Note that gracefulServer can be nil. It's only used for
// finding out if the server was interrupted (ctrl-c or killed, SIGINT/SIGTERM)
func (ac *Config) GenerateShutdownFunction(gracefulServer *graceful.Server) func() {
return func() {
if completed {
// The shutdown functions have already been called
return
}
if ac.verboseMode {
logrus.Info("Initiating shutdown")
}
// Call the shutdown functions in chronological order (FIFO)
for _, shutdownFunction := range shutdownFunctions {
serverServingMutex.Lock()
shutdownFunction()
serverServingMutex.Unlock()
}
completed = true
if ac.verboseMode {
logrus.Info("Shutdown complete")
}
serverServingMutex.Lock()
defer serverServingMutex.Unlock()
// Forced shutdown
if gracefulServer != nil {
if gracefulServer.Interrupted {
// gracefulServer.Stop(forcedShutdownTimeout)
ac.fatalExit(errors.New("Interrupted"))
}
}
// One final flush
os.Stdout.Sync()
}
}
// Serve HTTP, HTTP/2 and/or HTTPS. Returns an error if unable to serve, or nil when done serving.
func (ac *Config) Serve(handler http.Handler, done, ready chan bool) error {
// If we are not writing internal logs to a file, reduce the verbosity
http2.VerboseLogs = (ac.internalLogFilename != os.DevNull)
if ac.onlyLuaMode {
ready <- true // Send a "ready" message to the REPL
<-done // Wait for a "done" message from the REPL (or just keep waiting)
// Serve nothing
return nil // Done
}
// Channel to wait and see if we should just serve regular HTTP instead
justServeRegularHTTP := make(chan bool)
var (
servingHTTPS atomic.Bool
servingHTTP atomic.Bool
)
// Goroutine that wait for a message to just serve regular HTTP, if needed
go func() {
<-justServeRegularHTTP // Wait for a message to just serve regular HTTP
if strings.HasPrefix(ac.serverAddr, ":") {
logrus.Info("Serving HTTP on http://localhost" + ac.serverAddr + "/")
} else {
logrus.Info("Serving HTTP on http://" + ac.serverAddr + "/")
}
servingHTTP.Store(true)
HTTPserver := ac.NewGracefulServer(handler, false, ac.serverAddr)
// Open the URL before the serving has started, in a short delay
if ac.openURLAfterServing && ac.luaServerFilename != "" {
go func() {
time.Sleep(delayBeforeLaunchingBrowser)
ac.OpenURL(ac.serverHost, ac.serverAddr, false)
}()
}
// Start serving. Shut down gracefully at exit.
if err := HTTPserver.ListenAndServe(); err != nil {
servingHTTP.Store(false)
// If we can't serve regular HTTP on port 80, give up
ac.fatalExit(err)
}
}()
// Decide which protocol to listen to
switch {
case ac.useCertMagic:
if len(ac.certMagicDomains) == 0 {
logrus.Warnln("Found no directories looking like domains in the given directory.")
} else if len(ac.certMagicDomains) == 1 {
logrus.Infof("Serving one domain with CertMagic: %s", ac.certMagicDomains[0])
} else {
logrus.Infof("Serving %d domains with CertMagic: %s", len(ac.certMagicDomains), strings.Join(ac.certMagicDomains, ", "))
}
servingHTTPS.Store(true)
// TODO: Look at "Advanced use" at https://github.com/caddyserver/certmagic#examples
// Listen for HTTP and HTTPS requests, for specific domain(s)
go func() {
// If $XDG_CONFIG_DIR is not set, use $HOME.
// If $HOME is not set, use $TMPDIR.
// If $TMPDIR is not set, use /tmp.
certStorageDir := env.StrAlt("XDG_CONFIG_DIR", "HOME", env.Str("TMPDIR", "/tmp"))
defaultEmail := env.StrAlt("LOGNAME", "USER", "root") + "@localhost"
if len(ac.certMagicDomains) > 0 {
defaultEmail = "webmaster@" + ac.certMagicDomains[0]
}
certmagic.DefaultACME.Email = env.Str("EMAIL", defaultEmail)
// TODO: Find a way for Algernon users to agree on this manually
certmagic.DefaultACME.Agreed = true
certmagic.Default.Storage = &certmagic.FileStorage{Path: certStorageDir}
if err := certmagic.HTTPS(ac.certMagicDomains, handler); err != nil {
servingHTTPS.Store(false)
logrus.Error(err)
// Don't serve HTTP if CertMagic fails, just quit
// justServeRegularHTTP <- true
}
}()
case ac.serveJustQUIC: // Just serve QUIC, but fallback to HTTP
if strings.HasPrefix(ac.serverAddr, ":") {
logrus.Info("Serving QUIC on https://localhost" + ac.serverAddr + "/")
} else {
logrus.Info("Serving QUIC on https://" + ac.serverAddr + "/")
}
servingHTTPS.Store(true)
// Start serving over QUIC
go ac.ListenAndServeQUIC(handler, justServeRegularHTTP, &servingHTTPS)
case ac.productionMode:
// Listen for both HTTPS+HTTP/2 and HTTP requests, on different ports
if len(ac.serverHost) == 0 {
logrus.Info("Serving HTTP/2 on https://localhost/")
} else {
logrus.Info("Serving HTTP/2 on https://" + ac.serverHost + "/")
}
servingHTTPS.Store(true)
go func() {
// Start serving. Shut down gracefully at exit.
// Listen for HTTPS + HTTP/2 requests
HTTPS2server := ac.NewGracefulServer(handler, true, ac.serverHost+":443")
// Start serving. Shut down gracefully at exit.
if err := HTTPS2server.ListenAndServeTLS(ac.serverCert, ac.serverKey); err != nil {
servingHTTPS.Store(false)
logrus.Error(err)
}
}()
if len(ac.serverHost) == 0 {
logrus.Info("Serving HTTP on http://localhost/")
} else {
logrus.Info("Serving HTTP on http://" + ac.serverHost + "/")
}
servingHTTP.Store(true)
go func() {
if ac.redirectHTTP {
// Redirect HTTPS to HTTP
redirectFunc := func(w http.ResponseWriter, req *http.Request) {
http.Redirect(w, req, "https://"+req.Host+req.URL.String(), http.StatusMovedPermanently)
}
if err := http.ListenAndServe(ac.serverHost+":80", http.HandlerFunc(redirectFunc)); err != nil {
servingHTTP.Store(false)
// If we can't serve regular HTTP on port 80, give up
ac.fatalExit(err)
}
} else {
// Don't redirect, but serve the same contents as the HTTPS server as HTTP on port 80
HTTPserver := ac.NewGracefulServer(handler, false, ac.serverHost+":80")
if err := HTTPserver.ListenAndServe(); err != nil {
servingHTTP.Store(false)
// If we can't serve regular HTTP on port 80, give up
ac.fatalExit(err)
}
}
}()
case ac.serveJustHTTP2: // It's unusual to serve HTTP/2 without HTTPS
if strings.HasPrefix(ac.serverAddr, ":") {
logrus.Warn("Serving HTTP/2 without HTTPS (not recommended!) on http://localhost" + ac.serverAddr + "/")
} else {
logrus.Warn("Serving HTTP/2 without HTTPS (not recommended!) on http://" + ac.serverAddr + "/")
}
servingHTTPS.Store(true)
go func() {
// Listen for HTTP/2 requests
HTTP2server := ac.NewGracefulServer(handler, true, ac.serverAddr)
// Start serving. Shut down gracefully at exit.
if err := HTTP2server.ListenAndServe(); err != nil {
servingHTTPS.Store(false)
justServeRegularHTTP <- true
logrus.Error(err)
}
}()
case !ac.serveJustHTTP2 && !ac.serveJustHTTP:
if strings.HasPrefix(ac.serverAddr, ":") {
logrus.Info("Serving HTTP/2 on https://localhost" + ac.serverAddr + "/")
} else {
logrus.Info("Serving HTTP/2 on https://" + ac.serverAddr + "/")
}
servingHTTPS.Store(true)
// Listen for HTTPS + HTTP/2 requests
HTTPS2server := ac.NewGracefulServer(handler, true, ac.serverAddr)
// Start serving. Shut down gracefully at exit.
go func() {
if err := HTTPS2server.ListenAndServeTLS(ac.serverCert, ac.serverKey); err != nil {
logrus.Errorf("%s. Not serving HTTP/2.", err)
logrus.Info("Use the -t flag for serving regular HTTP.")
servingHTTPS.Store(false)
// If HTTPS failed (perhaps the key + cert are missing),
// serve plain HTTP instead
justServeRegularHTTP <- true
}
}()
default:
servingHTTP.Store(true)
justServeRegularHTTP <- true
}
// Wait just a tiny bit
time.Sleep(20 * time.Millisecond)
ready <- true // Send a "ready" message to the REPL
// Open the URL, if specified
if ac.openURLAfterServing {
if !servingHTTP.Load() && !servingHTTPS.Load() {
ac.fatalExit(errors.New("serving neither over http:// nor over https://"))
}
// Open the https:// URL if both http:// and https:// are being served
ac.OpenURL(ac.serverHost, ac.serverAddr, servingHTTPS.Load())
}
<-done // Wait for a "done" message from the REPL (or just keep waiting)
return nil // Done serving
}