Home | History | Annotate | Download | only in examples
      1 /***************************************************************************
      2  *                                  _   _ ____  _
      3  *  Project                     ___| | | |  _ \| |
      4  *                             / __| | | | |_) | |
      5  *                            | (__| |_| |  _ <| |___
      6  *                             \___|\___/|_| \_\_____|
      7  *
      8  * Copyright (C) 1998 - 2016, Daniel Stenberg, <daniel (at) 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  * Upload to FTP, resuming failed transfers.
     24  * </DESC>
     25  */
     26 
     27 #include <stdlib.h>
     28 #include <stdio.h>
     29 
     30 #include <curl/curl.h>
     31 
     32 #if defined(_MSC_VER) && (_MSC_VER < 1300)
     33 #  error _snscanf requires MSVC 7.0 or later.
     34 #endif
     35 
     36 /* The MinGW headers are missing a few Win32 function definitions,
     37    you shouldn't need this if you use VC++ */
     38 #if defined(__MINGW32__) && !defined(__MINGW64__)
     39 int __cdecl _snscanf(const char * input, size_t length,
     40                      const char * format, ...);
     41 #endif
     42 
     43 
     44 /* parse headers for Content-Length */
     45 size_t getcontentlengthfunc(void *ptr, size_t size, size_t nmemb, void *stream)
     46 {
     47   int r;
     48   long len = 0;
     49 
     50   /* _snscanf() is Win32 specific */
     51   r = _snscanf(ptr, size * nmemb, "Content-Length: %ld\n", &len);
     52 
     53   if(r) /* Microsoft: we don't read the specs */
     54     *((long *) stream) = len;
     55 
     56   return size * nmemb;
     57 }
     58 
     59 /* discard downloaded data */
     60 size_t discardfunc(void *ptr, size_t size, size_t nmemb, void *stream)
     61 {
     62   return size * nmemb;
     63 }
     64 
     65 /* read data to upload */
     66 size_t readfunc(void *ptr, size_t size, size_t nmemb, void *stream)
     67 {
     68   FILE *f = stream;
     69   size_t n;
     70 
     71   if(ferror(f))
     72     return CURL_READFUNC_ABORT;
     73 
     74   n = fread(ptr, size, nmemb, f) * size;
     75 
     76   return n;
     77 }
     78 
     79 
     80 int upload(CURL *curlhandle, const char * remotepath, const char * localpath,
     81            long timeout, long tries)
     82 {
     83   FILE *f;
     84   long uploaded_len = 0;
     85   CURLcode r = CURLE_GOT_NOTHING;
     86   int c;
     87 
     88   f = fopen(localpath, "rb");
     89   if(!f) {
     90     perror(NULL);
     91     return 0;
     92   }
     93 
     94   curl_easy_setopt(curlhandle, CURLOPT_UPLOAD, 1L);
     95 
     96   curl_easy_setopt(curlhandle, CURLOPT_URL, remotepath);
     97 
     98   if(timeout)
     99     curl_easy_setopt(curlhandle, CURLOPT_FTP_RESPONSE_TIMEOUT, timeout);
    100 
    101   curl_easy_setopt(curlhandle, CURLOPT_HEADERFUNCTION, getcontentlengthfunc);
    102   curl_easy_setopt(curlhandle, CURLOPT_HEADERDATA, &uploaded_len);
    103 
    104   curl_easy_setopt(curlhandle, CURLOPT_WRITEFUNCTION, discardfunc);
    105 
    106   curl_easy_setopt(curlhandle, CURLOPT_READFUNCTION, readfunc);
    107   curl_easy_setopt(curlhandle, CURLOPT_READDATA, f);
    108 
    109   /* disable passive mode */
    110   curl_easy_setopt(curlhandle, CURLOPT_FTPPORT, "-");
    111   curl_easy_setopt(curlhandle, CURLOPT_FTP_CREATE_MISSING_DIRS, 1L);
    112 
    113   curl_easy_setopt(curlhandle, CURLOPT_VERBOSE, 1L);
    114 
    115   for(c = 0; (r != CURLE_OK) && (c < tries); c++) {
    116     /* are we resuming? */
    117     if(c) { /* yes */
    118       /* determine the length of the file already written */
    119 
    120       /*
    121        * With NOBODY and NOHEADER, libcurl will issue a SIZE
    122        * command, but the only way to retrieve the result is
    123        * to parse the returned Content-Length header. Thus,
    124        * getcontentlengthfunc(). We need discardfunc() above
    125        * because HEADER will dump the headers to stdout
    126        * without it.
    127        */
    128       curl_easy_setopt(curlhandle, CURLOPT_NOBODY, 1L);
    129       curl_easy_setopt(curlhandle, CURLOPT_HEADER, 1L);
    130 
    131       r = curl_easy_perform(curlhandle);
    132       if(r != CURLE_OK)
    133         continue;
    134 
    135       curl_easy_setopt(curlhandle, CURLOPT_NOBODY, 0L);
    136       curl_easy_setopt(curlhandle, CURLOPT_HEADER, 0L);
    137 
    138       fseek(f, uploaded_len, SEEK_SET);
    139 
    140       curl_easy_setopt(curlhandle, CURLOPT_APPEND, 1L);
    141     }
    142     else { /* no */
    143       curl_easy_setopt(curlhandle, CURLOPT_APPEND, 0L);
    144     }
    145 
    146     r = curl_easy_perform(curlhandle);
    147   }
    148 
    149   fclose(f);
    150 
    151   if(r == CURLE_OK)
    152     return 1;
    153   else {
    154     fprintf(stderr, "%s\n", curl_easy_strerror(r));
    155     return 0;
    156   }
    157 }
    158 
    159 int main(int c, char **argv)
    160 {
    161   CURL *curlhandle = NULL;
    162 
    163   curl_global_init(CURL_GLOBAL_ALL);
    164   curlhandle = curl_easy_init();
    165 
    166   upload(curlhandle, "ftp://user:pass@example.com/path/file", "C:\\file",
    167          0, 3);
    168 
    169   curl_easy_cleanup(curlhandle);
    170   curl_global_cleanup();
    171 
    172   return 0;
    173 }
    174