2005-01-20 23:22:22 +00:00
|
|
|
// Berkeley Open Infrastructure for Network Computing
|
|
|
|
// http://boinc.berkeley.edu
|
|
|
|
// Copyright (C) 2005 University of California
|
2003-07-02 02:02:18 +00:00
|
|
|
//
|
2005-01-20 23:22:22 +00:00
|
|
|
// This 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 2.1 of the License, or (at your option) any later version.
|
2003-07-02 02:02:18 +00:00
|
|
|
//
|
2005-01-20 23:22:22 +00:00
|
|
|
// This software is distributed in the hope that it will be useful,
|
|
|
|
// 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-04-30 22:22:54 +00:00
|
|
|
//
|
2005-01-20 23:22:22 +00:00
|
|
|
// To view the GNU Lesser General Public License visit
|
|
|
|
// http://www.gnu.org/copyleft/lesser.html
|
|
|
|
// or write to the Free Software Foundation, Inc.,
|
|
|
|
// 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2003-10-16 19:03:49 +00:00
|
|
|
#include "cpp.h"
|
2002-06-06 18:42:01 +00:00
|
|
|
|
2004-03-05 04:37:53 +00:00
|
|
|
#ifdef _WIN32
|
2004-06-16 23:16:08 +00:00
|
|
|
#include "boinc_win.h"
|
2004-03-05 04:37:53 +00:00
|
|
|
#endif
|
|
|
|
|
2002-04-30 22:22:54 +00:00
|
|
|
#include "util.h"
|
2002-07-05 05:33:40 +00:00
|
|
|
#include "file_names.h"
|
2003-06-03 22:47:15 +00:00
|
|
|
#include "client_state.h"
|
2002-08-12 21:54:19 +00:00
|
|
|
#include "filesys.h"
|
2004-04-08 08:15:23 +00:00
|
|
|
#include "client_msgs.h"
|
2002-04-30 22:22:54 +00:00
|
|
|
#include "file_xfer.h"
|
2002-08-21 19:12:42 +00:00
|
|
|
#include "parse.h"
|
2002-07-11 01:09:53 +00:00
|
|
|
#include "error_numbers.h"
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2004-06-30 18:17:21 +00:00
|
|
|
using std::vector;
|
|
|
|
|
2002-04-30 22:22:54 +00:00
|
|
|
FILE_XFER::FILE_XFER() {
|
|
|
|
file_xfer_done = false;
|
|
|
|
file_xfer_retval = 0;
|
2002-07-15 23:21:20 +00:00
|
|
|
fip = NULL;
|
2003-03-19 01:15:46 +00:00
|
|
|
strcpy(pathname, "");
|
|
|
|
strcpy(header, "");
|
2002-08-21 19:12:42 +00:00
|
|
|
file_size_query = false;
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
FILE_XFER::~FILE_XFER() {
|
2003-07-23 22:24:28 +00:00
|
|
|
if (fip && fip->pers_file_xfer) {
|
|
|
|
fip->pers_file_xfer->fxp = NULL;
|
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
|
|
|
|
2002-07-05 05:33:40 +00:00
|
|
|
int FILE_XFER::init_download(FILE_INFO& file_info) {
|
2002-12-06 07:33:45 +00:00
|
|
|
double f_size;
|
2002-08-12 21:54:19 +00:00
|
|
|
|
2003-05-10 00:52:36 +00:00
|
|
|
is_upload = false;
|
2002-07-05 05:33:40 +00:00
|
|
|
fip = &file_info;
|
|
|
|
get_pathname(fip, pathname);
|
2002-08-12 21:54:19 +00:00
|
|
|
if (file_size(pathname, f_size)) {
|
|
|
|
f_size = 0;
|
|
|
|
}
|
2003-02-27 09:21:22 +00:00
|
|
|
bytes_xferred = f_size;
|
2003-06-09 22:33:23 +00:00
|
|
|
|
2004-07-06 17:37:58 +00:00
|
|
|
return HTTP_OP::init_get(fip->get_current_url(is_upload), pathname, false, (int)f_size);
|
2002-07-05 05:33:40 +00:00
|
|
|
}
|
|
|
|
|
2002-07-07 20:39:24 +00:00
|
|
|
// for uploads, we need to build a header with xml_signature etc.
|
2002-12-02 04:29:40 +00:00
|
|
|
// (see doc/upload.html)
|
2002-07-05 05:33:40 +00:00
|
|
|
// Do this in memory.
|
|
|
|
//
|
2004-07-13 10:36:18 +00:00
|
|
|
int FILE_XFER::init_upload(FILE_INFO& file_info) {
|
2002-08-12 21:54:19 +00:00
|
|
|
// If upload_offset < 0, we need to query the upload handler
|
|
|
|
// for the offset information
|
2003-03-13 21:49:52 +00:00
|
|
|
// TODO: give priority to unfinished upload if there are multiple choices
|
2002-08-12 21:54:19 +00:00
|
|
|
//
|
2002-07-05 05:33:40 +00:00
|
|
|
fip = &file_info;
|
|
|
|
get_pathname(fip, pathname);
|
2002-08-12 21:54:19 +00:00
|
|
|
|
2003-05-10 00:52:36 +00:00
|
|
|
is_upload = true;
|
2003-07-25 00:17:18 +00:00
|
|
|
if (file_info.upload_offset < 0) { // NOTE: Should this be <=?
|
2003-02-27 09:21:22 +00:00
|
|
|
bytes_xferred = 0;
|
2002-08-12 21:54:19 +00:00
|
|
|
sprintf(header,
|
2002-12-02 04:29:40 +00:00
|
|
|
"<data_server_request>\n"
|
|
|
|
" <core_client_major_version>%d</core_client_major_version>\n"
|
|
|
|
" <core_client_minor_version>%d</core_client_minor_version>\n"
|
|
|
|
" <get_file_size>%s</get_file_size>\n"
|
|
|
|
"</data_server_request>\n",
|
2004-07-13 10:36:18 +00:00
|
|
|
BOINC_MAJOR_VERSION, BOINC_MINOR_VERSION,
|
2002-08-12 21:54:19 +00:00
|
|
|
file_info.name
|
|
|
|
);
|
2002-08-21 19:12:42 +00:00
|
|
|
file_size_query = true;
|
2004-07-06 17:37:58 +00:00
|
|
|
return HTTP_OP::init_post2(fip->get_current_url(is_upload), header, NULL, 0);
|
2002-08-12 21:54:19 +00:00
|
|
|
} else {
|
2003-02-27 09:21:22 +00:00
|
|
|
bytes_xferred = file_info.upload_offset;
|
2002-08-12 21:54:19 +00:00
|
|
|
sprintf(header,
|
2002-12-02 04:29:40 +00:00
|
|
|
"<data_server_request>\n"
|
|
|
|
" <core_client_major_version>%d</core_client_major_version>\n"
|
|
|
|
" <core_client_minor_version>%d</core_client_minor_version>\n"
|
|
|
|
"<file_upload>\n"
|
2002-08-12 21:54:19 +00:00
|
|
|
"<file_info>\n"
|
|
|
|
"%s"
|
|
|
|
"<xml_signature>\n"
|
|
|
|
"%s"
|
|
|
|
"</xml_signature>\n"
|
|
|
|
"</file_info>\n"
|
2002-12-02 04:29:40 +00:00
|
|
|
"<nbytes>%.0f</nbytes>\n"
|
2002-08-12 21:54:19 +00:00
|
|
|
"<offset>%.0f</offset>\n"
|
|
|
|
"<data>\n",
|
2004-07-13 10:36:18 +00:00
|
|
|
BOINC_MAJOR_VERSION, BOINC_MINOR_VERSION,
|
2002-08-12 21:54:19 +00:00
|
|
|
file_info.signed_xml,
|
|
|
|
file_info.xml_signature,
|
|
|
|
file_info.nbytes,
|
|
|
|
file_info.upload_offset
|
|
|
|
);
|
2002-08-21 19:12:42 +00:00
|
|
|
file_size_query = false;
|
2002-08-26 22:57:17 +00:00
|
|
|
return HTTP_OP::init_post2(
|
2004-07-06 17:37:58 +00:00
|
|
|
fip->get_current_url(is_upload), header, pathname, fip->upload_offset
|
2002-12-02 04:29:40 +00:00
|
|
|
);
|
2002-08-12 21:54:19 +00:00
|
|
|
}
|
2002-07-05 05:33:40 +00:00
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2003-09-30 18:09:58 +00:00
|
|
|
// Parse the file upload handler response in req1
|
2002-08-21 19:12:42 +00:00
|
|
|
//
|
2003-09-30 18:09:58 +00:00
|
|
|
int FILE_XFER::parse_upload_response(double &nbytes) {
|
2003-05-06 21:43:26 +00:00
|
|
|
int status = ERR_UPLOAD_TRANSIENT, x;
|
2003-06-03 22:47:15 +00:00
|
|
|
char buf[256];
|
2002-12-02 04:29:40 +00:00
|
|
|
|
2003-05-06 21:43:26 +00:00
|
|
|
nbytes = -1;
|
2002-12-02 04:29:40 +00:00
|
|
|
parse_double(req1, "<file_size>", nbytes);
|
2003-05-06 21:43:26 +00:00
|
|
|
if (parse_int(req1, "<status>", x)) {
|
|
|
|
switch (x) {
|
|
|
|
case -1: status = ERR_UPLOAD_PERMANENT; break;
|
|
|
|
case 0: status = 0; break;
|
|
|
|
case 1: status = ERR_UPLOAD_TRANSIENT; break;
|
|
|
|
default: status = ERR_UPLOAD_TRANSIENT; break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
status = ERR_UPLOAD_TRANSIENT;
|
|
|
|
}
|
2003-07-02 02:02:18 +00:00
|
|
|
|
2002-12-02 04:29:40 +00:00
|
|
|
if (parse_str(req1, "<message>", buf, sizeof(buf))) {
|
2003-06-03 22:47:15 +00:00
|
|
|
msg_printf(fip->project, MSG_ERROR, "Error on file upload: %s", buf);
|
2002-12-02 04:29:40 +00:00
|
|
|
}
|
|
|
|
|
2002-08-21 19:12:42 +00:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2002-07-15 23:21:20 +00:00
|
|
|
// Create a new empty FILE_XFER_SET
|
|
|
|
//
|
2002-04-30 22:22:54 +00:00
|
|
|
FILE_XFER_SET::FILE_XFER_SET(HTTP_OP_SET* p) {
|
|
|
|
http_ops = p;
|
|
|
|
}
|
|
|
|
|
2004-02-19 16:43:28 +00:00
|
|
|
// start a FILE_XFER going (connect to server etc.)
|
|
|
|
// If successful, add to the set
|
2002-07-15 23:21:20 +00:00
|
|
|
//
|
2002-04-30 22:22:54 +00:00
|
|
|
int FILE_XFER_SET::insert(FILE_XFER* fxp) {
|
|
|
|
int retval;
|
2002-08-15 22:03:41 +00:00
|
|
|
|
2002-04-30 22:22:54 +00:00
|
|
|
retval = http_ops->insert(fxp);
|
|
|
|
if (retval) return retval;
|
2004-02-19 16:43:28 +00:00
|
|
|
file_xfers.push_back(fxp);
|
2002-04-30 22:22:54 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-07-15 23:21:20 +00:00
|
|
|
// Remove a FILE_XFER object from the set
|
|
|
|
//
|
2002-04-30 22:22:54 +00:00
|
|
|
int FILE_XFER_SET::remove(FILE_XFER* fxp) {
|
|
|
|
vector<FILE_XFER*>::iterator iter;
|
2002-08-15 22:03:41 +00:00
|
|
|
|
2002-04-30 22:22:54 +00:00
|
|
|
http_ops->remove(fxp);
|
|
|
|
|
|
|
|
iter = file_xfers.begin();
|
|
|
|
while (iter != file_xfers.end()) {
|
|
|
|
if (*iter == fxp) {
|
|
|
|
file_xfers.erase(iter);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
iter++;
|
|
|
|
}
|
2003-07-03 05:01:29 +00:00
|
|
|
msg_printf(NULL, MSG_ERROR, "FILE_XFER_SET::remove(): not found\n");
|
2004-01-30 22:19:19 +00:00
|
|
|
return ERR_NOT_FOUND;
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
|
|
|
|
2002-07-15 23:21:20 +00:00
|
|
|
// Run through the FILE_XFER_SET and determine if any of the file
|
|
|
|
// transfers are complete or had an error
|
|
|
|
//
|
2004-10-14 22:01:05 +00:00
|
|
|
bool FILE_XFER_SET::poll(double now) {
|
2002-04-30 22:22:54 +00:00
|
|
|
unsigned int i;
|
|
|
|
FILE_XFER* fxp;
|
|
|
|
bool action = false;
|
2004-10-14 22:01:05 +00:00
|
|
|
static double last_time=0;
|
|
|
|
|
|
|
|
if (now - last_time < 1.0) return false;
|
|
|
|
last_time = now;
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
SCOPE_MSG_LOG scope_messages(log_messages, CLIENT_MSG_LOG::DEBUG_FILE_XFER);
|
2003-07-02 02:02:18 +00:00
|
|
|
|
2002-04-30 22:22:54 +00:00
|
|
|
for (i=0; i<file_xfers.size(); i++) {
|
|
|
|
fxp = file_xfers[i];
|
|
|
|
if (fxp->http_op_done()) {
|
|
|
|
action = true;
|
|
|
|
fxp->file_xfer_done = true;
|
2003-07-02 02:02:18 +00:00
|
|
|
scope_messages.printf("FILE_XFER_SET::poll(): http op done; retval %d\n", fxp->http_op_retval);
|
2003-05-10 00:52:36 +00:00
|
|
|
fxp->file_xfer_retval = fxp->http_op_retval;
|
|
|
|
if (fxp->file_xfer_retval == 0) {
|
|
|
|
if (fxp->is_upload) {
|
2003-09-30 18:09:58 +00:00
|
|
|
fxp->file_xfer_retval = fxp->parse_upload_response(
|
2003-05-10 00:52:36 +00:00
|
|
|
fxp->fip->upload_offset
|
|
|
|
);
|
|
|
|
}
|
2003-02-26 00:47:57 +00:00
|
|
|
|
2002-08-21 19:12:42 +00:00
|
|
|
// If this was a file size query, restart the transfer
|
|
|
|
// using the remote file size information
|
2003-02-26 00:47:57 +00:00
|
|
|
//
|
2002-08-21 19:12:42 +00:00
|
|
|
if (fxp->file_size_query) {
|
2003-05-06 21:43:26 +00:00
|
|
|
if (fxp->file_xfer_retval) {
|
2003-05-12 21:35:17 +00:00
|
|
|
printf("ERROR: file upload returned %d\n", fxp->file_xfer_retval);
|
2002-08-21 19:12:42 +00:00
|
|
|
fxp->fip->upload_offset = -1;
|
|
|
|
} else {
|
2002-10-15 18:09:16 +00:00
|
|
|
|
2004-01-21 07:07:16 +00:00
|
|
|
// if the server's file size is bigger than ours,
|
|
|
|
// something bad has happened (like a result
|
|
|
|
// got sent to multiple users).
|
|
|
|
// Pretend the file was successfully uploaded
|
|
|
|
//
|
|
|
|
if (fxp->fip->upload_offset >= fxp->fip->nbytes) {
|
|
|
|
fxp->file_xfer_done = true;
|
|
|
|
fxp->file_xfer_retval = 0;
|
|
|
|
} else {
|
|
|
|
// Restart the upload, using the newly obtained
|
|
|
|
// upload_offset
|
2004-05-24 03:53:09 +00:00
|
|
|
fxp->close_socket();
|
2004-01-21 07:07:16 +00:00
|
|
|
fxp->file_xfer_retval = fxp->init_upload(*fxp->fip);
|
2002-08-21 23:01:12 +00:00
|
|
|
|
2003-05-06 21:43:26 +00:00
|
|
|
if (!fxp->file_xfer_retval) {
|
2005-03-10 00:43:22 +00:00
|
|
|
remove(fxp);
|
|
|
|
i--;
|
2004-01-21 07:07:16 +00:00
|
|
|
fxp->file_xfer_retval = insert(fxp);
|
|
|
|
if (!fxp->file_xfer_retval) {
|
|
|
|
fxp->file_xfer_done = false;
|
|
|
|
fxp->file_xfer_retval = 0;
|
|
|
|
fxp->http_op_retval = 0;
|
|
|
|
}
|
2002-08-21 23:01:12 +00:00
|
|
|
}
|
|
|
|
}
|
2002-08-21 19:12:42 +00:00
|
|
|
}
|
|
|
|
}
|
2005-03-10 00:43:22 +00:00
|
|
|
} else if (fxp->file_xfer_retval == HTTP_STATUS_RANGE_REQUEST_ERROR) {
|
|
|
|
fxp->fip->error_msg = "Local copy is at least as large as server copy";
|
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return action;
|
|
|
|
}
|
2004-12-08 00:40:19 +00:00
|
|
|
|
2005-01-02 18:29:53 +00:00
|
|
|
const char *BOINC_RCSID_31ba21bea3 = "$Id$";
|