113498266Sopenharmony_ci/*************************************************************************** 213498266Sopenharmony_ci * _ _ ____ _ 313498266Sopenharmony_ci * Project ___| | | | _ \| | 413498266Sopenharmony_ci * / __| | | | |_) | | 513498266Sopenharmony_ci * | (__| |_| | _ <| |___ 613498266Sopenharmony_ci * \___|\___/|_| \_\_____| 713498266Sopenharmony_ci * 813498266Sopenharmony_ci * Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al. 913498266Sopenharmony_ci * 1013498266Sopenharmony_ci * This software is licensed as described in the file COPYING, which 1113498266Sopenharmony_ci * you should have received as part of this distribution. The terms 1213498266Sopenharmony_ci * are also available at https://curl.se/docs/copyright.html. 1313498266Sopenharmony_ci * 1413498266Sopenharmony_ci * You may opt to use, copy, modify, merge, publish, distribute and/or sell 1513498266Sopenharmony_ci * copies of the Software, and permit persons to whom the Software is 1613498266Sopenharmony_ci * furnished to do so, under the terms of the COPYING file. 1713498266Sopenharmony_ci * 1813498266Sopenharmony_ci * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY 1913498266Sopenharmony_ci * KIND, either express or implied. 2013498266Sopenharmony_ci * 2113498266Sopenharmony_ci * SPDX-License-Identifier: curl 2213498266Sopenharmony_ci * 2313498266Sopenharmony_ci ***************************************************************************/ 2413498266Sopenharmony_ci 2513498266Sopenharmony_ci/* <DESC> 2613498266Sopenharmony_ci * multi_socket API using libuv 2713498266Sopenharmony_ci * </DESC> 2813498266Sopenharmony_ci */ 2913498266Sopenharmony_ci/* Example application using the multi socket interface to download multiple 3013498266Sopenharmony_ci files in parallel, powered by libuv. 3113498266Sopenharmony_ci 3213498266Sopenharmony_ci Requires libuv and (of course) libcurl. 3313498266Sopenharmony_ci 3413498266Sopenharmony_ci See https://nikhilm.github.io/uvbook/ for more information on libuv. 3513498266Sopenharmony_ci*/ 3613498266Sopenharmony_ci 3713498266Sopenharmony_ci#include <stdio.h> 3813498266Sopenharmony_ci#include <stdlib.h> 3913498266Sopenharmony_ci#include <uv.h> 4013498266Sopenharmony_ci#include <curl/curl.h> 4113498266Sopenharmony_ci 4213498266Sopenharmony_ciuv_loop_t *loop; 4313498266Sopenharmony_ciCURLM *curl_handle; 4413498266Sopenharmony_ciuv_timer_t timeout; 4513498266Sopenharmony_ci 4613498266Sopenharmony_citypedef struct curl_context_s { 4713498266Sopenharmony_ci uv_poll_t poll_handle; 4813498266Sopenharmony_ci curl_socket_t sockfd; 4913498266Sopenharmony_ci} curl_context_t; 5013498266Sopenharmony_ci 5113498266Sopenharmony_cistatic curl_context_t *create_curl_context(curl_socket_t sockfd) 5213498266Sopenharmony_ci{ 5313498266Sopenharmony_ci curl_context_t *context; 5413498266Sopenharmony_ci 5513498266Sopenharmony_ci context = (curl_context_t *) malloc(sizeof(*context)); 5613498266Sopenharmony_ci 5713498266Sopenharmony_ci context->sockfd = sockfd; 5813498266Sopenharmony_ci 5913498266Sopenharmony_ci uv_poll_init_socket(loop, &context->poll_handle, sockfd); 6013498266Sopenharmony_ci context->poll_handle.data = context; 6113498266Sopenharmony_ci 6213498266Sopenharmony_ci return context; 6313498266Sopenharmony_ci} 6413498266Sopenharmony_ci 6513498266Sopenharmony_cistatic void curl_close_cb(uv_handle_t *handle) 6613498266Sopenharmony_ci{ 6713498266Sopenharmony_ci curl_context_t *context = (curl_context_t *) handle->data; 6813498266Sopenharmony_ci free(context); 6913498266Sopenharmony_ci} 7013498266Sopenharmony_ci 7113498266Sopenharmony_cistatic void destroy_curl_context(curl_context_t *context) 7213498266Sopenharmony_ci{ 7313498266Sopenharmony_ci uv_close((uv_handle_t *) &context->poll_handle, curl_close_cb); 7413498266Sopenharmony_ci} 7513498266Sopenharmony_ci 7613498266Sopenharmony_cistatic void add_download(const char *url, int num) 7713498266Sopenharmony_ci{ 7813498266Sopenharmony_ci char filename[50]; 7913498266Sopenharmony_ci FILE *file; 8013498266Sopenharmony_ci CURL *handle; 8113498266Sopenharmony_ci 8213498266Sopenharmony_ci snprintf(filename, 50, "%d.download", num); 8313498266Sopenharmony_ci 8413498266Sopenharmony_ci file = fopen(filename, "wb"); 8513498266Sopenharmony_ci if(!file) { 8613498266Sopenharmony_ci fprintf(stderr, "Error opening %s\n", filename); 8713498266Sopenharmony_ci return; 8813498266Sopenharmony_ci } 8913498266Sopenharmony_ci 9013498266Sopenharmony_ci handle = curl_easy_init(); 9113498266Sopenharmony_ci curl_easy_setopt(handle, CURLOPT_WRITEDATA, file); 9213498266Sopenharmony_ci curl_easy_setopt(handle, CURLOPT_PRIVATE, file); 9313498266Sopenharmony_ci curl_easy_setopt(handle, CURLOPT_URL, url); 9413498266Sopenharmony_ci curl_multi_add_handle(curl_handle, handle); 9513498266Sopenharmony_ci fprintf(stderr, "Added download %s -> %s\n", url, filename); 9613498266Sopenharmony_ci} 9713498266Sopenharmony_ci 9813498266Sopenharmony_cistatic void check_multi_info(void) 9913498266Sopenharmony_ci{ 10013498266Sopenharmony_ci char *done_url; 10113498266Sopenharmony_ci CURLMsg *message; 10213498266Sopenharmony_ci int pending; 10313498266Sopenharmony_ci CURL *easy_handle; 10413498266Sopenharmony_ci FILE *file; 10513498266Sopenharmony_ci 10613498266Sopenharmony_ci while((message = curl_multi_info_read(curl_handle, &pending))) { 10713498266Sopenharmony_ci switch(message->msg) { 10813498266Sopenharmony_ci case CURLMSG_DONE: 10913498266Sopenharmony_ci /* Do not use message data after calling curl_multi_remove_handle() and 11013498266Sopenharmony_ci curl_easy_cleanup(). As per curl_multi_info_read() docs: 11113498266Sopenharmony_ci "WARNING: The data the returned pointer points to will not survive 11213498266Sopenharmony_ci calling curl_multi_cleanup, curl_multi_remove_handle or 11313498266Sopenharmony_ci curl_easy_cleanup." */ 11413498266Sopenharmony_ci easy_handle = message->easy_handle; 11513498266Sopenharmony_ci 11613498266Sopenharmony_ci curl_easy_getinfo(easy_handle, CURLINFO_EFFECTIVE_URL, &done_url); 11713498266Sopenharmony_ci curl_easy_getinfo(easy_handle, CURLINFO_PRIVATE, &file); 11813498266Sopenharmony_ci printf("%s DONE\n", done_url); 11913498266Sopenharmony_ci 12013498266Sopenharmony_ci curl_multi_remove_handle(curl_handle, easy_handle); 12113498266Sopenharmony_ci curl_easy_cleanup(easy_handle); 12213498266Sopenharmony_ci if(file) { 12313498266Sopenharmony_ci fclose(file); 12413498266Sopenharmony_ci } 12513498266Sopenharmony_ci break; 12613498266Sopenharmony_ci 12713498266Sopenharmony_ci default: 12813498266Sopenharmony_ci fprintf(stderr, "CURLMSG default\n"); 12913498266Sopenharmony_ci break; 13013498266Sopenharmony_ci } 13113498266Sopenharmony_ci } 13213498266Sopenharmony_ci} 13313498266Sopenharmony_ci 13413498266Sopenharmony_cistatic void curl_perform(uv_poll_t *req, int status, int events) 13513498266Sopenharmony_ci{ 13613498266Sopenharmony_ci int running_handles; 13713498266Sopenharmony_ci int flags = 0; 13813498266Sopenharmony_ci curl_context_t *context; 13913498266Sopenharmony_ci 14013498266Sopenharmony_ci if(events & UV_READABLE) 14113498266Sopenharmony_ci flags |= CURL_CSELECT_IN; 14213498266Sopenharmony_ci if(events & UV_WRITABLE) 14313498266Sopenharmony_ci flags |= CURL_CSELECT_OUT; 14413498266Sopenharmony_ci 14513498266Sopenharmony_ci context = (curl_context_t *) req->data; 14613498266Sopenharmony_ci 14713498266Sopenharmony_ci curl_multi_socket_action(curl_handle, context->sockfd, flags, 14813498266Sopenharmony_ci &running_handles); 14913498266Sopenharmony_ci 15013498266Sopenharmony_ci check_multi_info(); 15113498266Sopenharmony_ci} 15213498266Sopenharmony_ci 15313498266Sopenharmony_cistatic void on_timeout(uv_timer_t *req) 15413498266Sopenharmony_ci{ 15513498266Sopenharmony_ci int running_handles; 15613498266Sopenharmony_ci curl_multi_socket_action(curl_handle, CURL_SOCKET_TIMEOUT, 0, 15713498266Sopenharmony_ci &running_handles); 15813498266Sopenharmony_ci check_multi_info(); 15913498266Sopenharmony_ci} 16013498266Sopenharmony_ci 16113498266Sopenharmony_cistatic int start_timeout(CURLM *multi, long timeout_ms, void *userp) 16213498266Sopenharmony_ci{ 16313498266Sopenharmony_ci if(timeout_ms < 0) { 16413498266Sopenharmony_ci uv_timer_stop(&timeout); 16513498266Sopenharmony_ci } 16613498266Sopenharmony_ci else { 16713498266Sopenharmony_ci if(timeout_ms == 0) 16813498266Sopenharmony_ci timeout_ms = 1; /* 0 means directly call socket_action, but we will do it 16913498266Sopenharmony_ci in a bit */ 17013498266Sopenharmony_ci uv_timer_start(&timeout, on_timeout, timeout_ms, 0); 17113498266Sopenharmony_ci } 17213498266Sopenharmony_ci return 0; 17313498266Sopenharmony_ci} 17413498266Sopenharmony_ci 17513498266Sopenharmony_cistatic int handle_socket(CURL *easy, curl_socket_t s, int action, void *userp, 17613498266Sopenharmony_ci void *socketp) 17713498266Sopenharmony_ci{ 17813498266Sopenharmony_ci curl_context_t *curl_context; 17913498266Sopenharmony_ci int events = 0; 18013498266Sopenharmony_ci 18113498266Sopenharmony_ci switch(action) { 18213498266Sopenharmony_ci case CURL_POLL_IN: 18313498266Sopenharmony_ci case CURL_POLL_OUT: 18413498266Sopenharmony_ci case CURL_POLL_INOUT: 18513498266Sopenharmony_ci curl_context = socketp ? 18613498266Sopenharmony_ci (curl_context_t *) socketp : create_curl_context(s); 18713498266Sopenharmony_ci 18813498266Sopenharmony_ci curl_multi_assign(curl_handle, s, (void *) curl_context); 18913498266Sopenharmony_ci 19013498266Sopenharmony_ci if(action != CURL_POLL_IN) 19113498266Sopenharmony_ci events |= UV_WRITABLE; 19213498266Sopenharmony_ci if(action != CURL_POLL_OUT) 19313498266Sopenharmony_ci events |= UV_READABLE; 19413498266Sopenharmony_ci 19513498266Sopenharmony_ci uv_poll_start(&curl_context->poll_handle, events, curl_perform); 19613498266Sopenharmony_ci break; 19713498266Sopenharmony_ci case CURL_POLL_REMOVE: 19813498266Sopenharmony_ci if(socketp) { 19913498266Sopenharmony_ci uv_poll_stop(&((curl_context_t*)socketp)->poll_handle); 20013498266Sopenharmony_ci destroy_curl_context((curl_context_t*) socketp); 20113498266Sopenharmony_ci curl_multi_assign(curl_handle, s, NULL); 20213498266Sopenharmony_ci } 20313498266Sopenharmony_ci break; 20413498266Sopenharmony_ci default: 20513498266Sopenharmony_ci abort(); 20613498266Sopenharmony_ci } 20713498266Sopenharmony_ci 20813498266Sopenharmony_ci return 0; 20913498266Sopenharmony_ci} 21013498266Sopenharmony_ci 21113498266Sopenharmony_ciint main(int argc, char **argv) 21213498266Sopenharmony_ci{ 21313498266Sopenharmony_ci loop = uv_default_loop(); 21413498266Sopenharmony_ci 21513498266Sopenharmony_ci if(argc <= 1) 21613498266Sopenharmony_ci return 0; 21713498266Sopenharmony_ci 21813498266Sopenharmony_ci if(curl_global_init(CURL_GLOBAL_ALL)) { 21913498266Sopenharmony_ci fprintf(stderr, "Could not init curl\n"); 22013498266Sopenharmony_ci return 1; 22113498266Sopenharmony_ci } 22213498266Sopenharmony_ci 22313498266Sopenharmony_ci uv_timer_init(loop, &timeout); 22413498266Sopenharmony_ci 22513498266Sopenharmony_ci curl_handle = curl_multi_init(); 22613498266Sopenharmony_ci curl_multi_setopt(curl_handle, CURLMOPT_SOCKETFUNCTION, handle_socket); 22713498266Sopenharmony_ci curl_multi_setopt(curl_handle, CURLMOPT_TIMERFUNCTION, start_timeout); 22813498266Sopenharmony_ci 22913498266Sopenharmony_ci while(argc-- > 1) { 23013498266Sopenharmony_ci add_download(argv[argc], argc); 23113498266Sopenharmony_ci } 23213498266Sopenharmony_ci 23313498266Sopenharmony_ci uv_run(loop, UV_RUN_DEFAULT); 23413498266Sopenharmony_ci curl_multi_cleanup(curl_handle); 23513498266Sopenharmony_ci 23613498266Sopenharmony_ci return 0; 23713498266Sopenharmony_ci} 238