mirror of
https://github.com/v2fly/v2ray-core.git
synced 2024-12-21 09:36:34 -05:00
Optimize HTTP tunnel setup in TFO environment
This commit is contained in:
parent
197fe15d5a
commit
a5caa01cb6
@ -4,9 +4,9 @@ package http
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
|
"io"
|
||||||
"context"
|
"context"
|
||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
"io"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
@ -93,9 +93,7 @@ func (c *Client) Process(ctx context.Context, link *transport.Link, dialer inter
|
|||||||
p = c.policyManager.ForLevel(user.Level)
|
p = c.policyManager.ForLevel(user.Level)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := setUpHTTPTunnel(conn, &destination, user); err != nil {
|
conn = setUpHTTPTunnel(conn, &destination, user)
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx, cancel := context.WithCancel(ctx)
|
ctx, cancel := context.WithCancel(ctx)
|
||||||
timer := signal.CancelAfterInactivity(ctx, cancel, p.Timeouts.ConnectionIdle)
|
timer := signal.CancelAfterInactivity(ctx, cancel, p.Timeouts.ConnectionIdle)
|
||||||
@ -125,8 +123,13 @@ 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(writer io.Writer, destination *net.Destination, user *protocol.MemoryUser) error {
|
func setUpHTTPTunnel(conn internet.Connection, destination *net.Destination, user *protocol.MemoryUser) *tunnelConn {
|
||||||
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")
|
||||||
@ -140,11 +143,43 @@ func setUpHTTPTunnel(writer io.Writer, destination *net.Destination, user *proto
|
|||||||
|
|
||||||
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")
|
||||||
if err := buf.WriteAllBytes(writer, b.Bytes()); err != nil {
|
return &tunnelConn {
|
||||||
return err
|
Connection: conn,
|
||||||
|
header: b,
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return nil
|
func (c *tunnelConn) Write(b []byte) (n int, err error) {
|
||||||
|
if c.header == nil {
|
||||||
|
return c.Connection.Write(b)
|
||||||
|
}
|
||||||
|
buffer := c.header
|
||||||
|
lenheader := c.header.Len()
|
||||||
|
// Concate header and b
|
||||||
|
_, err = buffer.Write(b)
|
||||||
|
if err != nil {
|
||||||
|
c.header.Resize(0, lenheader)
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
// Write buffer
|
||||||
|
nc, err := io.Copy(c.Connection, buffer)
|
||||||
|
if int32(nc) < lenheader {
|
||||||
|
c.header.Resize(int32(nc), lenheader)
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
c.header.Release()
|
||||||
|
c.header = nil
|
||||||
|
n = int(nc) - int(lenheader)
|
||||||
|
if err != nil {
|
||||||
|
return n, err
|
||||||
|
}
|
||||||
|
// Write trailing bytes
|
||||||
|
if n < len(b) {
|
||||||
|
var nw int
|
||||||
|
nw, err = c.Connection.Write(b[:n])
|
||||||
|
n += nw
|
||||||
|
}
|
||||||
|
return n, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
|
Loading…
Reference in New Issue
Block a user