1
0

http_content.cpp 8.1 KB

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