Skip to content

Commit d81c19a

Browse files
committed
fix: grpc server panic
1 parent e2140e6 commit d81c19a

File tree

1 file changed

+50
-3
lines changed

1 file changed

+50
-3
lines changed

transport/gun/server.go

+50-3
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,10 @@ import (
55
"net"
66
"net/http"
77
"strings"
8+
"sync"
89
"time"
910

11+
"github.com/metacubex/mihomo/common/buf"
1012
N "github.com/metacubex/mihomo/common/net"
1113
C "github.com/metacubex/mihomo/constant"
1214

@@ -57,9 +59,14 @@ func NewServerHandler(options ServerOption) http.Handler {
5759
conn.localAddr = addr
5860
}
5961

60-
// gun.Conn can't correct handle ReadDeadline
61-
// so call N.NewDeadlineConn to add a safe wrapper
62-
connHandler(N.NewDeadlineConn(conn))
62+
wrapper := &h2ConnWrapper{
63+
// gun.Conn can't correct handle ReadDeadline
64+
// so call N.NewDeadlineConn to add a safe wrapper
65+
ExtendedConn: N.NewDeadlineConn(conn),
66+
}
67+
connHandler(wrapper)
68+
wrapper.CloseWrapper()
69+
6370
return
6471
}
6572

@@ -68,3 +75,43 @@ func NewServerHandler(options ServerOption) http.Handler {
6875
IdleTimeout: idleTimeout,
6976
})
7077
}
78+
79+
// h2ConnWrapper used to avoid "panic: Write called after Handler finished" for gun.Conn
80+
type h2ConnWrapper struct {
81+
N.ExtendedConn
82+
access sync.Mutex
83+
closed bool
84+
}
85+
86+
func (w *h2ConnWrapper) Write(p []byte) (n int, err error) {
87+
w.access.Lock()
88+
defer w.access.Unlock()
89+
if w.closed {
90+
return 0, net.ErrClosed
91+
}
92+
return w.ExtendedConn.Write(p)
93+
}
94+
95+
func (w *h2ConnWrapper) WriteBuffer(buffer *buf.Buffer) error {
96+
w.access.Lock()
97+
defer w.access.Unlock()
98+
if w.closed {
99+
return net.ErrClosed
100+
}
101+
return w.ExtendedConn.WriteBuffer(buffer)
102+
}
103+
104+
func (w *h2ConnWrapper) CloseWrapper() {
105+
w.access.Lock()
106+
defer w.access.Unlock()
107+
w.closed = true
108+
}
109+
110+
func (w *h2ConnWrapper) Close() error {
111+
w.CloseWrapper()
112+
return w.ExtendedConn.Close()
113+
}
114+
115+
func (w *h2ConnWrapper) Upstream() any {
116+
return w.ExtendedConn
117+
}

0 commit comments

Comments
 (0)