http.go 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  1. package http
  2. //go:generate errorgen
  3. import (
  4. "bufio"
  5. "bytes"
  6. "context"
  7. "io"
  8. "net"
  9. "net/http"
  10. "strings"
  11. "time"
  12. "v2ray.com/core/common"
  13. "v2ray.com/core/common/buf"
  14. )
  15. const (
  16. // CRLF is the line ending in HTTP header
  17. CRLF = "\r\n"
  18. // ENDING is the double line ending between HTTP header and body.
  19. ENDING = CRLF + CRLF
  20. // max length of HTTP header. Safety precaution for DDoS attack.
  21. maxHeaderLength = 8192
  22. )
  23. var (
  24. ErrHeaderToLong = newError("Header too long.")
  25. ErrHeaderMisMatch = newError("Header Mismatch.")
  26. )
  27. type Reader interface {
  28. Read(io.Reader) (*buf.Buffer, error)
  29. }
  30. type Writer interface {
  31. Write(io.Writer) error
  32. }
  33. type NoOpReader struct{}
  34. func (NoOpReader) Read(io.Reader) (*buf.Buffer, error) {
  35. return nil, nil
  36. }
  37. type NoOpWriter struct{}
  38. func (NoOpWriter) Write(io.Writer) error {
  39. return nil
  40. }
  41. type HeaderReader struct {
  42. req *http.Request
  43. expectedHeader *RequestConfig
  44. }
  45. func (h *HeaderReader) ExpectThisRequest(expectedHeader *RequestConfig) *HeaderReader {
  46. h.expectedHeader = expectedHeader
  47. return h
  48. }
  49. func (h *HeaderReader) Read(reader io.Reader) (*buf.Buffer, error) {
  50. buffer := buf.New()
  51. totalBytes := int32(0)
  52. endingDetected := false
  53. var headerBuf bytes.Buffer
  54. for totalBytes < maxHeaderLength {
  55. _, err := buffer.ReadFrom(reader)
  56. if err != nil {
  57. buffer.Release()
  58. return nil, err
  59. }
  60. if n := bytes.Index(buffer.Bytes(), []byte(ENDING)); n != -1 {
  61. headerBuf.Write(buffer.BytesRange(0, int32(n+len(ENDING))))
  62. buffer.Advance(int32(n + len(ENDING)))
  63. endingDetected = true
  64. break
  65. }
  66. lenEnding := int32(len(ENDING))
  67. if buffer.Len() >= lenEnding {
  68. totalBytes += buffer.Len() - lenEnding
  69. headerBuf.Write(buffer.BytesRange(0, buffer.Len()-lenEnding))
  70. leftover := buffer.BytesFrom(-lenEnding)
  71. buffer.Clear()
  72. copy(buffer.Extend(lenEnding), leftover)
  73. }
  74. }
  75. if !endingDetected {
  76. buffer.Release()
  77. return nil, ErrHeaderToLong
  78. }
  79. if h.expectedHeader == nil {
  80. if buffer.IsEmpty() {
  81. buffer.Release()
  82. return nil, nil
  83. }
  84. return buffer, nil
  85. }
  86. //Parse the request
  87. if req, err := readRequest(bufio.NewReader(bytes.NewReader(headerBuf.Bytes())), false); err != nil {
  88. return nil, err
  89. } else {
  90. h.req = req
  91. }
  92. //Check req
  93. path := h.req.URL.Path
  94. hasThisUri := false
  95. for _, u := range h.expectedHeader.Uri {
  96. if u == path {
  97. hasThisUri = true
  98. }
  99. }
  100. if hasThisUri == false {
  101. return nil, ErrHeaderMisMatch
  102. }
  103. if buffer.IsEmpty() {
  104. buffer.Release()
  105. return nil, nil
  106. }
  107. return buffer, nil
  108. }
  109. type HeaderWriter struct {
  110. header *buf.Buffer
  111. }
  112. func NewHeaderWriter(header *buf.Buffer) *HeaderWriter {
  113. return &HeaderWriter{
  114. header: header,
  115. }
  116. }
  117. func (w *HeaderWriter) Write(writer io.Writer) error {
  118. if w.header == nil {
  119. return nil
  120. }
  121. err := buf.WriteAllBytes(writer, w.header.Bytes())
  122. w.header.Release()
  123. w.header = nil
  124. return err
  125. }
  126. type HttpConn struct {
  127. net.Conn
  128. readBuffer *buf.Buffer
  129. oneTimeReader Reader
  130. oneTimeWriter Writer
  131. errorWriter Writer
  132. errorMismatchWriter Writer
  133. errorTooLongWriter Writer
  134. errReason error
  135. }
  136. func NewHttpConn(conn net.Conn, reader Reader, writer Writer, errorWriter Writer, errorMismatchWriter Writer, errorTooLongWriter Writer) *HttpConn {
  137. return &HttpConn{
  138. Conn: conn,
  139. oneTimeReader: reader,
  140. oneTimeWriter: writer,
  141. errorWriter: errorWriter,
  142. errorMismatchWriter: errorMismatchWriter,
  143. errorTooLongWriter: errorTooLongWriter,
  144. }
  145. }
  146. func (c *HttpConn) Read(b []byte) (int, error) {
  147. if c.oneTimeReader != nil {
  148. buffer, err := c.oneTimeReader.Read(c.Conn)
  149. if err != nil {
  150. c.errReason = err
  151. return 0, err
  152. }
  153. c.readBuffer = buffer
  154. c.oneTimeReader = nil
  155. }
  156. if !c.readBuffer.IsEmpty() {
  157. nBytes, _ := c.readBuffer.Read(b)
  158. if c.readBuffer.IsEmpty() {
  159. c.readBuffer.Release()
  160. c.readBuffer = nil
  161. }
  162. return nBytes, nil
  163. }
  164. return c.Conn.Read(b)
  165. }
  166. // Write implements io.Writer.
  167. func (c *HttpConn) Write(b []byte) (int, error) {
  168. if c.oneTimeWriter != nil {
  169. err := c.oneTimeWriter.Write(c.Conn)
  170. c.oneTimeWriter = nil
  171. if err != nil {
  172. return 0, err
  173. }
  174. }
  175. return c.Conn.Write(b)
  176. }
  177. // Close implements net.Conn.Close().
  178. func (c *HttpConn) Close() error {
  179. if c.oneTimeWriter != nil && c.errorWriter != nil {
  180. // Connection is being closed but header wasn't sent. This means the client request
  181. // is probably not valid. Sending back a server error header in this case.
  182. //Write response based on error reason
  183. if c.errReason == ErrHeaderMisMatch {
  184. c.errorMismatchWriter.Write(c.Conn)
  185. } else if c.errReason == ErrHeaderToLong {
  186. c.errorTooLongWriter.Write(c.Conn)
  187. } else {
  188. c.errorWriter.Write(c.Conn)
  189. }
  190. }
  191. return c.Conn.Close()
  192. }
  193. func formResponseHeader(config *ResponseConfig) *HeaderWriter {
  194. header := buf.New()
  195. common.Must2(header.WriteString(strings.Join([]string{config.GetFullVersion(), config.GetStatusValue().Code, config.GetStatusValue().Reason}, " ")))
  196. common.Must2(header.WriteString(CRLF))
  197. headers := config.PickHeaders()
  198. for _, h := range headers {
  199. common.Must2(header.WriteString(h))
  200. common.Must2(header.WriteString(CRLF))
  201. }
  202. if !config.HasHeader("Date") {
  203. common.Must2(header.WriteString("Date: "))
  204. common.Must2(header.WriteString(time.Now().Format(http.TimeFormat)))
  205. common.Must2(header.WriteString(CRLF))
  206. }
  207. common.Must2(header.WriteString(CRLF))
  208. return &HeaderWriter{
  209. header: header,
  210. }
  211. }
  212. type HttpAuthenticator struct {
  213. config *Config
  214. }
  215. func (a HttpAuthenticator) GetClientWriter() *HeaderWriter {
  216. header := buf.New()
  217. config := a.config.Request
  218. common.Must2(header.WriteString(strings.Join([]string{config.GetMethodValue(), config.PickUri(), config.GetFullVersion()}, " ")))
  219. common.Must2(header.WriteString(CRLF))
  220. headers := config.PickHeaders()
  221. for _, h := range headers {
  222. common.Must2(header.WriteString(h))
  223. common.Must2(header.WriteString(CRLF))
  224. }
  225. common.Must2(header.WriteString(CRLF))
  226. return &HeaderWriter{
  227. header: header,
  228. }
  229. }
  230. func (a HttpAuthenticator) GetServerWriter() *HeaderWriter {
  231. return formResponseHeader(a.config.Response)
  232. }
  233. func (a HttpAuthenticator) Client(conn net.Conn) net.Conn {
  234. if a.config.Request == nil && a.config.Response == nil {
  235. return conn
  236. }
  237. var reader Reader = NoOpReader{}
  238. if a.config.Request != nil {
  239. reader = new(HeaderReader)
  240. }
  241. var writer Writer = NoOpWriter{}
  242. if a.config.Response != nil {
  243. writer = a.GetClientWriter()
  244. }
  245. return NewHttpConn(conn, reader, writer, NoOpWriter{}, NoOpWriter{}, NoOpWriter{})
  246. }
  247. func (a HttpAuthenticator) Server(conn net.Conn) net.Conn {
  248. if a.config.Request == nil && a.config.Response == nil {
  249. return conn
  250. }
  251. return NewHttpConn(conn, new(HeaderReader).ExpectThisRequest(a.config.Request), a.GetServerWriter(),
  252. formResponseHeader(resp400),
  253. formResponseHeader(resp404),
  254. formResponseHeader(resp400))
  255. }
  256. func NewHttpAuthenticator(ctx context.Context, config *Config) (HttpAuthenticator, error) {
  257. return HttpAuthenticator{
  258. config: config,
  259. }, nil
  260. }
  261. func init() {
  262. common.Must(common.RegisterConfig((*Config)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {
  263. return NewHttpAuthenticator(ctx, config.(*Config))
  264. }))
  265. }