Thanks to visit codestin.com
Credit goes to github.com

Skip to content

Commit 81e99be

Browse files
authored
fix: close server pty connections on client disconnect (#15201)
1 parent 69c1d98 commit 81e99be

File tree

2 files changed

+23
-7
lines changed

2 files changed

+23
-7
lines changed

coderd/httpapi/websocket.go

+20-3
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,10 @@ package httpapi
22

33
import (
44
"context"
5+
"errors"
56
"time"
67

8+
"golang.org/x/xerrors"
79
"nhooyr.io/websocket"
810

911
"cdr.dev/slog"
@@ -31,7 +33,8 @@ func Heartbeat(ctx context.Context, conn *websocket.Conn) {
3133
// Heartbeat loops to ping a WebSocket to keep it alive. It calls `exit` on ping
3234
// failure.
3335
func HeartbeatClose(ctx context.Context, logger slog.Logger, exit func(), conn *websocket.Conn) {
34-
ticker := time.NewTicker(15 * time.Second)
36+
interval := 15 * time.Second
37+
ticker := time.NewTicker(interval)
3538
defer ticker.Stop()
3639

3740
for {
@@ -40,12 +43,26 @@ func HeartbeatClose(ctx context.Context, logger slog.Logger, exit func(), conn *
4043
return
4144
case <-ticker.C:
4245
}
43-
err := conn.Ping(ctx)
46+
err := pingWithTimeout(ctx, conn, interval)
4447
if err != nil {
48+
// context.DeadlineExceeded is expected when the client disconnects without sending a close frame
49+
if !errors.Is(err, context.DeadlineExceeded) {
50+
logger.Error(ctx, "failed to heartbeat ping", slog.Error(err))
51+
}
4552
_ = conn.Close(websocket.StatusGoingAway, "Ping failed")
46-
logger.Info(ctx, "failed to heartbeat ping", slog.Error(err))
4753
exit()
4854
return
4955
}
5056
}
5157
}
58+
59+
func pingWithTimeout(ctx context.Context, conn *websocket.Conn, timeout time.Duration) error {
60+
ctx, cancel := context.WithTimeout(ctx, timeout)
61+
defer cancel()
62+
err := conn.Ping(ctx)
63+
if err != nil {
64+
return xerrors.Errorf("failed to ping: %w", err)
65+
}
66+
67+
return nil
68+
}

coderd/workspaceapps/proxy.go

+3-4
Original file line numberDiff line numberDiff line change
@@ -593,7 +593,6 @@ func (s *Server) proxyWorkspaceApp(rw http.ResponseWriter, r *http.Request, appT
593593
tracing.EndHTTPSpan(r, http.StatusOK, trace.SpanFromContext(ctx))
594594

595595
report := newStatsReportFromSignedToken(appToken)
596-
s.collectStats(report)
597596
defer func() {
598597
// We must use defer here because ServeHTTP may panic.
599598
report.SessionEndedAt = dbtime.Now()
@@ -614,7 +613,8 @@ func (s *Server) proxyWorkspaceApp(rw http.ResponseWriter, r *http.Request, appT
614613
// @Success 101
615614
// @Router /workspaceagents/{workspaceagent}/pty [get]
616615
func (s *Server) workspaceAgentPTY(rw http.ResponseWriter, r *http.Request) {
617-
ctx := r.Context()
616+
ctx, cancel := context.WithCancel(r.Context())
617+
defer cancel()
618618

619619
s.websocketWaitMutex.Lock()
620620
s.websocketWaitGroup.Add(1)
@@ -670,12 +670,11 @@ func (s *Server) workspaceAgentPTY(rw http.ResponseWriter, r *http.Request) {
670670
})
671671
return
672672
}
673+
go httpapi.HeartbeatClose(ctx, s.Logger, cancel, conn)
673674

674675
ctx, wsNetConn := WebsocketNetConn(ctx, conn, websocket.MessageBinary)
675676
defer wsNetConn.Close() // Also closes conn.
676677

677-
go httpapi.Heartbeat(ctx, conn)
678-
679678
agentConn, release, err := s.AgentProvider.AgentConn(ctx, appToken.AgentID)
680679
if err != nil {
681680
log.Debug(ctx, "dial workspace agent", slog.Error(err))

0 commit comments

Comments
 (0)