2020-11-25 19:01:53 +08:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2024-09-14 00:26:36 +08:00
|
|
|
"context"
|
2020-11-25 19:01:53 +08:00
|
|
|
"errors"
|
|
|
|
"strings"
|
|
|
|
|
2020-12-04 09:36:16 +08:00
|
|
|
"github.com/xtls/xray-core/common"
|
|
|
|
"github.com/xtls/xray-core/common/net"
|
2024-09-14 00:26:36 +08:00
|
|
|
"github.com/xtls/xray-core/common/session"
|
2020-11-25 19:01:53 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
type version byte
|
|
|
|
|
|
|
|
const (
|
|
|
|
HTTP1 version = iota
|
|
|
|
HTTP2
|
|
|
|
)
|
|
|
|
|
|
|
|
type SniffHeader struct {
|
|
|
|
version version
|
|
|
|
host string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *SniffHeader) Protocol() string {
|
|
|
|
switch h.version {
|
|
|
|
case HTTP1:
|
|
|
|
return "http1"
|
|
|
|
case HTTP2:
|
|
|
|
return "http2"
|
|
|
|
default:
|
|
|
|
return "unknown"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *SniffHeader) Domain() string {
|
|
|
|
return h.host
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
methods = [...]string{"get", "post", "head", "put", "delete", "options", "connect"}
|
|
|
|
|
|
|
|
errNotHTTPMethod = errors.New("not an HTTP method")
|
|
|
|
)
|
|
|
|
|
|
|
|
func beginWithHTTPMethod(b []byte) error {
|
|
|
|
for _, m := range &methods {
|
|
|
|
if len(b) >= len(m) && strings.EqualFold(string(b[:len(m)]), m) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(b) < len(m) {
|
|
|
|
return common.ErrNoClue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return errNotHTTPMethod
|
|
|
|
}
|
|
|
|
|
2024-09-14 00:26:36 +08:00
|
|
|
func SniffHTTP(b []byte, c context.Context) (*SniffHeader, error) {
|
|
|
|
content := session.ContentFromContext(c)
|
|
|
|
ShouldSniffAttr := true
|
|
|
|
// If content.Attributes have information, that means it comes from HTTP inbound PlainHTTP mode.
|
|
|
|
// It will set attributes, so skip it.
|
2024-10-30 10:26:43 +08:00
|
|
|
if content == nil || content.AttributeLen() != 0 {
|
2024-09-14 00:26:36 +08:00
|
|
|
ShouldSniffAttr = false
|
|
|
|
}
|
2020-11-25 19:01:53 +08:00
|
|
|
if err := beginWithHTTPMethod(b); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
sh := &SniffHeader{
|
|
|
|
version: HTTP1,
|
|
|
|
}
|
|
|
|
|
|
|
|
headers := bytes.Split(b, []byte{'\n'})
|
|
|
|
for i := 1; i < len(headers); i++ {
|
|
|
|
header := headers[i]
|
|
|
|
if len(header) == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
parts := bytes.SplitN(header, []byte{':'}, 2)
|
|
|
|
if len(parts) != 2 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
key := strings.ToLower(string(parts[0]))
|
2024-09-14 00:26:36 +08:00
|
|
|
value := string(bytes.TrimSpace(parts[1]))
|
|
|
|
if ShouldSniffAttr {
|
|
|
|
content.SetAttribute(key, value) // Put header in attribute
|
|
|
|
}
|
2020-11-25 19:01:53 +08:00
|
|
|
if key == "host" {
|
2024-09-14 00:26:36 +08:00
|
|
|
rawHost := strings.ToLower(value)
|
2020-11-25 19:01:53 +08:00
|
|
|
dest, err := ParseHost(rawHost, net.Port(80))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
sh.host = dest.Address.String()
|
|
|
|
}
|
|
|
|
}
|
2024-09-14 00:26:36 +08:00
|
|
|
// Parse request line
|
|
|
|
// Request line is like this
|
|
|
|
// "GET /homo/114514 HTTP/1.1"
|
|
|
|
if len(headers) > 0 && ShouldSniffAttr {
|
|
|
|
RequestLineParts := bytes.Split(headers[0], []byte{' '})
|
|
|
|
if len(RequestLineParts) == 3 {
|
|
|
|
content.SetAttribute(":method", string(RequestLineParts[0]))
|
|
|
|
content.SetAttribute(":path", string(RequestLineParts[1]))
|
|
|
|
}
|
|
|
|
}
|
2020-11-25 19:01:53 +08:00
|
|
|
|
|
|
|
if len(sh.host) > 0 {
|
|
|
|
return sh, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, common.ErrNoClue
|
|
|
|
}
|