Skip to content

Commit ee57ba9

Browse files
authored
Merge pull request #406 from dolthub/macneale4/better-connecting
Avoid spin waits and dead connections in mysql server
2 parents 0e77d54 + 635bdc9 commit ee57ba9

File tree

1 file changed

+80
-19
lines changed

1 file changed

+80
-19
lines changed

go/mysql/server.go

+80-19
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ import (
2323
"io"
2424
"net"
2525
"strings"
26+
"sync/atomic"
2627
"time"
2728

2829
"github.com/dolthub/vitess/go/netutil"
@@ -198,6 +199,12 @@ type Listener struct {
198199
// Max limit for connections
199200
maxConns uint64
200201

202+
// maxWaitConns it the number of waiting connections allowed before new connections start getting rejected.
203+
maxWaitConns uint32
204+
205+
// maxWaitConnsTimeout is the amount of time to block a new connection before giving up and rejecting it.
206+
maxWaitConnsTimeout time.Duration
207+
201208
// The following parameters are read by multiple connection go
202209
// routines. They are not protected by a mutex, so they
203210
// should be set after NewListener, and not changed while
@@ -232,8 +239,8 @@ type Listener struct {
232239
// Reads are unbuffered if it's <=0.
233240
connReadBufferSize int
234241

235-
// shutdown indicates that Shutdown method was called.
236-
shutdown sync2.AtomicBool
242+
// shutdownCh - open channel until it's not. Used to block and handle shutdown without hanging
243+
shutdownCh chan struct{}
237244

238245
// RequireSecureTransport configures the server to reject connections from insecure clients
239246
RequireSecureTransport bool
@@ -274,6 +281,8 @@ type ListenerConfig struct {
274281
ConnWriteTimeout time.Duration
275282
ConnReadBufferSize int
276283
MaxConns uint64
284+
MaxWaitConns uint32
285+
MaxWaitConnsTimeout time.Duration
277286
AllowClearTextWithoutTLS bool
278287
}
279288

@@ -301,7 +310,10 @@ func NewListenerWithConfig(cfg ListenerConfig) (*Listener, error) {
301310
connWriteTimeout: cfg.ConnWriteTimeout,
302311
connReadBufferSize: cfg.ConnReadBufferSize,
303312
maxConns: cfg.MaxConns,
313+
maxWaitConns: cfg.MaxWaitConns,
314+
maxWaitConnsTimeout: cfg.MaxWaitConnsTimeout,
304315
AllowClearTextWithoutTLS: sync2.NewAtomicBool(cfg.AllowClearTextWithoutTLS),
316+
shutdownCh: make(chan struct{}),
305317
}, nil
306318
}
307319

@@ -312,6 +324,26 @@ func (l *Listener) Addr() net.Addr {
312324

313325
// Accept runs an accept loop until the listener is closed.
314326
func (l *Listener) Accept() {
327+
var sem chan struct{}
328+
if l.maxConns > 0 {
329+
sem = make(chan struct{}, l.maxConns)
330+
}
331+
332+
// don't spam the logs if we have a bunch of waiting connections come in at once
333+
warnOnWait := true
334+
var waitingConnections atomic.Int32
335+
336+
accepted := func(ctx context.Context, conn net.Conn, id uint32, acceptTime time.Time) {
337+
connCount.Add(1)
338+
connAccept.Add(1)
339+
go func() {
340+
if sem != nil {
341+
defer func() { <-sem }()
342+
}
343+
l.handle(ctx, conn, id, acceptTime)
344+
}()
345+
}
346+
315347
for {
316348
conn, err := l.listener.Accept()
317349
if err != nil {
@@ -320,24 +352,45 @@ func (l *Listener) Accept() {
320352
}
321353

322354
acceptTime := time.Now()
323-
324355
connectionID := l.connectionID
325356
l.connectionID++
326357

327-
maxConWarn := false
328-
for l.maxConns > 0 && uint64(connCount.Get()) >= l.maxConns {
329-
if !maxConWarn {
330-
log.Warning("max connections reached. Clients waiting. Increase server max connections")
331-
maxConWarn = true // Logging once for each connection seems adequate.
332-
}
333-
334-
// TODO: make this behavior configurable (wait v. reject)
335-
time.Sleep(500 * time.Millisecond)
358+
if sem == nil {
359+
accepted(context.Background(), conn, connectionID, acceptTime)
360+
continue
336361
}
337362

338-
connCount.Add(1)
339-
connAccept.Add(1)
340-
go l.handle(context.Background(), conn, connectionID, acceptTime)
363+
select {
364+
case sem <- struct{}{}:
365+
accepted(context.Background(), conn, connectionID, acceptTime)
366+
warnOnWait = true
367+
default:
368+
if warnOnWait {
369+
log.Warning("max connections reached. Clients waiting. Increase server max_connections")
370+
warnOnWait = false
371+
continue
372+
}
373+
waitNum := waitingConnections.Add(1)
374+
if uint32(waitNum) > l.maxWaitConns {
375+
log.Warning("max waiting connections reached. Client rejected. Increase server max_connections and back_log")
376+
conn.Close()
377+
waitingConnections.Add(-1)
378+
continue
379+
}
380+
go func(conn net.Conn, connectionID uint32, acceptTime time.Time) {
381+
select {
382+
case sem <- struct{}{}:
383+
waitingConnections.Add(-1)
384+
accepted(context.Background(), conn, connectionID, acceptTime)
385+
case <-l.shutdownCh:
386+
conn.Close()
387+
waitingConnections.Add(-1)
388+
case <-time.After(l.maxWaitConnsTimeout):
389+
conn.Close()
390+
waitingConnections.Add(-1)
391+
}
392+
}(conn, connectionID, acceptTime)
393+
}
341394
}
342395
}
343396

@@ -560,19 +613,27 @@ func (l *Listener) handleConnectionWarning(c *Conn, reason string) {
560613

561614
// Close stops the listener, which prevents accept of any new connections. Existing connections won't be closed.
562615
func (l *Listener) Close() {
563-
l.listener.Close()
616+
l.Shutdown()
564617
}
565618

566619
// Shutdown closes listener and fails any Ping requests from existing connections.
567620
// This can be used for graceful shutdown, to let clients know that they should reconnect to another server.
568621
func (l *Listener) Shutdown() {
569-
if l.shutdown.CompareAndSwap(false, true) {
570-
l.Close()
622+
select {
623+
case <-l.shutdownCh:
624+
default:
625+
close(l.shutdownCh)
626+
l.listener.Close()
571627
}
572628
}
573629

574630
func (l *Listener) isShutdown() bool {
575-
return l.shutdown.Get()
631+
select {
632+
case <-l.shutdownCh:
633+
return true
634+
default:
635+
return false
636+
}
576637
}
577638

578639
// writeHandshakeV10 writes the Initial Handshake Packet, server side.

0 commit comments

Comments
 (0)