move parseHost to http protocol

pull/1470/head^2
Darien Raymond 2018-12-10 23:08:16 +01:00
parent 867135d85a
commit 7e37d141e2
No known key found for this signature in database
GPG Key ID: 7251FFA14BB18169
4 changed files with 70 additions and 25 deletions

View File

@ -2,6 +2,7 @@ package http
import ( import (
"net/http" "net/http"
"strconv"
"strings" "strings"
"v2ray.com/core/common/net" "v2ray.com/core/common/net"
@ -42,3 +43,24 @@ func RemoveHopByHopHeaders(header http.Header) {
header.Del(strings.TrimSpace(h)) header.Del(strings.TrimSpace(h))
} }
} }
// ParseHost splits host and port from a raw string. Default port is used when raw string doesn't contain port.
func ParseHost(rawHost string, defaultPort net.Port) (net.Destination, error) {
port := defaultPort
host, rawPort, err := net.SplitHostPort(rawHost)
if err != nil {
if addrError, ok := err.(*net.AddrError); ok && strings.Contains(addrError.Err, "missing port") {
host = rawHost
} else {
return net.Destination{}, err
}
} else if len(rawPort) > 0 {
intPort, err := strconv.Atoi(rawPort)
if err != nil {
return net.Destination{}, err
}
port = net.Port(intPort)
}
return net.TCPDestination(net.ParseAddress(host), port), nil
}

View File

@ -6,6 +6,8 @@ import (
"strings" "strings"
"testing" "testing"
"v2ray.com/core/common/net"
. "v2ray.com/core/common/protocol/http" . "v2ray.com/core/common/protocol/http"
. "v2ray.com/ext/assert" . "v2ray.com/ext/assert"
) )
@ -53,3 +55,41 @@ Accept-Language: de,en;q=0.7,en-us;q=0.3
assert(req.Header.Get("Proxy-Connection"), IsEmpty) assert(req.Header.Get("Proxy-Connection"), IsEmpty)
assert(req.Header.Get("Proxy-Authenticate"), IsEmpty) assert(req.Header.Get("Proxy-Authenticate"), IsEmpty)
} }
func TestParseHost(t *testing.T) {
testCases := []struct {
RawHost string
DefaultPort net.Port
Destination net.Destination
Error bool
}{
{
RawHost: "v2ray.com:80",
DefaultPort: 443,
Destination: net.TCPDestination(net.DomainAddress("v2ray.com"), 80),
},
{
RawHost: "tls.v2ray.com",
DefaultPort: 443,
Destination: net.TCPDestination(net.DomainAddress("tls.v2ray.com"), 443),
},
{
RawHost: "[2401:1bc0:51f0:ec08::1]:80",
DefaultPort: 443,
Destination: net.TCPDestination(net.ParseAddress("[2401:1bc0:51f0:ec08::1]"), 80),
},
}
for _, testCase := range testCases {
dest, err := ParseHost(testCase.RawHost, testCase.DefaultPort)
if testCase.Error {
if err == nil {
t.Error("for test case: ", testCase.RawHost, " expected error, but actually nil")
}
} else {
if dest != testCase.Destination {
t.Error("for test case: ", testCase.RawHost, " expected host: ", testCase.Destination.String(), " but got ", dest.String())
}
}
}
}

View File

@ -6,6 +6,7 @@ import (
"strings" "strings"
"v2ray.com/core/common" "v2ray.com/core/common"
"v2ray.com/core/common/net"
) )
type version byte type version byte
@ -75,10 +76,13 @@ func SniffHTTP(b []byte) (*SniffHeader, error) {
continue continue
} }
key := strings.ToLower(string(parts[0])) key := strings.ToLower(string(parts[0]))
value := strings.ToLower(string(bytes.Trim(parts[1], " ")))
if key == "host" { if key == "host" {
domain := strings.Split(value, ":") rawHost := strings.ToLower(string(bytes.TrimSpace(parts[1])))
sh.host = strings.TrimSpace(domain[0]) dest, err := ParseHost(rawHost, net.Port(80))
if err != nil {
return nil, err
}
sh.host = dest.Address.String()
} }
} }

View File

@ -6,7 +6,6 @@ import (
"encoding/base64" "encoding/base64"
"io" "io"
"net/http" "net/http"
"strconv"
"strings" "strings"
"time" "time"
@ -56,26 +55,6 @@ func (*Server) Network() []net.Network {
return []net.Network{net.Network_TCP} return []net.Network{net.Network_TCP}
} }
func parseHost(rawHost string, defaultPort net.Port) (net.Destination, error) {
port := defaultPort
host, rawPort, err := net.SplitHostPort(rawHost)
if err != nil {
if addrError, ok := err.(*net.AddrError); ok && strings.Contains(addrError.Err, "missing port") {
host = rawHost
} else {
return net.Destination{}, err
}
} else if len(rawPort) > 0 {
intPort, err := strconv.Atoi(rawPort)
if err != nil {
return net.Destination{}, err
}
port = net.Port(intPort)
}
return net.TCPDestination(net.ParseAddress(host), port), nil
}
func isTimeout(err error) bool { func isTimeout(err error) bool {
nerr, ok := errors.Cause(err).(net.Error) nerr, ok := errors.Cause(err).(net.Error)
return ok && nerr.Timeout() return ok && nerr.Timeout()
@ -139,7 +118,7 @@ Start:
if len(host) == 0 { if len(host) == 0 {
host = request.URL.Host host = request.URL.Host
} }
dest, err := parseHost(host, defaultPort) dest, err := http_proto.ParseHost(host, defaultPort)
if err != nil { if err != nil {
return newError("malformed proxy host: ", host).AtWarning().Base(err) return newError("malformed proxy host: ", host).AtWarning().Base(err)
} }