. // Web RPCs for managing job input files on the server. // // Issues: // // 1) how are files named? // Their name is a function of their MD5. // This eliminates issues related to file immutability // // 2) how do we keep track of the files? // In the MySQL database, in a table called "job_file". // Each row describes a file currently on the server. // In addition, we maintain a table "batch_file_assoc" to record // that a file is used by a particular batch. // (Note: the association could be at the job level instead. // but this way is more efficient if many jobs in a batch use // a particular file.) // // 3) how do we clean up unused files? // A daemon (job_file_deleter) deletes files for which // - the delete date (if given) is in the past, and // - there are no associations to active batches // // 4) what are the RPC operations? // query_files // in: // authenticator // list of MD5s // batch ID (optional) // new delete time (optional) // out: // error message, // or list of files (indices in the MD5 list) not present on server // action: for each MD5 in in the input list: // if present on server // update delete time // create batch/file association // add MD5 to output list // upload_files // in: // authenticator // delete time (optional) // batch ID (optional) // list of MD5s // files (as multipart attachments) // out: // error message, or success // action: // for each file in list // move to project download dir w/ appropriate name // create job_files record // create batch_file_assoc record if needed error_reporting(E_ALL); ini_set('display_errors', true); ini_set('display_startup_errors', true); require_once("../inc/boinc_db.inc"); require_once("../inc/submit_db.inc"); require_once("../inc/dir_hier.inc"); require_once("../inc/xml.inc"); require_once("../inc/submit_util.inc"); function query_files($r) { list($user, $user_submit) = authenticate_user($r, null); $absent_files = array(); $now = time(); $delete_time = (int)$r->delete_time; $batch_id = (int)$r->batch_id; $fanout = parse_config(get_config(), ""); $i = 0; $md5s= array(); foreach($r->md5 as $f) { $md5 = (string)$f; $md5s[] = $md5; } $md5s = array_unique($md5s); foreach($md5s as $md5) { $fname = job_file_name($md5); $path = dir_hier_path($fname, project_dir() . "/download", $fanout); // if the job_file record is there, // update the delete time first to avoid race condition // with job file deleter // $job_file = BoincJobFile::lookup_md5($md5); if ($job_file && $job_file->delete_time < $delete_time) { $retval = $job_file->update("delete_time=$delete_time"); if ($retval) { xml_error(-1, "job_file->update() failed: ".BoincDb::error()); } } if (file_exists($path)) { // create the DB record if needed // if ($job_file) { $jf_id = $job_file->id; } else { $jf_id = BoincJobFile::insert( "(md5, create_time, delete_time) values ('$md5', $now, $delete_time)" ); if (!$jf_id) { xml_error(-1, "query_file(): BoincJobFile::insert($md5) failed: ".BoincDb::error()); } } // create batch association if needed // if ($batch_id) { $ret = BoincBatchFileAssoc::insert( "(batch_id, job_file_id) values ($batch_id, $jf_id)" ); if (!$ret) { xml_error(-1, "BoincBatchFileAssoc::insert() failed: ".BoincDb::error() ); } } } else { if ($job_file) { $ret = $job_file->delete(); if (!$ret) { xml_error(-1, "BoincJobFile::delete() failed: ".BoincDb::error() ); } } $absent_files[] = $i; } $i++; } echo "\n"; foreach ($absent_files as $i) { echo "$i\n"; } echo "\n"; } function upload_files($r) { list($user, $user_submit) = authenticate_user($r, null); $fanout = parse_config(get_config(), ""); $delete_time = (int)$r->delete_time; $batch_id = (int)$r->batch_id; //print_r($_FILES); $i = 0; foreach ($r->md5 as $f) { $md5 = (string)$f; $name = "file_$i"; $tmp_name = $_FILES[$name]['tmp_name']; if (!is_uploaded_file($tmp_name)) { xml_error(-1, "$tmp_name is not an uploaded file"); } $fname = job_file_name($md5); $path = dir_hier_path($fname, project_dir() . "/download", $fanout); rename($tmp_name, $path); $now = time(); $jf_id = BoincJobFile::insert( "(md5, create_time, delete_time) values ('$md5', $now, $delete_time)" ); if (!$jf_id) { xml_error(-1, "upload_files(): BoincJobFile::insert($md5) failed: ".BoincDb::error()); } if ($batch_id) { BoincBatchFileAssoc::insert( "(batch_id, job_file_id) values ($batch_id, $jf_id)" ); } $i++; } echo "\n"; } if (0) { $r = simplexml_load_string("\n0\n 80bf244b43fb5d39541ea7011883b7e0\n a6037b05afb05f36e6a85a7c5138cbc1\n\n "); submit_batch($r); exit; } if (0) { $r = simplexml_load_string("\n157f96a018b0b2f2b466e2ce3c7f54db\n1\n80bf244b43fb5d39541ea7011883b7e0\na6037b05afb05f36e6a85a7c5138cbc1\n"); upload_files($r); exit; } xml_header(); $r = simplexml_load_string($_POST['request']); if (!$r) { xml_error(-1, "can't parse request message", __FILE__, __LINE__); } switch($r->getName()) { case 'query_files': query_files($r); break; case 'upload_files': upload_files($r); break; default: xml_error(-1, "no such action"); } ?>