xref: /aosp_15_r20/external/curl/docs/examples/chkspeed.c (revision 6236dae45794135f37c4eb022389c904c8b0090d)
1*6236dae4SAndroid Build Coastguard Worker /***************************************************************************
2*6236dae4SAndroid Build Coastguard Worker  *                                  _   _ ____  _
3*6236dae4SAndroid Build Coastguard Worker  *  Project                     ___| | | |  _ \| |
4*6236dae4SAndroid Build Coastguard Worker  *                             / __| | | | |_) | |
5*6236dae4SAndroid Build Coastguard Worker  *                            | (__| |_| |  _ <| |___
6*6236dae4SAndroid Build Coastguard Worker  *                             \___|\___/|_| \_\_____|
7*6236dae4SAndroid Build Coastguard Worker  *
8*6236dae4SAndroid Build Coastguard Worker  * Copyright (C) Daniel Stenberg, <[email protected]>, et al.
9*6236dae4SAndroid Build Coastguard Worker  *
10*6236dae4SAndroid Build Coastguard Worker  * This software is licensed as described in the file COPYING, which
11*6236dae4SAndroid Build Coastguard Worker  * you should have received as part of this distribution. The terms
12*6236dae4SAndroid Build Coastguard Worker  * are also available at https://curl.se/docs/copyright.html.
13*6236dae4SAndroid Build Coastguard Worker  *
14*6236dae4SAndroid Build Coastguard Worker  * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15*6236dae4SAndroid Build Coastguard Worker  * copies of the Software, and permit persons to whom the Software is
16*6236dae4SAndroid Build Coastguard Worker  * furnished to do so, under the terms of the COPYING file.
17*6236dae4SAndroid Build Coastguard Worker  *
18*6236dae4SAndroid Build Coastguard Worker  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19*6236dae4SAndroid Build Coastguard Worker  * KIND, either express or implied.
20*6236dae4SAndroid Build Coastguard Worker  *
21*6236dae4SAndroid Build Coastguard Worker  * SPDX-License-Identifier: curl
22*6236dae4SAndroid Build Coastguard Worker  *
23*6236dae4SAndroid Build Coastguard Worker  ***************************************************************************/
24*6236dae4SAndroid Build Coastguard Worker /* <DESC>
25*6236dae4SAndroid Build Coastguard Worker  * Show transfer timing info after download completes.
26*6236dae4SAndroid Build Coastguard Worker  * </DESC>
27*6236dae4SAndroid Build Coastguard Worker  */
28*6236dae4SAndroid Build Coastguard Worker /* Example source code to show how the callback function can be used to
29*6236dae4SAndroid Build Coastguard Worker  * download data into a chunk of memory instead of storing it in a file.
30*6236dae4SAndroid Build Coastguard Worker  * After successful download we use curl_easy_getinfo() calls to get the
31*6236dae4SAndroid Build Coastguard Worker  * amount of downloaded bytes, the time used for the whole download, and
32*6236dae4SAndroid Build Coastguard Worker  * the average download speed.
33*6236dae4SAndroid Build Coastguard Worker  * On Linux you can create the download test files with:
34*6236dae4SAndroid Build Coastguard Worker  * dd if=/dev/urandom of=file_1M.bin bs=1M count=1
35*6236dae4SAndroid Build Coastguard Worker  *
36*6236dae4SAndroid Build Coastguard Worker  */
37*6236dae4SAndroid Build Coastguard Worker 
38*6236dae4SAndroid Build Coastguard Worker #include <stdio.h>
39*6236dae4SAndroid Build Coastguard Worker #include <stdlib.h>
40*6236dae4SAndroid Build Coastguard Worker #include <time.h>
41*6236dae4SAndroid Build Coastguard Worker 
42*6236dae4SAndroid Build Coastguard Worker #include <curl/curl.h>
43*6236dae4SAndroid Build Coastguard Worker 
44*6236dae4SAndroid Build Coastguard Worker #define URL_BASE "http://speedtest.your.domain/"
45*6236dae4SAndroid Build Coastguard Worker #define URL_1M   URL_BASE "file_1M.bin"
46*6236dae4SAndroid Build Coastguard Worker #define URL_2M   URL_BASE "file_2M.bin"
47*6236dae4SAndroid Build Coastguard Worker #define URL_5M   URL_BASE "file_5M.bin"
48*6236dae4SAndroid Build Coastguard Worker #define URL_10M  URL_BASE "file_10M.bin"
49*6236dae4SAndroid Build Coastguard Worker #define URL_20M  URL_BASE "file_20M.bin"
50*6236dae4SAndroid Build Coastguard Worker #define URL_50M  URL_BASE "file_50M.bin"
51*6236dae4SAndroid Build Coastguard Worker #define URL_100M URL_BASE "file_100M.bin"
52*6236dae4SAndroid Build Coastguard Worker 
53*6236dae4SAndroid Build Coastguard Worker #define CHKSPEED_VERSION "1.0"
54*6236dae4SAndroid Build Coastguard Worker 
WriteCallback(void * ptr,size_t size,size_t nmemb,void * data)55*6236dae4SAndroid Build Coastguard Worker static size_t WriteCallback(void *ptr, size_t size, size_t nmemb, void *data)
56*6236dae4SAndroid Build Coastguard Worker {
57*6236dae4SAndroid Build Coastguard Worker   /* we are not interested in the downloaded bytes itself,
58*6236dae4SAndroid Build Coastguard Worker      so we only return the size we would have saved ... */
59*6236dae4SAndroid Build Coastguard Worker   (void)ptr;  /* unused */
60*6236dae4SAndroid Build Coastguard Worker   (void)data; /* unused */
61*6236dae4SAndroid Build Coastguard Worker   return (size_t)(size * nmemb);
62*6236dae4SAndroid Build Coastguard Worker }
63*6236dae4SAndroid Build Coastguard Worker 
main(int argc,char * argv[])64*6236dae4SAndroid Build Coastguard Worker int main(int argc, char *argv[])
65*6236dae4SAndroid Build Coastguard Worker {
66*6236dae4SAndroid Build Coastguard Worker   CURL *curl_handle;
67*6236dae4SAndroid Build Coastguard Worker   CURLcode res;
68*6236dae4SAndroid Build Coastguard Worker   int prtall = 0, prtsep = 0, prttime = 0;
69*6236dae4SAndroid Build Coastguard Worker   const char *url = URL_1M;
70*6236dae4SAndroid Build Coastguard Worker   char *appname = argv[0];
71*6236dae4SAndroid Build Coastguard Worker 
72*6236dae4SAndroid Build Coastguard Worker   if(argc > 1) {
73*6236dae4SAndroid Build Coastguard Worker     /* parse input parameters */
74*6236dae4SAndroid Build Coastguard Worker     for(argc--, argv++; *argv; argc--, argv++) {
75*6236dae4SAndroid Build Coastguard Worker       if(argv[0][0] == '-') {
76*6236dae4SAndroid Build Coastguard Worker         switch(argv[0][1]) {
77*6236dae4SAndroid Build Coastguard Worker         case 'h':
78*6236dae4SAndroid Build Coastguard Worker         case 'H':
79*6236dae4SAndroid Build Coastguard Worker           fprintf(stderr,
80*6236dae4SAndroid Build Coastguard Worker                   "\rUsage: %s [-m=1|2|5|10|20|50|100] [-t] [-x] [url]\n",
81*6236dae4SAndroid Build Coastguard Worker                   appname);
82*6236dae4SAndroid Build Coastguard Worker           exit(1);
83*6236dae4SAndroid Build Coastguard Worker         case 'v':
84*6236dae4SAndroid Build Coastguard Worker         case 'V':
85*6236dae4SAndroid Build Coastguard Worker           fprintf(stderr, "\r%s %s - %s\n",
86*6236dae4SAndroid Build Coastguard Worker                   appname, CHKSPEED_VERSION, curl_version());
87*6236dae4SAndroid Build Coastguard Worker           exit(1);
88*6236dae4SAndroid Build Coastguard Worker         case 'a':
89*6236dae4SAndroid Build Coastguard Worker         case 'A':
90*6236dae4SAndroid Build Coastguard Worker           prtall = 1;
91*6236dae4SAndroid Build Coastguard Worker           break;
92*6236dae4SAndroid Build Coastguard Worker         case 'x':
93*6236dae4SAndroid Build Coastguard Worker         case 'X':
94*6236dae4SAndroid Build Coastguard Worker           prtsep = 1;
95*6236dae4SAndroid Build Coastguard Worker           break;
96*6236dae4SAndroid Build Coastguard Worker         case 't':
97*6236dae4SAndroid Build Coastguard Worker         case 'T':
98*6236dae4SAndroid Build Coastguard Worker           prttime = 1;
99*6236dae4SAndroid Build Coastguard Worker           break;
100*6236dae4SAndroid Build Coastguard Worker         case 'm':
101*6236dae4SAndroid Build Coastguard Worker         case 'M':
102*6236dae4SAndroid Build Coastguard Worker           if(argv[0][2] == '=') {
103*6236dae4SAndroid Build Coastguard Worker             long m = strtol((*argv) + 3, NULL, 10);
104*6236dae4SAndroid Build Coastguard Worker             switch(m) {
105*6236dae4SAndroid Build Coastguard Worker             case 1:
106*6236dae4SAndroid Build Coastguard Worker               url = URL_1M;
107*6236dae4SAndroid Build Coastguard Worker               break;
108*6236dae4SAndroid Build Coastguard Worker             case 2:
109*6236dae4SAndroid Build Coastguard Worker               url = URL_2M;
110*6236dae4SAndroid Build Coastguard Worker               break;
111*6236dae4SAndroid Build Coastguard Worker             case 5:
112*6236dae4SAndroid Build Coastguard Worker               url = URL_5M;
113*6236dae4SAndroid Build Coastguard Worker               break;
114*6236dae4SAndroid Build Coastguard Worker             case 10:
115*6236dae4SAndroid Build Coastguard Worker               url = URL_10M;
116*6236dae4SAndroid Build Coastguard Worker               break;
117*6236dae4SAndroid Build Coastguard Worker             case 20:
118*6236dae4SAndroid Build Coastguard Worker               url = URL_20M;
119*6236dae4SAndroid Build Coastguard Worker               break;
120*6236dae4SAndroid Build Coastguard Worker             case 50:
121*6236dae4SAndroid Build Coastguard Worker               url = URL_50M;
122*6236dae4SAndroid Build Coastguard Worker               break;
123*6236dae4SAndroid Build Coastguard Worker             case 100:
124*6236dae4SAndroid Build Coastguard Worker               url = URL_100M;
125*6236dae4SAndroid Build Coastguard Worker               break;
126*6236dae4SAndroid Build Coastguard Worker             default:
127*6236dae4SAndroid Build Coastguard Worker               fprintf(stderr, "\r%s: invalid parameter %s\n",
128*6236dae4SAndroid Build Coastguard Worker                       appname, *argv + 3);
129*6236dae4SAndroid Build Coastguard Worker               return 1;
130*6236dae4SAndroid Build Coastguard Worker             }
131*6236dae4SAndroid Build Coastguard Worker             break;
132*6236dae4SAndroid Build Coastguard Worker           }
133*6236dae4SAndroid Build Coastguard Worker           fprintf(stderr, "\r%s: invalid or unknown option %s\n",
134*6236dae4SAndroid Build Coastguard Worker                   appname, *argv);
135*6236dae4SAndroid Build Coastguard Worker           return 1;
136*6236dae4SAndroid Build Coastguard Worker         default:
137*6236dae4SAndroid Build Coastguard Worker           fprintf(stderr, "\r%s: invalid or unknown option %s\n",
138*6236dae4SAndroid Build Coastguard Worker                   appname, *argv);
139*6236dae4SAndroid Build Coastguard Worker           return 1;
140*6236dae4SAndroid Build Coastguard Worker         }
141*6236dae4SAndroid Build Coastguard Worker       }
142*6236dae4SAndroid Build Coastguard Worker       else {
143*6236dae4SAndroid Build Coastguard Worker         url = *argv;
144*6236dae4SAndroid Build Coastguard Worker       }
145*6236dae4SAndroid Build Coastguard Worker     }
146*6236dae4SAndroid Build Coastguard Worker   }
147*6236dae4SAndroid Build Coastguard Worker 
148*6236dae4SAndroid Build Coastguard Worker   /* print separator line */
149*6236dae4SAndroid Build Coastguard Worker   if(prtsep) {
150*6236dae4SAndroid Build Coastguard Worker     printf("-------------------------------------------------\n");
151*6236dae4SAndroid Build Coastguard Worker   }
152*6236dae4SAndroid Build Coastguard Worker   /* print localtime */
153*6236dae4SAndroid Build Coastguard Worker   if(prttime) {
154*6236dae4SAndroid Build Coastguard Worker     time_t t = time(NULL);
155*6236dae4SAndroid Build Coastguard Worker     printf("Localtime: %s", ctime(&t));
156*6236dae4SAndroid Build Coastguard Worker   }
157*6236dae4SAndroid Build Coastguard Worker 
158*6236dae4SAndroid Build Coastguard Worker   /* init libcurl */
159*6236dae4SAndroid Build Coastguard Worker   curl_global_init(CURL_GLOBAL_ALL);
160*6236dae4SAndroid Build Coastguard Worker 
161*6236dae4SAndroid Build Coastguard Worker   /* init the curl session */
162*6236dae4SAndroid Build Coastguard Worker   curl_handle = curl_easy_init();
163*6236dae4SAndroid Build Coastguard Worker 
164*6236dae4SAndroid Build Coastguard Worker   /* specify URL to get */
165*6236dae4SAndroid Build Coastguard Worker   curl_easy_setopt(curl_handle, CURLOPT_URL, url);
166*6236dae4SAndroid Build Coastguard Worker 
167*6236dae4SAndroid Build Coastguard Worker   /* send all data to this function  */
168*6236dae4SAndroid Build Coastguard Worker   curl_easy_setopt(curl_handle, CURLOPT_WRITEFUNCTION, WriteCallback);
169*6236dae4SAndroid Build Coastguard Worker 
170*6236dae4SAndroid Build Coastguard Worker   /* some servers do not like requests that are made without a user-agent
171*6236dae4SAndroid Build Coastguard Worker      field, so we provide one */
172*6236dae4SAndroid Build Coastguard Worker   curl_easy_setopt(curl_handle, CURLOPT_USERAGENT,
173*6236dae4SAndroid Build Coastguard Worker                    "libcurl-speedchecker/" CHKSPEED_VERSION);
174*6236dae4SAndroid Build Coastguard Worker 
175*6236dae4SAndroid Build Coastguard Worker   /* get it! */
176*6236dae4SAndroid Build Coastguard Worker   res = curl_easy_perform(curl_handle);
177*6236dae4SAndroid Build Coastguard Worker 
178*6236dae4SAndroid Build Coastguard Worker   if(CURLE_OK == res) {
179*6236dae4SAndroid Build Coastguard Worker     curl_off_t val;
180*6236dae4SAndroid Build Coastguard Worker 
181*6236dae4SAndroid Build Coastguard Worker     /* check for bytes downloaded */
182*6236dae4SAndroid Build Coastguard Worker     res = curl_easy_getinfo(curl_handle, CURLINFO_SIZE_DOWNLOAD_T, &val);
183*6236dae4SAndroid Build Coastguard Worker     if((CURLE_OK == res) && (val > 0))
184*6236dae4SAndroid Build Coastguard Worker       printf("Data downloaded: %lu bytes.\n", (unsigned long)val);
185*6236dae4SAndroid Build Coastguard Worker 
186*6236dae4SAndroid Build Coastguard Worker     /* check for total download time */
187*6236dae4SAndroid Build Coastguard Worker     res = curl_easy_getinfo(curl_handle, CURLINFO_TOTAL_TIME_T, &val);
188*6236dae4SAndroid Build Coastguard Worker     if((CURLE_OK == res) && (val > 0))
189*6236dae4SAndroid Build Coastguard Worker       printf("Total download time: %lu.%06lu sec.\n",
190*6236dae4SAndroid Build Coastguard Worker              (unsigned long)(val / 1000000), (unsigned long)(val % 1000000));
191*6236dae4SAndroid Build Coastguard Worker 
192*6236dae4SAndroid Build Coastguard Worker     /* check for average download speed */
193*6236dae4SAndroid Build Coastguard Worker     res = curl_easy_getinfo(curl_handle, CURLINFO_SPEED_DOWNLOAD_T, &val);
194*6236dae4SAndroid Build Coastguard Worker     if((CURLE_OK == res) && (val > 0))
195*6236dae4SAndroid Build Coastguard Worker       printf("Average download speed: %lu kbyte/sec.\n",
196*6236dae4SAndroid Build Coastguard Worker              (unsigned long)(val / 1024));
197*6236dae4SAndroid Build Coastguard Worker 
198*6236dae4SAndroid Build Coastguard Worker     if(prtall) {
199*6236dae4SAndroid Build Coastguard Worker       /* check for name resolution time */
200*6236dae4SAndroid Build Coastguard Worker       res = curl_easy_getinfo(curl_handle, CURLINFO_NAMELOOKUP_TIME_T, &val);
201*6236dae4SAndroid Build Coastguard Worker       if((CURLE_OK == res) && (val > 0))
202*6236dae4SAndroid Build Coastguard Worker         printf("Name lookup time: %lu.%06lu sec.\n",
203*6236dae4SAndroid Build Coastguard Worker                (unsigned long)(val / 1000000), (unsigned long)(val % 1000000));
204*6236dae4SAndroid Build Coastguard Worker 
205*6236dae4SAndroid Build Coastguard Worker       /* check for connect time */
206*6236dae4SAndroid Build Coastguard Worker       res = curl_easy_getinfo(curl_handle, CURLINFO_CONNECT_TIME_T, &val);
207*6236dae4SAndroid Build Coastguard Worker       if((CURLE_OK == res) && (val > 0))
208*6236dae4SAndroid Build Coastguard Worker         printf("Connect time: %lu.%06lu sec.\n",
209*6236dae4SAndroid Build Coastguard Worker                (unsigned long)(val / 1000000), (unsigned long)(val % 1000000));
210*6236dae4SAndroid Build Coastguard Worker     }
211*6236dae4SAndroid Build Coastguard Worker   }
212*6236dae4SAndroid Build Coastguard Worker   else {
213*6236dae4SAndroid Build Coastguard Worker     fprintf(stderr, "Error while fetching '%s' : %s\n",
214*6236dae4SAndroid Build Coastguard Worker             url, curl_easy_strerror(res));
215*6236dae4SAndroid Build Coastguard Worker   }
216*6236dae4SAndroid Build Coastguard Worker 
217*6236dae4SAndroid Build Coastguard Worker   /* cleanup curl stuff */
218*6236dae4SAndroid Build Coastguard Worker   curl_easy_cleanup(curl_handle);
219*6236dae4SAndroid Build Coastguard Worker 
220*6236dae4SAndroid Build Coastguard Worker   /* we are done with libcurl, so clean it up */
221*6236dae4SAndroid Build Coastguard Worker   curl_global_cleanup();
222*6236dae4SAndroid Build Coastguard Worker 
223*6236dae4SAndroid Build Coastguard Worker   return 0;
224*6236dae4SAndroid Build Coastguard Worker }
225