Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
caddy
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Łukasz Nowak
caddy
Commits
139a3cfb
Commit
139a3cfb
authored
Jan 24, 2017
by
Matthew Holt
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Replace our old faithful gracefulListener with Go 1.8's Shutdown()
parent
16250da3
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
11 additions
and
130 deletions
+11
-130
caddyhttp/httpserver/graceful.go
caddyhttp/httpserver/graceful.go
+0
-80
caddyhttp/httpserver/server.go
caddyhttp/httpserver/server.go
+11
-50
No files found.
caddyhttp/httpserver/graceful.go
deleted
100644 → 0
View file @
16250da3
package
httpserver
import
(
"net"
"sync"
"syscall"
)
// TODO: Should this be a generic graceful listener available in its own package or something?
// Also, passing in a WaitGroup is a little awkward. Why can't this listener just keep
// the waitgroup internal to itself?
// newGracefulListener returns a gracefulListener that wraps l and
// uses wg (stored in the host server) to count connections.
func
newGracefulListener
(
l
net
.
Listener
,
wg
*
sync
.
WaitGroup
)
*
gracefulListener
{
gl
:=
&
gracefulListener
{
Listener
:
l
,
stop
:
make
(
chan
error
),
connWg
:
wg
}
go
func
()
{
<-
gl
.
stop
gl
.
Lock
()
gl
.
stopped
=
true
gl
.
Unlock
()
gl
.
stop
<-
gl
.
Listener
.
Close
()
}()
return
gl
}
// gracefuListener is a net.Listener which can
// count the number of connections on it. Its
// methods mainly wrap net.Listener to be graceful.
type
gracefulListener
struct
{
net
.
Listener
stop
chan
error
stopped
bool
sync
.
Mutex
// protects the stopped flag
connWg
*
sync
.
WaitGroup
// pointer to the host's wg used for counting connections
}
// Accept accepts a connection.
func
(
gl
*
gracefulListener
)
Accept
()
(
c
net
.
Conn
,
err
error
)
{
c
,
err
=
gl
.
Listener
.
Accept
()
if
err
!=
nil
{
return
}
c
=
gracefulConn
{
Conn
:
c
,
connWg
:
gl
.
connWg
}
gl
.
connWg
.
Add
(
1
)
return
}
// Close immediately closes the listener.
func
(
gl
*
gracefulListener
)
Close
()
error
{
gl
.
Lock
()
if
gl
.
stopped
{
gl
.
Unlock
()
return
syscall
.
EINVAL
}
gl
.
Unlock
()
gl
.
stop
<-
nil
return
<-
gl
.
stop
}
// gracefulConn represents a connection on a
// gracefulListener so that we can keep track
// of the number of connections, thus facilitating
// a graceful shutdown.
type
gracefulConn
struct
{
net
.
Conn
connWg
*
sync
.
WaitGroup
// pointer to the host server's connection waitgroup
}
// Close closes c's underlying connection while updating the wg count.
func
(
c
gracefulConn
)
Close
()
error
{
err
:=
c
.
Conn
.
Close
()
if
err
!=
nil
{
return
err
}
// close can fail on http2 connections (as of Oct. 2015, before http2 in std lib)
// so don't decrement count unless close succeeds
c
.
connWg
.
Done
()
return
nil
}
caddyhttp/httpserver/server.go
View file @
139a3cfb
...
...
@@ -2,6 +2,7 @@
package
httpserver
import
(
"context"
"crypto/tls"
"fmt"
"io"
...
...
@@ -27,9 +28,8 @@ type Server struct {
listener
net
.
Listener
listenerMu
sync
.
Mutex
sites
[]
*
SiteConfig
connTimeout
time
.
Duration
// max time to wait for a connection before force stop
connWg
sync
.
WaitGroup
// one increment per connection
tlsGovChan
chan
struct
{}
// close to stop the TLS maintenance goroutine
connTimeout
time
.
Duration
// max time to wait for a connection before force stop
tlsGovChan
chan
struct
{}
// close to stop the TLS maintenance goroutine
vhosts
*
vhostTrie
}
...
...
@@ -46,16 +46,6 @@ func NewServer(addr string, group []*SiteConfig) (*Server, error) {
connTimeout
:
GracefulTimeout
,
}
s
.
Server
.
Handler
=
s
// this is weird, but whatever
s
.
Server
.
ConnState
=
func
(
c
net
.
Conn
,
cs
http
.
ConnState
)
{
if
cs
==
http
.
StateIdle
{
s
.
listenerMu
.
Lock
()
// server stopped, close idle connection
if
s
.
listener
==
nil
{
c
.
Close
()
}
s
.
listenerMu
.
Unlock
()
}
}
// Disable HTTP/2 if desired
if
!
HTTP2
{
...
...
@@ -68,14 +58,6 @@ func NewServer(addr string, group []*SiteConfig) (*Server, error) {
s
.
Server
.
Handler
=
s
.
wrapWithSvcHeaders
(
s
.
Server
.
Handler
)
}
// We have to bound our wg with one increment
// to prevent a "race condition" that is hard-coded
// into sync.WaitGroup.Wait() - basically, an add
// with a positive delta must be guaranteed to
// occur before Wait() is called on the wg.
// In a way, this kind of acts as a safety barrier.
s
.
connWg
.
Add
(
1
)
// Set up TLS configuration
var
tlsConfigs
[]
*
caddytls
.
Config
for
_
,
site
:=
range
group
{
...
...
@@ -154,8 +136,6 @@ func (s *Server) Serve(ln net.Listener) error {
ln
=
tcpKeepAliveListener
{
TCPListener
:
tcpLn
}
}
ln
=
newGracefulListener
(
ln
,
&
s
.
connWg
)
s
.
listenerMu
.
Lock
()
s
.
listener
=
ln
s
.
listenerMu
.
Unlock
()
...
...
@@ -300,40 +280,21 @@ func (s *Server) Address() string {
// Stop stops s gracefully (or forcefully after timeout) and
// closes its listener.
func
(
s
*
Server
)
Stop
()
(
err
error
)
{
s
.
Server
.
SetKeepAlivesEnabled
(
false
)
if
runtime
.
GOOS
!=
"windows"
{
// force connections to close after timeout
done
:=
make
(
chan
struct
{})
go
func
()
{
s
.
connWg
.
Done
()
// decrement our initial increment used as a barrier
s
.
connWg
.
Wait
()
close
(
done
)
}()
// Wait for remaining connections to finish or
// force them all to close after timeout
select
{
case
<-
time
.
After
(
s
.
connTimeout
)
:
case
<-
done
:
}
}
func
(
s
*
Server
)
Stop
()
error
{
ctx
,
cancel
:=
context
.
WithTimeout
(
context
.
Background
(),
s
.
connTimeout
)
defer
cancel
()
// Close the listener now; this stops the server without delay
s
.
listenerMu
.
Lock
()
if
s
.
listener
!=
nil
{
err
=
s
.
listener
.
Close
()
s
.
listener
=
nil
err
:=
s
.
Server
.
Shutdown
(
ctx
)
if
err
!=
nil
{
return
err
}
s
.
listenerMu
.
Unlock
()
//
Closing this signals
any TLS governor goroutines to exit
//
signal
any TLS governor goroutines to exit
if
s
.
tlsGovChan
!=
nil
{
close
(
s
.
tlsGovChan
)
}
return
return
nil
}
// sanitizePath collapses any ./ ../ /// madness
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment