1
0

Http1Parser.cpp 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. #include "Http1Parser.h"
  2. #define MAX_CONTENT_LENGTH (1 << 24) // 16M
  3. static int on_url(http_parser* parser, const char *at, size_t length);
  4. static int on_status(http_parser* parser, const char *at, size_t length);
  5. static int on_header_field(http_parser* parser, const char *at, size_t length);
  6. static int on_header_value(http_parser* parser, const char *at, size_t length);
  7. static int on_body(http_parser* parser, const char *at, size_t length);
  8. static int on_message_begin(http_parser* parser);
  9. static int on_headers_complete(http_parser* parser);
  10. static int on_message_complete(http_parser* parser);
  11. static int on_chunk_header(http_parser* parser);
  12. static int on_chunk_complete(http_parser* parser);
  13. http_parser_settings Http1Parser::cbs = {
  14. on_message_begin,
  15. on_url,
  16. on_status,
  17. on_header_field,
  18. on_header_value,
  19. on_headers_complete,
  20. on_body,
  21. on_message_complete,
  22. on_chunk_header,
  23. on_chunk_complete
  24. };
  25. Http1Parser::Http1Parser(http_session_type type) {
  26. http_parser_init(&parser, HTTP_BOTH);
  27. parser.data = this;
  28. flags = 0;
  29. state = HP_START_REQ_OR_RES;
  30. submited = NULL;
  31. parsed = NULL;
  32. }
  33. Http1Parser::~Http1Parser() {
  34. }
  35. int on_url(http_parser* parser, const char *at, size_t length) {
  36. printd("on_url:%.*s\n", (int)length, at);
  37. Http1Parser* hp = (Http1Parser*)parser->data;
  38. hp->state = HP_URL;
  39. hp->url.append(at, length);
  40. return 0;
  41. }
  42. int on_status(http_parser* parser, const char *at, size_t length) {
  43. printd("on_status:%d %.*s\n", (int)parser->status_code, (int)length, at);
  44. Http1Parser* hp = (Http1Parser*)parser->data;
  45. hp->state = HP_STATUS;
  46. return 0;
  47. }
  48. int on_header_field(http_parser* parser, const char *at, size_t length) {
  49. printd("on_header_field:%.*s\n", (int)length, at);
  50. Http1Parser* hp = (Http1Parser*)parser->data;
  51. hp->handle_header();
  52. hp->state = HP_HEADER_FIELD;
  53. hp->header_field.append(at, length);
  54. return 0;
  55. }
  56. int on_header_value(http_parser* parser, const char *at, size_t length) {
  57. printd("on_header_value:%.*s\n", (int)length, at);
  58. Http1Parser* hp = (Http1Parser*)parser->data;
  59. hp->state = HP_HEADER_VALUE;
  60. hp->header_value.append(at, length);
  61. return 0;
  62. }
  63. int on_body(http_parser* parser, const char *at, size_t length) {
  64. printd("on_body:%d\n", (int)length);
  65. // printd("on_body:%.*s\n", (int)length, at);
  66. Http1Parser* hp = (Http1Parser*)parser->data;
  67. hp->state = HP_BODY;
  68. if (hp->parsed->body_cb) {
  69. hp->parsed->body_cb(at, length);
  70. } else {
  71. hp->parsed->body.append(at, length);
  72. }
  73. return 0;
  74. }
  75. int on_message_begin(http_parser* parser) {
  76. printd("on_message_begin\n");
  77. Http1Parser* hp = (Http1Parser*)parser->data;
  78. hp->state = HP_MESSAGE_BEGIN;
  79. return 0;
  80. }
  81. int on_headers_complete(http_parser* parser) {
  82. printd("on_headers_complete\n");
  83. Http1Parser* hp = (Http1Parser*)parser->data;
  84. hp->handle_header();
  85. bool skip_body = false;
  86. hp->parsed->http_major = parser->http_major;
  87. hp->parsed->http_minor = parser->http_minor;
  88. if (hp->parsed->type == HTTP_REQUEST) {
  89. HttpRequest* req = (HttpRequest*)hp->parsed;
  90. req->method = (http_method)parser->method;
  91. req->url = hp->url;
  92. }
  93. else if (hp->parsed->type == HTTP_RESPONSE) {
  94. HttpResponse* res = (HttpResponse*)hp->parsed;
  95. res->status_code = (http_status)parser->status_code;
  96. // response to HEAD
  97. if (hp->flags & F_SKIPBODY) {
  98. skip_body = true;
  99. }
  100. }
  101. auto iter = hp->parsed->headers.find("content-type");
  102. if (iter != hp->parsed->headers.end()) {
  103. hp->parsed->content_type = http_content_type_enum(iter->second.c_str());
  104. }
  105. iter = hp->parsed->headers.find("content-length");
  106. if (iter != hp->parsed->headers.end()) {
  107. int content_length = atoi(iter->second.c_str());
  108. hp->parsed->content_length = content_length;
  109. int reserve_length = MIN(content_length + 1, MAX_CONTENT_LENGTH);
  110. if ((!skip_body) && reserve_length > hp->parsed->body.capacity()) {
  111. hp->parsed->body.reserve(reserve_length);
  112. }
  113. }
  114. hp->state = HP_HEADERS_COMPLETE;
  115. if (hp->parsed->head_cb) {
  116. hp->parsed->head_cb(hp->parsed->headers);
  117. }
  118. return skip_body ? 1 : 0;
  119. }
  120. int on_message_complete(http_parser* parser) {
  121. printd("on_message_complete\n");
  122. Http1Parser* hp = (Http1Parser*)parser->data;
  123. hp->state = HP_MESSAGE_COMPLETE;
  124. return 0;
  125. }
  126. int on_chunk_header(http_parser* parser) {
  127. printd("on_chunk_header:%llu\n", parser->content_length);
  128. Http1Parser* hp = (Http1Parser*)parser->data;
  129. hp->state = HP_CHUNK_HEADER;
  130. int chunk_size = parser->content_length;
  131. int reserve_size = MIN(chunk_size + 1, MAX_CONTENT_LENGTH);
  132. if (reserve_size > hp->parsed->body.capacity()) {
  133. hp->parsed->body.reserve(reserve_size);
  134. }
  135. return 0;
  136. }
  137. int on_chunk_complete(http_parser* parser) {
  138. printd("on_chunk_complete\n");
  139. Http1Parser* hp = (Http1Parser*)parser->data;
  140. hp->state = HP_CHUNK_COMPLETE;
  141. if (hp->parsed->chunked_cb) {
  142. hp->parsed->chunked_cb(hp->parsed->body.c_str(), hp->parsed->body.size());
  143. hp->parsed->body.clear();
  144. }
  145. return 0;
  146. }