request.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. package detection
  2. import (
  3. "bytes"
  4. "errors"
  5. "fmt"
  6. "io"
  7. "net"
  8. "net/http"
  9. "strconv"
  10. "github.com/chaitin/t1k-go/misc"
  11. )
  12. type Request interface {
  13. Header() ([]byte, error)
  14. Body() (uint32, io.ReadCloser, error)
  15. Extra() ([]byte, error)
  16. }
  17. type HttpRequest struct {
  18. req *http.Request
  19. dc *DetectionContext // this is optional
  20. }
  21. func MakeHttpRequest(req *http.Request) *HttpRequest {
  22. return &HttpRequest{
  23. req: req,
  24. }
  25. }
  26. func MakeHttpRequestInCtx(req *http.Request, dc *DetectionContext) *HttpRequest {
  27. ret := &HttpRequest{
  28. req: req,
  29. dc: dc,
  30. }
  31. dc.Request = ret
  32. if dc.ReqBeginTime == 0 {
  33. dc.ReqBeginTime = misc.Now()
  34. }
  35. return ret
  36. }
  37. func (r *HttpRequest) GetUpstreamAddress() (string, error) {
  38. if r.req.Host == "" {
  39. return "", errors.New("empty Host in request")
  40. }
  41. host, _, err := net.SplitHostPort(r.req.Host)
  42. if err != nil {
  43. return r.req.Host, nil // OK; there probably was no port
  44. }
  45. return host, nil
  46. }
  47. func (r *HttpRequest) GetUpstreamPort() (uint16, error) {
  48. _, port, err := net.SplitHostPort(r.req.Host)
  49. if err != nil {
  50. if r.req.TLS != nil {
  51. return 443, nil
  52. } else {
  53. return 80, nil
  54. }
  55. }
  56. if portNum, err := strconv.Atoi(port); err == nil {
  57. return uint16(portNum), nil
  58. }
  59. return 0, errors.New("wrong value of port")
  60. }
  61. func (r *HttpRequest) GetRemoteIP() (string, error) {
  62. host, _, err := net.SplitHostPort(r.req.RemoteAddr)
  63. if err != nil {
  64. return r.req.RemoteAddr, nil
  65. }
  66. return host, nil
  67. }
  68. func (r *HttpRequest) GetRemotePort() (uint16, error) {
  69. _, port, _ := net.SplitHostPort(r.req.RemoteAddr)
  70. if portNum, err := strconv.Atoi(port); err == nil {
  71. return uint16(portNum), nil
  72. }
  73. return 0, errors.New("wrong value of port")
  74. }
  75. func (r *HttpRequest) Header() ([]byte, error) {
  76. var buf bytes.Buffer
  77. proto := r.req.Proto
  78. if r.dc != nil {
  79. if r.dc.Protocol != "" {
  80. proto = r.dc.Protocol
  81. } else {
  82. r.dc.Protocol = proto
  83. }
  84. }
  85. startLine := fmt.Sprintf("%s %s %s\r\n", r.req.Method, r.req.URL.RequestURI(), proto)
  86. _, err := buf.Write([]byte(startLine))
  87. if err != nil {
  88. return nil, err
  89. }
  90. _, err = buf.Write([]byte(fmt.Sprintf("Host: %s\r\n", r.req.Host)))
  91. if err != nil {
  92. return nil, err
  93. }
  94. err = r.req.Header.Write(&buf)
  95. if err != nil {
  96. return nil, err
  97. }
  98. _, err = buf.Write([]byte("\r\n"))
  99. if err != nil {
  100. return nil, err
  101. }
  102. return buf.Bytes(), nil
  103. }
  104. func (r *HttpRequest) Body() (uint32, io.ReadCloser, error) {
  105. bodyBytes, err := io.ReadAll(r.req.Body)
  106. if err != nil {
  107. return 0, nil, err
  108. }
  109. r.req.Body = io.NopCloser(bytes.NewReader(bodyBytes))
  110. return uint32(len(bodyBytes)), io.NopCloser(bytes.NewReader(bodyBytes)), nil
  111. }
  112. func (r *HttpRequest) Extra() ([]byte, error) {
  113. if r.dc == nil {
  114. return PlaceholderRequestExtra(misc.GenUUID()), nil
  115. }
  116. return GenRequestExtra(r.dc), nil
  117. }