about summary refs log tree commit diff
path: root/pkg
diff options
context:
space:
mode:
authorMel <einebeere@gmail.com>2022-04-08 20:32:55 +0200
committerMel <einebeere@gmail.com>2022-04-08 20:32:55 +0200
commit81096040e0cf19ed1ff1ef25739ab11b99e0a7c9 (patch)
tree46fa4ca983be4ed38c2dd194e2a521f4023cd814 /pkg
parent63a5c949f14c28ed29c2f2dc2490ff2542d249ef (diff)
downloadjinx-81096040e0cf19ed1ff1ef25739ab11b99e0a7c9.tar.zst
jinx-81096040e0cf19ed1ff1ef25739ab11b99e0a7c9.zip
Graceful closing
Diffstat (limited to 'pkg')
-rw-r--r--pkg/discord/gateway/gateway.go9
-rw-r--r--pkg/libs/cancellablewebsocket/cancellablewebsocket.go44
2 files changed, 40 insertions, 13 deletions
diff --git a/pkg/discord/gateway/gateway.go b/pkg/discord/gateway/gateway.go
index 6fe0087..9c86bb6 100644
--- a/pkg/discord/gateway/gateway.go
+++ b/pkg/discord/gateway/gateway.go
@@ -76,7 +76,14 @@ func (g *GatewayImpl) Start(ctx context.Context, url string, token string) error
 }
 
 func (g *GatewayImpl) Close() error {
-	return g.conn.Close()
+	// Try closing gracefully.
+	g.logger.Debug().Msg("closing gateway gracefully...")
+	if err := g.conn.Close(1000); err != nil {
+		g.logger.Error().Err(err).Msg("error closing gateway gracefully, trying to murder...")
+		return g.conn.Kill()
+	}
+
+	return nil
 }
 
 func (g *GatewayImpl) Heartbeat() error {
diff --git a/pkg/libs/cancellablewebsocket/cancellablewebsocket.go b/pkg/libs/cancellablewebsocket/cancellablewebsocket.go
index 5dcffc1..accc180 100644
--- a/pkg/libs/cancellablewebsocket/cancellablewebsocket.go
+++ b/pkg/libs/cancellablewebsocket/cancellablewebsocket.go
@@ -9,10 +9,11 @@ import (
 )
 
 type CancellableWebSocket struct {
-	conn   *websocket.Conn
-	ctx    context.Context
-	cancel context.CancelFunc
-	errors chan error
+	conn      *websocket.Conn
+	ctx       context.Context
+	cancel    context.CancelFunc
+	closeCode int // 0 means killed
+	errors    chan error
 }
 
 func Dial(dialer *websocket.Dialer, ctx context.Context, url string, requestHeader http.Header) (*CancellableWebSocket, error) {
@@ -24,10 +25,11 @@ func Dial(dialer *websocket.Dialer, ctx context.Context, url string, requestHead
 	childCtx, cancel := context.WithCancel(ctx)
 
 	cws := &CancellableWebSocket{
-		conn:   conn,
-		ctx:    childCtx,
-		cancel: cancel,
-		errors: make(chan error),
+		conn:      conn,
+		ctx:       childCtx,
+		cancel:    cancel,
+		closeCode: 0,
+		errors:    make(chan error),
 	}
 
 	go cws.listenForCancel()
@@ -57,7 +59,7 @@ func (cws *CancellableWebSocket) WriteJSON(v interface{}) error {
 	return nil
 }
 
-func (cws *CancellableWebSocket) Close() error {
+func (cws *CancellableWebSocket) Kill() error {
 	cws.cancel()
 	err, ok := <-cws.errors
 
@@ -68,8 +70,16 @@ func (cws *CancellableWebSocket) Close() error {
 	return nil
 }
 
-func (cws *CancellableWebSocket) listenForCancel() {
+func (cws *CancellableWebSocket) Close(code int) error {
+	cws.closeCode = code
+	if err := cws.Kill(); err != nil {
+		cws.closeCode = 0 // Reset close code
+		return err
+	}
+	return nil
+}
 
+func (cws *CancellableWebSocket) listenForCancel() {
 	<-cws.ctx.Done()
 
 	var err error
@@ -83,8 +93,18 @@ func (cws *CancellableWebSocket) listenForCancel() {
 		cws.errors <- err
 	}
 
-	if err = cws.conn.Close(); err != nil {
-		cws.errors <- err
+	if cws.closeCode == 0 {
+		// Close without sending close.
+		if err = cws.conn.Close(); err != nil {
+			cws.errors <- err
+		}
+	} else {
+		// Send close with code.
+		// NOTE: The SetWriteDeadline above does not affect this. :)
+		err = cws.conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(cws.closeCode, ""), time.Now().Add(time.Second))
+		if err != nil {
+			cws.errors <- err
+		}
 	}
 
 	close(cws.errors)