2008-08-06 18:36:30 +00:00
|
|
|
// This file is part of BOINC.
|
2005-01-20 23:22:22 +00:00
|
|
|
// http://boinc.berkeley.edu
|
2008-08-06 18:36:30 +00:00
|
|
|
// Copyright (C) 2008 University of California
|
2003-07-02 02:05:44 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// BOINC is free software; you can redistribute it and/or modify it
|
|
|
|
// under the terms of the GNU Lesser General Public License
|
|
|
|
// as published by the Free Software Foundation,
|
|
|
|
// either version 3 of the License, or (at your option) any later version.
|
2003-07-02 02:05:44 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// BOINC is distributed in the hope that it will be useful,
|
2005-01-20 23:22:22 +00:00
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
// See the GNU Lesser General Public License for more details.
|
2002-06-10 06:14:18 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with BOINC. If not, see <http://www.gnu.org/licenses/>.
|
2002-06-10 06:14:18 +00:00
|
|
|
|
2003-10-16 19:03:49 +00:00
|
|
|
#include "cpp.h"
|
2002-08-09 21:43:19 +00:00
|
|
|
|
2004-03-04 11:41:43 +00:00
|
|
|
#ifdef _WIN32
|
2004-06-16 23:16:08 +00:00
|
|
|
#include "boinc_win.h"
|
2010-05-11 19:10:29 +00:00
|
|
|
#else
|
2005-11-21 18:34:44 +00:00
|
|
|
#include "config.h"
|
2004-07-13 13:54:09 +00:00
|
|
|
#include <cmath>
|
|
|
|
#include <cstdlib>
|
2004-03-04 11:41:43 +00:00
|
|
|
#endif
|
2002-08-07 22:52:10 +00:00
|
|
|
|
|
|
|
#include "error_numbers.h"
|
2003-06-16 21:23:58 +00:00
|
|
|
#include "md5_file.h"
|
2002-08-07 22:52:10 +00:00
|
|
|
#include "parse.h"
|
2010-05-07 20:08:59 +00:00
|
|
|
#include "util.h"
|
2007-02-21 16:26:51 +00:00
|
|
|
#include "str_util.h"
|
2003-08-01 21:20:20 +00:00
|
|
|
#include "filesys.h"
|
2002-08-07 22:52:10 +00:00
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
#include "log_flags.h"
|
|
|
|
#include "file_names.h"
|
|
|
|
#include "client_state.h"
|
|
|
|
#include "client_types.h"
|
|
|
|
#include "client_msgs.h"
|
|
|
|
|
2004-06-30 18:17:21 +00:00
|
|
|
using std::vector;
|
|
|
|
|
2003-02-04 21:47:12 +00:00
|
|
|
PERS_FILE_XFER::PERS_FILE_XFER() {
|
2002-08-07 22:52:10 +00:00
|
|
|
nretry = 0;
|
2005-06-07 19:22:50 +00:00
|
|
|
first_request_time = gstate.now;
|
2011-12-06 16:56:27 +00:00
|
|
|
is_upload = false;
|
2002-08-07 22:52:10 +00:00
|
|
|
next_request_time = first_request_time;
|
2003-02-06 03:39:10 +00:00
|
|
|
time_so_far = 0;
|
2011-12-06 16:56:27 +00:00
|
|
|
last_time = 0;
|
2006-02-18 01:57:24 +00:00
|
|
|
last_bytes_xferred = 0;
|
2007-01-19 20:56:49 +00:00
|
|
|
pers_xfer_done = false;
|
2003-07-23 22:24:28 +00:00
|
|
|
fxp = NULL;
|
2011-12-06 16:56:27 +00:00
|
|
|
fip = NULL;
|
2003-07-23 22:24:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
PERS_FILE_XFER::~PERS_FILE_XFER() {
|
|
|
|
if (fip) {
|
|
|
|
fip->pers_file_xfer = NULL;
|
|
|
|
}
|
2003-02-04 21:47:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int PERS_FILE_XFER::init(FILE_INFO* f, bool is_file_upload) {
|
|
|
|
fxp = NULL;
|
|
|
|
fip = f;
|
2002-08-07 22:52:10 +00:00
|
|
|
is_upload = is_file_upload;
|
2004-09-04 05:21:33 +00:00
|
|
|
pers_xfer_done = false;
|
2011-07-20 19:12:10 +00:00
|
|
|
URL_LIST ul = f->get_url_list(is_upload);
|
|
|
|
const char* p = ul.get_init_url();
|
2004-07-06 18:31:56 +00:00
|
|
|
if (!p) {
|
2007-01-25 23:39:06 +00:00
|
|
|
msg_printf(NULL, MSG_INTERNAL_ERROR, "No URL for file transfer of %s", f->name);
|
2004-07-06 18:31:56 +00:00
|
|
|
return ERR_NULL;
|
|
|
|
}
|
2002-08-07 22:52:10 +00:00
|
|
|
return 0;
|
2002-06-10 06:14:18 +00:00
|
|
|
}
|
|
|
|
|
2003-05-16 19:22:57 +00:00
|
|
|
int PERS_FILE_XFER::start_xfer() {
|
2007-09-24 18:32:55 +00:00
|
|
|
if (is_upload) {
|
|
|
|
if (gstate.exit_before_upload) {
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
return fxp->init_upload(*fip);
|
|
|
|
} else {
|
|
|
|
return fxp->init_download(*fip);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Possibly create and start a file transfer
|
|
|
|
//
|
|
|
|
int PERS_FILE_XFER::create_xfer() {
|
2002-08-07 22:52:10 +00:00
|
|
|
FILE_XFER *file_xfer;
|
|
|
|
int retval;
|
2003-02-26 00:47:57 +00:00
|
|
|
|
2002-08-07 22:52:10 +00:00
|
|
|
// Decide whether to start a new file transfer
|
2002-08-14 20:31:27 +00:00
|
|
|
//
|
2003-12-02 22:47:32 +00:00
|
|
|
if (!gstate.start_new_file_xfer(*this)) {
|
2003-05-16 19:22:57 +00:00
|
|
|
return ERR_IDLE_PERIOD;
|
2002-08-21 19:12:42 +00:00
|
|
|
}
|
2003-06-09 23:50:49 +00:00
|
|
|
|
2009-07-16 20:14:57 +00:00
|
|
|
// if download, see if file already exists and is valid
|
2003-12-02 22:47:32 +00:00
|
|
|
//
|
2003-08-01 21:20:20 +00:00
|
|
|
if (!is_upload) {
|
|
|
|
char pathname[256];
|
2007-03-13 19:33:27 +00:00
|
|
|
get_pathname(fip, pathname, sizeof(pathname));
|
2005-04-06 19:41:31 +00:00
|
|
|
|
2007-01-19 20:56:49 +00:00
|
|
|
if (!fip->verify_file(true, false)) {
|
2005-04-12 21:54:40 +00:00
|
|
|
retval = fip->set_permissions();
|
|
|
|
fip->status = FILE_PRESENT;
|
|
|
|
pers_xfer_done = true;
|
2003-08-01 21:20:20 +00:00
|
|
|
|
2006-12-06 20:59:29 +00:00
|
|
|
if (log_flags.file_xfer) {
|
|
|
|
msg_printf(
|
|
|
|
fip->project, MSG_INFO,
|
|
|
|
"File %s exists already, skipping download", fip->name
|
|
|
|
);
|
|
|
|
}
|
2004-07-13 13:54:09 +00:00
|
|
|
|
2005-04-12 21:54:40 +00:00
|
|
|
return 0;
|
2005-08-16 17:43:32 +00:00
|
|
|
} else {
|
2009-09-23 19:44:21 +00:00
|
|
|
// Mark file as not present but don't delete it.
|
2012-02-06 06:06:44 +00:00
|
|
|
// It might be partly downloaded.
|
2009-09-23 19:44:21 +00:00
|
|
|
//
|
2005-08-16 17:43:32 +00:00
|
|
|
fip->status = FILE_NOT_PRESENT;
|
2003-08-01 21:20:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-20 19:12:10 +00:00
|
|
|
URL_LIST& ul = fip->get_url_list(is_upload);
|
2002-08-21 19:12:42 +00:00
|
|
|
file_xfer = new FILE_XFER;
|
2004-06-03 21:19:18 +00:00
|
|
|
fxp = file_xfer;
|
2007-09-24 18:32:55 +00:00
|
|
|
retval = start_xfer();
|
2004-10-25 20:16:30 +00:00
|
|
|
if (!retval) retval = gstate.file_xfers->insert(file_xfer);
|
2002-08-21 19:12:42 +00:00
|
|
|
if (retval) {
|
2007-01-25 23:39:06 +00:00
|
|
|
if (log_flags.http_debug) {
|
|
|
|
msg_printf(
|
2010-04-29 20:32:51 +00:00
|
|
|
fip->project, MSG_INFO, "[file_xfer] Couldn't start %s of %s",
|
2007-01-25 23:39:06 +00:00
|
|
|
(is_upload ? "upload" : "download"), fip->name
|
|
|
|
);
|
|
|
|
msg_printf(
|
2010-04-29 20:32:51 +00:00
|
|
|
fip->project, MSG_INFO, "[file_xfer] URL %s: %s",
|
2011-07-20 19:12:10 +00:00
|
|
|
ul.get_current_url(*fip), boincerror(retval)
|
2007-01-25 23:39:06 +00:00
|
|
|
);
|
|
|
|
}
|
2004-10-25 20:16:30 +00:00
|
|
|
|
2003-05-16 19:22:57 +00:00
|
|
|
fxp->file_xfer_retval = retval;
|
2012-02-07 07:58:08 +00:00
|
|
|
if (retval == ERR_HTTP_PERMANENT) {
|
2011-12-17 05:36:45 +00:00
|
|
|
permanent_failure(retval);
|
|
|
|
} else {
|
|
|
|
transient_failure(retval);
|
|
|
|
}
|
2003-05-20 00:03:39 +00:00
|
|
|
delete fxp;
|
2003-05-16 19:22:57 +00:00
|
|
|
fxp = NULL;
|
|
|
|
return retval;
|
2002-08-07 22:52:10 +00:00
|
|
|
}
|
2003-05-16 19:22:57 +00:00
|
|
|
if (log_flags.file_xfer) {
|
2003-07-03 05:01:29 +00:00
|
|
|
msg_printf(
|
2007-10-17 23:30:05 +00:00
|
|
|
fip->project, MSG_INFO, "Started %s of %s",
|
2003-07-03 05:01:29 +00:00
|
|
|
(is_upload ? "upload" : "download"), fip->name
|
|
|
|
);
|
2003-05-16 19:22:57 +00:00
|
|
|
}
|
2006-06-22 19:40:30 +00:00
|
|
|
if (log_flags.file_xfer_debug) {
|
2009-02-22 20:54:33 +00:00
|
|
|
msg_printf(fip->project, MSG_INFO,
|
2010-04-29 20:32:51 +00:00
|
|
|
"[file_xfer] URL: %s\n",
|
2011-07-20 19:12:10 +00:00
|
|
|
ul.get_current_url(*fip)
|
2006-06-22 19:40:30 +00:00
|
|
|
);
|
|
|
|
}
|
2003-05-16 19:22:57 +00:00
|
|
|
return 0;
|
2002-06-10 06:14:18 +00:00
|
|
|
}
|
|
|
|
|
2002-08-14 20:31:27 +00:00
|
|
|
// Poll the status of this persistent file transfer.
|
|
|
|
// If it's time to start it, then attempt to start it.
|
2011-11-04 20:25:30 +00:00
|
|
|
// If it has finished or failed:
|
|
|
|
// handle the success or failure
|
|
|
|
// remove the FILE_XFER from gstate.file_xfers and delete it
|
2012-02-06 06:06:44 +00:00
|
|
|
// Return true if it finished
|
2002-08-07 22:52:10 +00:00
|
|
|
//
|
2005-06-07 19:22:50 +00:00
|
|
|
bool PERS_FILE_XFER::poll() {
|
2004-09-04 05:21:33 +00:00
|
|
|
if (pers_xfer_done) {
|
2002-08-22 00:16:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!fxp) {
|
2002-08-14 20:31:27 +00:00
|
|
|
// No file xfer is active.
|
2003-12-02 22:47:32 +00:00
|
|
|
// Either initial or resume after failure.
|
2002-08-14 20:31:27 +00:00
|
|
|
// See if it's time to try again.
|
|
|
|
//
|
2005-09-30 21:29:31 +00:00
|
|
|
if (gstate.now < next_request_time) {
|
|
|
|
return false;
|
|
|
|
}
|
2009-07-16 16:35:57 +00:00
|
|
|
FILE_XFER_BACKOFF& fxb = fip->project->file_xfer_backoff(is_upload);
|
2011-10-06 06:13:33 +00:00
|
|
|
if (!fxb.ok_to_transfer() && nretry>0) {
|
2009-07-16 19:11:26 +00:00
|
|
|
#if 0
|
2009-07-10 17:06:06 +00:00
|
|
|
if (log_flags.file_xfer_debug) {
|
|
|
|
msg_printf(fip->project, MSG_INFO,
|
2010-04-29 20:32:51 +00:00
|
|
|
"[file_xfer] delaying %s of %s: project-wide backoff %f sec",
|
2009-07-16 16:35:57 +00:00
|
|
|
is_upload?"upload":"download", fip->name,
|
|
|
|
fxb.next_xfer_time - gstate.now
|
2009-07-10 17:06:06 +00:00
|
|
|
);
|
|
|
|
}
|
2009-07-16 19:11:26 +00:00
|
|
|
#endif
|
2002-08-20 00:30:13 +00:00
|
|
|
return false;
|
|
|
|
}
|
2005-09-30 21:29:31 +00:00
|
|
|
last_time = gstate.now;
|
2012-02-06 06:06:44 +00:00
|
|
|
create_xfer();
|
|
|
|
return false;
|
2002-08-14 20:31:27 +00:00
|
|
|
}
|
2002-08-07 22:52:10 +00:00
|
|
|
|
2006-02-18 01:57:24 +00:00
|
|
|
// copy bytes_xferred for use in GUI
|
2007-09-24 18:32:55 +00:00
|
|
|
//
|
2006-02-18 01:57:24 +00:00
|
|
|
last_bytes_xferred = fxp->bytes_xferred;
|
2011-02-23 23:11:59 +00:00
|
|
|
if (is_upload) {
|
2006-02-23 20:37:36 +00:00
|
|
|
last_bytes_xferred += fxp->file_offset;
|
|
|
|
}
|
2006-02-18 01:57:24 +00:00
|
|
|
|
2004-09-02 23:06:14 +00:00
|
|
|
// don't count suspended periods in total time
|
|
|
|
//
|
2005-06-07 19:22:50 +00:00
|
|
|
double diff = gstate.now - last_time;
|
2004-09-02 23:06:14 +00:00
|
|
|
if (diff <= 2) {
|
|
|
|
time_so_far += diff;
|
2003-02-04 21:47:12 +00:00
|
|
|
}
|
2005-06-07 19:22:50 +00:00
|
|
|
last_time = gstate.now;
|
2003-02-03 22:47:08 +00:00
|
|
|
|
2002-08-14 20:31:27 +00:00
|
|
|
if (fxp->file_xfer_done) {
|
2006-06-22 19:40:30 +00:00
|
|
|
if (log_flags.file_xfer_debug) {
|
2009-02-22 20:54:33 +00:00
|
|
|
msg_printf(fip->project, MSG_INFO,
|
2011-05-12 05:01:11 +00:00
|
|
|
"[file_xfer] file transfer status %d (%s)",
|
|
|
|
fxp->file_xfer_retval, boincerror(fxp->file_xfer_retval)
|
2006-06-22 19:40:30 +00:00
|
|
|
);
|
|
|
|
}
|
2007-09-24 18:32:55 +00:00
|
|
|
switch (fxp->file_xfer_retval) {
|
|
|
|
case 0:
|
2009-07-16 16:35:57 +00:00
|
|
|
fip->project->file_xfer_backoff(is_upload).file_xfer_succeeded();
|
2004-09-02 23:06:14 +00:00
|
|
|
if (log_flags.file_xfer) {
|
|
|
|
msg_printf(
|
2007-10-17 23:30:05 +00:00
|
|
|
fip->project, MSG_INFO, "Finished %s of %s",
|
2004-09-02 23:06:14 +00:00
|
|
|
is_upload?"upload":"download", fip->name
|
2005-01-13 18:57:27 +00:00
|
|
|
);
|
2007-10-17 23:30:05 +00:00
|
|
|
}
|
|
|
|
if (log_flags.file_xfer_debug) {
|
2004-09-02 23:06:14 +00:00
|
|
|
if (fxp->xfer_speed < 0) {
|
2010-04-29 20:32:51 +00:00
|
|
|
msg_printf(fip->project, MSG_INFO, "[file_xfer] No data transferred");
|
2004-09-02 23:06:14 +00:00
|
|
|
} else {
|
2004-07-14 21:28:59 +00:00
|
|
|
msg_printf(
|
2010-04-29 20:32:51 +00:00
|
|
|
fip->project, MSG_INFO, "[file_xfer] Throughput %d bytes/sec",
|
2004-09-02 23:06:14 +00:00
|
|
|
(int)fxp->xfer_speed
|
2004-08-19 15:36:30 +00:00
|
|
|
);
|
2004-07-14 21:28:59 +00:00
|
|
|
}
|
|
|
|
}
|
2004-09-04 05:21:33 +00:00
|
|
|
pers_xfer_done = true;
|
2007-09-24 18:32:55 +00:00
|
|
|
break;
|
|
|
|
case ERR_UPLOAD_PERMANENT:
|
|
|
|
permanent_failure(fxp->file_xfer_retval);
|
|
|
|
break;
|
|
|
|
case ERR_NOT_FOUND:
|
2012-02-07 07:58:08 +00:00
|
|
|
case ERR_HTTP_PERMANENT:
|
2007-11-20 22:58:14 +00:00
|
|
|
if (is_upload) {
|
|
|
|
// if we get a "not found" on an upload,
|
|
|
|
// the project must not have a file_upload_handler.
|
|
|
|
// Treat this as a transient error.
|
|
|
|
//
|
|
|
|
msg_printf(fip->project, MSG_INFO,
|
|
|
|
"Project file upload handler is missing"
|
|
|
|
);
|
|
|
|
transient_failure(fxp->file_xfer_retval);
|
|
|
|
} else {
|
|
|
|
permanent_failure(fxp->file_xfer_retval);
|
|
|
|
}
|
2007-09-24 18:32:55 +00:00
|
|
|
break;
|
|
|
|
default:
|
2004-09-02 23:06:14 +00:00
|
|
|
if (log_flags.file_xfer) {
|
|
|
|
msg_printf(
|
2007-10-17 23:30:05 +00:00
|
|
|
fip->project, MSG_INFO, "Temporarily failed %s of %s: %s",
|
2005-03-09 23:47:45 +00:00
|
|
|
is_upload?"upload":"download", fip->name,
|
2005-07-05 09:02:55 +00:00
|
|
|
boincerror(fxp->file_xfer_retval)
|
2004-09-02 23:06:14 +00:00
|
|
|
);
|
|
|
|
}
|
2007-09-24 18:32:55 +00:00
|
|
|
transient_failure(fxp->file_xfer_retval);
|
2004-08-19 15:36:30 +00:00
|
|
|
}
|
2004-09-02 23:06:14 +00:00
|
|
|
|
2009-07-16 20:14:57 +00:00
|
|
|
// If we transferred any bytes, or there are >1 URLs,
|
|
|
|
// set upload_offset back to -1
|
2009-07-16 19:41:56 +00:00
|
|
|
// so that we'll query file size on next retry.
|
|
|
|
// Otherwise leave it as is, avoiding unnecessary size query.
|
|
|
|
//
|
2011-07-20 19:12:10 +00:00
|
|
|
if (last_bytes_xferred || (fip->upload_urls.urls.size() > 1)) {
|
2009-07-16 19:41:56 +00:00
|
|
|
fip->upload_offset = -1;
|
|
|
|
}
|
|
|
|
|
2007-09-24 18:32:55 +00:00
|
|
|
// fxp could have already been freed and zeroed above
|
2005-03-01 19:56:30 +00:00
|
|
|
// so check before trying to remove
|
|
|
|
//
|
|
|
|
if (fxp) {
|
|
|
|
gstate.file_xfers->remove(fxp);
|
|
|
|
delete fxp;
|
|
|
|
fxp = NULL;
|
|
|
|
}
|
2011-02-23 23:11:59 +00:00
|
|
|
|
|
|
|
if (is_upload && !fip->project->uploading()) {
|
|
|
|
gstate.request_work_fetch("project finished uploading");
|
|
|
|
}
|
|
|
|
|
2004-09-02 23:06:14 +00:00
|
|
|
return true;
|
2002-06-10 06:14:18 +00:00
|
|
|
}
|
2002-08-07 22:52:10 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-09-24 18:32:55 +00:00
|
|
|
void PERS_FILE_XFER::permanent_failure(int retval) {
|
2006-06-19 04:29:26 +00:00
|
|
|
gstate.file_xfers->remove(fxp);
|
|
|
|
delete fxp;
|
|
|
|
fxp = NULL;
|
2007-09-24 18:32:55 +00:00
|
|
|
fip->status = retval;
|
2006-06-19 04:29:26 +00:00
|
|
|
pers_xfer_done = true;
|
2007-01-25 23:39:06 +00:00
|
|
|
if (log_flags.file_xfer) {
|
|
|
|
msg_printf(
|
2007-10-17 23:30:05 +00:00
|
|
|
fip->project, MSG_INFO, "Giving up on %s of %s: %s",
|
2007-09-24 18:32:55 +00:00
|
|
|
is_upload?"upload":"download", fip->name, boincerror(retval)
|
2007-01-25 23:39:06 +00:00
|
|
|
);
|
|
|
|
}
|
2007-09-24 18:32:55 +00:00
|
|
|
fip->error_msg = boincerror(retval);
|
2006-06-19 04:29:26 +00:00
|
|
|
}
|
|
|
|
|
2007-09-24 18:32:55 +00:00
|
|
|
// Handle a transient failure
|
2004-07-27 23:29:45 +00:00
|
|
|
//
|
2007-09-24 18:32:55 +00:00
|
|
|
void PERS_FILE_XFER::transient_failure(int retval) {
|
2003-03-13 21:49:52 +00:00
|
|
|
|
2002-08-21 19:12:42 +00:00
|
|
|
// If it was a bad range request, delete the file and start over
|
2003-03-13 21:49:52 +00:00
|
|
|
//
|
2007-09-24 18:32:55 +00:00
|
|
|
if (retval == HTTP_STATUS_RANGE_REQUEST_ERROR) {
|
2002-08-21 19:12:42 +00:00
|
|
|
fip->delete_file();
|
2004-07-27 23:29:45 +00:00
|
|
|
return;
|
2002-08-21 19:12:42 +00:00
|
|
|
}
|
2003-02-04 21:47:12 +00:00
|
|
|
|
2007-09-24 18:32:55 +00:00
|
|
|
// If too much time has elapsed, give up
|
2002-08-21 19:12:42 +00:00
|
|
|
//
|
2005-06-07 19:22:50 +00:00
|
|
|
if ((gstate.now - first_request_time) > gstate.file_xfer_giveup_period) {
|
2007-09-24 18:32:55 +00:00
|
|
|
permanent_failure(ERR_TIMEOUT);
|
2002-08-30 20:56:02 +00:00
|
|
|
}
|
2006-01-19 01:13:49 +00:00
|
|
|
|
2007-09-24 18:32:55 +00:00
|
|
|
// Cycle to the next URL to try.
|
|
|
|
// If we reach the URL that we started at, then back off.
|
2003-02-26 00:47:57 +00:00
|
|
|
//
|
2004-07-06 18:31:56 +00:00
|
|
|
|
2011-07-20 19:12:10 +00:00
|
|
|
URL_LIST& ul = fip->get_url_list(is_upload);
|
2011-11-08 05:49:24 +00:00
|
|
|
if (!ul.get_next_url()) {
|
2005-09-30 21:29:31 +00:00
|
|
|
do_backoff();
|
2002-08-21 19:12:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-09-30 21:29:31 +00:00
|
|
|
// per-file backoff policy: sets next_request_time
|
|
|
|
//
|
|
|
|
void PERS_FILE_XFER::do_backoff() {
|
|
|
|
double backoff = 0;
|
|
|
|
|
2006-04-18 23:30:32 +00:00
|
|
|
// don't count it as a server failure if network is down
|
|
|
|
//
|
2006-07-17 22:18:17 +00:00
|
|
|
if (!net_status.need_physical_connection) {
|
2006-04-18 23:30:32 +00:00
|
|
|
nretry++;
|
|
|
|
}
|
2005-09-30 21:29:31 +00:00
|
|
|
|
2007-09-24 18:32:55 +00:00
|
|
|
// keep track of transient failures per project (not currently used)
|
|
|
|
//
|
2009-07-16 16:35:57 +00:00
|
|
|
PROJECT* p = fip->project;
|
|
|
|
p->file_xfer_backoff(is_upload).file_xfer_failed(p);
|
2007-09-24 18:32:55 +00:00
|
|
|
|
2005-09-30 21:29:31 +00:00
|
|
|
// Do an exponential backoff of e^nretry seconds,
|
|
|
|
// keeping within the bounds of pers_retry_delay_min and
|
|
|
|
// pers_retry_delay_max
|
|
|
|
//
|
|
|
|
backoff = calculate_exponential_backoff(
|
|
|
|
nretry, gstate.pers_retry_delay_min, gstate.pers_retry_delay_max
|
|
|
|
);
|
|
|
|
next_request_time = gstate.now + backoff;
|
|
|
|
msg_printf(fip->project, MSG_INFO,
|
2007-10-17 23:30:05 +00:00
|
|
|
"Backing off %s on %s of %s",
|
2005-09-30 21:29:31 +00:00
|
|
|
timediff_format(backoff).c_str(),
|
|
|
|
is_upload?"upload":"download",
|
|
|
|
fip->name
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2004-09-26 04:16:52 +00:00
|
|
|
void PERS_FILE_XFER::abort() {
|
|
|
|
if (fxp) {
|
|
|
|
gstate.file_xfers->remove(fxp);
|
|
|
|
delete fxp;
|
|
|
|
fxp = NULL;
|
|
|
|
}
|
2007-09-24 18:32:55 +00:00
|
|
|
fip->status = ERR_ABORTED_VIA_GUI;
|
2004-09-26 04:16:52 +00:00
|
|
|
fip->error_msg = "user requested transfer abort";
|
|
|
|
pers_xfer_done = true;
|
|
|
|
}
|
|
|
|
|
2005-03-01 19:56:30 +00:00
|
|
|
// Parse XML information about a persistent file transfer
|
2002-08-07 22:52:10 +00:00
|
|
|
//
|
2011-08-09 21:44:14 +00:00
|
|
|
int PERS_FILE_XFER::parse(XML_PARSER& xp) {
|
2011-08-11 06:17:33 +00:00
|
|
|
while (!xp.get_tag()) {
|
|
|
|
if (xp.match_tag("/persistent_file_xfer")) return 0;
|
|
|
|
else if (xp.parse_int("num_retries", nretry)) continue;
|
|
|
|
else if (xp.parse_double("first_request_time", first_request_time)) {
|
2004-12-01 20:56:20 +00:00
|
|
|
continue;
|
|
|
|
}
|
2011-08-11 06:17:33 +00:00
|
|
|
else if (xp.parse_double("next_request_time", next_request_time)) {
|
2004-12-01 20:56:20 +00:00
|
|
|
continue;
|
|
|
|
}
|
2011-08-11 06:17:33 +00:00
|
|
|
else if (xp.parse_double("time_so_far", time_so_far)) continue;
|
|
|
|
else if (xp.parse_double("last_bytes_xferred", last_bytes_xferred)) continue;
|
|
|
|
else if (xp.parse_bool("is_upload", is_upload)) continue;
|
2003-03-13 21:49:52 +00:00
|
|
|
else {
|
2006-12-06 20:59:29 +00:00
|
|
|
if (log_flags.unparsed_xml) {
|
2007-01-25 23:39:06 +00:00
|
|
|
msg_printf(NULL, MSG_INFO,
|
2011-08-11 06:17:33 +00:00
|
|
|
"[unparsed_xml] Unparsed line in file transfer info: %s",
|
|
|
|
xp.parsed_tag
|
2006-12-06 20:59:29 +00:00
|
|
|
);
|
|
|
|
}
|
2003-03-13 21:49:52 +00:00
|
|
|
}
|
2002-08-07 22:52:10 +00:00
|
|
|
}
|
2003-10-21 18:18:41 +00:00
|
|
|
return ERR_XML_PARSE;
|
2002-08-07 22:52:10 +00:00
|
|
|
}
|
|
|
|
|
2005-03-01 19:56:30 +00:00
|
|
|
// Write XML information about a persistent file transfer
|
2002-08-07 22:52:10 +00:00
|
|
|
//
|
2004-06-12 04:45:36 +00:00
|
|
|
int PERS_FILE_XFER::write(MIOFILE& fout) {
|
|
|
|
fout.printf(
|
2002-08-20 00:30:13 +00:00
|
|
|
" <persistent_file_xfer>\n"
|
|
|
|
" <num_retries>%d</num_retries>\n"
|
2004-10-25 20:16:30 +00:00
|
|
|
" <first_request_time>%f</first_request_time>\n"
|
|
|
|
" <next_request_time>%f</next_request_time>\n"
|
2003-02-03 22:47:08 +00:00
|
|
|
" <time_so_far>%f</time_so_far>\n"
|
2006-02-18 01:57:24 +00:00
|
|
|
" <last_bytes_xferred>%f</last_bytes_xferred>\n"
|
2011-07-20 19:12:10 +00:00
|
|
|
" <is_upload>%d</is_upload>\n"
|
2002-08-20 00:30:13 +00:00
|
|
|
" </persistent_file_xfer>\n",
|
2011-07-20 19:12:10 +00:00
|
|
|
nretry,
|
|
|
|
first_request_time,
|
|
|
|
next_request_time,
|
|
|
|
time_so_far,
|
|
|
|
last_bytes_xferred,
|
|
|
|
is_upload?1:0
|
2002-08-14 20:31:27 +00:00
|
|
|
);
|
2004-01-31 23:21:07 +00:00
|
|
|
if (fxp) {
|
2004-06-12 04:45:36 +00:00
|
|
|
fout.printf(
|
2004-01-31 23:21:07 +00:00
|
|
|
" <file_xfer>\n"
|
|
|
|
" <bytes_xferred>%f</bytes_xferred>\n"
|
|
|
|
" <file_offset>%f</file_offset>\n"
|
|
|
|
" <xfer_speed>%f</xfer_speed>\n"
|
2006-02-25 00:02:55 +00:00
|
|
|
" <url>%s</url>\n"
|
2004-01-31 23:21:07 +00:00
|
|
|
" </file_xfer>\n",
|
|
|
|
fxp->bytes_xferred,
|
|
|
|
fxp->file_offset,
|
|
|
|
fxp->xfer_speed,
|
2006-02-25 00:02:55 +00:00
|
|
|
fxp->m_url
|
2004-01-31 23:21:07 +00:00
|
|
|
);
|
|
|
|
}
|
2002-08-07 22:52:10 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-06-08 23:55:21 +00:00
|
|
|
// suspend file transfers by killing them.
|
|
|
|
// They'll restart automatically later.
|
|
|
|
//
|
2003-07-29 23:26:32 +00:00
|
|
|
void PERS_FILE_XFER::suspend() {
|
|
|
|
if (fxp) {
|
2006-02-18 01:57:24 +00:00
|
|
|
last_bytes_xferred = fxp->bytes_xferred; // save bytes transferred
|
2006-02-23 20:37:36 +00:00
|
|
|
if (fxp->is_upload) {
|
|
|
|
last_bytes_xferred += fxp->file_offset;
|
|
|
|
}
|
2003-07-29 23:26:32 +00:00
|
|
|
gstate.file_xfers->remove(fxp); // this removes from http_op_set too
|
|
|
|
delete fxp;
|
|
|
|
fxp = 0;
|
|
|
|
}
|
2010-02-27 01:04:14 +00:00
|
|
|
fip->upload_offset = -1;
|
2003-07-29 23:26:32 +00:00
|
|
|
}
|
|
|
|
|
2002-08-07 22:52:10 +00:00
|
|
|
PERS_FILE_XFER_SET::PERS_FILE_XFER_SET(FILE_XFER_SET* p) {
|
|
|
|
file_xfers = p;
|
2002-06-10 06:14:18 +00:00
|
|
|
}
|
|
|
|
|
2002-08-21 19:12:42 +00:00
|
|
|
// Run through the set, starting any transfers that need to be
|
|
|
|
// started and deleting any that have finished
|
|
|
|
//
|
2005-06-07 19:22:50 +00:00
|
|
|
bool PERS_FILE_XFER_SET::poll() {
|
2002-08-07 22:52:10 +00:00
|
|
|
unsigned int i;
|
2002-06-10 06:14:18 +00:00
|
|
|
bool action = false;
|
2004-10-14 22:01:05 +00:00
|
|
|
static double last_time=0;
|
|
|
|
|
2009-02-26 03:24:39 +00:00
|
|
|
if (gstate.now - last_time < PERS_FILE_XFER_POLL_PERIOD) return false;
|
2005-06-07 19:22:50 +00:00
|
|
|
last_time = gstate.now;
|
2002-06-10 06:14:18 +00:00
|
|
|
|
2010-04-01 05:54:29 +00:00
|
|
|
// try to finish ones we've already started
|
|
|
|
//
|
|
|
|
for (i=0; i<pers_file_xfers.size(); i++) {
|
|
|
|
if (!pers_file_xfers[i]->last_bytes_xferred) continue;
|
|
|
|
action |= pers_file_xfers[i]->poll();
|
|
|
|
}
|
2002-06-10 06:14:18 +00:00
|
|
|
for (i=0; i<pers_file_xfers.size(); i++) {
|
2010-04-01 05:54:29 +00:00
|
|
|
if (pers_file_xfers[i]->last_bytes_xferred) continue;
|
2005-06-07 19:22:50 +00:00
|
|
|
action |= pers_file_xfers[i]->poll();
|
2002-08-07 22:52:10 +00:00
|
|
|
}
|
|
|
|
|
2002-08-22 21:29:58 +00:00
|
|
|
if (action) gstate.set_client_state_dirty("pers_file_xfer_set poll");
|
2002-08-21 19:12:42 +00:00
|
|
|
|
2002-08-07 22:52:10 +00:00
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
2002-08-14 20:31:27 +00:00
|
|
|
// Insert a PERS_FILE_XFER object into the set.
|
|
|
|
// We will decide which ones to start when we hit the polling loop
|
2002-08-07 22:52:10 +00:00
|
|
|
//
|
|
|
|
int PERS_FILE_XFER_SET::insert(PERS_FILE_XFER* pfx) {
|
|
|
|
pers_file_xfers.push_back(pfx);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-08-14 20:31:27 +00:00
|
|
|
// Remove a PERS_FILE_XFER object from the set.
|
|
|
|
// What should the action here be?
|
2002-08-07 22:52:10 +00:00
|
|
|
//
|
|
|
|
int PERS_FILE_XFER_SET::remove(PERS_FILE_XFER* pfx) {
|
|
|
|
vector<PERS_FILE_XFER*>::iterator iter;
|
|
|
|
|
|
|
|
iter = pers_file_xfers.begin();
|
|
|
|
while (iter != pers_file_xfers.end()) {
|
|
|
|
if (*iter == pfx) {
|
2006-10-02 17:44:27 +00:00
|
|
|
iter = pers_file_xfers.erase(iter);
|
2002-08-07 22:52:10 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
iter++;
|
2002-06-10 06:14:18 +00:00
|
|
|
}
|
2003-06-03 22:47:15 +00:00
|
|
|
msg_printf(
|
2007-01-25 23:39:06 +00:00
|
|
|
pfx->fip->project, MSG_INTERNAL_ERROR,
|
2006-01-17 22:48:09 +00:00
|
|
|
"Persistent file transfer object not found"
|
2003-05-14 21:17:58 +00:00
|
|
|
);
|
2004-01-30 22:19:19 +00:00
|
|
|
return ERR_NOT_FOUND;
|
2002-06-10 06:14:18 +00:00
|
|
|
}
|
2003-07-29 23:26:32 +00:00
|
|
|
|
|
|
|
// suspend all PERS_FILE_XFERs
|
|
|
|
//
|
|
|
|
void PERS_FILE_XFER_SET::suspend() {
|
|
|
|
unsigned int i;
|
|
|
|
for (i=0; i<pers_file_xfers.size(); i++) {
|
|
|
|
pers_file_xfers[i]->suspend();
|
|
|
|
}
|
|
|
|
}
|
2004-12-08 00:40:19 +00:00
|
|
|
|
2010-05-07 20:08:59 +00:00
|
|
|
// add a random delay 0..x to all transfers
|
|
|
|
// (used when emerging from bandwidth quota suspension)
|
|
|
|
//
|
|
|
|
void PERS_FILE_XFER_SET::add_random_delay(double x) {
|
|
|
|
unsigned int i;
|
2010-05-12 04:00:14 +00:00
|
|
|
double y = gstate.now + x*drand();
|
2010-05-07 20:08:59 +00:00
|
|
|
for (i=0; i<pers_file_xfers.size(); i++) {
|
|
|
|
if (y > pers_file_xfers[i]->next_request_time) {
|
|
|
|
pers_file_xfers[i]->next_request_time = y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|