sniff.go 1.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  1. package http
  2. import (
  3. "bytes"
  4. "errors"
  5. "strings"
  6. "github.com/v2fly/v2ray-core/v5/common"
  7. "github.com/v2fly/v2ray-core/v5/common/net"
  8. )
  9. type version byte
  10. type SniffHeader struct {
  11. host string
  12. }
  13. func (h *SniffHeader) Protocol() string {
  14. return "http1"
  15. }
  16. func (h *SniffHeader) Domain() string {
  17. return h.host
  18. }
  19. var (
  20. // refer to https://pkg.go.dev/net/http@master#pkg-constants
  21. methods = [...]string{"get", "post", "head", "put", "delete", "options", "connect", "patch", "trace"}
  22. errNotHTTPMethod = errors.New("not an HTTP method")
  23. )
  24. func beginWithHTTPMethod(b []byte) error {
  25. for _, m := range &methods {
  26. if len(b) >= len(m) && strings.EqualFold(string(b[:len(m)]), m) {
  27. return nil
  28. }
  29. if len(b) < len(m) {
  30. return common.ErrNoClue
  31. }
  32. }
  33. return errNotHTTPMethod
  34. }
  35. func SniffHTTP(b []byte) (*SniffHeader, error) {
  36. if err := beginWithHTTPMethod(b); err != nil {
  37. return nil, err
  38. }
  39. sh := &SniffHeader{}
  40. headers := bytes.Split(b, []byte{'\n'})
  41. for i := 1; i < len(headers); i++ {
  42. header := headers[i]
  43. if len(header) == 0 {
  44. break
  45. }
  46. parts := bytes.SplitN(header, []byte{':'}, 2)
  47. if len(parts) != 2 {
  48. continue
  49. }
  50. key := strings.ToLower(string(parts[0]))
  51. if key == "host" {
  52. rawHost := strings.ToLower(string(bytes.TrimSpace(parts[1])))
  53. dest, err := ParseHost(rawHost, net.Port(80))
  54. if err != nil {
  55. return nil, err
  56. }
  57. sh.host = dest.Address.String()
  58. }
  59. }
  60. if len(sh.host) > 0 {
  61. return sh, nil
  62. }
  63. return nil, common.ErrNoClue
  64. }