http_content.cpp 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. #include "http_content.h"
  2. #include "hurl.h"
  3. #include <string.h>
  4. BEGIN_NAMESPACE_HV
  5. std::string dump_query_params(const QueryParams& query_params) {
  6. std::string query_string;
  7. for (auto& pair : query_params) {
  8. if (query_string.size() != 0) {
  9. query_string += '&';
  10. }
  11. query_string += HUrl::escape(pair.first);
  12. query_string += '=';
  13. query_string += HUrl::escape(pair.second);
  14. }
  15. return query_string;
  16. }
  17. int parse_query_params(const char* query_string, QueryParams& query_params) {
  18. const char* p = strchr(query_string, '?');
  19. p = p ? p+1 : query_string;
  20. enum {
  21. s_key,
  22. s_value,
  23. } state = s_key;
  24. const char* key = p;
  25. const char* value = NULL;
  26. int key_len = 0;
  27. int value_len = 0;
  28. while (*p != '\0') {
  29. if (*p == '&') {
  30. if (key_len /* && value_len */) {
  31. std::string strkey = std::string(key, key_len);
  32. std::string strvalue = std::string(value, value_len);
  33. query_params[HUrl::unescape(strkey)] = HUrl::unescape(strvalue);
  34. key_len = value_len = 0;
  35. }
  36. state = s_key;
  37. key = p+1;
  38. }
  39. else if (*p == '=') {
  40. state = s_value;
  41. value = p+1;
  42. }
  43. else {
  44. state == s_key ? ++key_len : ++value_len;
  45. }
  46. ++p;
  47. }
  48. if (key_len /* && value_len */) {
  49. std::string strkey = std::string(key, key_len);
  50. std::string strvalue = std::string(value, value_len);
  51. query_params[HUrl::unescape(strkey)] = HUrl::unescape(strvalue);
  52. key_len = value_len = 0;
  53. }
  54. return query_params.size() == 0 ? -1 : 0;
  55. }
  56. #ifndef WITHOUT_HTTP_CONTENT
  57. #include "hstring.h" // for split
  58. #include "hfile.h"
  59. #include "httpdef.h" // for http_content_type_str_by_suffix
  60. std::string dump_multipart(MultiPart& mp, const char* boundary) {
  61. char c_str[256] = {0};
  62. std::string str;
  63. for (auto& pair : mp) {
  64. str += "--";
  65. str += boundary;
  66. str += "\r\n";
  67. str += "Content-Disposition: form-data";
  68. snprintf(c_str, sizeof(c_str), "; name=\"%s\"", pair.first.c_str());
  69. str += c_str;
  70. auto& form = pair.second;
  71. if (form.filename.size() != 0) {
  72. if (form.content.size() == 0) {
  73. HFile file;
  74. if (file.open(form.filename.c_str(), "rb") == 0) {
  75. file.readall(form.content);
  76. }
  77. }
  78. snprintf(c_str, sizeof(c_str), "; filename=\"%s\"", hv_basename(form.filename.c_str()));
  79. str += c_str;
  80. const char* suffix = strrchr(form.filename.c_str(), '.');
  81. if (suffix) {
  82. const char* stype = http_content_type_str_by_suffix(++suffix);
  83. if (stype && *stype != '\0') {
  84. str += "\r\n";
  85. str += "Content-Type: ";
  86. str += stype;
  87. }
  88. }
  89. }
  90. str += "\r\n\r\n";
  91. str += form.content;
  92. str += "\r\n";
  93. }
  94. str += "--";
  95. str += boundary;
  96. str += "--\r\n";
  97. return str;
  98. }
  99. #include "multipart_parser.h"
  100. enum multipart_parser_state_e {
  101. MP_START,
  102. MP_PART_DATA_BEGIN,
  103. MP_HEADER_FIELD,
  104. MP_HEADER_VALUE,
  105. MP_HEADERS_COMPLETE,
  106. MP_PART_DATA,
  107. MP_PART_DATA_END,
  108. MP_BODY_END
  109. };
  110. struct multipart_parser_userdata {
  111. MultiPart* mp;
  112. // tmp
  113. multipart_parser_state_e state;
  114. std::string header_field;
  115. std::string header_value;
  116. std::string part_data;
  117. std::string name;
  118. std::string filename;
  119. void handle_header() {
  120. if (header_field.size() == 0 || header_value.size() == 0) return;
  121. if (stricmp(header_field.c_str(), "Content-Disposition") == 0) {
  122. StringList strlist = split(header_value, ';');
  123. for (auto& str : strlist) {
  124. StringList kv = split(trim(str, " "), '=');
  125. if (kv.size() == 2) {
  126. const char* key = kv.begin()->c_str();
  127. std::string value = *(kv.begin() + 1);
  128. value = trim_pairs(value, "\"\"\'\'");
  129. if (strcmp(key, "name") == 0) {
  130. name = value;
  131. }
  132. else if (strcmp(key, "filename") == 0) {
  133. filename = value;
  134. }
  135. }
  136. }
  137. }
  138. header_field.clear();
  139. header_value.clear();
  140. }
  141. void handle_data() {
  142. if (name.size() != 0) {
  143. FormData formdata;
  144. formdata.content = part_data;
  145. formdata.filename = filename;
  146. (*mp)[name] = formdata;
  147. }
  148. name.clear();
  149. filename.clear();
  150. part_data.clear();
  151. }
  152. };
  153. static int on_header_field(multipart_parser* parser, const char *at, size_t length) {
  154. //printf("on_header_field:%.*s\n", (int)length, at);
  155. multipart_parser_userdata* userdata = (multipart_parser_userdata*)multipart_parser_get_data(parser);
  156. userdata->handle_header();
  157. userdata->state = MP_HEADER_FIELD;
  158. userdata->header_field.append(at, length);
  159. return 0;
  160. }
  161. static int on_header_value(multipart_parser* parser, const char *at, size_t length) {
  162. //printf("on_header_value:%.*s\n", (int)length, at);
  163. multipart_parser_userdata* userdata = (multipart_parser_userdata*)multipart_parser_get_data(parser);
  164. userdata->state = MP_HEADER_VALUE;
  165. userdata->header_value.append(at, length);
  166. return 0;
  167. }
  168. static int on_part_data(multipart_parser* parser, const char *at, size_t length) {
  169. //printf("on_part_data:%.*s\n", (int)length, at);
  170. multipart_parser_userdata* userdata = (multipart_parser_userdata*)multipart_parser_get_data(parser);
  171. userdata->state = MP_PART_DATA;
  172. userdata->part_data.append(at, length);
  173. return 0;
  174. }
  175. static int on_part_data_begin(multipart_parser* parser) {
  176. //printf("on_part_data_begin\n");
  177. multipart_parser_userdata* userdata = (multipart_parser_userdata*)multipart_parser_get_data(parser);
  178. userdata->state = MP_PART_DATA_BEGIN;
  179. return 0;
  180. }
  181. static int on_headers_complete(multipart_parser* parser) {
  182. //printf("on_headers_complete\n");
  183. multipart_parser_userdata* userdata = (multipart_parser_userdata*)multipart_parser_get_data(parser);
  184. userdata->handle_header();
  185. userdata->state = MP_HEADERS_COMPLETE;
  186. return 0;
  187. }
  188. static int on_part_data_end(multipart_parser* parser) {
  189. //printf("on_part_data_end\n");
  190. multipart_parser_userdata* userdata = (multipart_parser_userdata*)multipart_parser_get_data(parser);
  191. userdata->state = MP_PART_DATA_END;
  192. userdata->handle_data();
  193. return 0;
  194. }
  195. static int on_body_end(multipart_parser* parser) {
  196. //printf("on_body_end\n");
  197. multipart_parser_userdata* userdata = (multipart_parser_userdata*)multipart_parser_get_data(parser);
  198. userdata->state = MP_BODY_END;
  199. return 0;
  200. }
  201. int parse_multipart(const std::string& str, MultiPart& mp, const char* boundary) {
  202. //printf("boundary=%s\n", boundary);
  203. std::string __boundary("--");
  204. __boundary += boundary;
  205. multipart_parser_settings settings;
  206. settings.on_header_field = on_header_field;
  207. settings.on_header_value = on_header_value;
  208. settings.on_part_data = on_part_data;
  209. settings.on_part_data_begin = on_part_data_begin;
  210. settings.on_headers_complete = on_headers_complete;
  211. settings.on_part_data_end = on_part_data_end;
  212. settings.on_body_end = on_body_end;
  213. multipart_parser* parser = multipart_parser_init(__boundary.c_str(), &settings);
  214. multipart_parser_userdata userdata;
  215. userdata.state = MP_START;
  216. userdata.mp = &mp;
  217. multipart_parser_set_data(parser, &userdata);
  218. size_t nparse = multipart_parser_execute(parser, str.c_str(), str.size());
  219. multipart_parser_free(parser);
  220. return nparse == str.size() ? 0 : -1;
  221. }
  222. std::string dump_json(const hv::Json& json, int indent) {
  223. return json.dump(indent);
  224. }
  225. int parse_json(const char* str, hv::Json& json, std::string& errmsg) {
  226. try {
  227. json = nlohmann::json::parse(str);
  228. }
  229. catch(nlohmann::detail::exception e) {
  230. errmsg = e.what();
  231. return -1;
  232. }
  233. return (json.is_discarded() || json.is_null()) ? -1 : 0;
  234. }
  235. #endif
  236. END_NAMESPACE_HV