#include "../multiif.h"
#include "../select.h"
#include "../curlx/warnless.h"
-#include "curl_path.h"
+#include "vssh.h"
#ifdef HAVE_UNISTD_H
#include <unistd.h>
return myssh_to_ERROR(data, sshc, CURLE_BAD_DOWNLOAD_RESUME);
}
if(data->state.use_range) {
- curl_off_t from, to;
- const char *p = data->state.range;
- int from_t, to_t;
-
- from_t = curlx_str_number(&p, &from, CURL_OFF_T_MAX);
- if(from_t == STRE_OVERFLOW)
- return myssh_to_ERROR(data, sshc, CURLE_RANGE_ERROR);
-
- curlx_str_passblanks(&p);
- (void)curlx_str_single(&p, '-');
-
- to_t = curlx_str_numblanks(&p, &to);
- if(to_t == STRE_OVERFLOW)
- return myssh_to_ERROR(data, sshc, CURLE_RANGE_ERROR);
-
- if((to_t == STRE_NO_NUM) || (to >= size)) {
- to = size - 1;
- }
-
- if(from_t == STRE_NO_NUM) {
- /* from is relative to end of file */
- from = size - to;
- to = size - 1;
- }
- if(from > size) {
- failf(data, "Offset (%" FMT_OFF_T ") was beyond file size (%"
- FMT_OFF_T ")", from, size);
- return myssh_to_ERROR(data, sshc, CURLE_BAD_DOWNLOAD_RESUME);
- }
- if(from > to) {
- from = to;
- size = 0;
- }
- else {
- if((to - from) == CURL_OFF_T_MAX)
- return myssh_to_ERROR(data, sshc, CURLE_RANGE_ERROR);
- size = to - from + 1;
- }
+ curl_off_t from;
+ CURLcode result = Curl_ssh_range(data, data->state.range, size,
+ &from, &size);
+ if(result)
+ return myssh_to_ERROR(data, sshc, result);
rc = sftp_seek64(sshc->sftp_file, from);
if(rc)
#include "../select.h"
#include "../curlx/fopen.h"
#include "../curlx/warnless.h"
-#include "curl_path.h"
+#include "vssh.h"
#include "../curlx/strparse.h"
#include "../curlx/base64.h" /* for base64 encoding/decoding */
return CURLE_BAD_DOWNLOAD_RESUME;
}
if(data->state.use_range) {
- curl_off_t from, to;
- const char *p = data->state.range;
- int to_t, from_t;
-
- from_t = curlx_str_number(&p, &from, CURL_OFF_T_MAX);
- if(from_t == STRE_OVERFLOW)
- return CURLE_RANGE_ERROR;
- curlx_str_passblanks(&p);
- (void)curlx_str_single(&p, '-');
-
- to_t = curlx_str_numblanks(&p, &to);
- if(to_t == STRE_OVERFLOW)
- return CURLE_RANGE_ERROR;
- if((to_t == STRE_NO_NUM) /* no "to" value given */
- || (to >= size)) {
- to = size - 1;
- }
- if(from_t) {
- /* from is relative to end of file */
- from = size - to;
- to = size - 1;
- }
- if(from > size) {
- failf(data, "Offset (%" FMT_OFF_T ") was beyond file size (%"
- FMT_OFF_T ")", from, (curl_off_t)attrs.filesize);
- return CURLE_BAD_DOWNLOAD_RESUME;
- }
- if(from > to) {
- from = to;
- size = 0;
- }
- else {
- if((to - from) == CURL_OFF_T_MAX)
- return CURLE_RANGE_ERROR;
- size = to - from + 1;
- }
+ curl_off_t from;
+ CURLcode result = Curl_ssh_range(data, data->state.range, size,
+ &from, &size);
+ if(result)
+ return result;
libssh2_sftp_seek64(sshc->sftp_handle, (libssh2_uint64_t)from);
}
#ifdef USE_SSH
-#include "curl_path.h"
+#include "vssh.h"
#include <curl/curl.h>
#include "../curlx/strparse.h"
+#include "../curl_trc.h"
#include "../curl_memory.h"
#include "../escape.h"
#include "../memdebug.h"
return CURLE_QUOTE_ERROR;
}
+CURLcode Curl_ssh_range(struct Curl_easy *data,
+ const char *p, curl_off_t filesize,
+ curl_off_t *startp, curl_off_t *sizep)
+{
+ curl_off_t from, to;
+ int to_t;
+ int from_t = curlx_str_number(&p, &from, CURL_OFF_T_MAX);
+ if(from_t == STRE_OVERFLOW)
+ return CURLE_RANGE_ERROR;
+ curlx_str_passblanks(&p);
+ (void)curlx_str_single(&p, '-');
+
+ to_t = curlx_str_numblanks(&p, &to);
+ if((to_t == STRE_OVERFLOW) || (to_t && from_t) || *p)
+ return CURLE_RANGE_ERROR;
+
+ if(from_t) {
+ /* no start point given, set from relative to end of file */
+ if(!to)
+ /* "-0" is not a fine range */
+ return CURLE_RANGE_ERROR;
+ else if(to > filesize)
+ to = filesize;
+ from = filesize - to;
+ to = filesize - 1;
+ }
+ else if(from > filesize) {
+ failf(data, "Offset (%" FMT_OFF_T ") was beyond file size (%"
+ FMT_OFF_T ")", from, filesize);
+ return CURLE_RANGE_ERROR;
+ }
+ else if((to_t == STRE_NO_NUM) || (to >= filesize))
+ to = filesize - 1;
+
+ if(from > to) {
+ failf(data, "Bad range: start offset larger than end offset");
+ return CURLE_RANGE_ERROR;
+ }
+ if((to - from) == CURL_OFF_T_MAX)
+ return CURLE_RANGE_ERROR;
+
+ *startp = from;
+ *sizep = to - from + 1;
+ return CURLE_OK;
+}
+
#endif /* if SSH is used */
--- /dev/null
+/***************************************************************************
+ * _ _ ____ _
+ * Project ___| | | | _ \| |
+ * / __| | | | |_) | |
+ * | (__| |_| | _ <| |___
+ * \___|\___/|_| \_\_____|
+ *
+ * Copyright (C) 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.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.
+ *
+ * SPDX-License-Identifier: curl
+ *
+ ***************************************************************************/
+#include "unitcheck.h"
+#include "vssh/vssh.h"
+#include "memdebug.h"
+
+static CURLcode test_unit2605(const char *arg)
+{
+ UNITTEST_BEGIN_SIMPLE
+
+#ifdef USE_SSH
+ CURL *curl;
+ struct range {
+ const char *r;
+ curl_off_t filesize;
+ curl_off_t start;
+ curl_off_t size;
+ CURLcode result;
+ };
+
+ int i;
+ struct range list[] = {
+ { "0-9", 100, 0, 10, CURLE_OK},
+ { "1-10", 100, 1, 10, CURLE_OK},
+ { "222222-222222", 300000, 222222, 1, CURLE_OK},
+ { "4294967296 - 4294967297", 4294967298, 4294967296, 2, CURLE_OK},
+ { "-10", 100, 90, 10, CURLE_OK},
+ { "-20", 100, 80, 20, CURLE_OK},
+ { "-1", 100, 99, 1, CURLE_OK},
+ { "-0", 100, 0, 0, CURLE_RANGE_ERROR},
+ { "--2", 100, 0, 0, CURLE_RANGE_ERROR},
+ { "-100", 100, 0, 100, CURLE_OK},
+ { "-101", 100, 0, 100, CURLE_OK},
+ { "-1000", 100, 0, 100, CURLE_OK},
+ { "2-1000", 100, 2, 98, CURLE_OK},
+ { ".2-3", 100, 0, 0, CURLE_RANGE_ERROR},
+ { "+2-3", 100, 0, 0, CURLE_RANGE_ERROR},
+ { "2 - 3", 100, 2, 2, CURLE_OK},
+ { " 2 - 3", 100, 2, 2, CURLE_RANGE_ERROR}, /* no leading space */
+ { "2 - 3 ", 100, 2, 2, CURLE_RANGE_ERROR}, /* no trailing space */
+ { "3-2", 100, 0, 0, CURLE_RANGE_ERROR},
+ { "2.-3", 100, 0, 0, CURLE_RANGE_ERROR},
+ { "-3-2", 100, 0, 0, CURLE_RANGE_ERROR},
+ { "101-102", 100, 0, 0, CURLE_RANGE_ERROR},
+ { "0-", 100, 0, 100, CURLE_OK},
+ { "1-", 100, 1, 99, CURLE_OK},
+ { "99-", 100, 99, 1, CURLE_OK},
+ { "100-", 100, 0, 0, CURLE_RANGE_ERROR},
+ { NULL, 0, 0, 0, CURLE_OK }
+ };
+
+ curl_global_init(CURL_GLOBAL_ALL);
+ curl = curl_easy_init();
+ if(curl) {
+ curl_easy_setopt(curl, CURLOPT_VERBOSE, 1L);
+
+ for(i = 0; list[i].r; i++) {
+ curl_off_t start;
+ curl_off_t size;
+ CURLcode result;
+ curl_mprintf("%u: '%s' (file size: %" FMT_OFF_T ")\n",
+ i, list[i].r, list[i].filesize);
+ result = Curl_ssh_range(curl, list[i].r, list[i].filesize,
+ &start, &size);
+ if(result != list[i].result) {
+ curl_mprintf("... returned %d\n", result);
+ unitfail++;
+ }
+ if(!result) {
+ if(start != list[i].start) {
+ curl_mprintf("... start (%" FMT_OFF_T ") was not %" FMT_OFF_T " \n",
+ start, list[i].start);
+ unitfail++;
+ }
+ if(size != list[i].size) {
+ curl_mprintf("... size (%" FMT_OFF_T ") was not %" FMT_OFF_T " \n",
+ size, list[i].size);
+ unitfail++;
+ }
+ }
+ }
+ curl_easy_cleanup(curl);
+ }
+ curl_global_cleanup();
+
+ if(!unitfail)
+ curl_mprintf("ok\n");
+
+#endif
+
+ UNITTEST_END_SIMPLE
+}