| lh | 9ed821d | 2023-04-07 01:36:19 -0700 | [diff] [blame] | 1 | /*************************************************************************** | 
|  | 2 | *                                  _   _ ____  _ | 
|  | 3 | *  Project                     ___| | | |  _ \| | | 
|  | 4 | *                             / __| | | | |_) | | | 
|  | 5 | *                            | (__| |_| |  _ <| |___ | 
|  | 6 | *                             \___|\___/|_| \_\_____| | 
|  | 7 | * | 
|  | 8 | * Copyright (C) 1998 - 2016, 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 | * 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 | } |