// This file is part of BOINC. // http://boinc.berkeley.edu // Copyright (C) 2019 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 . // Utility classes for the BOINC scheduler #include "config.h" #include #include #include #include #include #include "parse.h" #include "error_numbers.h" #include "str_util.h" #include "util.h" #include "boinc_db.h" #include "sched_main.h" #include "sched_util.h" #include "sched_msgs.h" #include "sched_send.h" #include "time_stats_log.h" #include "sched_types.h" #ifdef _USING_FCGI_ #include "boinc_fcgi.h" #endif using std::string; SCHEDULER_REQUEST* g_request; SCHEDULER_REPLY* g_reply; WORK_REQ* g_wreq; // remove (by truncating) any quotes from the given string. // This is for things (e.g. authenticator) that will be used in // a SQL query, to prevent SQL injection attacks // void remove_quotes(char* p) { int i, n=strlen(p); for (i=0; ilookup_app_name(app_name); if (!app) return ERR_NOT_FOUND; double pf = host_usage.avg_ncpus * g_reply->host.p_fpops; if (host_usage.proc_type != PROC_TYPE_CPU) { COPROC* cp = g_request->coprocs.proc_type_to_coproc(host_usage.proc_type); if (cp) { pf += host_usage.gpu_usage*cp->peak_flops; } } host_usage.peak_flops = pf; return 0; } if (xp.parse_str("app_name", app_name, 256)) continue; if (xp.parse_str("platform", platform, 256)) continue; if (xp.parse_str("plan_class", plan_class, 256)) continue; if (xp.parse_int("version_num", version_num)) continue; if (xp.parse_double("avg_ncpus", x)) { if (x>0) host_usage.avg_ncpus = x; continue; } if (xp.parse_double("flops", x)) { if (x>0) host_usage.projected_flops = x; continue; } if (xp.match_tag("coproc")) { COPROC_REQ coproc_req; int retval = coproc_req.parse(xp); if (!retval) { int rt = coproc_type_name_to_num(coproc_req.type); if (rt <= 0) { log_messages.printf(MSG_NORMAL, "UNKNOWN COPROC TYPE %s\n", coproc_req.type ); continue; } host_usage.proc_type = rt; host_usage.gpu_usage = coproc_req.count; } continue; } } return ERR_XML_PARSE; } int FILE_INFO::parse(XML_PARSER& xp) { memset(this, 0, sizeof(*this)); while (!xp.get_tag()) { if (xp.match_tag("/file_info")) { if (!strlen(name)) return ERR_XML_PARSE; return 0; } if (xp.parse_str("name", name, 256)) continue; if (xp.parse_double("nbytes", nbytes)) continue; if (xp.parse_int("status", status)) continue; if (xp.parse_bool("sticky", sticky)) continue; } return ERR_XML_PARSE; } int OTHER_RESULT::parse(XML_PARSER& xp) { strcpy(name, ""); have_plan_class = false; app_version = -1; while (!xp.get_tag()) { if (xp.match_tag("/other_result")) { if (!strcmp(name, "")) return ERR_XML_PARSE; return 0; } if (xp.parse_str("name", name, sizeof(name))) continue; if (xp.parse_int("app_version", app_version)) continue; if (xp.parse_str("plan_class", plan_class, sizeof(plan_class))) { have_plan_class = true; continue; } } return ERR_XML_PARSE; } int IP_RESULT::parse(XML_PARSER& xp) { report_deadline = 0; cpu_time_remaining = 0; strcpy(name, ""); while (!xp.get_tag()) { if (xp.match_tag("/ip_result")) return 0; if (xp.parse_str("name", name, sizeof(name))) continue; if (xp.parse_double("report_deadline", report_deadline)) continue; if (xp.parse_double("cpu_time_remaining", cpu_time_remaining)) continue; } return ERR_XML_PARSE; } int CLIENT_PLATFORM::parse(XML_PARSER& xp) { strcpy(name, ""); while (!xp.get_tag()) { if (xp.match_tag("/alt_platform")) return 0; if (xp.parse_str("name", name, sizeof(name))) continue; } return ERR_XML_PARSE; } // Parse user's project preferences. // TODO: use XML_PARSER // void PROJECT_PREFS::parse() { char buf[8096]; std::string str; unsigned int pos = 0; int temp_int=0; bool flag; extract_venue(g_reply->user.project_prefs, g_reply->host.venue, buf, sizeof(buf)); str = buf; // scan user's project prefs for elements of the form N, // indicating the apps they want to run. // selected_apps.clear(); while (parse_int(str.substr(pos,str.length()-pos).c_str(), "", temp_int)) { APP_INFO ai; ai.appid = temp_int; ai.work_available = false; selected_apps.push_back(ai); pos = str.find("", pos) + 1; } if (parse_bool(buf,"allow_non_preferred_apps", flag)) { allow_non_preferred_apps = flag; } if (parse_bool(buf,"allow_beta_work", flag)) { allow_beta_work = flag; } if (parse_bool(buf,"no_gpus", flag)) { // deprecated, but need to handle if (flag) { for (int i=1; i", temp_int)) { max_cpus = temp_int; } if (parse_int(buf, "", temp_int)) { max_jobs_in_progress = temp_int; } } void WORK_REQ::add_no_work_message(const char* message) { for (unsigned int i=0; iinsert_message(buf, "notice"); } else { g_reply->insert_message( "Syntax error in app_info.xml", "notice" ); } } else { // this case happens if the app version // refers to a deprecated app } cav.app = 0; } // store the CLIENT_APP_VERSION even if it didn't parse. // This is necessary to maintain the correspondence // with result.app_version // client_app_versions.push_back(cav); } } continue; } if (xp.parse_int("core_client_major_version", core_client_major_version)) continue; if (xp.parse_int("core_client_minor_version", core_client_minor_version)) continue; if (xp.parse_int("core_client_release", core_client_release)) continue; if (xp.parse_double("work_req_seconds", work_req_seconds)) continue; if (xp.parse_double("cpu_req_secs", cpu_req_secs)) continue; if (xp.parse_double("cpu_req_instances", cpu_req_instances)) continue; if (xp.parse_double("resource_share_fraction", resource_share_fraction)) continue; if (xp.parse_double("rrs_fraction", rrs_fraction)) continue; if (xp.parse_double("prrs_fraction", prrs_fraction)) continue; if (xp.parse_double("estimated_delay", cpu_estimated_delay)) continue; if (xp.parse_double("duration_correction_factor", host.duration_correction_factor)) continue; if (xp.parse_bool("dont_send_work", dont_send_work)) continue; if (xp.match_tag("global_preferences")) { safe_strcpy(global_prefs_xml, "\n"); char buf[BLOB_SIZE]; retval = xp.element_contents( "", buf, sizeof(buf) ); if (retval) return "error copying global prefs"; safe_strcat(global_prefs_xml, buf); // xp.element_contents() strips the linebreak from buf so we add it back because it is essential safe_strcat(global_prefs_xml, "\n\n"); continue; } if (xp.match_tag("working_global_preferences")) { retval = xp.element_contents( "", working_global_prefs_xml, sizeof(working_global_prefs_xml) ); if (retval) return "error copying working global prefs"; continue; } if (xp.parse_str("global_prefs_source_email_hash", global_prefs_source_email_hash, sizeof(global_prefs_source_email_hash))) continue; if (xp.match_tag("host_info")) { host.parse(xp); continue; } if (xp.match_tag("time_stats")) { host.parse_time_stats(xp); continue; } if (xp.match_tag("time_stats_log")) { if (handle_time_stats_log(xp.f->f)) { log_messages.printf(MSG_NORMAL, "SCHEDULER_REQUEST::parse(): Couldn't parse contents of . Ignoring it."); } else { have_time_stats_log = true; } continue; } if (xp.match_tag("net_stats")) { host.parse_net_stats(xp); continue; } if (xp.match_tag("disk_usage")) { host.parse_disk_usage(xp); continue; } if (xp.match_tag("result")) { retval = result.parse_from_client(xp); if (retval) continue; if (strstr(result.name, "download") || strstr(result.name, "upload")) { file_xfer_results.push_back(result); continue; } if (config.max_results_accepted && (int)(results.size()) >= config.max_results_accepted) { results_truncated = true; continue; } // check if client is sending the same result twice. // Shouldn't happen, but if it does bad things will happen // bool found = false; for (unsigned int i=0; if, "", code_sign_key, sizeof(code_sign_key)); strip_whitespace(code_sign_key); continue; } if (xp.match_tag("msg_from_host")) { MSG_FROM_HOST_DESC md; retval = md.parse(xp); if (!retval) { msgs_from_host.push_back(md); } continue; } if (xp.match_tag("file_info")) { FILE_INFO fi; retval = fi.parse(xp); if (!retval) { file_infos.push_back(fi); } continue; } if (xp.match_tag("host_venue")) { continue; } if (xp.match_tag("other_results")) { have_other_results_list = true; while (!xp.get_tag()) { if (xp.match_tag("/other_results")) break; if (xp.match_tag("other_result")) { OTHER_RESULT o_r; retval = o_r.parse(xp); if (!retval) { other_results.push_back(o_r); } } } continue; } if (xp.match_tag("in_progress_results")) { have_ip_results_list = true; int i = 0; double now = time(0); while (!xp.get_tag()) { if (xp.match_tag("/in_progress_results")) break; if (xp.match_tag("ip_result")) { IP_RESULT ir; retval = ir.parse(xp); if (!retval) { if (!strlen(ir.name)) { sprintf(ir.name, "ip%d", i++); } ir.report_deadline -= now; ip_results.push_back(ir); } } } continue; } if (xp.match_tag("coprocs")) { coprocs.parse(xp); continue; } if (xp.parse_bool("client_cap_plan_class", client_cap_plan_class)) continue; if (xp.parse_int("sandbox", sandbox)) continue; if (xp.parse_int("allow_multiple_clients", allow_multiple_clients)) continue; if (xp.match_tag("user_keywords")) { user_keywords.parse(xp); continue; } if (xp.parse_str("client_brand", client_brand, sizeof(client_brand))) continue; // unused or deprecated stuff follows if (xp.match_tag("active_task_set")) continue; if (xp.match_tag("app")) continue; if (xp.match_tag("app_version")) continue; if (xp.match_tag("duration_variability")) continue; if (xp.match_tag("new_version_check_time")) continue; if (xp.match_tag("newer_version")) continue; if (xp.match_tag("project")) continue; if (xp.match_tag("project_files")) continue; if (xp.match_tag("proxy_info")) continue; if (xp.match_tag("user_network_request")) continue; if (xp.match_tag("user_run_request")) continue; if (xp.match_tag("master_url")) continue; if (xp.match_tag("project_name")) continue; if (xp.match_tag("user_name")) continue; if (xp.match_tag("team_name")) continue; if (xp.match_tag("email_hash")) continue; if (xp.match_tag("user_total_credit")) continue; if (xp.match_tag("user_expavg_credit")) continue; if (xp.match_tag("user_create_time")) continue; if (xp.match_tag("host_total_credit")) continue; if (xp.match_tag("host_expavg_credit")) continue; if (xp.match_tag("host_create_time")) continue; if (xp.match_tag("nrpc_failures")) continue; if (xp.match_tag("master_fetch_failures")) continue; if (xp.match_tag("min_rpc_time")) continue; if (xp.match_tag("short_term_debt")) continue; if (xp.match_tag("long_term_debt")) continue; if (xp.match_tag("resource_share")) continue; if (xp.match_tag("scheduler_url")) continue; if (xp.match_tag("/project")) continue; if (xp.match_tag("?xml")) continue; log_messages.printf(MSG_NORMAL, "SCHEDULER_REQUEST::parse(): unexpected: %s\n", xp.parsed_tag ); xp.skip_unexpected(); } return "no end tag"; } // I'm not real sure why this is here. // Why not copy the request message directly? // int SCHEDULER_REQUEST::write(FILE* fout) { unsigned int i; fprintf(fout, "\n" " %s\n" " %s\n" " %s\n" " %lu\n" " %d\n" " %d\n" " %d\n" " %d\n" " %.15f\n" " %.15f\n" " %.15f\n" " %.15f\n" " %.15f\n" " %s\n" " %s\n", authenticator, platform.name, cross_project_id, hostid, core_client_major_version, core_client_minor_version, core_client_release, rpc_seqno, work_req_seconds, resource_share_fraction, rrs_fraction, prrs_fraction, cpu_estimated_delay, code_sign_key, is_anonymous(platforms.list[0])?"true":"false" ); for (i=0; i\n" " %s\n" " %d\n" " \n", client_app_versions[i].app_name, client_app_versions[i].version_num ); } fprintf(fout, " \n" " %s" " \n", global_prefs_xml ); fprintf(fout, " %s\n", global_prefs_source_email_hash ); fprintf(fout, " \n" " %lu\n" " %d\n" " %d\n" " %.15f\n" " %.15f\n" " %.15f\n" " %.15f\n" " %.15f\n", host.id, host.rpc_time, host.timezone, host.d_total, host.d_free, host.d_boinc_used_total, host.d_boinc_used_project, host.d_boinc_max ); for (i=0; i\n" " %s\n" " %d\n" " %.15f\n" " %d\n" " %d\n" " \n", results[i].name, results[i].client_state, results[i].cpu_time, results[i].exit_status, results[i].app_version_num ); } for (i=0; i\n" " %s\n" " %s\n" " \n", msgs_from_host[i].variety, msgs_from_host[i].msg_text.c_str() ); } for (i=0; i\n" " %s\n" " \n", file_infos[i].name ); fprintf(fout, "\n"); } return 0; } int MSG_FROM_HOST_DESC::parse(XML_PARSER& xp) { char buf[256]; msg_text = ""; strcpy(variety, ""); MIOFILE& in = *(xp.f); while (in.fgets(buf, sizeof(buf))) { if (match_tag(buf, "")) return 0; if (parse_str(buf, "", variety, sizeof(variety))) continue; msg_text += buf; } return ERR_XML_PARSE; } SCHEDULER_REPLY::SCHEDULER_REPLY() { wreq.clear(); memset(&disk_limits, 0, sizeof(disk_limits)); request_delay = 0; hostid = 0; lockfile_fd = -1; send_global_prefs = false; strcpy(code_sign_key, ""); strcpy(code_sign_key_signature, ""); memset(&user, 0, sizeof(user)); memset(&host, 0, sizeof(host)); memset(&team, 0, sizeof(team)); nucleus_only = false; project_is_down = false; send_msg_ack = false; strcpy(email_hash, ""); } static bool have_apps_for_client() { for (int i=0; ihave_apps_for_proc_type[i]) { if (!i) return true; COPROC* cp = g_request->coprocs.proc_type_to_coproc(i); if (cp && cp->count) return true; } } return false; } int SCHEDULER_REPLY::write(FILE* fout, SCHEDULER_REQUEST& sreq) { unsigned int i; char buf[BLOB_SIZE]; // Note: at one point we had // "\n" // after the Content-type (to make it legit XML), // but this broke 4.19 clients // fprintf(fout, "Content-type: text/xml\n\n" "\n" "%d\n", BOINC_MAJOR_VERSION*100+BOINC_MINOR_VERSION ); if (sreq.core_client_version >= 70028) { fprintf(fout, "\n"); } if (strlen(config.master_url)) { fprintf(fout, "%s\n", config.master_url ); } if (config.ended) { fprintf(fout, " 1\n"); } // if the scheduler has requested a delay OR the sysadmin has configured // the scheduler with a minimum time between RPCs, send a delay request. // Make it 1% larger than the min required to take care of time skew. // If this is less than one second bigger, bump up by one sec. // if (request_delay || config.min_sendwork_interval) { double min_delay_needed = 1.01*config.min_sendwork_interval; if (min_delay_needed < config.min_sendwork_interval+1) { min_delay_needed = config.min_sendwork_interval+1; } if (request_delay%f\n", request_delay); } log_messages.printf(MSG_NORMAL, "Sending reply to [HOST#%lu]: %d results, delay req %.2f\n", host.id, wreq.njobs_sent, request_delay ); if (sreq.core_client_version <= 41900) { string msg; string pri = "low"; for (i=0; i0) { // any newlines will break message printing under 4.19 and under! // replace them with spaces. // while (1) { string::size_type pos = msg.find("\n", 0); if (pos == string::npos) break; msg.replace(pos, 1, " "); } fprintf(fout, "%s\n", pri.c_str(), msg.c_str() ); } } else if (sreq.core_client_version <= 61100) { char prio[256]; for (i=0; i%s\n", prio, um.message.c_str() ); } } else { for (i=0; i%s\n", um.priority.c_str(), um.message.c_str() ); } } fprintf(fout, "%s\n", config.long_name ); if (config.request_time_stats_log) { if (!have_time_stats_log()) { fprintf(fout, "1\n"); } } if (project_is_down) { fprintf(fout,"\n"); goto end; } if (config.workload_sim) { fprintf(fout, "\n"); } if (nucleus_only) goto end; if (strlen(config.symstore)) { fprintf(fout, "%s\n", config.symstore); } if (config.next_rpc_delay) { fprintf(fout, "%f\n", config.next_rpc_delay); } if (user.id) { xml_escape(user.name, buf, sizeof(buf)); fprintf(fout, "%lu\n" "%s\n" "%f\n" "%f\n" "%d\n", user.id, buf, user.total_credit, user.expavg_credit, user.create_time ); // be paranoid about the following to avoid sending null // if (strlen(email_hash)) { fprintf(fout, "%s\n", email_hash ); } if (strlen(user.cross_project_id)) { char external_cpid[MD5_LEN]; safe_strcpy(buf, user.cross_project_id); safe_strcat(buf, user.email_addr); md5_block((unsigned char*)buf, strlen(buf), external_cpid); fprintf(fout, "%s\n" "%s\n", user.cross_project_id, external_cpid ); } if (send_global_prefs) { fputs(user.global_prefs, fout); fputs("\n", fout); } // always send project prefs // fputs(user.project_prefs, fout); fputs("\n", fout); } if (hostid) { fprintf(fout, "%lu\n", hostid ); } fprintf(fout, "%f\n" "%f\n" "%s\n" "%d\n", host.total_credit, host.expavg_credit, host.venue, host.create_time ); // might want to send team credit too. // if (team.id) { xml_escape(team.name, buf, sizeof(buf)); fprintf(fout, "%lu\n" "%s\n", team.id, buf ); } else { fprintf(fout, "\n" ); } // acknowledge results // for (i=0; i\n" " %s\n" "\n", result_acks[i].c_str() ); } // abort results // for (i=0; i\n" " %s\n" "\n", result_aborts[i].c_str() ); } // abort results not started // for (i=0; i\n" " %s\n" "\n", result_abort_if_not_starteds[i].c_str() ); } for (i=0; i\n", fout); fputs(code_sign_key, fout); fputs("\n\n", fout); } if (strlen(code_sign_key_signature)) { fputs("\n", fout); fputs(code_sign_key_signature, fout); fputs("\n", fout); } if (send_msg_ack) { fputs("\n", fout); } for (i=0; i\n"); } if (config.verify_files_on_app_start) { fprintf(fout, "\n"); } for (i=0; i%s\n", file_deletes[i].name ); } for (i=0; icore_client_version < 70040) { fprintf(fout, "%d\n" "%d\n" "%d\n", ssp->have_apps_for_proc_type[PROC_TYPE_CPU]?0:1, ssp->have_apps_for_proc_type[PROC_TYPE_NVIDIA_GPU]?0:1, ssp->have_apps_for_proc_type[PROC_TYPE_AMD_GPU]?0:1 ); } // write modern form. // for (i=0; i0) { // skip types that the client doesn't have // COPROC* cp = g_request->coprocs.proc_type_to_coproc(i); if (!cp || !cp->count) continue; } if (!ssp->have_apps_for_proc_type[i]) { fprintf(fout, "%s\n", proc_type_name_xml(i) ); } } } else { if (config.debug_send) { log_messages.printf(MSG_NORMAL, "[send] no app versions for client resources; suppressing no_rsc_apps\n" ); } } gui_urls.get_gui_urls(user, host, team, buf, sizeof(buf)); fputs(buf, fout); if (project_files.text) { fputs(project_files.text, fout); fprintf(fout, "\n"); } end: fprintf(fout, "\n" ); return 0; } // set delay to the MAX of the existing value or the requested value // never send a delay request longer than two days. // void SCHEDULER_REPLY::set_delay(double delay) { if (request_delay < delay) { request_delay = delay; } if (request_delay > DELAY_MAX) { request_delay = DELAY_MAX; } } void SCHEDULER_REPLY::insert_app_unique(APP& app) { unsigned int i; for (i=0; icore_client_version < 61200) { char buf[1024]; safe_strcpy(buf, m); strip_translation(buf); message = buf; } else { message = m; } priority = p; } int APP::write(FILE* fout) { fprintf(fout, "\n" " %s\n" " %s\n" " %d\n" " %d\n" "\n", name, user_friendly_name, non_cpu_intensive?1:0, fraction_done_exact?1:0 ); return 0; } int APP_VERSION::write(FILE* fout) { char buf[APP_VERSION_XML_BLOB_SIZE]; safe_strcpy(buf, xml_doc); char* p = strstr(buf, ""); if (!p) { fprintf(stderr, "ERROR: app version %lu XML has no end tag!\n", id); return -1; } *p = 0; fputs(buf, fout); PLATFORM* pp = ssp->lookup_platform_id(platformid); fprintf(fout, " %s\n", pp->name); if (strlen(plan_class)) { fprintf(fout, " %s\n", plan_class); } fprintf(fout, " %f\n" " %f\n", bavp->host_usage.avg_ncpus, bavp->host_usage.projected_flops ); if (strlen(bavp->host_usage.cmdline)) { fprintf(fout, " %s\n", bavp->host_usage.cmdline ); } int pt = bavp->host_usage.proc_type; if (pt != PROC_TYPE_CPU) { const char* nm; if (pt == PROC_TYPE_NVIDIA_GPU) { // KLUDGE: older clients use "CUDA", newer ones use "NVIDIA" // if (g_request->core_client_version < 70000) { nm = "CUDA"; } else { nm = proc_type_name_xml(pt); } } else { nm = proc_type_name_xml(pt); } fprintf(fout, " \n" " %s\n" " %f\n" " \n", nm, bavp->host_usage.gpu_usage ); } if (strlen(bavp->host_usage.custom_coproc_type)) { fprintf(fout, " \n" " %s\n" " %f\n" " \n", bavp->host_usage.custom_coproc_type, bavp->host_usage.gpu_usage ); } if (bavp->host_usage.gpu_ram) { fprintf(fout, " %f\n", bavp->host_usage.gpu_ram ); } fputs("\n", fout); return 0; } int SCHED_DB_RESULT::write_to_client(FILE* fout) { char buf[BLOB_SIZE]; safe_strcpy(buf, xml_doc_in); char* p = strstr(buf, ""); if (!p) { fprintf(stderr, "ERROR: result %lu XML has no end tag!\n", id); return -1; } *p = 0; fputs(buf, fout); fputs("\n", fout); // for old clients APP_VERSION* avp = bav.avp; CLIENT_APP_VERSION* cavp = bav.cavp; if (avp) { PLATFORM* pp = ssp->lookup_platform_id(avp->platformid); fprintf(fout, " %s\n" " %d\n" " %s\n", pp->name, avp->version_num, avp->plan_class ); } else if (cavp) { fprintf(fout, " %s\n" " %d\n" " %s\n", cavp->platform, cavp->version_num, cavp->plan_class ); } fputs("\n", fout); return 0; } int SCHED_DB_RESULT::parse_from_client(XML_PARSER& xp) { double dtemp; bool btemp; string stemp; int itemp; // should be non-zero if exit_status is not found exit_status = ERR_NO_EXIT_STATUS; memset(this, 0, sizeof(*this)); while (!xp.get_tag()) { if (xp.match_tag("/result")) { return 0; } if (xp.parse_str("name", name, sizeof(name))) continue; if (xp.parse_int("state", client_state)) continue; if (xp.parse_double("final_cpu_time", cpu_time)) { if (!boinc_is_finite(cpu_time)) { cpu_time = 0; } continue; } if (xp.parse_double("final_elapsed_time", elapsed_time)) { if (!boinc_is_finite(elapsed_time)) { elapsed_time = 0; } continue; } if (xp.parse_double("final_peak_working_set_size", peak_working_set_size)) { if (!boinc_is_finite(peak_working_set_size)) { peak_working_set_size = 0; } continue; } if (xp.parse_double("final_peak_swap_size", peak_swap_size)) { if (!boinc_is_finite(peak_swap_size)) { peak_swap_size = 0; } continue; } if (xp.parse_double("final_peak_disk_usage", peak_disk_usage)) { if (!boinc_is_finite(peak_disk_usage)) { peak_disk_usage = 0; } continue; } if (xp.parse_int("exit_status", exit_status)) continue; if (xp.parse_int("app_version_num", app_version_num)) continue; if (xp.match_tag("file_info")) { string s; xp.copy_element(s); safe_strcat(xml_doc_out, s.c_str()); continue; } if (xp.match_tag("stderr_out" )) { copy_element_contents(xp.f->f, "", stderr_out, sizeof(stderr_out)); continue; } if (xp.parse_string("platform", stemp)) continue; if (xp.parse_int("version_num", itemp)) continue; if (xp.parse_string("plan_class", stemp)) continue; if (xp.parse_double("completed_time", dtemp)) continue; if (xp.parse_string("file_name", stemp)) continue; if (xp.match_tag("file_ref")) { xp.copy_element(stemp); continue; } if (xp.parse_string("open_name", stemp)) continue; if (xp.parse_bool("ready_to_report", btemp)) continue; if (xp.parse_double("report_deadline", dtemp)) continue; if (xp.parse_string("wu_name", stemp)) continue; // deprecated stuff if (xp.parse_double("fpops_per_cpu_sec", dtemp)) continue; if (xp.parse_double("fpops_cumulative", dtemp)) continue; if (xp.parse_double("intops_per_cpu_sec", dtemp)) continue; if (xp.parse_double("intops_cumulative", dtemp)) continue; log_messages.printf(MSG_NORMAL, "RESULT::parse_from_client(): unrecognized: %s\n", xp.parsed_tag ); } return ERR_XML_PARSE; } int HOST::parse(XML_PARSER& xp) { p_ncpus = 1; double dtemp; string stemp; int x; while (!xp.get_tag()) { if (xp.match_tag("/host_info")) return 0; if (xp.parse_int("timezone", timezone)) continue; if (xp.parse_str("domain_name", domain_name, sizeof(domain_name))) continue; if (xp.parse_str("ip_addr", last_ip_addr, sizeof(last_ip_addr))) continue; if (xp.parse_str("host_cpid", host_cpid, sizeof(host_cpid))) continue; if (xp.parse_int("p_ncpus", p_ncpus)) continue; if (xp.parse_str("p_vendor", p_vendor, sizeof(p_vendor))) continue; if (xp.parse_str("p_model", p_model, sizeof(p_model))) continue; if (xp.parse_double("p_fpops", p_fpops)) continue; if (xp.parse_double("p_iops", p_iops)) continue; if (xp.parse_double("p_membw", p_membw)) continue; if (xp.parse_str("os_name", os_name, sizeof(os_name))) continue; if (xp.parse_str("os_version", os_version, sizeof(os_version))) continue; if (xp.parse_str("product_name", product_name, sizeof(product_name))) continue; if (xp.parse_double("m_nbytes", m_nbytes)) continue; if (xp.parse_double("m_cache", m_cache)) continue; if (xp.parse_double("m_swap", m_swap)) continue; if (xp.parse_double("d_total", d_total)) continue; if (xp.parse_double("d_free", d_free)) continue; if (xp.parse_double("n_bwup", n_bwup)) continue; if (xp.parse_double("n_bwdown", n_bwdown)) continue; if (xp.parse_str("p_features", p_features, sizeof(p_features))) continue; if (xp.parse_str("virtualbox_version", virtualbox_version, sizeof(virtualbox_version))) continue; if (xp.parse_bool("p_vm_extensions_disabled", p_vm_extensions_disabled)) continue; if (xp.match_tag("opencl_cpu_prop")) { int retval = opencl_cpu_prop[num_opencl_cpu_platforms].parse(xp); if (!retval) num_opencl_cpu_platforms++; continue; } if (xp.parse_bool("wsl_available", wsl_available)) continue; if (xp.match_tag("wsl")) { wsls.parse(xp); continue; } // unused fields // if (xp.parse_int("n_usable_coprocs", x)) continue; // parse deprecated fields to avoid error messages // if (xp.parse_double("p_calculated", dtemp)) continue; if (xp.match_tag("p_fpop_err")) continue; if (xp.match_tag("p_iop_err")) continue; if (xp.match_tag("p_membw_err")) continue; // fields reported by 5.5+ clients, not currently used // if (xp.parse_string("p_capabilities", stemp)) continue; if (xp.parse_string("accelerators", stemp)) continue; #if 1 // deprecated items // if (xp.parse_string("cpu_caps", stemp)) continue; if (xp.parse_string("cache_l1", stemp)) continue; if (xp.parse_string("cache_l2", stemp)) continue; if (xp.parse_string("cache_l3", stemp)) continue; #endif log_messages.printf(MSG_NORMAL, "HOST::parse(): unrecognized: %s\n", xp.parsed_tag ); } return ERR_XML_PARSE; } int HOST::parse_time_stats(XML_PARSER& xp) { while (!xp.get_tag()) { if (xp.match_tag("/time_stats")) return 0; if (xp.parse_double("on_frac", on_frac)) continue; if (xp.parse_double("connected_frac", connected_frac)) continue; if (xp.parse_double("active_frac", active_frac)) continue; if (xp.parse_double("gpu_active_frac", gpu_active_frac)) continue; if (xp.parse_double("cpu_and_network_available_frac", cpu_and_network_available_frac)) continue; if (xp.parse_double("client_start_time", client_start_time)) continue; if (xp.parse_double("previous_uptime", previous_uptime)) continue; #if 0 if (xp.match_tag("outages")) continue; if (xp.match_tag("outage")) continue; if (xp.match_tag("start")) continue; if (xp.match_tag("end")) continue; log_messages.printf(MSG_NORMAL, "HOST::parse_time_stats(): unrecognized: %s\n", xp.parsed_tag ); #endif } return ERR_XML_PARSE; } int HOST::parse_net_stats(XML_PARSER& xp) { double dtemp; while (!xp.get_tag()) { if (xp.match_tag("/net_stats")) return 0; if (xp.parse_double("bwup", n_bwup)) continue; if (xp.parse_double("bwdown", n_bwdown)) continue; // items reported by 5.10+ clients, not currently used // if (xp.parse_double("avg_time_up", dtemp)) continue; if (xp.parse_double("avg_up", dtemp)) continue; if (xp.parse_double("avg_time_down", dtemp)) continue; if (xp.parse_double("avg_down", dtemp)) continue; log_messages.printf(MSG_NORMAL, "HOST::parse_net_stats(): unrecognized: %s\n", xp.parsed_tag ); } return ERR_XML_PARSE; } int HOST::parse_disk_usage(XML_PARSER& xp) { while (!xp.get_tag()) { if (xp.match_tag("/disk_usage")) return 0; if (xp.parse_double("d_boinc_used_total", d_boinc_used_total)) continue; if (xp.parse_double("d_boinc_used_project", d_boinc_used_project)) continue; if (xp.parse_double("d_project_share", d_boinc_max)) continue; log_messages.printf(MSG_NORMAL, "HOST::parse_disk_usage(): unrecognized: %s\n", xp.parsed_tag ); } return ERR_XML_PARSE; } void GLOBAL_PREFS::parse(const char* buf, const char* venue) { char buf2[BLOB_SIZE]; double dtemp; defaults(); if (parse_double(buf, "", mod_time)) { // mod_time is outside of venue if (mod_time > dtime()) mod_time = dtime(); } extract_venue(buf, venue, buf2, sizeof(buf2)); parse_double(buf2, "", disk_max_used_gb); parse_double(buf2, "", disk_max_used_pct); parse_double(buf2, "", disk_min_free_gb); parse_double(buf2, "", work_buf_min_days); if (parse_double(buf2, "", dtemp)) { ram_max_used_busy_frac = dtemp/100.; } if (parse_double(buf2, "", dtemp)) { ram_max_used_idle_frac = dtemp/100.; } parse_double(buf2, "", max_ncpus_pct); } void GLOBAL_PREFS::defaults() { memset(this, 0, sizeof(GLOBAL_PREFS)); } void GUI_URLS::init() { text = 0; read_file_malloc(config.project_path("gui_urls.xml"), text); if (text) text = lf_terminate(text); } void GUI_URLS::get_gui_urls(USER& user, HOST& host, TEAM& team, char* buf, int len) { bool found; char userid[256], teamid[256], hostid[256], weak_auth[256], rss_auth[256]; strcpy(buf, ""); if (!text) return; strlcpy(buf, text, len); sprintf(userid, "%lu", user.id); sprintf(hostid, "%lu", host.id); if (user.teamid) { sprintf(teamid, "%lu", team.id); } else { strcpy(teamid, "0"); while (remove_element(buf, "", "")) { continue; } } get_weak_auth(user, weak_auth); get_rss_auth(user, rss_auth); while (1) { found = false; found |= str_replace(buf, "", user.authenticator); found |= str_replace(buf, "", hostid); found |= str_replace(buf, "", config.master_url); found |= str_replace(buf, "", config.long_name); found |= str_replace(buf, "", rss_auth); found |= str_replace(buf, "", teamid); found |= str_replace(buf, "", team.name); found |= str_replace(buf, "", userid); found |= str_replace(buf, "", user.name); found |= str_replace(buf, "", weak_auth); if (!found) break; } } void PROJECT_FILES::init() { text = 0; read_file_malloc(config.project_path("project_files.xml"), text); if (text) text = lf_terminate(text); } void get_weak_auth(USER& user, char* buf) { char buf2[1024], out[256]; sprintf(buf2, "%s%s", user.authenticator, user.passwd_hash); md5_block((unsigned char*)buf2, strlen(buf2), out); sprintf(buf, "%lu_%s", user.id, out); } void get_rss_auth(USER& user, char* buf) { char buf2[256], out[256]; sprintf(buf2, "%s%s%s", user.authenticator, user.passwd_hash, "notify_rss"); md5_block((unsigned char*)buf2, strlen(buf2), out); sprintf(buf, "%lu_%s", user.id, out); } void read_host_app_versions() { DB_HOST_APP_VERSION hav; char clause[256]; sprintf(clause, "where host_id=%lu", g_reply->host.id); while (!hav.enumerate(clause)) { g_wreq->host_app_versions.push_back(hav); } g_wreq->host_app_versions_orig = g_wreq->host_app_versions; } DB_HOST_APP_VERSION* gavid_to_havp(DB_ID_TYPE gavid) { for (unsigned int i=0; ihost_app_versions.size(); i++) { DB_HOST_APP_VERSION& hav = g_wreq->host_app_versions[i]; if (hav.app_version_id == gavid) return &hav; } return NULL; } void write_host_app_versions() { for (unsigned int i=0; ihost_app_versions.size(); i++) { DB_HOST_APP_VERSION& hav = g_wreq->host_app_versions[i]; DB_HOST_APP_VERSION& hav_orig = g_wreq->host_app_versions_orig[i]; int retval = hav.update_scheduler(hav_orig); if (retval) { log_messages.printf(MSG_CRITICAL, "CRITICAL: hav.update_sched() error: %s\n", boincerror(retval) ); } } } DB_HOST_APP_VERSION* BEST_APP_VERSION::host_app_version() { if (cavp) { return gavid_to_havp( generalized_app_version_id(host_usage.resource_type(), appid) ); } else { return gavid_to_havp(avp->id); } } // return some HAV for which quota was exceeded // DB_HOST_APP_VERSION* quota_exceeded_version() { for (unsigned int i=0; ihost_app_versions.size(); i++) { DB_HOST_APP_VERSION& hav = g_wreq->host_app_versions[i]; if (hav.daily_quota_exceeded) return &hav; } return NULL; } double capped_host_fpops() { double x = g_request->host.p_fpops; if (!ssp) return x; if (x <= 0) { return ssp->perf_info.host_fpops_50_percentile; } if (x > ssp->perf_info.host_fpops_95_percentile*1.1) { return ssp->perf_info.host_fpops_95_percentile*1.1; } return x; } bool HOST::get_opencl_cpu_prop(const char* platform, OPENCL_CPU_PROP& ocp) { for (int i=0; i