http: send HTTP Connect if no content is written after 5ms.

pull/2531/head
Anonymous-Someneese 5 years ago committed by kslr
parent 82dee01833
commit 99cf401867

@ -4,11 +4,13 @@ package http
import ( import (
"bufio" "bufio"
"io"
"context" "context"
"encoding/base64" "encoding/base64"
"io"
"net/http" "net/http"
"strings" "strings"
"sync"
"time"
"v2ray.com/core" "v2ray.com/core"
"v2ray.com/core/common" "v2ray.com/core/common"
@ -123,13 +125,8 @@ func (c *Client) Process(ctx context.Context, link *transport.Link, dialer inter
return nil return nil
} }
type tunnelConn struct {
internet.Connection
header *buf.Buffer
}
// setUpHTTPTunnel will create a socket tunnel via HTTP CONNECT method // setUpHTTPTunnel will create a socket tunnel via HTTP CONNECT method
func setUpHTTPTunnel(conn internet.Connection, destination *net.Destination, user *protocol.MemoryUser) *tunnelConn { func setUpHTTPTunnel(conn internet.Connection, destination *net.Destination, user *protocol.MemoryUser) *tunConn {
var headers []string var headers []string
destNetAddr := destination.NetAddr() destNetAddr := destination.NetAddr()
headers = append(headers, "CONNECT "+destNetAddr+" HTTP/1.1") headers = append(headers, "CONNECT "+destNetAddr+" HTTP/1.1")
@ -143,40 +140,59 @@ func setUpHTTPTunnel(conn internet.Connection, destination *net.Destination, use
b := buf.New() b := buf.New()
b.WriteString(strings.Join(headers, "\r\n") + "\r\n\r\n") b.WriteString(strings.Join(headers, "\r\n") + "\r\n\r\n")
return &tunnelConn { return newTunConn(conn, b, 5 * time.Millisecond)
}
// tunConn is a connection that writes header before content,
// the header will be written during the next Write call or after
// specified delay.
type tunConn struct {
internet.Connection
header *buf.Buffer
once sync.Once
timer *time.Timer
}
func newTunConn(conn internet.Connection, header *buf.Buffer, delay time.Duration) *tunConn {
tc := &tunConn{
Connection: conn, Connection: conn,
header: b, header: header,
}
if delay > 0 {
tc.timer = time.AfterFunc(delay, func() {
tc.Write([]byte{})
})
} }
return tc
} }
func (c *tunnelConn) Write(b []byte) (n int, err error) { func (c *tunConn) Write(b []byte) (n int, err error) {
// fallback to normal write if header is sent
if c.header == nil { if c.header == nil {
return c.Connection.Write(b) return c.Connection.Write(b)
} }
buffer := c.header // Prevent timer and writer race condition
lenheader := c.header.Len() c.once.Do(func() {
// Concate header and b if c.timer != nil {
_, err = buffer.Write(b) c.timer.Stop()
if err != nil { c.timer = nil
c.header.Resize(0, lenheader) }
return 0, err lenheader := c.header.Len()
} // Concate header and b
// Write buffer common.Must2(c.header.Write(b))
nc, err := io.Copy(c.Connection, buffer) // Write buffer
if int32(nc) < lenheader { var nc int64
c.header.Resize(int32(nc), lenheader) nc, err = io.Copy(c.Connection, c.header)
return 0, err c.header.Release()
} c.header = nil
c.header.Release() n = int(nc) - int(lenheader)
c.header = nil if n < 0 { n = 0 }
n = int(nc) - int(lenheader) b = b[n:]
if err != nil { })
return n, err // Write Trailing bytes
} if len(b) > 0 && err == nil {
// Write trailing bytes
if n < len(b) {
var nw int var nw int
nw, err = c.Connection.Write(b[:n]) nw, err = c.Connection.Write(b)
n += nw n += nw
} }
return n, err return n, err

Loading…
Cancel
Save