mirror of https://github.com/v2ray/v2ray-core
cleanup websocket code
parent
fc1b37d809
commit
3be6c72211
|
@ -6,8 +6,8 @@ import (
|
||||||
"net"
|
"net"
|
||||||
|
|
||||||
"github.com/gorilla/websocket"
|
"github.com/gorilla/websocket"
|
||||||
"v2ray.com/core/common"
|
|
||||||
"v2ray.com/core/app/log"
|
"v2ray.com/core/app/log"
|
||||||
|
"v2ray.com/core/common"
|
||||||
v2net "v2ray.com/core/common/net"
|
v2net "v2ray.com/core/common/net"
|
||||||
"v2ray.com/core/transport/internet"
|
"v2ray.com/core/transport/internet"
|
||||||
"v2ray.com/core/transport/internet/internal"
|
"v2ray.com/core/transport/internet/internal"
|
||||||
|
@ -24,12 +24,9 @@ func Dial(ctx context.Context, dest v2net.Destination) (internet.Connection, err
|
||||||
wsSettings := internet.TransportSettingsFromContext(ctx).(*Config)
|
wsSettings := internet.TransportSettingsFromContext(ctx).(*Config)
|
||||||
|
|
||||||
id := internal.NewConnectionID(src, dest)
|
id := internal.NewConnectionID(src, dest)
|
||||||
var conn *wsconn
|
var conn net.Conn
|
||||||
if dest.Network == v2net.Network_TCP && wsSettings.IsConnectionReuse() {
|
if dest.Network == v2net.Network_TCP && wsSettings.IsConnectionReuse() {
|
||||||
connt := globalCache.Get(id)
|
conn = globalCache.Get(id)
|
||||||
if connt != nil {
|
|
||||||
conn = connt.(*wsconn)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if conn == nil {
|
if conn == nil {
|
||||||
var err error
|
var err error
|
||||||
|
@ -46,7 +43,7 @@ func init() {
|
||||||
common.Must(internet.RegisterTransportDialer(internet.TransportProtocol_WebSocket, Dial))
|
common.Must(internet.RegisterTransportDialer(internet.TransportProtocol_WebSocket, Dial))
|
||||||
}
|
}
|
||||||
|
|
||||||
func wsDial(ctx context.Context, dest v2net.Destination) (*wsconn, error) {
|
func wsDial(ctx context.Context, dest v2net.Destination) (net.Conn, error) {
|
||||||
src := internet.DialerSourceFromContext(ctx)
|
src := internet.DialerSourceFromContext(ctx)
|
||||||
wsSettings := internet.TransportSettingsFromContext(ctx).(*Config)
|
wsSettings := internet.TransportSettingsFromContext(ctx).(*Config)
|
||||||
|
|
||||||
|
@ -87,13 +84,12 @@ func wsDial(ctx context.Context, dest v2net.Destination) (*wsconn, error) {
|
||||||
}
|
}
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return func() internet.Connection {
|
return func() net.Conn {
|
||||||
connv2ray := &wsconn{
|
connv2ray := &wsconn{
|
||||||
wsc: conn,
|
wsc: conn,
|
||||||
connClosing: false,
|
connClosing: false,
|
||||||
config: wsSettings,
|
|
||||||
}
|
}
|
||||||
connv2ray.setup()
|
connv2ray.setup()
|
||||||
return connv2ray
|
return connv2ray
|
||||||
}().(*wsconn), nil
|
}(), nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,15 +8,14 @@ import (
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/gorilla/websocket"
|
||||||
|
"v2ray.com/core/app/log"
|
||||||
"v2ray.com/core/common"
|
"v2ray.com/core/common"
|
||||||
"v2ray.com/core/common/errors"
|
"v2ray.com/core/common/errors"
|
||||||
"v2ray.com/core/app/log"
|
|
||||||
v2net "v2ray.com/core/common/net"
|
v2net "v2ray.com/core/common/net"
|
||||||
"v2ray.com/core/transport/internet"
|
"v2ray.com/core/transport/internet"
|
||||||
v2tls "v2ray.com/core/transport/internet/tls"
|
|
||||||
|
|
||||||
"github.com/gorilla/websocket"
|
|
||||||
"v2ray.com/core/transport/internet/internal"
|
"v2ray.com/core/transport/internet/internal"
|
||||||
|
v2tls "v2ray.com/core/transport/internet/tls"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -68,7 +67,7 @@ func ListenWS(address v2net.Address, port v2net.Port, options internet.ListenOpt
|
||||||
|
|
||||||
func (wsl *WSListener) listenws(address v2net.Address, port v2net.Port) error {
|
func (wsl *WSListener) listenws(address v2net.Address, port v2net.Port) error {
|
||||||
http.HandleFunc("/"+wsl.config.Path, func(w http.ResponseWriter, r *http.Request) {
|
http.HandleFunc("/"+wsl.config.Path, func(w http.ResponseWriter, r *http.Request) {
|
||||||
con, err := wsl.converttovws(w, r)
|
conn, err := wsl.converttovws(w, r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warning("WebSocket|Listener: Failed to convert connection: ", err)
|
log.Warning("WebSocket|Listener: Failed to convert connection: ", err)
|
||||||
return
|
return
|
||||||
|
@ -76,17 +75,17 @@ func (wsl *WSListener) listenws(address v2net.Address, port v2net.Port) error {
|
||||||
|
|
||||||
select {
|
select {
|
||||||
case wsl.awaitingConns <- &ConnectionWithError{
|
case wsl.awaitingConns <- &ConnectionWithError{
|
||||||
conn: con,
|
conn: conn,
|
||||||
}:
|
}:
|
||||||
default:
|
default:
|
||||||
if con != nil {
|
if conn != nil {
|
||||||
con.Close()
|
conn.Close()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
})
|
})
|
||||||
|
|
||||||
errchan := make(chan error)
|
errchan := make(chan error, 1)
|
||||||
|
|
||||||
listenerfunc := func() error {
|
listenerfunc := func() error {
|
||||||
ol, err := net.Listen("tcp", address.String()+":"+strconv.Itoa(int(port.Value())))
|
ol, err := net.Listen("tcp", address.String()+":"+strconv.Itoa(int(port.Value())))
|
||||||
|
@ -114,14 +113,13 @@ func (wsl *WSListener) listenws(address v2net.Address, port v2net.Port) error {
|
||||||
go func() {
|
go func() {
|
||||||
err := listenerfunc()
|
err := listenerfunc()
|
||||||
errchan <- err
|
errchan <- err
|
||||||
return
|
|
||||||
}()
|
}()
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
select {
|
select {
|
||||||
case err = <-errchan:
|
case err = <-errchan:
|
||||||
case <-time.After(time.Second * 2):
|
case <-time.After(time.Second * 5):
|
||||||
//Should this listen fail after 2 sec, it could gone untracked.
|
//Should this listen fail after 5 sec, it could gone untracked.
|
||||||
}
|
}
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -158,7 +156,7 @@ func (v *WSListener) Accept() (internet.Connection, error) {
|
||||||
if connErr.err != nil {
|
if connErr.err != nil {
|
||||||
return nil, connErr.err
|
return nil, connErr.err
|
||||||
}
|
}
|
||||||
return internal.NewConnection(internal.ConnectionID{}, connErr.conn.(*wsconn), v, internal.ReuseConnection(v.config.IsConnectionReuse())), nil
|
return internal.NewConnection(internal.ConnectionID{}, connErr.conn, v, internal.ReuseConnection(v.config.IsConnectionReuse())), nil
|
||||||
case <-time.After(time.Second * 2):
|
case <-time.After(time.Second * 2):
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,18 +8,16 @@ import (
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/gorilla/websocket"
|
"github.com/gorilla/websocket"
|
||||||
"v2ray.com/core/common/errors"
|
|
||||||
"v2ray.com/core/app/log"
|
"v2ray.com/core/app/log"
|
||||||
|
"v2ray.com/core/common/errors"
|
||||||
)
|
)
|
||||||
|
|
||||||
type wsconn struct {
|
type wsconn struct {
|
||||||
wsc *websocket.Conn
|
wsc *websocket.Conn
|
||||||
readBuffer *bufio.Reader
|
readBuffer *bufio.Reader
|
||||||
connClosing bool
|
connClosing bool
|
||||||
reusable bool
|
|
||||||
rlock *sync.Mutex
|
rlock *sync.Mutex
|
||||||
wlock *sync.Mutex
|
wlock *sync.Mutex
|
||||||
config *Config
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ws *wsconn) Read(b []byte) (n int, err error) {
|
func (ws *wsconn) Read(b []byte) (n int, err error) {
|
||||||
|
@ -42,7 +40,7 @@ func (ws *wsconn) read(b []byte) (n int, err error) {
|
||||||
func (ws *wsconn) getNewReadBuffer() error {
|
func (ws *wsconn) getNewReadBuffer() error {
|
||||||
_, r, err := ws.wsc.NextReader()
|
_, r, err := ws.wsc.NextReader()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warning("WS transport: ws connection NewFrameReader return ", err)
|
log.Warning("WebSocket|Connection: Failed to get reader.", err)
|
||||||
ws.connClosing = true
|
ws.connClosing = true
|
||||||
ws.Close()
|
ws.Close()
|
||||||
return err
|
return err
|
||||||
|
@ -90,7 +88,7 @@ func (ws *wsconn) Write(b []byte) (n int, err error) {
|
||||||
func (ws *wsconn) write(b []byte) (n int, err error) {
|
func (ws *wsconn) write(b []byte) (n int, err error) {
|
||||||
wr, err := ws.wsc.NextWriter(websocket.BinaryMessage)
|
wr, err := ws.wsc.NextWriter(websocket.BinaryMessage)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warning("WS transport: ws connection NewFrameReader return ", err)
|
log.Warning("WebSocket|Connection: Failed to get writer.", err)
|
||||||
ws.connClosing = true
|
ws.connClosing = true
|
||||||
ws.Close()
|
ws.Close()
|
||||||
return 0, err
|
return 0, err
|
||||||
|
@ -150,14 +148,6 @@ func (ws *wsconn) setup() {
|
||||||
ws.pingPong()
|
ws.pingPong()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ws *wsconn) Reusable() bool {
|
|
||||||
return ws.config.IsConnectionReuse() && ws.reusable && !ws.connClosing
|
|
||||||
}
|
|
||||||
|
|
||||||
func (ws *wsconn) SetReusable(reusable bool) {
|
|
||||||
ws.reusable = reusable
|
|
||||||
}
|
|
||||||
|
|
||||||
func (ws *wsconn) pingPong() {
|
func (ws *wsconn) pingPong() {
|
||||||
pongRcv := make(chan int, 1)
|
pongRcv := make(chan int, 1)
|
||||||
ws.wsc.SetPongHandler(func(data string) error {
|
ws.wsc.SetPongHandler(func(data string) error {
|
||||||
|
|
Loading…
Reference in New Issue