Http1Parser.h 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. #ifndef HV_HTTP1_PARSER_H_
  2. #define HV_HTTP1_PARSER_H_
  3. #include "HttpParser.h"
  4. #include "http_parser.h"
  5. enum http_parser_state {
  6. HP_START_REQ_OR_RES,
  7. HP_MESSAGE_BEGIN,
  8. HP_URL,
  9. HP_STATUS,
  10. HP_HEADER_FIELD,
  11. HP_HEADER_VALUE,
  12. HP_HEADERS_COMPLETE,
  13. HP_BODY,
  14. HP_MESSAGE_COMPLETE
  15. };
  16. class Http1Parser : public HttpParser {
  17. public:
  18. static http_parser_settings* cbs;
  19. http_parser parser;
  20. int flags;
  21. http_parser_state state;
  22. HttpMessage* submited;
  23. HttpMessage* parsed;
  24. // tmp
  25. std::string url; // for on_url
  26. std::string header_field; // for on_header_field
  27. std::string header_value; // for on_header_value
  28. std::string sendbuf; // for GetSendData
  29. Http1Parser(http_session_type type = HTTP_CLIENT);
  30. virtual ~Http1Parser();
  31. void handle_header() {
  32. if (header_field.size() != 0 && header_value.size() != 0) {
  33. if (stricmp(header_field.c_str(), "Set-CooKie") == 0 ||
  34. stricmp(header_field.c_str(), "Cookie") == 0) {
  35. HttpCookie cookie;
  36. if (cookie.parse(header_value)) {
  37. parsed->cookies.emplace_back(cookie);
  38. }
  39. }
  40. parsed->headers[header_field] = header_value;
  41. header_field.clear();
  42. header_value.clear();
  43. }
  44. }
  45. virtual int GetSendData(char** data, size_t* len) {
  46. if (!submited) {
  47. *data = NULL;
  48. *len = 0;
  49. return 0;
  50. }
  51. sendbuf = submited->Dump(true, true);
  52. submited = NULL;
  53. *data = (char*)sendbuf.data();
  54. *len = sendbuf.size();
  55. return sendbuf.size();
  56. }
  57. virtual int FeedRecvData(const char* data, size_t len) {
  58. return http_parser_execute(&parser, cbs, data, len);
  59. }
  60. virtual int GetState() {
  61. return (int)state;
  62. }
  63. virtual bool WantRecv() {
  64. return state != HP_MESSAGE_COMPLETE;
  65. }
  66. virtual bool WantSend() {
  67. return state == HP_MESSAGE_COMPLETE;
  68. }
  69. virtual bool IsComplete() {
  70. return state == HP_MESSAGE_COMPLETE;
  71. }
  72. virtual int GetError() {
  73. return parser.http_errno;
  74. }
  75. virtual const char* StrError(int error) {
  76. return http_errno_description((enum http_errno)error);
  77. }
  78. // client
  79. // SubmitRequest -> while(GetSendData) {send} -> InitResponse -> do {recv -> FeedRecvData} while(WantRecv)
  80. virtual int SubmitRequest(HttpRequest* req) {
  81. submited = req;
  82. if (req) {
  83. if (req->method == HTTP_HEAD) {
  84. flags |= F_SKIPBODY;
  85. } else {
  86. flags &= ~F_SKIPBODY;
  87. }
  88. }
  89. return 0;
  90. }
  91. virtual int InitResponse(HttpResponse* res) {
  92. res->Reset();
  93. parsed = res;
  94. http_parser_init(&parser, HTTP_RESPONSE);
  95. url.clear();
  96. header_field.clear();
  97. header_value.clear();
  98. return 0;
  99. }
  100. // server
  101. // InitRequest -> do {recv -> FeedRecvData} while(WantRecv) -> SubmitResponse -> while(GetSendData) {send}
  102. virtual int InitRequest(HttpRequest* req) {
  103. req->Reset();
  104. parsed = req;
  105. http_parser_init(&parser, HTTP_REQUEST);
  106. state = HP_START_REQ_OR_RES;
  107. url.clear();
  108. header_field.clear();
  109. header_value.clear();
  110. return 0;
  111. }
  112. virtual int SubmitResponse(HttpResponse* res) {
  113. submited = res;
  114. return 0;
  115. }
  116. };
  117. #endif // HV_HTTP1_PARSER_H_