@@ -5,8 +5,10 @@ import (
5
5
"net"
6
6
"net/http"
7
7
"strings"
8
+ "sync"
8
9
"time"
9
10
11
+ "github.com/metacubex/mihomo/common/buf"
10
12
N "github.com/metacubex/mihomo/common/net"
11
13
C "github.com/metacubex/mihomo/constant"
12
14
@@ -57,9 +59,14 @@ func NewServerHandler(options ServerOption) http.Handler {
57
59
conn .localAddr = addr
58
60
}
59
61
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
+
63
70
return
64
71
}
65
72
@@ -68,3 +75,43 @@ func NewServerHandler(options ServerOption) http.Handler {
68
75
IdleTimeout : idleTimeout ,
69
76
})
70
77
}
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