// The contents of this file are subject to the BOINC Public License // Version 1.0 (the "License"); you may not use this file except in // compliance with the License. You may obtain a copy of the License at // http://boinc.berkeley.edu/license_1.0.txt // // Software distributed under the License is distributed on an "AS IS" // basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the // License for the specific language governing rights and limitations // under the License. // // The Original Code is the Berkeley Open Infrastructure for Network Computing. // // The Initial Developer of the Original Code is the SETI@home project. // Portions created by the SETI@home project are Copyright (C) 2002 // University of California at Berkeley. All Rights Reserved. // // Contributor(s): // // High-level logic for communicating with scheduling servers, // and for merging the result of a scheduler RPC into the client state // The scheduler RPC mechanism is in scheduler_op.C #include "cpp.h" #ifdef _WIN32 #include "boinc_win.h" #endif #ifndef _WIN32 #include #include #include #include #endif #include "crypt.h" #include "error_numbers.h" #include "file_names.h" #include "filesys.h" #include "parse.h" #include "util.h" #include "client_msgs.h" #include "scheduler_op.h" #include "client_state.h" using std::max; using std::vector; using std::string; static double trs; // quantities like avg CPU time decay by a factor of e every week // #define EXP_DECAY_RATE (1./(SECONDS_PER_DAY*7)) // how often to show user "backing off" messages // const int SECONDS_BEFORE_REPORTING_MIN_RPC_TIME_AGAIN = 60*60; // try to report results this much before their deadline // #define REPORT_DEADLINE_CUSHION SECONDS_PER_DAY static int proj_min_results(PROJECT* p, int ncpus) { return (int)(ceil(ncpus*p->resource_share/trs)); } void PROJECT::set_min_rpc_time(double future_time) { if (future_time > min_rpc_time) { min_rpc_time = future_time; } min_report_min_rpc_time = 0; } // Return true iff we should not contact the project yet. // Print a message to the user if we haven't recently // bool PROJECT::waiting_until_min_rpc_time(double now) { if (min_rpc_time > now ) { if (now >= min_report_min_rpc_time) { min_report_min_rpc_time = now + SECONDS_BEFORE_REPORTING_MIN_RPC_TIME_AGAIN; msg_printf( this, MSG_ERROR, "Deferring communication with project for %s\n", timediff_format(min_rpc_time - now).c_str() ); } return true; } return false; } // find a project that needs to have its master file fetched // PROJECT* CLIENT_STATE::next_project_master_pending() { unsigned int i; PROJECT* p; double now = dtime(); for (i=0; iwaiting_until_min_rpc_time(now)) continue; if (p->suspended_via_gui) continue; if (p->master_url_fetch_pending) { return p; } } return 0; } // find a project that needs to contact its scheduling server // PROJECT* CLIENT_STATE::next_project_sched_rpc_pending() { unsigned int i; double now = dtime(); PROJECT* p; for (i=0; iwaiting_until_min_rpc_time(now)) continue; if (p->suspended_via_gui) continue; if (p->sched_rpc_pending) { return p; } } return 0; } // return the next project after "old" that // 1) is eligible for a scheduler RPC // 2) has work_request > 0 // 3) has master_url_fetch_pending == false // 4) has dont_request_more_work == false // PROJECT* CLIENT_STATE::next_project_need_work(PROJECT *old) { PROJECT *p; double now = dtime(); unsigned int i; bool found_old = (old == 0); for (i=0; imaster_url_fetch_pending) continue; if (p->waiting_until_min_rpc_time(now)) continue; if (p->suspended_via_gui) continue; if (p->dont_request_more_work) continue; if (found_old && p->work_request > 0) { return p; } } return 0; } // Write a scheduler request to a disk file // (later sent to the scheduling server) // int CLIENT_STATE::make_scheduler_request(PROJECT* p, double work_req) { char buf[1024]; get_sched_request_filename(*p, buf); FILE* f = boinc_fopen(buf, "wb"); MIOFILE mf; unsigned int i; RESULT* rp; int retval; #if 0 double free, possible; #endif char cross_project_id[MD5_LEN]; trs = total_resource_share(); if (!f) return ERR_FOPEN; mf.init_file(f); fprintf(f, "\n" " %s\n" " %d\n" " %d\n" " %s\n" " %d\n" " %d\n" " %f\n" " %f\n" " %f\n", p->authenticator, p->hostid, p->rpc_seqno, p->anonymous_platform?"anonymous":platform_name, core_client_major_version, core_client_minor_version, work_req, p->resource_share / trs, ettprc(p, proj_min_results(p, ncpus)-1) ); if (p->anonymous_platform) { fprintf(f, " \n"); for (i=0; iproject != p) continue; avp->write(mf); } fprintf(f, " \n"); } #if 0 anything_free(free); fprintf(f, " %f\n", free); total_potential_offender(p, possible); fprintf(f, " %f\n", possible); total_potential_self(p, possible); fprintf(f, " %f\n", possible); #endif if (strlen(p->code_sign_key)) { fprintf(f, " \n%s\n", p->code_sign_key); } // insert global preferences if present // if (boinc_file_exists(GLOBAL_PREFS_FILE_NAME)) { FILE* fprefs = fopen(GLOBAL_PREFS_FILE_NAME, "r"); if (fprefs) { copy_stream(fprefs, f); fclose(fprefs); } PROJECT* pp = lookup_project(global_prefs.source_project); if (pp && strlen(pp->email_hash)) { fprintf(f, "%s\n", pp->email_hash ); } } // send the maximum of cross_project_id over projects // with the same email hash as this one // strcpy(cross_project_id, p->cross_project_id); for (i=0; iemail_hash, p->email_hash)) continue; if (strcmp(project->cross_project_id, cross_project_id) > 0) { strcpy(cross_project_id, project->cross_project_id); } } fprintf(f, "%s\n", cross_project_id); retval = time_stats.write(mf, true); if (retval) return retval; retval = net_stats.write(mf); if (retval) return retval; retval = host_info.write(mf); if (retval) return retval; for (i=0; iproject == p && rp->ready_to_report) { rp->write(mf, true); } } read_trickle_files(p, f); // report sticky files as needed // for (i=0; iproject != p) continue; if (!fip->report_on_rpc) continue; fprintf(f, " \n" " %s\n" " \n" " \n", fip->name ); } fprintf(f, "\n"); fclose(f); return 0; } // find a project with results that are overdue to report, // and which we're allowed to contact. // PROJECT* CLIENT_STATE::find_project_with_overdue_results() { unsigned int i; RESULT* r; double now = dtime(); for (i=0; iproject; if (p->waiting_until_min_rpc_time(now)) continue; if (p->suspended_via_gui) continue; if (!r->ready_to_report) continue; if (return_results_immediately || (r->report_deadline <= (now + REPORT_DEADLINE_CUSHION)) ) { return p; } } return 0; } #if 0 // return true if we're allowed to do a scheduler RPC to at least one project // bool CLIENT_STATE::some_project_rpc_ok() { unsigned int i; double now = dtime(); for (i=0; imin_rpc_time < now) return true; } return false; } #endif // return the average number of CPU seconds completed by the client // for project p in a second of (wall-clock) time // double CLIENT_STATE::avg_proc_rate(PROJECT *p) { return (p->resource_share / trs) * ncpus * time_stats.active_frac; } // "estimated time to project result count" // return the estimated amount of time that will elapse until the // number of results for project p will reach k // double CLIENT_STATE::ettprc(PROJECT *p, int k) { int num_results_to_skip = k; double est = 0; // total up the estimated time for this project's unstarted // and partially completed results, // omitting the last k // for (vector::reverse_iterator iter = results.rbegin(); iter != results.rend(); iter++ ) { RESULT *rp = *iter; if (rp->project != p || rp->state > RESULT_FILES_DOWNLOADED || rp->ready_to_report ) continue; if (num_results_to_skip > 0) { --num_results_to_skip; continue; } est += rp->estimated_cpu_time_remaining(); } est /= avg_proc_rate(p); return est; } // set work_request for each project and return the urgency level for // requesting more work // only set non-zero work requests for projects that are allowed to do // a scheduler RPC // int CLIENT_STATE::compute_work_requests() { int urgency = DONT_NEED_WORK; unsigned int i; double work_min_period = global_prefs.work_buf_min_days * SECONDS_PER_DAY; double now = dtime(); trs = total_resource_share(); // for each project, compute // min_results = min # of results for project needed by CPU scheduling, // to avoid "starvation". // Then estimate how long it's going to be until we have fewer // than this # of results remaining. // for (i=0; iwork_request = 0; if (p->min_rpc_time >= now) continue; if (p->dont_request_more_work) continue; if (p->suspended_via_gui) continue; int min_results = proj_min_results(p, ncpus); double estimated_time_to_starvation = ettprc(p, min_results-1); // determine urgency // if (estimated_time_to_starvation < work_min_period) { if (estimated_time_to_starvation == 0) { //msg_printf(p, MSG_INFO, "is starved"); urgency = NEED_WORK_IMMEDIATELY; } else { #if 0 msg_printf(p, MSG_INFO, "will starve in %.2f sec", estimated_time_to_starvation ); #endif urgency = max(NEED_WORK, urgency); } } // determine work requests for each project // NOTE: don't need to divide by active_frac etc.; // the scheduler does that (see sched/sched_send.C) // p->work_request = max(0.0, //(2*work_min_period - estimated_time_to_starvation) (work_min_period - estimated_time_to_starvation) * ncpus ); //msg_printf(p, MSG_INFO, "work req: %f sec", p->work_request); } if (urgency == DONT_NEED_WORK) { for (i=0; iwork_request = 0; } } return urgency; } // called from the client's polling loop. // initiate scheduler RPC activity if needed and possible // bool CLIENT_STATE::scheduler_rpc_poll(double now) { int urgency = DONT_NEED_WORK; PROJECT *p; bool action=false; static double last_time=0; if (now - last_time < 1.0) return false; last_time = now; switch(scheduler_op->state) { case SCHEDULER_OP_STATE_IDLE: if (network_suspended) break; urgency = compute_work_requests(); // highest priority is to report overdue results // p = find_project_with_overdue_results(); if (p) { scheduler_op->init_return_results(p); action = true; } else if (!(exit_when_idle && contacted_sched_server) && urgency != DONT_NEED_WORK) { if (urgency == NEED_WORK) { msg_printf(NULL, MSG_INFO, "May run out of work in %.2f days; requesting more", global_prefs.work_buf_min_days ); } else if (urgency == NEED_WORK_IMMEDIATELY) { msg_printf(NULL, MSG_INFO, "Insufficient work; requesting more" ); } scheduler_op->init_get_work(); action = true; } else if ((p=next_project_master_pending())) { scheduler_op->init_get_work(); action = true; } else if ((p=next_project_sched_rpc_pending())) { scheduler_op->init_return_results(p); action = true; } break; default: scheduler_op->poll(); if (scheduler_op->state == SCHEDULER_OP_STATE_IDLE) { action = true; } break; } return action; } // Handle the reply from a scheduler // int CLIENT_STATE::handle_scheduler_reply( PROJECT* project, char* scheduler_url, int& nresults ) { SCHEDULER_REPLY sr; FILE* f; int retval; unsigned int i; bool signature_valid, update_global_prefs=false, update_project_prefs=false; char buf[256], filename[256]; nresults = 0; contacted_sched_server = true; SCOPE_MSG_LOG scope_messages(log_messages, CLIENT_MSG_LOG::DEBUG_SCHED_OP); get_sched_reply_filename(*project, filename); scope_messages.printf_file(filename, "reply: "); f = fopen(filename, "r"); if (!f) return ERR_FOPEN; retval = sr.parse(f, project); fclose(f); if (retval) return retval; if (sr.request_delay) { double x = dtime() + sr.request_delay; if (x > project->min_rpc_time) project->min_rpc_time = x; } if (strlen(sr.message)) { sprintf(buf, "Message from server: %s", sr.message); int prio = (!strcmp(sr.message_priority, "high"))?MSG_ERROR:MSG_INFO; show_message(project, buf, prio); } // if project is down, return error (so that we back off) // and don't do anything else // if (sr.project_is_down) { return ERR_PROJECT_DOWN; } if (sr.hostid) { project->hostid = sr.hostid; project->rpc_seqno = 0; } // see if we have a new venue from this project // if (strlen(sr.host_venue) && strcmp(project->host_venue, sr.host_venue)) { safe_strcpy(project->host_venue, sr.host_venue); msg_printf(project, MSG_INFO, "New host venue: %s", sr.host_venue); update_project_prefs = true; if (project == global_prefs_source_project()) { strcpy(main_host_venue, sr.host_venue); update_global_prefs = true; } } // if the scheduler reply includes global preferences, // insert extra elements, write to disk, and parse // if (sr.global_prefs_xml) { f = boinc_fopen(GLOBAL_PREFS_FILE_NAME, "w"); if (!f) return ERR_FOPEN; fprintf(f, "\n" ); // tag with the project and scheduler URL, // but only if not already tagged // if (!strstr(sr.global_prefs_xml, "")) { fprintf(f, " %s\n" " %s\n", project->master_url, scheduler_url ); } fprintf(f, "%s" "\n", sr.global_prefs_xml ); fclose(f); update_global_prefs = true; } if (update_global_prefs) { bool found_venue; retval = global_prefs.parse_file( GLOBAL_PREFS_FILE_NAME, project->host_venue, found_venue ); if (retval) { msg_printf(project, MSG_ERROR, "Can't parse general preferences"); } else { show_global_prefs_source(found_venue); install_global_prefs(); } } // deal with project preferences (should always be there) // If they've changed, write to account file, // then parse to get our venue, and pass to running apps // if (sr.project_prefs_xml) { if (strcmp(project->project_prefs.c_str(), sr.project_prefs_xml)) { project->project_prefs = string(sr.project_prefs_xml); update_project_prefs = true; } } if (update_project_prefs) { retval = project->write_account_file(); if (retval) { msg_printf(project, MSG_ERROR, "Can't write account file: %d", retval); return retval; } project->parse_account_file(); project->parse_preferences_for_user_files(); active_tasks.request_reread_prefs(project); } // if the scheduler reply includes a code-signing key, // accept it if we don't already have one from the project. // Otherwise verify its signature, using the key we already have. // if (sr.code_sign_key) { if (!strlen(project->code_sign_key)) { safe_strcpy(project->code_sign_key, sr.code_sign_key); } else { if (sr.code_sign_key_signature) { retval = verify_string2( sr.code_sign_key, sr.code_sign_key_signature, project->code_sign_key, signature_valid ); if (!retval && signature_valid) { safe_strcpy(project->code_sign_key, sr.code_sign_key); } else { msg_printf(project, MSG_ERROR, "New code signing key doesn't validate"); } } else { msg_printf(project, MSG_ERROR, "Missing code sign key signature"); } } } // copy new entities to client state // for (i=0; iname ); delete app; } else { apps.push_back(app); } } } FILE_INFO* fip; for (i=0; imerge_info(sr.file_infos[i]); } else { fip = new FILE_INFO; *fip = sr.file_infos[i]; retval = link_file_info(project, fip); if (retval) { msg_printf(project, MSG_ERROR, "Can't link file_info %s in sched reply", fip->name ); delete fip; } else { file_infos.push_back(fip); } } } for (i=0; iname); fip->sticky = false; } } for (i=0; iclear_errors(); continue; } avp = new APP_VERSION; *avp = sr.app_versions[i]; retval = link_app_version(project, avp); if (retval) { msg_printf(project, MSG_ERROR, "Can't link app version %s %d in sched reply", avp->app_name, avp->version_num ); delete avp; continue; } app_versions.push_back(avp); } for (i=0; iapp_name, sr); if (vnum < 0) { msg_printf(project, MSG_ERROR, "Can't find app version for WU %s", wup->name ); delete wup; continue; } wup->version_num = vnum; retval = link_workunit(project, wup); if (retval) { msg_printf(project, MSG_ERROR, "Can't link workunit %s in sched reply", wup->name ); delete wup; continue; } wup->clear_errors(); workunits.push_back(wup); } for (i=0; iname ); delete rp; continue; } results.push_back(rp); rp->state = RESULT_NEW; nresults++; } // update records for ack'ed results // for (i=0; igot_server_ack = true; } else { msg_printf(project, MSG_ERROR, "Got ack for result %s, can't find", sr.result_acks[i].name ); } } // remove acked trickle files // if (sr.message_ack) { remove_trickle_files(project); } if (sr.send_file_list) { project->send_file_list = true; } project->sched_rpc_pending = false; set_client_state_dirty("handle_scheduler_reply"); scope_messages.printf("CLIENT_STATE::handle_scheduler_reply(): State after handle_scheduler_reply():\n"); print_summary(); return 0; } const char *BOINC_RCSID_d35a4a7711 = "$Id$";