2011-03-10 18:48:02 +08:00
|
|
|
/***************************************************************************
|
2010-06-24 23:22:47 +08:00
|
|
|
* _ _ ____ _
|
|
|
|
* Project ___| | | | _ \| |
|
|
|
|
* / __| | | | |_) | |
|
|
|
|
* | (__| |_| | _ <| |___
|
|
|
|
* \___|\___/|_| \_\_____|
|
|
|
|
*
|
2023-01-02 20:51:48 +08:00
|
|
|
* Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
|
2011-03-10 18:48:02 +08:00
|
|
|
*
|
|
|
|
* This software is licensed as described in the file COPYING, which
|
|
|
|
* you should have received as part of this distribution. The terms
|
2020-11-04 21:02:01 +08:00
|
|
|
* are also available at https://curl.se/docs/copyright.html.
|
2011-03-10 18:48:02 +08:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: curl
|
2022-05-17 17:16:50 +08:00
|
|
|
*
|
2011-03-10 18:48:02 +08:00
|
|
|
***************************************************************************/
|
2015-06-18 17:38:54 +08:00
|
|
|
/* <DESC>
|
|
|
|
* FTP wildcard pattern matching
|
|
|
|
* </DESC>
|
|
|
|
*/
|
2010-06-24 23:22:47 +08:00
|
|
|
#include <curl/curl.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
struct callback_data {
|
|
|
|
FILE *output;
|
|
|
|
};
|
|
|
|
|
2014-09-09 17:10:04 +08:00
|
|
|
static long file_is_coming(struct curl_fileinfo *finfo,
|
|
|
|
struct callback_data *data,
|
|
|
|
int remains);
|
2010-06-24 23:22:47 +08:00
|
|
|
|
|
|
|
static long file_is_downloaded(struct callback_data *data);
|
|
|
|
|
|
|
|
static size_t write_it(char *buff, size_t size, size_t nmemb,
|
2012-04-13 23:58:41 +08:00
|
|
|
void *cb_data);
|
2010-06-24 23:22:47 +08:00
|
|
|
|
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
/* curl easy handle */
|
|
|
|
CURL *handle;
|
|
|
|
|
|
|
|
/* help data */
|
|
|
|
struct callback_data data = { 0 };
|
|
|
|
|
|
|
|
/* global initialization */
|
2019-05-25 16:06:08 +08:00
|
|
|
int rc = curl_global_init(CURL_GLOBAL_ALL);
|
2010-06-24 23:22:47 +08:00
|
|
|
if(rc)
|
|
|
|
return rc;
|
|
|
|
|
|
|
|
/* initialization of easy handle */
|
|
|
|
handle = curl_easy_init();
|
|
|
|
if(!handle) {
|
|
|
|
curl_global_cleanup();
|
|
|
|
return CURLE_OUT_OF_MEMORY;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* turn on wildcard matching */
|
|
|
|
curl_easy_setopt(handle, CURLOPT_WILDCARDMATCH, 1L);
|
|
|
|
|
|
|
|
/* callback is called before download of concrete file started */
|
2014-09-09 17:10:04 +08:00
|
|
|
curl_easy_setopt(handle, CURLOPT_CHUNK_BGN_FUNCTION, file_is_coming);
|
2010-06-24 23:22:47 +08:00
|
|
|
|
|
|
|
/* callback is called after data from the file have been transferred */
|
|
|
|
curl_easy_setopt(handle, CURLOPT_CHUNK_END_FUNCTION, file_is_downloaded);
|
|
|
|
|
|
|
|
/* this callback will write contents into files */
|
|
|
|
curl_easy_setopt(handle, CURLOPT_WRITEFUNCTION, write_it);
|
|
|
|
|
|
|
|
/* put transfer data into callbacks */
|
|
|
|
curl_easy_setopt(handle, CURLOPT_CHUNK_DATA, &data);
|
|
|
|
curl_easy_setopt(handle, CURLOPT_WRITEDATA, &data);
|
|
|
|
|
|
|
|
/* curl_easy_setopt(handle, CURLOPT_VERBOSE, 1L); */
|
|
|
|
|
2022-10-11 22:01:37 +08:00
|
|
|
/* set a URL containing wildcard pattern (only in the last part) */
|
2010-06-24 23:22:47 +08:00
|
|
|
if(argc == 2)
|
|
|
|
curl_easy_setopt(handle, CURLOPT_URL, argv[1]);
|
|
|
|
else
|
2010-10-05 21:00:19 +08:00
|
|
|
curl_easy_setopt(handle, CURLOPT_URL, "ftp://example.com/test/*");
|
2010-06-24 23:22:47 +08:00
|
|
|
|
|
|
|
/* and start transfer! */
|
|
|
|
rc = curl_easy_perform(handle);
|
|
|
|
|
|
|
|
curl_easy_cleanup(handle);
|
|
|
|
curl_global_cleanup();
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:10:04 +08:00
|
|
|
static long file_is_coming(struct curl_fileinfo *finfo,
|
|
|
|
struct callback_data *data,
|
|
|
|
int remains)
|
2010-06-24 23:22:47 +08:00
|
|
|
{
|
|
|
|
printf("%3d %40s %10luB ", remains, finfo->filename,
|
|
|
|
(unsigned long)finfo->size);
|
|
|
|
|
|
|
|
switch(finfo->filetype) {
|
|
|
|
case CURLFILETYPE_DIRECTORY:
|
|
|
|
printf(" DIR\n");
|
|
|
|
break;
|
|
|
|
case CURLFILETYPE_FILE:
|
|
|
|
printf("FILE ");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
printf("OTHER\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(finfo->filetype == CURLFILETYPE_FILE) {
|
|
|
|
/* do not transfer files >= 50B */
|
|
|
|
if(finfo->size > 50) {
|
|
|
|
printf("SKIPPED\n");
|
|
|
|
return CURL_CHUNK_BGN_FUNC_SKIP;
|
|
|
|
}
|
|
|
|
|
2016-02-11 16:42:38 +08:00
|
|
|
data->output = fopen(finfo->filename, "wb");
|
2010-06-24 23:22:47 +08:00
|
|
|
if(!data->output) {
|
|
|
|
return CURL_CHUNK_BGN_FUNC_FAIL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return CURL_CHUNK_BGN_FUNC_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static long file_is_downloaded(struct callback_data *data)
|
|
|
|
{
|
|
|
|
if(data->output) {
|
|
|
|
printf("DOWNLOADED\n");
|
|
|
|
fclose(data->output);
|
|
|
|
data->output = 0x0;
|
|
|
|
}
|
|
|
|
return CURL_CHUNK_END_FUNC_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t write_it(char *buff, size_t size, size_t nmemb,
|
2012-04-13 23:58:41 +08:00
|
|
|
void *cb_data)
|
2010-06-24 23:22:47 +08:00
|
|
|
{
|
2012-04-13 23:58:41 +08:00
|
|
|
struct callback_data *data = cb_data;
|
2010-06-24 23:22:47 +08:00
|
|
|
size_t written = 0;
|
|
|
|
if(data->output)
|
|
|
|
written = fwrite(buff, size, nmemb, data->output);
|
|
|
|
else
|
|
|
|
/* listing output */
|
|
|
|
written = fwrite(buff, size, nmemb, stdout);
|
|
|
|
return written;
|
|
|
|
}
|