sniff.go 1.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. package http
  2. import (
  3. "bytes"
  4. "errors"
  5. "strings"
  6. "v2ray.com/core"
  7. )
  8. type version byte
  9. const (
  10. HTTP1 version = iota
  11. HTTP2
  12. )
  13. type SniffHeader struct {
  14. version version
  15. host string
  16. }
  17. func (h *SniffHeader) Protocol() string {
  18. switch h.version {
  19. case HTTP1:
  20. return "http1"
  21. case HTTP2:
  22. return "http2"
  23. default:
  24. return "unknown"
  25. }
  26. }
  27. func (h *SniffHeader) Domain() string {
  28. return h.host
  29. }
  30. var (
  31. methods = [...]string{"get", "post", "head", "put", "delete", "options", "connect"}
  32. errNotHTTPMethod = errors.New("not an HTTP method")
  33. )
  34. func beginWithHTTPMethod(b []byte) error {
  35. for _, m := range &methods {
  36. if len(b) >= len(m) && strings.ToLower(string(b[:len(m)])) == m {
  37. return nil
  38. }
  39. if len(b) < len(m) {
  40. return core.ErrNoClue
  41. }
  42. }
  43. return errNotHTTPMethod
  44. }
  45. func SniffHTTP(b []byte) (*SniffHeader, error) {
  46. if err := beginWithHTTPMethod(b); err != nil {
  47. return nil, err
  48. }
  49. sh := &SniffHeader{
  50. version: HTTP1,
  51. }
  52. headers := bytes.Split(b, []byte{'\n'})
  53. for i := 1; i < len(headers); i++ {
  54. header := headers[i]
  55. if len(header) == 0 {
  56. break
  57. }
  58. parts := bytes.SplitN(header, []byte{':'}, 2)
  59. if len(parts) != 2 {
  60. continue
  61. }
  62. key := strings.ToLower(string(parts[0]))
  63. value := strings.ToLower(string(bytes.Trim(parts[1], " ")))
  64. if key == "host" {
  65. domain := strings.Split(value, ":")
  66. sh.host = strings.TrimSpace(domain[0])
  67. }
  68. }
  69. if len(sh.host) > 0 {
  70. return sh, nil
  71. }
  72. return nil, core.ErrNoClue
  73. }