add ping-pong support for HTTP2
This commit is contained in:
parent
e476a9f700
commit
b244ce00c9
45
chain.go
45
chain.go
@ -1,19 +1,21 @@
|
|||||||
package gost
|
package gost
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"crypto/rand"
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
"errors"
|
"errors"
|
||||||
"github.com/golang/glog"
|
"github.com/golang/glog"
|
||||||
"golang.org/x/net/http2"
|
"golang.org/x/net/http2"
|
||||||
"io"
|
"io"
|
||||||
//"io/ioutil"
|
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/http/httputil"
|
"net/http/httputil"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Proxy chain holds a list of proxy nodes
|
// Proxy chain holds a list of proxy nodes
|
||||||
@ -134,7 +136,15 @@ func (c *ProxyChain) initHttp2Client(config *tls.Config, nodes ...ProxyNode) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return conn, err
|
return conn, err
|
||||||
}
|
}
|
||||||
return tls.Client(conn, cfg), nil
|
conn = tls.Client(conn, cfg)
|
||||||
|
|
||||||
|
// enable HTTP2 ping-pong
|
||||||
|
pingIntvl, _ := strconv.Atoi(http2Node.Get("ping"))
|
||||||
|
if pingIntvl > 0 {
|
||||||
|
enablePing(conn, time.Duration(pingIntvl)*time.Second)
|
||||||
|
}
|
||||||
|
|
||||||
|
return conn, nil
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
c.http2Client = &http.Client{Transport: &tr}
|
c.http2Client = &http.Client{Transport: &tr}
|
||||||
@ -142,6 +152,37 @@ func (c *ProxyChain) initHttp2Client(config *tls.Config, nodes ...ProxyNode) {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func enablePing(conn net.Conn, interval time.Duration) {
|
||||||
|
if conn == nil || interval == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
glog.V(LINFO).Infoln("[http2] ping enabled, interval:", interval)
|
||||||
|
go func() {
|
||||||
|
t := time.NewTicker(interval)
|
||||||
|
var framer *http2.Framer
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-t.C:
|
||||||
|
if framer == nil {
|
||||||
|
framer = http2.NewFramer(conn, conn)
|
||||||
|
}
|
||||||
|
|
||||||
|
var p [8]byte
|
||||||
|
rand.Read(p[:])
|
||||||
|
err := framer.WritePing(false, p)
|
||||||
|
if err != nil {
|
||||||
|
t.Stop()
|
||||||
|
framer = nil
|
||||||
|
glog.V(LWARNING).Infoln("[http2] ping:", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
glog.V(LINFO).Infoln("[http2] ping OK")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
||||||
// Connect to addr through proxy chain
|
// Connect to addr through proxy chain
|
||||||
func (c *ProxyChain) Dial(addr string) (net.Conn, error) {
|
func (c *ProxyChain) Dial(addr string) (net.Conn, error) {
|
||||||
if !strings.Contains(addr, ":") {
|
if !strings.Contains(addr, ":") {
|
||||||
|
Loading…
Reference in New Issue
Block a user