123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144 |
- /***************************************************************************
- * _ _ ____ _
- * Project ___| | | | _ \| |
- * / __| | | | |_) | |
- * | (__| |_| | _ <| |___
- * \___|\___/|_| \_\_____|
- *
- * Copyright (C) 1998 - 2018, Daniel Stenberg, <daniel@haxx.se>, et al.
- *
- * This software is licensed as described in the file COPYING, which
- * you should have received as part of this distribution. The terms
- * are also available at https://curl.haxx.se/docs/copyright.html.
- *
- * You may opt to use, copy, modify, merge, publish, distribute and/or sell
- * copies of the Software, and permit persons to whom the Software is
- * furnished to do so, under the terms of the COPYING file.
- *
- * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
- * KIND, either express or implied.
- *
- ***************************************************************************/
- /* <DESC>
- * Use the progress callbacks, old and/or new one depending on available
- * libcurl version.
- * </DESC>
- */
- #include <stdio.h>
- #include <curl/curl.h>
- #if LIBCURL_VERSION_NUM >= 0x073d00
- /* In libcurl 7.61.0, support was added for extracting the time in plain
- microseconds. Older libcurl versions are stuck in using 'double' for this
- information so we complicate this example a bit by supporting either
- approach. */
- #define TIME_IN_US 1 /* microseconds */
- #define TIMETYPE curl_off_t
- #define TIMEOPT CURLINFO_TOTAL_TIME_T
- #define MINIMAL_PROGRESS_FUNCTIONALITY_INTERVAL 3000000
- #else
- #define TIMETYPE double
- #define TIMEOPT CURLINFO_TOTAL_TIME
- #define MINIMAL_PROGRESS_FUNCTIONALITY_INTERVAL 3
- #endif
- #define STOP_DOWNLOAD_AFTER_THIS_MANY_BYTES 6000
- struct myprogress {
- TIMETYPE lastruntime; /* type depends on version, see above */
- CURL *curl;
- };
- /* this is how the CURLOPT_XFERINFOFUNCTION callback works */
- static int xferinfo(void *p,
- curl_off_t dltotal, curl_off_t dlnow,
- curl_off_t ultotal, curl_off_t ulnow)
- {
- struct myprogress *myp = (struct myprogress *)p;
- CURL *curl = myp->curl;
- TIMETYPE curtime = 0;
- curl_easy_getinfo(curl, TIMEOPT, &curtime);
- /* under certain circumstances it may be desirable for certain functionality
- to only run every N seconds, in order to do this the transaction time can
- be used */
- if((curtime - myp->lastruntime) >= MINIMAL_PROGRESS_FUNCTIONALITY_INTERVAL) {
- myp->lastruntime = curtime;
- #ifdef TIME_IN_US
- fprintf(stderr, "TOTAL TIME: %" CURL_FORMAT_CURL_OFF_T ".%06ld\r\n",
- (curtime / 1000000), (long)(curtime % 1000000));
- #else
- fprintf(stderr, "TOTAL TIME: %f \r\n", curtime);
- #endif
- }
- fprintf(stderr, "UP: %" CURL_FORMAT_CURL_OFF_T " of %" CURL_FORMAT_CURL_OFF_T
- " DOWN: %" CURL_FORMAT_CURL_OFF_T " of %" CURL_FORMAT_CURL_OFF_T
- "\r\n",
- ulnow, ultotal, dlnow, dltotal);
- if(dlnow > STOP_DOWNLOAD_AFTER_THIS_MANY_BYTES)
- return 1;
- return 0;
- }
- #if LIBCURL_VERSION_NUM < 0x072000
- /* for libcurl older than 7.32.0 (CURLOPT_PROGRESSFUNCTION) */
- static int older_progress(void *p,
- double dltotal, double dlnow,
- double ultotal, double ulnow)
- {
- return xferinfo(p,
- (curl_off_t)dltotal,
- (curl_off_t)dlnow,
- (curl_off_t)ultotal,
- (curl_off_t)ulnow);
- }
- #endif
- int main(void)
- {
- CURL *curl;
- CURLcode res = CURLE_OK;
- struct myprogress prog;
- curl = curl_easy_init();
- if(curl) {
- prog.lastruntime = 0;
- prog.curl = curl;
- curl_easy_setopt(curl, CURLOPT_URL, "https://example.com/");
- #if LIBCURL_VERSION_NUM >= 0x072000
- /* xferinfo was introduced in 7.32.0, no earlier libcurl versions will
- compile as they won't have the symbols around.
- If built with a newer libcurl, but running with an older libcurl:
- curl_easy_setopt() will fail in run-time trying to set the new
- callback, making the older callback get used.
- New libcurls will prefer the new callback and instead use that one even
- if both callbacks are set. */
- curl_easy_setopt(curl, CURLOPT_XFERINFOFUNCTION, xferinfo);
- /* pass the struct pointer into the xferinfo function, note that this is
- an alias to CURLOPT_PROGRESSDATA */
- curl_easy_setopt(curl, CURLOPT_XFERINFODATA, &prog);
- #else
- curl_easy_setopt(curl, CURLOPT_PROGRESSFUNCTION, older_progress);
- /* pass the struct pointer into the progress function */
- curl_easy_setopt(curl, CURLOPT_PROGRESSDATA, &prog);
- #endif
- curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0L);
- res = curl_easy_perform(curl);
- if(res != CURLE_OK)
- fprintf(stderr, "%s\n", curl_easy_strerror(res));
- /* always cleanup */
- curl_easy_cleanup(curl);
- }
- return (int)res;
- }
|