aos_http_io.c 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. #include "aos_log.h"
  2. #include "aos_http_io.h"
  3. #include "aos_define.h"
  4. #include <apr_thread_mutex.h>
  5. #include <apr_file_io.h>
  6. aos_pool_t *aos_global_pool = NULL;
  7. apr_file_t *aos_stderr_file = NULL;
  8. aos_http_request_options_t *aos_default_http_request_options = NULL;
  9. aos_http_transport_options_t *aos_default_http_transport_options = NULL;
  10. aos_http_transport_create_pt aos_http_transport_create = aos_curl_http_transport_create;
  11. aos_http_transport_perform_pt aos_http_transport_perform = aos_curl_http_transport_perform;
  12. static apr_thread_mutex_t* requestStackMutexG = NULL;
  13. static CURL *requestStackG[AOS_REQUEST_STACK_SIZE];
  14. static int requestStackCountG;
  15. static char aos_user_agent[256];
  16. static aos_http_transport_options_t *aos_http_transport_options_create(aos_pool_t *p);
  17. CURL *aos_request_get()
  18. {
  19. CURL *request = NULL;
  20. apr_thread_mutex_lock(requestStackMutexG);
  21. if (requestStackCountG > 0) {
  22. request = requestStackG[--requestStackCountG];
  23. }
  24. apr_thread_mutex_unlock(requestStackMutexG);
  25. // If we got one, deinitialize it for re-use
  26. if (request) {
  27. curl_easy_reset(request);
  28. }
  29. else {
  30. request = curl_easy_init();
  31. }
  32. return request;
  33. }
  34. void request_release(CURL *request)
  35. {
  36. apr_thread_mutex_lock(requestStackMutexG);
  37. // If the request stack is full, destroy this one
  38. // else put this one at the front of the request stack; we do this because
  39. // we want the most-recently-used curl handle to be re-used on the next
  40. // request, to maximize our chances of re-using a TCP connection before it
  41. // times out
  42. if (requestStackCountG == AOS_REQUEST_STACK_SIZE) {
  43. apr_thread_mutex_unlock(requestStackMutexG);
  44. curl_easy_cleanup(request);
  45. }
  46. else {
  47. requestStackG[requestStackCountG++] = request;
  48. apr_thread_mutex_unlock(requestStackMutexG);
  49. }
  50. }
  51. void aos_set_default_request_options(aos_http_request_options_t *op)
  52. {
  53. aos_default_http_request_options = op;
  54. }
  55. void aos_set_default_transport_options(aos_http_transport_options_t *op)
  56. {
  57. aos_default_http_transport_options = op;
  58. }
  59. aos_http_request_options_t *aos_http_request_options_create(aos_pool_t *p)
  60. {
  61. aos_http_request_options_t *options;
  62. options = (aos_http_request_options_t *)aos_pcalloc(p, sizeof(aos_http_request_options_t));
  63. options->speed_limit = AOS_MIN_SPEED_LIMIT;
  64. options->speed_time = AOS_MIN_SPEED_TIME;
  65. options->connect_timeout = AOS_CONNECT_TIMEOUT;
  66. options->dns_cache_timeout = AOS_DNS_CACHE_TIMOUT;
  67. options->max_memory_size = AOS_MAX_MEMORY_SIZE;
  68. options->enable_crc = AOS_TRUE;
  69. options->proxy_auth = NULL;
  70. options->proxy_host = NULL;
  71. return options;
  72. }
  73. aos_http_transport_options_t *aos_http_transport_options_create(aos_pool_t *p)
  74. {
  75. return (aos_http_transport_options_t *)aos_pcalloc(p, sizeof(aos_http_transport_options_t));
  76. }
  77. aos_http_controller_t *aos_http_controller_create(aos_pool_t *p, int owner)
  78. {
  79. int s;
  80. aos_http_controller_t *ctl;
  81. if(p == NULL) {
  82. if ((s = aos_pool_create(&p, NULL)) != APR_SUCCESS) {
  83. aos_fatal_log("aos_pool_create failure.");
  84. return NULL;
  85. }
  86. }
  87. ctl = (aos_http_controller_t *)aos_pcalloc(p, sizeof(aos_http_controller_ex_t));
  88. ctl->pool = p;
  89. ctl->owner = owner;
  90. ctl->options = aos_default_http_request_options;
  91. return ctl;
  92. }
  93. aos_http_request_t *aos_http_request_create(aos_pool_t *p)
  94. {
  95. aos_http_request_t *req;
  96. req = (aos_http_request_t *)aos_pcalloc(p, sizeof(aos_http_request_t));
  97. req->method = HTTP_GET;
  98. req->headers = aos_table_make(p, 5);
  99. req->query_params = aos_table_make(p, 3);
  100. aos_list_init(&req->body);
  101. req->type = BODY_IN_MEMORY;
  102. req->body_len = 0;
  103. req->pool = p;
  104. req->read_body = aos_read_http_body_memory;
  105. return req;
  106. }
  107. aos_http_response_t *aos_http_response_create(aos_pool_t *p)
  108. {
  109. aos_http_response_t *resp;
  110. resp = (aos_http_response_t *)aos_pcalloc(p, sizeof(aos_http_response_t));
  111. resp->status = -1;
  112. resp->headers = aos_table_make(p, 10);
  113. aos_list_init(&resp->body);
  114. resp->type = BODY_IN_MEMORY;
  115. resp->body_len = 0;
  116. resp->pool = p;
  117. resp->write_body = aos_write_http_body_memory;
  118. return resp;
  119. }
  120. int aos_read_http_body_memory(aos_http_request_t *req, char *buffer, int len)
  121. {
  122. int wsize;
  123. int bytes = 0;
  124. aos_buf_t *b;
  125. aos_buf_t *n;
  126. aos_list_for_each_entry_safe(aos_buf_t, b, n, &req->body, node) {
  127. wsize = aos_buf_size(b);
  128. if (wsize == 0) {
  129. aos_list_del(&b->node);
  130. continue;
  131. }
  132. wsize = aos_min(len - bytes, wsize);
  133. if (wsize == 0) {
  134. break;
  135. }
  136. memcpy(buffer + bytes, b->pos, wsize);
  137. b->pos += wsize;
  138. bytes += wsize;
  139. if (b->pos == b->last) {
  140. aos_list_del(&b->node);
  141. }
  142. }
  143. return bytes;
  144. }
  145. int aos_read_http_body_file(aos_http_request_t *req, char *buffer, int len)
  146. {
  147. int s;
  148. char buf[256];
  149. apr_size_t nbytes = len;
  150. apr_size_t bytes_left;
  151. if (req->file_buf == NULL || req->file_buf->file == NULL) {
  152. aos_error_log("request body arg invalid file_buf NULL.");
  153. return AOSE_INVALID_ARGUMENT;
  154. }
  155. if (req->file_buf->file_pos >= req->file_buf->file_last) {
  156. aos_debug_log("file read finish.");
  157. return 0;
  158. }
  159. bytes_left = (apr_size_t)(req->file_buf->file_last - req->file_buf->file_pos);
  160. if (nbytes > bytes_left) {
  161. nbytes = bytes_left;
  162. }
  163. if ((s = apr_file_read(req->file_buf->file, buffer, &nbytes)) != APR_SUCCESS) {
  164. aos_error_log("apr_file_read filure, code:%d %s.", s, apr_strerror(s, buf, sizeof(buf)));
  165. return AOSE_FILE_READ_ERROR;
  166. }
  167. req->file_buf->file_pos += nbytes;
  168. return nbytes;
  169. }
  170. int aos_write_http_body_memory(aos_http_response_t *resp, const char *buffer, int len)
  171. {
  172. aos_buf_t *b;
  173. b = aos_create_buf(resp->pool, len);
  174. memcpy(b->pos, buffer, len);
  175. b->last += len;
  176. aos_list_add_tail(&b->node, &resp->body);
  177. resp->body_len += len;
  178. return len;
  179. }
  180. int aos_write_http_body_file(aos_http_response_t *resp, const char *buffer, int len)
  181. {
  182. int elen;
  183. int s;
  184. char buf[256];
  185. apr_size_t nbytes = len;
  186. if (resp->file_buf == NULL) {
  187. resp->file_buf = aos_create_file_buf(resp->pool);
  188. }
  189. if (resp->file_buf->file == NULL) {
  190. if (resp->file_path == NULL) {
  191. aos_error_log("resp body file arg NULL.");
  192. return AOSE_INVALID_ARGUMENT;
  193. }
  194. aos_trace_log("open file %s.", resp->file_path);
  195. if ((elen = aos_open_file_for_write(resp->pool, resp->file_path, resp->file_buf)) != AOSE_OK) {
  196. return elen;
  197. }
  198. }
  199. assert(resp->file_buf->file != NULL);
  200. if ((s = apr_file_write(resp->file_buf->file, buffer, &nbytes)) != APR_SUCCESS) {
  201. aos_error_log("apr_file_write fialure, code:%d %s.", s, apr_strerror(s, buf, sizeof(buf)));
  202. return AOSE_FILE_WRITE_ERROR;
  203. }
  204. resp->file_buf->file_last += nbytes;
  205. resp->body_len += nbytes;
  206. return nbytes;
  207. }
  208. int aos_http_io_initialize(const char *user_agent_info, int flags)
  209. {
  210. CURLcode ecode;
  211. int s;
  212. char buf[256];
  213. aos_http_request_options_t *req_options;
  214. aos_http_transport_options_t *trans_options;
  215. if ((ecode = curl_global_init(CURL_GLOBAL_ALL &
  216. ~((flags & AOS_INIT_WINSOCK) ? 0: CURL_GLOBAL_WIN32))) != CURLE_OK)
  217. {
  218. aos_error_log("curl_global_init failure, code:%d %s.\n", ecode, curl_easy_strerror(ecode));
  219. return AOSE_INTERNAL_ERROR;
  220. }
  221. if ((s = apr_initialize()) != APR_SUCCESS) {
  222. aos_error_log("apr_initialize failue.\n");
  223. return AOSE_INTERNAL_ERROR;
  224. }
  225. if (!user_agent_info || !*user_agent_info) {
  226. user_agent_info = "Unknown";
  227. }
  228. if ((s = aos_pool_create(&aos_global_pool, NULL)) != APR_SUCCESS) {
  229. aos_error_log("aos_pool_create failure, code:%d %s.\n", s, apr_strerror(s, buf, sizeof(buf)));
  230. return AOSE_INTERNAL_ERROR;
  231. }
  232. if ((s = apr_thread_mutex_create(&requestStackMutexG, APR_THREAD_MUTEX_DEFAULT, aos_global_pool)) != APR_SUCCESS) {
  233. aos_error_log("apr_thread_mutex_create failure, code:%d %s.\n", s, apr_strerror(s, buf, sizeof(buf)));
  234. return AOSE_INTERNAL_ERROR;
  235. }
  236. requestStackCountG = 0;
  237. apr_snprintf(aos_user_agent, sizeof(aos_user_agent)-1, "%s(Compatible %s)",
  238. AOS_VER, user_agent_info);
  239. req_options = aos_http_request_options_create(aos_global_pool);
  240. trans_options = aos_http_transport_options_create(aos_global_pool);
  241. trans_options->user_agent = aos_user_agent;
  242. aos_set_default_request_options(req_options);
  243. aos_set_default_transport_options(trans_options);
  244. return AOSE_OK;
  245. }
  246. void aos_http_io_deinitialize()
  247. {
  248. apr_thread_mutex_destroy(requestStackMutexG);
  249. while (requestStackCountG--) {
  250. curl_easy_cleanup(requestStackG[requestStackCountG]);
  251. }
  252. if (aos_stderr_file != NULL) {
  253. apr_file_close(aos_stderr_file);
  254. aos_stderr_file = NULL;
  255. }
  256. if (aos_global_pool != NULL) {
  257. aos_pool_destroy(aos_global_pool);
  258. aos_global_pool = NULL;
  259. }
  260. apr_terminate();
  261. }
  262. int aos_http_send_request(aos_http_controller_t *ctl, aos_http_request_t *req, aos_http_response_t *resp)
  263. {
  264. aos_http_transport_t *t;
  265. t = aos_http_transport_create(ctl->pool);
  266. t->req = req;
  267. t->resp = resp;
  268. t->controller = (aos_http_controller_ex_t *)ctl;
  269. return aos_http_transport_perform(t);
  270. }