http_content.cpp 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. #include "http_content.h"
  2. #include "hurl.h"
  3. std::string dump_query_params(QueryParams& query_params) {
  4. std::string query_string;
  5. for (auto& pair : query_params) {
  6. if (query_string.size() != 0) {
  7. query_string += '&';
  8. }
  9. query_string += url_escape(pair.first.c_str());
  10. query_string += '=';
  11. query_string += url_escape(pair.second.c_str());
  12. }
  13. return query_string;
  14. }
  15. int parse_query_params(const char* query_string, QueryParams& query_params) {
  16. const char* p = strchr(query_string, '?');
  17. p = p ? p+1 : query_string;
  18. enum {
  19. s_key,
  20. s_value,
  21. } state = s_key;
  22. const char* key = p;
  23. const char* value = NULL;
  24. int key_len = 0;
  25. int value_len = 0;
  26. while (*p != '\0') {
  27. if (*p == '&') {
  28. if (key_len && value_len) {
  29. std::string strkey = std::string(key, key_len);
  30. std::string strvalue = std::string(value, value_len);
  31. query_params[url_unescape(strkey.c_str())] = url_unescape(strvalue.c_str());
  32. key_len = value_len = 0;
  33. }
  34. state = s_key;
  35. key = p+1;
  36. }
  37. else if (*p == '=') {
  38. state = s_value;
  39. value = p+1;
  40. }
  41. else {
  42. state == s_key ? ++key_len : ++value_len;
  43. }
  44. ++p;
  45. }
  46. if (key_len && value_len) {
  47. std::string strkey = std::string(key, key_len);
  48. std::string strvalue = std::string(value, value_len);
  49. query_params[url_unescape(strkey.c_str())] = url_unescape(strvalue.c_str());
  50. key_len = value_len = 0;
  51. }
  52. return query_params.size() == 0 ? -1 : 0;
  53. }
  54. #ifndef WITHOUT_HTTP_CONTENT
  55. #include <sys/types.h>
  56. #include <sys/stat.h>
  57. #include "hstring.h" // for split
  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. FILE* fp = fopen(form.filename.c_str(), "r");
  73. if (fp) {
  74. struct stat st;
  75. if (stat(form.filename.c_str(), &st) == 0 && st.st_size != 0) {
  76. form.content.resize(st.st_size);
  77. fread((void*)form.content.data(), 1, st.st_size, fp);
  78. }
  79. fclose(fp);
  80. }
  81. }
  82. snprintf(c_str, sizeof(c_str), "; filename=\"%s\"", basename(form.filename).c_str());
  83. str += c_str;
  84. const char* suffix = strrchr(form.filename.c_str(), '.');
  85. if (suffix) {
  86. const char* stype = http_content_type_str_by_suffix(++suffix);
  87. if (stype && *stype != '\0') {
  88. str += "\r\n";
  89. str += "Content-Type: ";
  90. str += stype;
  91. }
  92. }
  93. }
  94. str += "\r\n\r\n";
  95. str += form.content;
  96. str += "\r\n";
  97. }
  98. str += "--";
  99. str += boundary;
  100. str += "--";
  101. return str;
  102. }
  103. #include "multipart_parser.h"
  104. enum multipart_parser_state_e {
  105. MP_START,
  106. MP_PART_DATA_BEGIN,
  107. MP_HEADER_FIELD,
  108. MP_HEADER_VALUE,
  109. MP_HEADERS_COMPLETE,
  110. MP_PART_DATA,
  111. MP_PART_DATA_END,
  112. MP_BODY_END
  113. };
  114. struct multipart_parser_userdata {
  115. MultiPart* mp;
  116. // tmp
  117. multipart_parser_state_e state;
  118. std::string header_field;
  119. std::string header_value;
  120. std::string part_data;
  121. std::string name;
  122. std::string filename;
  123. void handle_header() {
  124. if (header_field.size() == 0 || header_value.size() == 0) return;
  125. if (stricmp(header_field.c_str(), "Content-Disposition") == 0) {
  126. StringList strlist = split(header_value, ';');
  127. for (auto& str : strlist) {
  128. StringList kv = split(trim(str, " "), '=');
  129. if (kv.size() == 2) {
  130. const char* key = kv.begin()->c_str();
  131. const char* value = trim_pairs(*(kv.begin()+1), "\"\"").c_str();
  132. if (strcmp(key, "name") == 0) {
  133. name = value;
  134. }
  135. else if (strcmp(key, "filename") == 0) {
  136. filename = value;
  137. }
  138. }
  139. }
  140. }
  141. header_field.clear();
  142. header_value.clear();
  143. }
  144. void handle_data() {
  145. if (name.size() != 0) {
  146. (*mp)[name] = FormData(part_data.c_str(), filename.c_str());
  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.insert(userdata->header_field.size(), 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.insert(userdata->header_value.size(), 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.insert(userdata->part_data.size(), 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(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(Json& json) {
  223. return json.dump();
  224. }
  225. std::string g_parse_json_errmsg;
  226. int parse_json(const char* str, Json& json, std::string& errmsg) {
  227. try {
  228. json = Json::parse(str);
  229. }
  230. catch(nlohmann::detail::exception e) {
  231. errmsg = e.what();
  232. return -1;
  233. }
  234. return (json.is_discarded() || json.is_null()) ? -1 : 0;
  235. }
  236. #endif