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