http.go 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  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. if _, err := readRequest(bufio.NewReader(bytes.NewReader(headerBuf.Bytes())), false); err != io.ErrUnexpectedEOF {
  74. return nil, err
  75. }
  76. }
  77. }
  78. if !endingDetected {
  79. buffer.Release()
  80. return nil, ErrHeaderToLong
  81. }
  82. if h.expectedHeader == nil {
  83. if buffer.IsEmpty() {
  84. buffer.Release()
  85. return nil, nil
  86. }
  87. return buffer, nil
  88. }
  89. //Parse the request
  90. if req, err := readRequest(bufio.NewReader(bytes.NewReader(headerBuf.Bytes())), false); err != nil {
  91. return nil, err
  92. } else {
  93. h.req = req
  94. }
  95. //Check req
  96. path := h.req.URL.Path
  97. hasThisUri := false
  98. for _, u := range h.expectedHeader.Uri {
  99. if u == path {
  100. hasThisUri = true
  101. }
  102. }
  103. if !hasThisUri {
  104. return nil, ErrHeaderMisMatch
  105. }
  106. if buffer.IsEmpty() {
  107. buffer.Release()
  108. return nil, nil
  109. }
  110. return buffer, nil
  111. }
  112. type HeaderWriter struct {
  113. header *buf.Buffer
  114. }
  115. func NewHeaderWriter(header *buf.Buffer) *HeaderWriter {
  116. return &HeaderWriter{
  117. header: header,
  118. }
  119. }
  120. func (w *HeaderWriter) Write(writer io.Writer) error {
  121. if w.header == nil {
  122. return nil
  123. }
  124. err := buf.WriteAllBytes(writer, w.header.Bytes())
  125. w.header.Release()
  126. w.header = nil
  127. return err
  128. }
  129. type HttpConn struct {
  130. net.Conn
  131. readBuffer *buf.Buffer
  132. oneTimeReader Reader
  133. oneTimeWriter Writer
  134. errorWriter Writer
  135. errorMismatchWriter Writer
  136. errorTooLongWriter Writer
  137. errReason error
  138. }
  139. func NewHttpConn(conn net.Conn, reader Reader, writer Writer, errorWriter Writer, errorMismatchWriter Writer, errorTooLongWriter Writer) *HttpConn {
  140. return &HttpConn{
  141. Conn: conn,
  142. oneTimeReader: reader,
  143. oneTimeWriter: writer,
  144. errorWriter: errorWriter,
  145. errorMismatchWriter: errorMismatchWriter,
  146. errorTooLongWriter: errorTooLongWriter,
  147. }
  148. }
  149. func (c *HttpConn) Read(b []byte) (int, error) {
  150. if c.oneTimeReader != nil {
  151. buffer, err := c.oneTimeReader.Read(c.Conn)
  152. if err != nil {
  153. c.errReason = err
  154. return 0, err
  155. }
  156. c.readBuffer = buffer
  157. c.oneTimeReader = nil
  158. }
  159. if !c.readBuffer.IsEmpty() {
  160. nBytes, _ := c.readBuffer.Read(b)
  161. if c.readBuffer.IsEmpty() {
  162. c.readBuffer.Release()
  163. c.readBuffer = nil
  164. }
  165. return nBytes, nil
  166. }
  167. return c.Conn.Read(b)
  168. }
  169. // Write implements io.Writer.
  170. func (c *HttpConn) Write(b []byte) (int, error) {
  171. if c.oneTimeWriter != nil {
  172. err := c.oneTimeWriter.Write(c.Conn)
  173. c.oneTimeWriter = nil
  174. if err != nil {
  175. return 0, err
  176. }
  177. }
  178. return c.Conn.Write(b)
  179. }
  180. // Close implements net.Conn.Close().
  181. func (c *HttpConn) Close() error {
  182. if c.oneTimeWriter != nil && c.errorWriter != nil {
  183. // Connection is being closed but header wasn't sent. This means the client request
  184. // is probably not valid. Sending back a server error header in this case.
  185. //Write response based on error reason
  186. if c.errReason == ErrHeaderMisMatch {
  187. c.errorMismatchWriter.Write(c.Conn)
  188. } else if c.errReason == ErrHeaderToLong {
  189. c.errorTooLongWriter.Write(c.Conn)
  190. } else {
  191. c.errorWriter.Write(c.Conn)
  192. }
  193. }
  194. return c.Conn.Close()
  195. }
  196. func formResponseHeader(config *ResponseConfig) *HeaderWriter {
  197. header := buf.New()
  198. common.Must2(header.WriteString(strings.Join([]string{config.GetFullVersion(), config.GetStatusValue().Code, config.GetStatusValue().Reason}, " ")))
  199. common.Must2(header.WriteString(CRLF))
  200. headers := config.PickHeaders()
  201. for _, h := range headers {
  202. common.Must2(header.WriteString(h))
  203. common.Must2(header.WriteString(CRLF))
  204. }
  205. if !config.HasHeader("Date") {
  206. common.Must2(header.WriteString("Date: "))
  207. common.Must2(header.WriteString(time.Now().Format(http.TimeFormat)))
  208. common.Must2(header.WriteString(CRLF))
  209. }
  210. common.Must2(header.WriteString(CRLF))
  211. return &HeaderWriter{
  212. header: header,
  213. }
  214. }
  215. type HttpAuthenticator struct {
  216. config *Config
  217. }
  218. func (a HttpAuthenticator) GetClientWriter() *HeaderWriter {
  219. header := buf.New()
  220. config := a.config.Request
  221. common.Must2(header.WriteString(strings.Join([]string{config.GetMethodValue(), config.PickUri(), config.GetFullVersion()}, " ")))
  222. common.Must2(header.WriteString(CRLF))
  223. headers := config.PickHeaders()
  224. for _, h := range headers {
  225. common.Must2(header.WriteString(h))
  226. common.Must2(header.WriteString(CRLF))
  227. }
  228. common.Must2(header.WriteString(CRLF))
  229. return &HeaderWriter{
  230. header: header,
  231. }
  232. }
  233. func (a HttpAuthenticator) GetServerWriter() *HeaderWriter {
  234. return formResponseHeader(a.config.Response)
  235. }
  236. func (a HttpAuthenticator) Client(conn net.Conn) net.Conn {
  237. if a.config.Request == nil && a.config.Response == nil {
  238. return conn
  239. }
  240. var reader Reader = NoOpReader{}
  241. if a.config.Request != nil {
  242. reader = new(HeaderReader)
  243. }
  244. var writer Writer = NoOpWriter{}
  245. if a.config.Response != nil {
  246. writer = a.GetClientWriter()
  247. }
  248. return NewHttpConn(conn, reader, writer, NoOpWriter{}, NoOpWriter{}, NoOpWriter{})
  249. }
  250. func (a HttpAuthenticator) Server(conn net.Conn) net.Conn {
  251. if a.config.Request == nil && a.config.Response == nil {
  252. return conn
  253. }
  254. return NewHttpConn(conn, new(HeaderReader).ExpectThisRequest(a.config.Request), a.GetServerWriter(),
  255. formResponseHeader(resp400),
  256. formResponseHeader(resp404),
  257. formResponseHeader(resp400))
  258. }
  259. func NewHttpAuthenticator(ctx context.Context, config *Config) (HttpAuthenticator, error) {
  260. return HttpAuthenticator{
  261. config: config,
  262. }, nil
  263. }
  264. func init() {
  265. common.Must(common.RegisterConfig((*Config)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {
  266. return NewHttpAuthenticator(ctx, config.(*Config))
  267. }))
  268. }