Http1Parser.h 3.7 KB

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