// This file is part of BOINC. // http://boinc.berkeley.edu // Copyright (C) 2008 University of California // // 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. // // BOINC 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. // // You should have received a copy of the GNU Lesser General Public License // along with BOINC. If not, see . #include "cpp.h" #ifdef _WIN32 #include "boinc_win.h" #else #include "config.h" #include #include #include #include #endif #ifdef _MSC_VER #define snprintf _snprintf #endif #include "error_numbers.h" #include "filesys.h" #include "parse.h" #include "str_util.h" #include "str_replace.h" #include "util.h" #include "client_state.h" #include "client_types.h" #include "client_msgs.h" #include "file_names.h" #include "log_flags.h" #include "main.h" #include "project.h" #include "result.h" #include "scheduler_op.h" using std::vector; SCHEDULER_OP::SCHEDULER_OP(HTTP_OP_SET* h) { scheduler_op_retval = 0; http_op.http_op_state = HTTP_STATE_IDLE; http_ops = h; safe_strcpy(scheduler_url,""); url_index = 0; cur_proj = NULL; state = SCHEDULER_OP_STATE_IDLE; reason = 0; url_random = 0.0; } // See if there's a pending master file fetch. // If so, start it and return true. // bool SCHEDULER_OP::check_master_fetch_start() { int retval; PROJECT* p = gstate.next_project_master_pending(); if (!p) return false; retval = init_master_fetch(p); if (retval) { msg_printf(p, MSG_INTERNAL_ERROR, "Couldn't start download of scheduler list: %s", boincerror(retval) ); p->master_fetch_failures++; project_rpc_backoff(p, "scheduler list fetch failed\n"); return false; } msg_printf(p, MSG_INFO, "Fetching scheduler list"); return true; } #ifndef SIM // try to initiate an RPC to the given project. // If there are multiple schedulers, start with a random one. // User messages and project RPC backoff is done at this level. // int SCHEDULER_OP::init_op_project(PROJECT* p, int r) { int retval; char err_msg[256]; reason = r; if (log_flags.sched_op_debug) { msg_printf(p, MSG_INFO, "[sched_op] Starting scheduler request" ); } // if project has no schedulers, // skip everything else and just get its master file. // if (p->scheduler_urls.size() == 0) { retval = init_master_fetch(p); if (retval) { snprintf(err_msg, sizeof(err_msg), "Scheduler list fetch initialization failed: %d\n", retval ); project_rpc_backoff(p, err_msg); } return retval; } if (reason == RPC_REASON_INIT) { work_fetch.set_initial_work_request(p); if (!gstate.cpu_benchmarks_done()) { gstate.cpu_benchmarks_set_defaults(); } } url_index = 0; retval = gstate.make_scheduler_request(p); if (!retval) { retval = start_rpc(p); } if (retval) { snprintf(err_msg, sizeof(err_msg), "scheduler request to %s failed: %s\n", p->get_scheduler_url(url_index, url_random), boincerror(retval) ); project_rpc_backoff(p, err_msg); } else { // RPC started OK, so we must have network connectivity. // Now's a good time to check for new BOINC versions // and project list // if (!cc_config.no_info_fetch) { gstate.new_version_check(); gstate.all_projects_list_check(); } } return retval; } #endif // One of the following errors occurred: // - connection failure in fetching master file // - connection failure in scheduler RPC // - got master file, but it didn't have any elements // - tried all schedulers, none responded // - sent nonzero work request, got a reply with no work // // Back off contacting this project's schedulers, // and output an error msg if needed // void SCHEDULER_OP::project_rpc_backoff(PROJECT* p, const char *reason_msg) { char buf[1024]; if (gstate.in_abort_sequence) { return; } if (p->master_fetch_failures >= gstate.master_fetch_retry_cap) { snprintf(buf, sizeof(buf), "%d consecutive failures fetching scheduler list", p->master_fetch_failures ); p->master_url_fetch_pending = true; p->set_min_rpc_time(gstate.now + gstate.master_fetch_interval, buf); return; } // if nrpc failures is master_fetch_period, // then set master_url_fetch_pending and initialize again // if (p->nrpc_failures == gstate.master_fetch_period) { p->master_url_fetch_pending = true; p->min_rpc_time = 0; p->nrpc_failures = 0; p->master_fetch_failures++; } // if network is down, don't count it as RPC failure // if (!net_status.need_physical_connection) { p->nrpc_failures++; } //msg_printf(p, MSG_INFO, "nrpc_failures %d need_conn %d", p->nrpc_failures, gstate.need_physical_connection); int n = p->nrpc_failures; if (n > gstate.retry_cap) n = gstate.retry_cap; double exp_backoff = calculate_exponential_backoff( n, gstate.sched_retry_delay_min, gstate.sched_retry_delay_max ); //msg_printf(p, MSG_INFO, "simulating backoff of %f", exp_backoff); p->set_min_rpc_time(gstate.now + exp_backoff, reason_msg); } // RPC failed, either on startup or later. // If RPC was requested by project or acct mgr, or init, // keep trying (subject to backoff); otherwise give up. // The other cases (results_dur, need_work, project req, and trickle_up) // will be retriggered automatically // void SCHEDULER_OP::rpc_failed(const char* msg) { project_rpc_backoff(cur_proj, msg); switch (cur_proj->sched_rpc_pending) { case RPC_REASON_INIT: case RPC_REASON_ACCT_MGR_REQ: break; default: cur_proj->sched_rpc_pending = 0; } cur_proj = 0; } static void request_string(char* buf, int len) { bool first = true; strlcpy(buf, "", len); for (int i=0; iget_scheduler_url(url_index, url_random)); if (log_flags.sched_ops) { msg_printf(p, MSG_INFO, "Sending scheduler request: %s.", rpc_reason_string(reason) ); if (p->trickle_up_pending && reason != RPC_REASON_TRICKLE_UP) { msg_printf(p, MSG_INFO, "Sending trickle-up message"); } if (p->nresults_returned) { msg_printf(p, MSG_INFO, "Reporting %d completed tasks", p->nresults_returned ); } request_string(buf, sizeof(buf)); if (strlen(buf)) { msg_printf(p, MSG_INFO, "Requesting new tasks for %s", buf); } else { if (p->pwf.project_reason) { msg_printf(p, MSG_INFO, "Not requesting tasks: %s", project_reason_string(p, buf, sizeof(buf)) ); } else { msg_printf(p, MSG_INFO, "Not requesting tasks"); } } } if (log_flags.sched_op_debug) { for (int i=0; iinsert(&http_op); p->rpc_seqno++; state = SCHEDULER_OP_STATE_RPC; return 0; } // initiate a fetch of a project's master URL file // int SCHEDULER_OP::init_master_fetch(PROJECT* p) { int retval; char master_filename[256]; get_master_filename(*p, master_filename, sizeof(master_filename)); if (log_flags.sched_op_debug) { msg_printf(p, MSG_INFO, "[sched_op] Fetching master file"); } cur_proj = p; retval = http_op.init_get(p, p->master_url, master_filename, true, 0, 0); if (retval) { if (log_flags.sched_ops) { msg_printf(p, MSG_INFO, "Master file fetch failed: %s", boincerror(retval) ); } rpc_failed("Master file fetch initialization failed"); return retval; } http_ops->insert(&http_op); state = SCHEDULER_OP_STATE_GET_MASTER; return 0; } // parse a master file. // int SCHEDULER_OP::parse_master_file(PROJECT* p, vector &urls) { char buf[256], buf2[256]; char master_filename[256]; string str; FILE* f; int n; get_master_filename(*p, master_filename, sizeof(master_filename)); f = boinc_fopen(master_filename, "r"); if (!f) { msg_printf(p, MSG_INTERNAL_ERROR, "Can't open scheduler list file"); return ERR_FOPEN; } p->scheduler_urls.clear(); while (fgets(buf, 256, f)) { // allow for the possibility of > 1 tag per line here // (UMTS may collapse lines) // char* q = buf; while (q && parse_str(q, "", str)) { push_unique(str, urls); q = strstr(q, ""); if (q) q += strlen(""); } // check for new syntax: // q = buf; while (q) { n = sscanf(q, "'); if (q) q = strchr(q, '<'); } } fclose(f); if (log_flags.sched_op_debug) { msg_printf(p, MSG_INFO, "[sched_op] Found %d scheduler URLs in master file\n", (int)urls.size() ); } // couldn't find any scheduler URLs in the master file? // if ((int) urls.size() == 0) { msg_printf(p, MSG_INTERNAL_ERROR, "No scheduler URLs found in master file\n" ); p->sched_rpc_pending = 0; return ERR_XML_PARSE; } return 0; } // A master file has just been read. // transfer scheduler URLs to project. // Return true if any of them is new // bool SCHEDULER_OP::update_urls(PROJECT* p, vector &urls) { unsigned int i, j; bool found, any_new; any_new = false; for (i=0; ischeduler_urls.size(); j++) { if (urls[i] == p->scheduler_urls[j]) { found = true; break; } } if (!found) any_new = true; } p->scheduler_urls.clear(); for (i=0; ischeduler_urls.push_back(urls[i]); } return any_new; } #ifndef SIM // poll routine. If an operation is in progress, check for completion // bool SCHEDULER_OP::poll() { int retval; vector urls; bool changed; switch(state) { case SCHEDULER_OP_STATE_GET_MASTER: // here we're fetching the master file for a project // if (http_op.http_op_state == HTTP_STATE_DONE) { state = SCHEDULER_OP_STATE_IDLE; cur_proj->master_url_fetch_pending = false; http_ops->remove(&http_op); if (http_op.http_op_retval == 0) { if (log_flags.sched_op_debug) { msg_printf(cur_proj, MSG_INFO, "[sched_op] Got master file; parsing" ); } retval = parse_master_file(cur_proj, urls); if (retval || (urls.size()==0)) { // master file parse failed. // cur_proj->master_fetch_failures++; rpc_failed("Couldn't parse scheduler list"); } else { // parse succeeded // msg_printf(cur_proj, MSG_INFO, "Master file download succeeded"); cur_proj->master_fetch_failures = 0; changed = update_urls(cur_proj, urls); // reenable scheduler RPCs if have new URLs // if (changed) { cur_proj->min_rpc_time = 0; cur_proj->nrpc_failures = 0; } } } else { // master file fetch failed. // char buf[256]; snprintf(buf, sizeof(buf), "Scheduler list fetch failed: %s", boincerror(http_op.http_op_retval) ); cur_proj->master_fetch_failures++; rpc_failed("Master file request failed"); } gstate.set_client_state_dirty("Master fetch complete"); gstate.request_work_fetch("Master fetch complete"); cur_proj = NULL; return true; } break; case SCHEDULER_OP_STATE_RPC: // here we're doing a scheduler RPC // if (http_op.http_op_state == HTTP_STATE_DONE) { state = SCHEDULER_OP_STATE_IDLE; http_ops->remove(&http_op); if (http_op.http_op_retval) { if (log_flags.sched_ops) { msg_printf(cur_proj, MSG_INFO, "Scheduler request failed: %s", http_op.error_msg ); } // scheduler RPC failed. Try another scheduler if one exists // while (1) { url_index++; if (url_index == (int)cur_proj->scheduler_urls.size()) { break; } retval = start_rpc(cur_proj); if (!retval) return true; } if (url_index == (int) cur_proj->scheduler_urls.size()) { rpc_failed("Scheduler request failed"); } } else { retval = gstate.handle_scheduler_reply(cur_proj, scheduler_url); switch (retval) { case 0: break; case ERR_PROJECT_DOWN: project_rpc_backoff(cur_proj, "project is down"); break; default: project_rpc_backoff(cur_proj, "can't parse scheduler reply"); break; } cur_proj->sched_rpc_pending = 0; // do this after handle_scheduler_reply() } cur_proj = NULL; gstate.set_client_state_dirty("RPC complete"); gstate.request_work_fetch("RPC complete"); return true; } } return false; } #endif void SCHEDULER_OP::abort(PROJECT* p) { if (state != SCHEDULER_OP_STATE_IDLE && cur_proj == p) { gstate.http_ops->remove(&http_op); state = SCHEDULER_OP_STATE_IDLE; cur_proj = NULL; } } void SCHEDULER_REPLY::clear() { hostid = 0; request_delay = 0; next_rpc_delay = 0; messages.clear(); global_prefs_xml = 0; project_prefs_xml = 0; safe_strcpy(master_url, ""); safe_strcpy(host_venue, ""); user_create_time = 0; code_sign_key = 0; code_sign_key_signature = 0; message_ack = false; project_is_down = false; send_file_list = false; send_full_workload = false; dont_use_dcf = false; send_time_stats_log = 0; send_job_log = 0; scheduler_version = 0; got_rss_feeds = false; } SCHEDULER_REPLY::SCHEDULER_REPLY() { clear(); } SCHEDULER_REPLY::~SCHEDULER_REPLY() { if (global_prefs_xml) free(global_prefs_xml); if (project_prefs_xml) free(project_prefs_xml); if (code_sign_key) free(code_sign_key); if (code_sign_key_signature) free(code_sign_key_signature); } #ifndef SIM static void handle_no_rsc_apps(const char* name, PROJECT* p, bool value) { int i = rsc_index(name); if (i < 0) return; p->no_rsc_apps[i] = value; } // parse a scheduler reply. // Some of the items go into the SCHEDULER_REPLY object. // Others are copied straight to the PROJECT // int SCHEDULER_REPLY::parse(FILE* in, PROJECT* project) { char buf[256], msg_buf[1024], pri_buf[256], attr_buf[256]; int retval; MIOFILE mf; XML_PARSER xp(&mf); string delete_file_name; bool verify_files_on_app_start = false; bool non_cpu_intensive = false; bool ended = false; mf.init_file(in); bool found_start_tag = false, btemp; double cpid_time = 0; clear(); safe_strcpy(host_venue, project->host_venue); // the project won't send us a venue if it's doing maintenance // or doesn't check the DB because no work. // Don't overwrite the host venue in that case. sr_feeds.clear(); trickle_up_urls.clear(); if (!project->anonymous_platform) { for (int i=0; ino_rsc_apps[i] = false; } } // First line should either be tag (HTTP 1.0) or // hex length of response (HTTP 1.1) // while (!xp.get_tag(attr_buf, sizeof(attr_buf))) { if (!found_start_tag) { if (xp.match_tag("scheduler_reply")) { found_start_tag = true; } continue; } if (xp.match_tag("/scheduler_reply")) { // update statistics after parsing the scheduler reply // add new record if vector is empty or we have a new day // if (project->statistics.empty() || project->statistics.back().day!=dday()) { project->trim_statistics(); DAILY_STATS nds; project->statistics.push_back(nds); } DAILY_STATS& ds = project->statistics.back(); ds.day=dday(); ds.user_total_credit=project->user_total_credit; ds.user_expavg_credit=project->user_expavg_credit; ds.host_total_credit=project->host_total_credit; ds.host_expavg_credit=project->host_expavg_credit; project->write_statistics_file(); if (cpid_time) { project->cpid_time = cpid_time; } else { project->cpid_time = project->user_create_time; } if (project->dont_use_dcf) { project->duration_correction_factor = 1; } // boolean project attributes. // If the scheduler reply didn't specify them, they're not set. // project->verify_files_on_app_start = verify_files_on_app_start; project->non_cpu_intensive = non_cpu_intensive; project->ended = ended; return 0; } else if (xp.parse_str("project_name", project->project_name, sizeof(project->project_name))) { continue; } else if (xp.parse_str("master_url", master_url, sizeof(master_url))) { continue; } else if (xp.parse_str("symstore", project->symstore, sizeof(project->symstore))) continue; else if (xp.parse_str("user_name", project->user_name, sizeof(project->user_name))) continue; else if (xp.parse_double("user_total_credit", project->user_total_credit)) continue; else if (xp.parse_double("user_expavg_credit", project->user_expavg_credit)) continue; else if (xp.parse_double("user_create_time", project->user_create_time)) continue; else if (xp.parse_double("cpid_time", cpid_time)) continue; else if (xp.parse_str("team_name", project->team_name, sizeof(project->team_name))) continue; else if (xp.parse_int("hostid", hostid)) continue; else if (xp.parse_double("host_total_credit", project->host_total_credit)) continue; else if (xp.parse_double("host_expavg_credit", project->host_expavg_credit)) continue; else if (xp.parse_str("host_venue", host_venue, sizeof(host_venue))) continue; else if (xp.parse_double("host_create_time", project->host_create_time)) continue; else if (xp.parse_double("request_delay", request_delay)) continue; else if (xp.parse_double("next_rpc_delay", next_rpc_delay)) continue; else if (xp.match_tag("global_preferences")) { retval = dup_element_contents( xp.f->f, "", &global_prefs_xml ); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse global prefs in scheduler reply: %s", boincerror(retval) ); return retval; } } else if (xp.match_tag("project_preferences")) { retval = dup_element_contents( xp.f->f, "", &project_prefs_xml ); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse project prefs in scheduler reply: %s", boincerror(retval) ); return retval; } } else if (xp.match_tag("gui_urls")) { string foo; retval = copy_element_contents(xp.f->f, "", foo); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse GUI URLs in scheduler reply: %s", boincerror(retval) ); return retval; } project->gui_urls = "\n"+foo+"\n"; } else if (xp.match_tag("code_sign_key")) { retval = dup_element_contents( xp.f->f, "", &code_sign_key ); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse code sign key in scheduler reply: %s", boincerror(retval) ); return ERR_XML_PARSE; } strip_whitespace(code_sign_key); } else if (xp.match_tag("code_sign_key_signature")) { retval = dup_element_contents( xp.f->f, "", &code_sign_key_signature ); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse code sign key signature in scheduler reply: %s", boincerror(retval) ); return ERR_XML_PARSE; } } else if (xp.match_tag("app")) { APP app; retval = app.parse(xp); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse application in scheduler reply: %s", boincerror(retval) ); } else { apps.push_back(app); } } else if (xp.match_tag("file_info")) { FILE_INFO file_info; retval = file_info.parse(xp); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse file info in scheduler reply: %s", boincerror(retval) ); } else { file_infos.push_back(file_info); } } else if (xp.match_tag("app_version")) { APP_VERSION av; retval = av.parse(xp); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse application version in scheduler reply: %s", boincerror(retval) ); } else { app_versions.push_back(av); } } else if (xp.match_tag("workunit")) { WORKUNIT wu; retval = wu.parse(xp); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse workunit in scheduler reply: %s", boincerror(retval) ); } else { workunits.push_back(wu); } } else if (xp.match_tag("result")) { RESULT result; // make sure this is here so constructor // gets called each time retval = result.parse_server(xp); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse task in scheduler reply: %s", boincerror(retval) ); } else { results.push_back(result); } } else if (xp.match_tag("result_ack")) { RESULT result; retval = result.parse_name(xp, "/result_ack"); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse ack in scheduler reply: %s", boincerror(retval) ); } else { result_acks.push_back(result); } } else if (xp.match_tag("result_abort")) { RESULT result; retval = result.parse_name(xp, "/result_abort"); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse result abort in scheduler reply: %s", boincerror(retval) ); } else { result_abort.push_back(result); } } else if (xp.match_tag("result_abort_if_not_started")) { RESULT result; retval = result.parse_name(xp, "/result_abort_if_not_started"); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "Can't parse result abort-if-not-started in scheduler reply: %s", boincerror(retval) ); } else { result_abort_if_not_started.push_back(result); } } else if (xp.parse_string("delete_file_info", delete_file_name)) { file_deletes.push_back(delete_file_name); } else if (xp.parse_str("message", msg_buf, sizeof(msg_buf))) { parse_attr(attr_buf, "priority", pri_buf, sizeof(pri_buf)); USER_MESSAGE um(msg_buf, pri_buf); messages.push_back(um); continue; } else if (xp.parse_bool("message_ack", message_ack)) { continue; } else if (xp.parse_bool("project_is_down", project_is_down)) { continue; } else if (xp.parse_str("email_hash", project->email_hash, sizeof(project->email_hash))) { continue; } else if (xp.parse_str("cross_project_id", project->cross_project_id, sizeof(project->cross_project_id))) { continue; } else if (xp.parse_str("external_cpid", project->external_cpid, sizeof(project->external_cpid))) { continue; } else if (xp.match_tag("trickle_down")) { retval = gstate.handle_trickle_down(project, in); if (retval) { msg_printf(project, MSG_INTERNAL_ERROR, "handle_trickle_down failed: %s", boincerror(retval) ); } continue; } else if (xp.parse_bool("non_cpu_intensive", non_cpu_intensive)) { continue; } else if (xp.parse_bool("ended", ended)) { continue; } else if (xp.parse_bool("no_cpu_apps", btemp)) { if (!project->anonymous_platform) { handle_no_rsc_apps("CPU", project, btemp); } continue; // deprecated syntax } else if (xp.parse_bool("no_cuda_apps", btemp)) { if (!project->anonymous_platform) { handle_no_rsc_apps(GPU_TYPE_NVIDIA, project, btemp); } continue; } else if (xp.parse_bool("no_ati_apps", btemp)) { if (!project->anonymous_platform) { handle_no_rsc_apps(GPU_TYPE_ATI, project, btemp); } continue; } else if (xp.parse_str("no_rsc_apps", buf, sizeof(buf))) { if (!project->anonymous_platform) { handle_no_rsc_apps(buf, project, true); } continue; } else if (xp.parse_bool("verify_files_on_app_start", verify_files_on_app_start)) { continue; } else if (xp.parse_bool("send_full_workload", send_full_workload)) { continue; } else if (xp.parse_bool("dont_use_dcf", dont_use_dcf)) { continue; } else if (xp.parse_int("send_time_stats_log", send_time_stats_log)){ continue; } else if (xp.parse_int("send_job_log", send_job_log)) { continue; } else if (xp.parse_int("scheduler_version", scheduler_version)) { continue; } else if (xp.match_tag("project_files")) { retval = parse_project_files(xp, project_files); #ifdef ENABLE_AUTO_UPDATE } else if (xp.match_tag("auto_update")) { retval = auto_update.parse(xp); if (!retval) auto_update.present = true; #endif } else if (xp.match_tag("rss_feeds")) { got_rss_feeds = true; parse_rss_feed_descs(xp, sr_feeds); continue; } else if (xp.match_tag("trickle_up_urls")) { parse_trickle_up_urls(xp, trickle_up_urls); continue; } else if (xp.parse_int("userid", project->userid)) { continue; } else if (xp.parse_int("teamid", project->teamid)) { continue; } else if (xp.parse_double("desired_disk_usage", project->desired_disk_usage)) { continue; } else { if (log_flags.unparsed_xml) { msg_printf(project, MSG_INFO, "[unparsed_xml] SCHEDULER_REPLY::parse(): unrecognized %s\n", xp.parsed_tag ); } } } if (found_start_tag) { msg_printf(project, MSG_INTERNAL_ERROR, "No close tag in scheduler reply"); } else { msg_printf(project, MSG_INTERNAL_ERROR, "No start tag in scheduler reply"); } return ERR_XML_PARSE; } #endif USER_MESSAGE::USER_MESSAGE(char* m, char* p) { message = m; priority = p; }