// This file is part of BOINC. // https://boinc.berkeley.edu // Copyright (C) 2022 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 . // This file contains: // 1) functions to clear and parse the various structs // 2) actual GUI RPCs // The core client expects all data to be formatted in the "C" locale, // so each GUI RPC should get the current locale, then switch to the // "C" locale before formatting messages or parsing results. // After all work is completed, revert back to the original locale. // // Template: // // int RPC_CLIENT::template_function( args ) { // int retval; // SET_LOCALE sl; // char buf[256]; // RPC rpc(this); // // // // return retval; // } // // NOTE: Failing to revert back to the original locale will cause // formatting failures for any software that has been localized or // displays localized data. #ifdef _WIN32 #include "boinc_win.h" #include "../version.h" #else #include "config.h" #include #include #include #include #include #include #include #include #include #endif #include "diagnostics.h" #include "parse.h" #include "str_util.h" #include "str_replace.h" #include "util.h" #include "error_numbers.h" #include "miofile.h" #include "md5_file.h" #include "network.h" #include "common_defs.h" #include "gui_rpc_client.h" using std::string; using std::vector; using std::sort; int OLD_RESULT::parse(XML_PARSER& xp) { memset(this, 0, sizeof(OLD_RESULT)); while (!xp.get_tag()) { if (xp.match_tag("/old_result")) return 0; if (xp.parse_str("project_url", project_url, sizeof(project_url))) continue; if (xp.parse_str("result_name", result_name, sizeof(result_name))) continue; if (xp.parse_str("app_name", app_name, sizeof(app_name))) continue; if (xp.parse_int("exit_status", exit_status)) continue; if (xp.parse_double("elapsed_time", elapsed_time)) continue; if (xp.parse_double("cpu_time", cpu_time)) continue; if (xp.parse_double("completed_time", completed_time)) continue; if (xp.parse_double("create_time", create_time)) continue; } return ERR_XML_PARSE; } int TIME_STATS::parse(XML_PARSER& xp) { memset(this, 0, sizeof(TIME_STATS)); while (!xp.get_tag()) { if (xp.match_tag("/time_stats")) return 0; if (xp.parse_double("now", now)) continue; if (xp.parse_double("on_frac", on_frac)) continue; if (xp.parse_double("connected_frac", connected_frac)) continue; if (xp.parse_double("cpu_and_network_available_frac", cpu_and_network_available_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("client_start_time", client_start_time)) continue; if (xp.parse_double("previous_uptime", previous_uptime)) continue; if (xp.parse_double("session_active_duration", session_active_duration)) continue; if (xp.parse_double("session_gpu_active_duration", session_gpu_active_duration)) continue; if (xp.parse_double("total_start_time", total_start_time)) continue; if (xp.parse_double("total_duration", total_duration)) continue; if (xp.parse_double("total_active_duration", total_active_duration)) continue; if (xp.parse_double("total_gpu_active_duration", total_gpu_active_duration)) continue; } return ERR_XML_PARSE; } int DAILY_XFER::parse(XML_PARSER& xp) { while (!xp.get_tag()) { if (xp.match_tag("/dx")) return 0; if (xp.parse_int("when", when)) continue; if (xp.parse_double("up", up)) continue; if (xp.parse_double("down", down)) continue; } return ERR_XML_PARSE; } int DAILY_XFER_HISTORY::parse(XML_PARSER& xp) { while (!xp.get_tag()) { if (!xp.is_tag) continue; if (xp.match_tag("dx")) { DAILY_XFER dx; int retval = dx.parse(xp); if (!retval) { daily_xfers.push_back(dx); } } } return 0; } int GUI_URL::parse(XML_PARSER& xp) { while (!xp.get_tag()) { if (!xp.is_tag) continue; if (xp.match_tag("/gui_url")) return 0; if (xp.match_tag("/gui_urls")) break; if (xp.parse_string("name", name)) continue; if (xp.parse_string("description", description)) continue; if (xp.parse_string("url", url)) continue; } return ERR_XML_PARSE; } PROJECT_LIST_ENTRY::PROJECT_LIST_ENTRY() { clear(); } int PROJECT_LIST_ENTRY::parse(XML_PARSER& xp) { string platform; while (!xp.get_tag()) { if (xp.match_tag("/project")) return 0; if (xp.parse_string("name", name)) continue; if (xp.parse_string("url", url)) { continue; } if (xp.parse_string("web_url", web_url)) { continue; } if (xp.parse_string("general_area", general_area)) continue; if (xp.parse_string("specific_area", specific_area)) continue; if (xp.parse_string("description", description)) { continue; } if (xp.parse_string("home", home)) continue; if (xp.parse_string("image", image)) continue; if (xp.match_tag("platforms")) { while (!xp.get_tag()) { if (xp.match_tag("/platforms")) break; if (xp.parse_string("name", platform)) { platforms.push_back(platform); } } } xp.skip_unexpected(false, ""); } return ERR_XML_PARSE; } void PROJECT_LIST_ENTRY::clear() { name.clear(); url.clear(); web_url.clear(); general_area.clear(); specific_area.clear(); description.clear(); platforms.clear(); home.clear(); image.clear(); } AM_LIST_ENTRY::AM_LIST_ENTRY() { clear(); } int AM_LIST_ENTRY::parse(XML_PARSER& xp) { while (!xp.get_tag()) { if (xp.match_tag("/account_manager")) return 0; if (xp.parse_string("name", name)) continue; if (xp.parse_string("url", url)) continue; if (xp.parse_string("description", description)) continue; if (xp.parse_string("image", image)) continue; } return 0; } void AM_LIST_ENTRY::clear() { name.clear(); url.clear(); description.clear(); image.clear(); } ALL_PROJECTS_LIST::ALL_PROJECTS_LIST() { } bool compare_project_list_entry( const PROJECT_LIST_ENTRY* a, const PROJECT_LIST_ENTRY* b ) { #ifdef _WIN32 return _stricmp(a->name.c_str(), b->name.c_str()) < 0; #else return strcasecmp(a->name.c_str(), b->name.c_str()) < 0; #endif } bool compare_am_list_entry(const AM_LIST_ENTRY* a, const AM_LIST_ENTRY* b) { #ifdef _WIN32 return _stricmp(a->name.c_str(), b->name.c_str()) < 0; #else return strcasecmp(a->name.c_str(), b->name.c_str()) < 0; #endif } void ALL_PROJECTS_LIST::alpha_sort() { sort(projects.begin(), projects.end(), compare_project_list_entry); sort(account_managers.begin(), account_managers.end(), compare_am_list_entry); } void ALL_PROJECTS_LIST::clear() { unsigned int i; for (i=0; i", buf); stderr_out = buf; continue; } #endif if (xp.parse_int("app_version_num", app_version_num)) continue; if (xp.parse_int("slot", slot)) continue; if (xp.parse_int("pid", pid)) continue; if (xp.parse_double("checkpoint_cpu_time", checkpoint_cpu_time)) continue; if (xp.parse_double("current_cpu_time", current_cpu_time)) continue; if (xp.parse_double("elapsed_time", elapsed_time)) continue; if (xp.parse_double("progress_rate", progress_rate)) continue; if (xp.parse_double("swap_size", swap_size)) continue; if (xp.parse_double("working_set_size_smoothed", working_set_size_smoothed)) continue; if (xp.parse_double("fraction_done", fraction_done)) continue; if (xp.parse_double("estimated_cpu_time_remaining", estimated_cpu_time_remaining)) continue; if (xp.parse_double("bytes_sent", bytes_sent)) continue; if (xp.parse_double("bytes_received", bytes_received)) continue; if (xp.parse_bool("too_large", too_large)) continue; if (xp.parse_bool("needs_shmem", needs_shmem)) continue; if (xp.parse_bool("edf_scheduled", edf_scheduled)) continue; if (xp.parse_str("graphics_exec_path", graphics_exec_path, sizeof(graphics_exec_path))) continue; if (xp.parse_str("web_graphics_url", web_graphics_url, sizeof(web_graphics_url))) continue; if (xp.parse_str("remote_desktop_addr", remote_desktop_addr, sizeof(remote_desktop_addr))) continue; if (xp.parse_str("slot_path", slot_path, sizeof(slot_path))) continue; if (xp.parse_str("resources", resources, sizeof(resources))) continue; } return ERR_XML_PARSE; } void RESULT::clear() { safe_strcpy(name, ""); safe_strcpy(wu_name, ""); version_num = 0; safe_strcpy(plan_class, ""); safe_strcpy(project_url, ""); safe_strcpy(platform, ""); safe_strcpy(graphics_exec_path, ""); safe_strcpy(web_graphics_url, ""); safe_strcpy(remote_desktop_addr, ""); safe_strcpy(slot_path, ""); safe_strcpy(resources, ""); report_deadline = 0; received_time = 0; ready_to_report = false; got_server_ack = false; final_cpu_time = 0; final_elapsed_time = 0; state = 0; scheduler_state = 0; exit_status = 0; signal = 0; //stderr_out.clear(); suspended_via_gui = false; project_suspended_via_gui = false; coproc_missing = false; scheduler_wait = false; safe_strcpy(scheduler_wait_reason, ""); network_wait = false; active_task = false; active_task_state = 0; app_version_num = 0; slot = -1; pid = 0; checkpoint_cpu_time = 0; current_cpu_time = 0; fraction_done = 0; elapsed_time = 0; progress_rate = 0; swap_size = 0; working_set_size_smoothed = 0; estimated_cpu_time_remaining = 0; bytes_sent = 0; bytes_received = 0; too_large = false; needs_shmem = false; edf_scheduled = false; app = NULL; wup = NULL; project = NULL; avp = NULL; } FILE_TRANSFER::FILE_TRANSFER() { clear(); } int FILE_TRANSFER::parse(XML_PARSER& xp) { while (!xp.get_tag()) { if (xp.match_tag("/file_transfer")) return 0; if (xp.parse_string("name", name)) continue; if (xp.parse_string("project_url", project_url)) continue; if (xp.parse_string("project_name", project_name)) continue; if (xp.parse_double("nbytes", nbytes)) continue; if (xp.parse_bool("sticky", sticky)) continue; if (xp.match_tag("persistent_file_xfer")) { pers_xfer_active = true; continue; } if (xp.match_tag("file_xfer")) { xfer_active = true; continue; } if (xp.parse_bool("is_upload", is_upload)) { generated_locally = is_upload; continue; } if (xp.parse_bool("generated_locally", generated_locally)) { is_upload = generated_locally; } if (xp.parse_int("num_retries", num_retries)) continue; if (xp.parse_double("first_request_time", first_request_time)) continue; if (xp.parse_double("next_request_time", next_request_time)) continue; if (xp.parse_int("status", status)) continue; if (xp.parse_double("time_so_far", time_so_far)) continue; if (xp.parse_double("estimated_xfer_time_remaining", estimated_xfer_time_remaining)) continue; if (xp.parse_double("last_bytes_xferred", bytes_xferred)) continue; if (xp.parse_double("file_offset", file_offset)) continue; if (xp.parse_double("xfer_speed", xfer_speed)) continue; if (xp.parse_string("hostname", hostname)) continue; if (xp.parse_double("project_backoff", project_backoff)) continue; } return ERR_XML_PARSE; } void FILE_TRANSFER::clear() { name.clear(); project_url.clear(); project_name.clear(); nbytes = 0; uploaded = false; is_upload = false; generated_locally = false; sticky = false; pers_xfer_active = false; xfer_active = false; num_retries = 0; first_request_time = 0; next_request_time = 0; status = 0; time_so_far = 0; estimated_xfer_time_remaining = 0; bytes_xferred = 0; file_offset = 0; xfer_speed = 0; hostname.clear(); project = NULL; project_backoff = 0; } MESSAGE::MESSAGE() { clear(); } int MESSAGE::parse(XML_PARSER& xp) { while (!xp.get_tag()) { if (xp.match_tag("/msg")) return 0; if (xp.parse_string("project", project)) continue; if (xp.parse_string("body", body)) continue; if (xp.parse_int("pri", priority)) continue; if (xp.parse_int("time", timestamp)) continue; if (xp.parse_int("seqno", seqno)) continue; } return ERR_XML_PARSE; } void MESSAGE::clear() { project.clear(); priority = 0; seqno = 0; timestamp = 0; body.clear(); } GR_PROXY_INFO::GR_PROXY_INFO() { clear(); } int GR_PROXY_INFO::parse(XML_PARSER& xp) { use_http_proxy = false; use_socks_proxy = false; use_http_authentication = false; while (!xp.get_tag()) { if (xp.match_tag("/proxy_info")) return 0; if (xp.parse_string("socks_server_name", socks_server_name)) continue; if (xp.parse_int("socks_server_port", socks_server_port)) continue; if (xp.parse_string("socks5_user_name", socks5_user_name)) continue; if (xp.parse_string("socks5_user_passwd", socks5_user_passwd)) continue; if (xp.parse_bool("socks5_remote_dns", socks5_remote_dns)) continue; if (xp.parse_string("http_server_name", http_server_name)) continue; if (xp.parse_int("http_server_port", http_server_port)) continue; if (xp.parse_string("http_user_name", http_user_name)) continue; if (xp.parse_string("http_user_passwd", http_user_passwd)) continue; if (xp.parse_bool("use_http_proxy", use_http_proxy)) continue; if (xp.parse_bool("use_socks_proxy", use_socks_proxy)) continue; if (xp.parse_bool("use_http_auth", use_http_authentication)) continue; if (xp.parse_string("no_proxy", noproxy_hosts)) continue; } return ERR_XML_PARSE; } void GR_PROXY_INFO::clear() { use_http_proxy = false; use_socks_proxy = false; use_http_authentication = false; socks_server_name.clear(); http_server_name.clear(); socks_server_port = 0; http_server_port = 0; http_user_name.clear(); http_user_passwd.clear(); socks5_user_name.clear(); socks5_user_passwd.clear(); socks5_remote_dns = false; noproxy_hosts.clear(); } CC_STATE::CC_STATE() { clear(); } int CC_STATE::parse(XML_PARSER& xp) { string platform; PROJECT* project = NULL; int retval; while (!xp.get_tag()) { if (xp.match_tag("unauthorized")) { return ERR_AUTHENTICATOR; } if (xp.match_tag("/client_state")) break; if (xp.parse_bool("executing_as_daemon", executing_as_daemon)) continue; if (xp.match_tag("project")) { project = new PROJECT(); retval = project->parse(xp); if (retval) { // should never happen delete project; project = NULL; continue; } projects.push_back(project); continue; } if (xp.match_tag("app")) { APP* app = new APP(); retval = app->parse(xp); if (retval || !project) { delete app; continue; } app->project = project; apps.push_back(app); continue; } if (xp.match_tag("app_version")) { APP_VERSION* app_version = new APP_VERSION(); retval = app_version->parse(xp); if (retval || !project) { delete app_version; continue; } app_version->project = project; app_version->app = lookup_app(project, app_version->app_name); if (!app_version->app) { delete app_version; continue; } app_versions.push_back(app_version); continue; } if (xp.match_tag("workunit")) { WORKUNIT* wu = new WORKUNIT(); retval = wu->parse(xp); if (retval || !project) { delete wu; continue; } wu->project = project; wu->app = lookup_app(project, wu->app_name); if (!wu->app) { delete wu; continue; } wus.push_back(wu); continue; } if (xp.match_tag("result")) { RESULT* result = new RESULT(); retval = result->parse(xp); if (retval || !project) { delete result; continue; } result->project = project; result->wup = lookup_wu(project, result->wu_name); if (!result->wup) { delete result; continue; } result->app = result->wup->app; APP_VERSION* avp; if (strlen(result->platform)) { avp = lookup_app_version( project, result->app, result->platform, result->version_num, result->plan_class ); } else if (result->version_num) { avp = lookup_app_version( project, result->app, result->version_num, result->plan_class ); } else { avp = lookup_app_version( project, result->app, result->wup->version_num ); } if (!avp) { delete result; continue; } result->avp = avp; results.push_back(result); continue; } if (xp.match_tag("global_preferences")) { bool flag = false; GLOBAL_PREFS_MASK mask; global_prefs.parse(xp, "", flag, mask); continue; } if (xp.parse_string("platform", platform)) { platforms.push_back(platform); continue; } if (xp.match_tag("host_info")) { host_info.parse(xp); continue; } if (xp.match_tag("time_stats")) { time_stats.parse(xp); continue; } if (xp.parse_bool("have_cuda", have_nvidia)) continue; if (xp.parse_bool("have_ati", have_ati)) continue; } return 0; } void CC_STATE::clear() { unsigned int i; for (i=0; imaster_url, url)) return projects[i]; } return 0; } APP* CC_STATE::lookup_app(PROJECT* project, const char* name) { unsigned int i; for (i=0; iproject != project) continue; if (!strcmp(apps[i]->name, name)) return apps[i]; } return 0; } APP_VERSION* CC_STATE::lookup_app_version( PROJECT* project, APP* app, char* platform, int version_num, char* plan_class ) { unsigned int i; for (i=0; iproject != project) continue; if (app_versions[i]->app != app) continue; if (strcmp(app_versions[i]->platform, platform)) continue; if (app_versions[i]->version_num != version_num) continue; if (strcmp(app_versions[i]->plan_class, plan_class)) continue; return app_versions[i]; } return 0; } APP_VERSION* CC_STATE::lookup_app_version( PROJECT* project, APP* app, int version_num, char* plan_class ) { unsigned int i; for (i=0; iproject != project) continue; if (app_versions[i]->app != app) continue; if (app_versions[i]->version_num != version_num) continue; if (strcmp(app_versions[i]->plan_class, plan_class)) continue; return app_versions[i]; } return 0; } APP_VERSION* CC_STATE::lookup_app_version( PROJECT* project, APP* app, int version_num ) { unsigned int i; for (i=0; iproject != project) continue; if (app_versions[i]->app != app) continue; if (app_versions[i]->version_num != version_num) continue; return app_versions[i]; } return 0; } WORKUNIT* CC_STATE::lookup_wu(PROJECT* project, const char* name) { unsigned int i; for (i=0; iproject != project) continue; if (!strcmp(wus[i]->name, name)) return wus[i]; } return 0; } RESULT* CC_STATE::lookup_result(PROJECT* project, const char* name) { unsigned int i; for (i=0; iproject != project) continue; if (!strcmp(results[i]->name, name)) return results[i]; } return 0; } RESULT* CC_STATE::lookup_result(const char* url, const char* name) { unsigned int i; for (i=0; iproject_url, url)) continue; if (!strcmp(results[i]->name, name)) return results[i]; } return 0; } void PROJECTS::clear() { unsigned int i; for (i=0; i", buf, sizeof(buf))) { // HTML TOU can have no open/close html tags // so I see no proper way to identify // whether it is html or plain text // and I have to use this dirty hack const string tou = buf; xml_unescape(buf); terms_of_use = buf; terms_of_use_is_html = terms_of_use != tou; } continue; } if (xp.parse_int("min_client_version", min_client_version)) continue; if (xp.parse_bool("web_stopped", web_stopped)) continue; if (xp.parse_bool("sched_stopped", sched_stopped)) continue; if (xp.parse_string("platform_name", msg)) { platforms.push_back(msg); continue; } if (xp.parse_bool("ldap_auth", ldap_auth)) continue; } return ERR_XML_PARSE; } void PROJECT_CONFIG::clear() { error_num = 0; name.clear(); master_url.clear(); web_rpc_url_base.clear(); error_msg.clear(); terms_of_use_is_html = false; terms_of_use.clear(); local_revision = 0; min_passwd_length = 6; account_manager = false; uses_username = false; account_creation_disabled = false; client_account_creation_disabled = false; platforms.clear(); sched_stopped = false; web_stopped = false; min_client_version = 0; ldap_auth = false; } ACCOUNT_IN::ACCOUNT_IN() { clear(); } void ACCOUNT_IN::clear() { url.clear(); email_addr.clear(); user_name.clear(); passwd.clear(); team_name.clear(); ldap_auth = false; consented_to_terms = false; } ACCOUNT_OUT::ACCOUNT_OUT() { clear(); } int ACCOUNT_OUT::parse(XML_PARSER& xp) { clear(); while (!xp.get_tag()) { if (xp.parse_int("error_num", error_num)) continue; if (xp.parse_string("error_msg", error_msg)) continue; if (xp.parse_string("authenticator", authenticator)) continue; } return 0; } void ACCOUNT_OUT::clear() { error_num = 0; error_msg = ""; authenticator.clear(); } CC_STATUS::CC_STATUS() { clear(); } int CC_STATUS::parse(XML_PARSER& xp) { while (!xp.get_tag()) { if (xp.match_tag("/cc_status")) return 0; if (xp.parse_int("network_status", network_status)) continue; if (xp.parse_bool("ams_password_error", ams_password_error)) continue; if (xp.parse_bool("manager_must_quit", manager_must_quit)) continue; if (xp.parse_int("task_suspend_reason", task_suspend_reason)) continue; if (xp.parse_int("task_mode", task_mode)) continue; if (xp.parse_int("task_mode_perm", task_mode_perm)) continue; if (xp.parse_double("task_mode_delay", task_mode_delay)) continue; if (xp.parse_int("gpu_suspend_reason", gpu_suspend_reason)) continue; if (xp.parse_int("gpu_mode", gpu_mode)) continue; if (xp.parse_int("gpu_mode_perm", gpu_mode_perm)) continue; if (xp.parse_double("gpu_mode_delay", gpu_mode_delay)) continue; if (xp.parse_int("network_suspend_reason", network_suspend_reason)) continue; if (xp.parse_int("network_mode", network_mode)) continue; if (xp.parse_int("network_mode_perm", network_mode_perm)) continue; if (xp.parse_double("network_mode_delay", network_mode_delay)) continue; if (xp.parse_bool("disallow_attach", disallow_attach)) continue; if (xp.parse_bool("simple_gui_only", simple_gui_only)) continue; if (xp.parse_int("max_event_log_lines", max_event_log_lines)) continue; } return ERR_XML_PARSE; } void CC_STATUS::clear() { network_status = -1; ams_password_error = false; manager_must_quit = false; task_suspend_reason = -1; task_mode = -1; task_mode_perm = -1; task_mode_delay = 0; network_suspend_reason = -1; network_mode = -1; network_mode_perm = -1; network_mode_delay = 0; gpu_suspend_reason = -1; gpu_mode = -1; gpu_mode_perm = -1; gpu_mode_delay = 0; disallow_attach = false; simple_gui_only = false; max_event_log_lines = 0; } /////////// END OF PARSING FUNCTIONS. RPCS START HERE //////////////// int RPC_CLIENT::exchange_versions(string client_name, VERSION_INFO& server) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); snprintf(buf, sizeof(buf), "\n" " %d\n" " %d\n" " %d\n" " %s\n" "\n", BOINC_MAJOR_VERSION, BOINC_MINOR_VERSION, BOINC_RELEASE, client_name.c_str() ); retval = rpc.do_rpc(buf); if (!retval) { while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) break; else if (parse_int(buf, "", server.major)) continue; else if (parse_int(buf, "", server.minor)) continue; else if (parse_int(buf, "", server.release)) continue; } } return retval; } int RPC_CLIENT::get_state(CC_STATE& state) { int retval; SET_LOCALE sl; RPC rpc(this); state.clear(); retval = rpc.do_rpc("\n"); if (retval) return retval; return state.parse(rpc.xp); } int RPC_CLIENT::get_results(RESULTS& t, bool active_only) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); t.clear(); snprintf(buf, sizeof(buf), "\n%d\n\n", active_only?1:0 ); retval = rpc.do_rpc(buf); if (!retval) { while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) break; else if (match_tag(buf, "")) { RESULT* rp = new RESULT(); rp->parse(rpc.xp); t.results.push_back(rp); continue; } } } return retval; } int RPC_CLIENT::get_old_results(vector& r) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); r.clear(); retval = rpc.do_rpc("\n"); if (retval) return retval; while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) break; if (match_tag(buf, "")) { OLD_RESULT ores; retval = ores.parse(rpc.xp); if (!retval) { r.push_back(ores); } continue; } } return 0; } int RPC_CLIENT::get_file_transfers(FILE_TRANSFERS& t) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); t.clear(); retval = rpc.do_rpc("\n"); if (!retval) { while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) break; else if (match_tag(buf, "")) { FILE_TRANSFER* fip = new FILE_TRANSFER(); fip->parse(rpc.xp); t.file_transfers.push_back(fip); continue; } } } return retval; } int RPC_CLIENT::get_simple_gui_info(SIMPLE_GUI_INFO& info) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); info.projects.clear(); info.results.clear(); retval = rpc.do_rpc("\n"); if (!retval) { while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) break; else if (match_tag(buf, "")) { PROJECT* project = new PROJECT(); project->parse(rpc.xp); info.projects.push_back(project); continue; } else if (match_tag(buf, "")) { RESULT* result = new RESULT(); result->parse(rpc.xp); info.results.push_back(result); continue; } } } return retval; } // creates new array of PROJECTs // int RPC_CLIENT::get_project_status(PROJECTS& p) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); p.clear(); retval = rpc.do_rpc("\n"); if (!retval) { while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) break; else if (match_tag(buf, "")) { PROJECT* project = new PROJECT(); project->parse(rpc.xp); p.projects.push_back(project); continue; } } } return retval; } int RPC_CLIENT::get_all_projects_list(ALL_PROJECTS_LIST& pl) { int retval = 0; SET_LOCALE sl; MIOFILE mf; PROJECT_LIST_ENTRY* project; AM_LIST_ENTRY* am; RPC rpc(this); pl.clear(); retval = rpc.do_rpc("\n"); if (retval) return retval; while (!rpc.xp.get_tag()) { if (rpc.xp.match_tag("/projects")) break; else if (rpc.xp.match_tag("project")) { project = new PROJECT_LIST_ENTRY(); retval = project->parse(rpc.xp); if (!retval) { pl.projects.push_back(project); } else { delete project; } continue; } else if (rpc.xp.match_tag("account_manager")) { am = new AM_LIST_ENTRY(); retval = am->parse(rpc.xp); if (!retval) { pl.account_managers.push_back(am); } else { delete am; } continue; } } pl.alpha_sort(); return 0; } int RPC_CLIENT::get_disk_usage(DISK_USAGE& du) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); du.clear(); retval = rpc.do_rpc("\n"); if (!retval) { while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) break; if (match_tag(buf, "")) { PROJECT* project = new PROJECT(); project->parse(rpc.xp); du.projects.push_back(project); continue; } if (parse_double(buf, "", du.d_total)) continue; if (parse_double(buf, "", du.d_free)) continue; if (parse_double(buf, "", du.d_boinc)) continue; if (parse_double(buf, "", du.d_allowed)) continue; } } return retval; } int DAILY_STATS::parse(XML_PARSER& xp) { while (!xp.get_tag()) { if (xp.match_tag("/daily_statistics")) return 0; if (xp.parse_double("day", day)) continue; if (xp.parse_double("user_total_credit", user_total_credit)) continue; if (xp.parse_double("user_expavg_credit", user_expavg_credit)) continue; if (xp.parse_double("host_total_credit", host_total_credit)) continue; if (xp.parse_double("host_expavg_credit", host_expavg_credit)) continue; } return ERR_XML_PARSE; } int RPC_CLIENT::get_statistics(PROJECTS& p) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); retval = rpc.do_rpc("\n"); if (!retval) { p.clear(); while (rpc.fin.fgets(buf, 256)) { if (retval) break; if (match_tag(buf, "")) break; if (match_tag(buf, "")) { PROJECT* project = new PROJECT(); p.projects.push_back(project); while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) break; if (parse_str(buf, "", p.projects.back()->master_url, sizeof(project->master_url))) continue; if (match_tag(buf, "")) { DAILY_STATS ds; retval = ds.parse(rpc.xp); if (retval) break; p.projects.back()->statistics.push_back(ds); continue; } } continue; } } } return retval; } int RPC_CLIENT::get_cc_status(CC_STATUS& status) { SET_LOCALE sl; char buf[256]; RPC rpc(this); int retval = rpc.do_rpc("\n"); if (!retval) { while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) { retval = status.parse(rpc.xp); if (retval) break; } } } return retval; } int RPC_CLIENT::network_available() { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc("\n"); return retval; } int RPC_CLIENT::project_op(PROJECT& project, const char* op) { int retval; SET_LOCALE sl; char buf[512]; const char *tag; RPC rpc(this); if (!strcmp(op, "reset")) { tag = "project_reset"; } else if (!strcmp(op, "detach")) { tag = "project_detach"; } else if (!strcmp(op, "update")) { tag = "project_update"; } else if (!strcmp(op, "suspend")) { tag = "project_suspend"; project.suspended_via_gui = true; } else if (!strcmp(op, "resume")) { tag = "project_resume"; project.suspended_via_gui = false; } else if (!strcmp(op, "allowmorework")) { tag = "project_allowmorework"; project.dont_request_more_work = false; } else if (!strcmp(op, "nomorework")) { tag = "project_nomorework"; project.dont_request_more_work = true; } else if (!strcmp(op, "detach_when_done")) { tag = "project_detach_when_done"; } else if (!strcmp(op, "dont_detach_when_done")) { tag = "project_dont_detach_when_done"; } else { return -1; } snprintf(buf, sizeof(buf), "<%s>\n" " %s\n" "\n", tag, project.master_url, tag ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::project_attach_from_file() { int retval; SET_LOCALE sl; char buf[768]; RPC rpc(this); snprintf(buf, sizeof(buf), "\n" " \n" "\n" ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::project_attach( const char* url, const char* auth, const char* name ) { int retval; SET_LOCALE sl; char buf[768]; RPC rpc(this); snprintf(buf, sizeof(buf), "\n" " %s\n" " %s\n" " %s\n" "\n", url, auth, name ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::project_attach_poll(PROJECT_ATTACH_REPLY& reply) { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc("\n"); if (!retval) { retval = reply.parse(rpc.xp); } return retval; } const char* RPC_CLIENT::mode_name(int mode) { const char* p = NULL; switch (mode) { case RUN_MODE_ALWAYS: p=""; break; case RUN_MODE_NEVER: p=""; break; case RUN_MODE_AUTO: p=""; break; case RUN_MODE_RESTORE: p=""; break; } return p; } int RPC_CLIENT::set_run_mode(int mode, double duration) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); snprintf(buf, sizeof(buf), "\n" "%s\n" " %f\n" "\n", mode_name(mode), duration ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::set_gpu_mode(int mode, double duration) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); snprintf(buf, sizeof(buf), "\n" "%s\n" " %f\n" "\n", mode_name(mode), duration ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::set_network_mode(int mode, double duration) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); snprintf(buf, sizeof(buf), "\n" "%s\n" " %f\n" "\n", mode_name(mode), duration ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::get_screensaver_tasks(int& suspend_reason, RESULTS& t) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); t.clear(); retval = rpc.do_rpc("\n"); if (!retval) { while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) break; if (parse_int(buf, "", suspend_reason)) continue; if (match_tag(buf, "")) { RESULT* rp = new RESULT(); rp->parse(rpc.xp); t.results.push_back(rp); continue; } } } return retval; } int RPC_CLIENT::run_benchmarks() { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc("\n"); if (retval) return retval; return rpc.parse_reply(); } // start or stop a graphics app on behalf of the screensaver. // (needed for Mac OS X 10.15+) // // \n"); if (!strcmp(operation, "run")) { snprintf(buf, sizeof(buf), "\n%d\n\n%s\n", operand, screensaverLoginUser ); } else if (!strcmp(operation, "runfullscreen")) { snprintf(buf, sizeof(buf), "\n%d\n\n%s\n", operand, screensaverLoginUser ); } else if (!strcmp(operation, "stop")) { snprintf(buf, sizeof(buf), "\n%d\n\n%s\n", operand, screensaverLoginUser ); } else if (!strcmp(operation, "test")) { snprintf(buf, sizeof(buf), "\n%d\n\n", operand ); test = true; } else { operand = -1; return -1; } safe_strcat(buf, "\n"); int retval = rpc.do_rpc(buf); if (retval) { operand = -1; } else if (test) { while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) break; if (parse_int(buf, "", thePID)) { operand = thePID; continue; } } } return retval; } int RPC_CLIENT::set_proxy_settings(GR_PROXY_INFO& procinfo) { int retval; SET_LOCALE sl; char buf[1792]; RPC rpc(this); snprintf(buf, sizeof(buf), "\n" " \n" "%s%s%s" " %s\n" " %d\n" " %s\n" " %s\n" " %s\n" " %d\n" " %s\n" " %s\n" " %d\n" " %s\n" " \n" "\n", procinfo.use_http_proxy?" \n":"", procinfo.use_socks_proxy?" \n":"", procinfo.use_http_authentication?" \n":"", procinfo.http_server_name.c_str(), procinfo.http_server_port, procinfo.http_user_name.c_str(), procinfo.http_user_passwd.c_str(), procinfo.socks_server_name.c_str(), procinfo.socks_server_port, procinfo.socks5_user_name.c_str(), procinfo.socks5_user_passwd.c_str(), procinfo.socks5_remote_dns?1:0, procinfo.noproxy_hosts.c_str() ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::get_proxy_settings(GR_PROXY_INFO& p) { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc(""); if (!retval) { retval = p.parse(rpc.xp); } return retval; } int RPC_CLIENT::get_message_count(int& seqno) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); retval = rpc.do_rpc(""); if (retval) return retval; while (rpc.fin.fgets(buf, 256)) { if (parse_int(buf, "", seqno)) { return 0; } } return ERR_XML_PARSE; } int RPC_CLIENT::get_messages(int seqno, MESSAGES& msgs, bool translatable) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); snprintf(buf, sizeof(buf), "\n" " %d\n" "%s" "\n", seqno, translatable?" \n":"" ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (!retval) { while (!rpc.xp.get_tag()) { if (rpc.xp.match_tag("/msgs")) { return 0; } if (rpc.xp.match_tag("msg")) { MESSAGE* message = new MESSAGE(); message->parse(rpc.xp); msgs.messages.push_back(message); continue; } if (rpc.xp.match_tag("boinc_gui_rpc_reply")) continue; if (rpc.xp.match_tag("msgs")) continue; //fprintf(stderr, "bad tag %s\n", buf); } } return retval; } int RPC_CLIENT::file_transfer_op(FILE_TRANSFER& ft, const char* op) { int retval; SET_LOCALE sl; char buf[768]; const char *tag; RPC rpc(this); if (!strcmp(op, "retry")) { tag = "retry_file_transfer"; } else if (!strcmp(op, "abort")) { tag = "abort_file_transfer"; } else { return -1; } snprintf(buf, sizeof(buf), "<%s>\n" " %s\n" " %s\n" "\n", tag, ft.project_url.c_str(), ft.name.c_str(), tag ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::result_op(RESULT& result, const char* op) { int retval; SET_LOCALE sl; char buf[768]; const char *tag; RPC rpc(this); if (!strcmp(op, "abort")) { tag = "abort_result"; } else if (!strcmp(op, "suspend")) { tag = "suspend_result"; result.suspended_via_gui = true; } else if (!strcmp(op, "resume")) { tag = "resume_result"; result.suspended_via_gui = false; } else { return -1; } snprintf(buf, sizeof(buf), "<%s>\n" " %s\n" " %s\n" "\n", tag, result.project_url, result.name, tag ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::get_host_info(HOST_INFO& h) { int retval; char buf[256]; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc(""); if (retval) return retval; while (rpc.fin.fgets(buf, 256)) { if (match_tag(buf, "")) { return h.parse(rpc.xp); } if (match_tag(buf, "\n" " \n" " %s\n" " \n" "\n", h.product_name ); buf[sizeof(buf)-1] = 0; int retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::reset_host_info() { SET_LOCALE sl; RPC rpc(this); char buf[1024]; snprintf(buf, sizeof(buf), "\n\n"); int retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::quit() { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc("\n"); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::acct_mgr_rpc( const char* url, const char* name, const char* password, bool use_config_file ) { int retval; SET_LOCALE sl; char buf[1024]; RPC rpc(this); if (use_config_file) { snprintf(buf, sizeof(buf), "\n" " \n" "\n" ); buf[sizeof(buf)-1] = 0; } else { snprintf(buf, sizeof(buf), "\n" " %s\n" " %s\n" " %s\n" "\n", url, name, password ); buf[sizeof(buf)-1] = 0; } retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::acct_mgr_rpc_poll(ACCT_MGR_RPC_REPLY& r) { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc("\n"); if (!retval) { retval = r.parse(rpc.xp); } return retval; } int RPC_CLIENT::acct_mgr_info(ACCT_MGR_INFO& ami) { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc("\n"); if (!retval) { retval = ami.parse(rpc.xp); } return retval; } int RPC_CLIENT::get_project_init_status(PROJECT_INIT_STATUS& pis) { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc("\n"); if (!retval) { retval = pis.parse(rpc.xp); } return retval; } int RPC_CLIENT::get_project_config(std::string url) { int retval; char buf[512]; SET_LOCALE sl; RPC rpc(this); snprintf(buf, sizeof(buf), "\n" " %s\n" "\n", url.c_str() ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::get_project_config_poll(PROJECT_CONFIG& pc) { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc("\n"); if (!retval) { retval = pc.parse(rpc.xp); } return retval; } static string get_passwd_hash(string passwd, string email_addr) { return md5_string(passwd+email_addr); } int RPC_CLIENT::lookup_account(ACCOUNT_IN& ai) { int retval; SET_LOCALE sl; char buf[1024]; RPC rpc(this); string passwd_hash; if (ai.ldap_auth && !strchr(ai.email_addr.c_str(), '@')) { // LDAP case // passwd_hash = ai.passwd; } else { downcase_string(ai.email_addr); passwd_hash = get_passwd_hash(ai.passwd, ai.email_addr); } snprintf(buf, sizeof(buf), "\n" " %s\n" " %s\n" " %s\n" " %d\n" "\n", ai.url.c_str(), ai.email_addr.c_str(), passwd_hash.c_str(), ai.ldap_auth?1:0 ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::lookup_account_poll(ACCOUNT_OUT& ao) { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc("\n"); if (!retval) { retval = ao.parse(rpc.xp); } return retval; } int RPC_CLIENT::create_account(ACCOUNT_IN& ai) { int retval; SET_LOCALE sl; char buf[1280]; RPC rpc(this); downcase_string(ai.email_addr); string passwd_hash = get_passwd_hash(ai.passwd, ai.email_addr); snprintf(buf, sizeof(buf), "\n" " %s\n" " %s\n" " %s\n" " %s\n" " %s\n" " %s" "\n", ai.url.c_str(), ai.email_addr.c_str(), passwd_hash.c_str(), ai.user_name.c_str(), ai.team_name.c_str(), ai.consented_to_terms ? "\n" : "" ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::create_account_poll(ACCOUNT_OUT& ao) { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc("\n"); if (!retval) { retval = ao.parse(rpc.xp); } return retval; } int RPC_CLIENT::get_newer_version(std::string& version, std::string& version_download_url) { int retval; SET_LOCALE sl; char buf[256]; RPC rpc(this); version = ""; version_download_url = ""; retval = rpc.do_rpc("\n"); if (!retval) { while (rpc.fin.fgets(buf, 256)) { if (!version.empty() && !version_download_url.empty()) { break; } if (parse_str(buf, "", version)) { continue; } if (parse_str(buf, "", version_download_url)) { continue; } } } return retval; } int RPC_CLIENT::read_global_prefs_override() { SET_LOCALE sl; RPC rpc(this); int retval = rpc.do_rpc(""); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::get_global_prefs_file(string& s) { int retval; SET_LOCALE sl; RPC rpc(this); char buf[1024]; bool found = false; bool in_prefs = false; s = ""; retval = rpc.do_rpc(""); if (retval) return retval; while (rpc.fin.fgets(buf, 256)) { if (in_prefs) { s += buf; if (match_tag(buf, "")) { in_prefs = false; } } else { if (match_tag(buf, "")) { s += buf; in_prefs = true; found = true; } } } if (!found) return ERR_NOT_FOUND; return 0; } int RPC_CLIENT::get_global_prefs_working(string& s) { int retval; SET_LOCALE sl; RPC rpc(this); char buf[1024]; bool found = false; bool in_prefs = false; s = ""; retval = rpc.do_rpc(""); if (retval) return retval; while (rpc.fin.fgets(buf, 256)) { if (in_prefs) { s += buf; if (match_tag(buf, "")) { in_prefs = false; } } else { if (match_tag(buf, "")) { s += buf; in_prefs = true; found = true; } } } if (!found) return ERR_NOT_FOUND; return 0; } int RPC_CLIENT::get_global_prefs_working_struct(GLOBAL_PREFS& prefs, GLOBAL_PREFS_MASK& mask) { int retval; SET_LOCALE sl; string s; MIOFILE mf; bool found_venue; retval = get_global_prefs_working(s); if (retval) return retval; mf.init_buf_read(s.c_str()); XML_PARSER xp(&mf); prefs.parse(xp, "", found_venue, mask); if (!mask.are_prefs_set()) { return ERR_NOT_FOUND; } return 0; } int RPC_CLIENT::get_global_prefs_override(string& s) { int retval; SET_LOCALE sl; RPC rpc(this); char buf[1024]; bool found = false; bool in_prefs = false; s = ""; retval = rpc.do_rpc(""); if (retval) return retval; while (rpc.fin.fgets(buf, 256)) { if (in_prefs) { s += buf; if (match_tag(buf, "")) { in_prefs = false; } } else { if (match_tag(buf, "")) { s += buf; in_prefs = true; found = true; } } } if (!found) return ERR_NOT_FOUND; return 0; } int RPC_CLIENT::set_global_prefs_override(string& s) { int retval; RPC rpc(this); char buf[64000]; snprintf(buf, sizeof(buf), "\n" "%s\n" "\n", s.c_str() ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::get_global_prefs_override_struct(GLOBAL_PREFS& prefs, GLOBAL_PREFS_MASK& mask) { int retval; SET_LOCALE sl; string s; MIOFILE mf; bool found_venue; retval = get_global_prefs_override(s); if (retval) return retval; mf.init_buf_read(s.c_str()); XML_PARSER xp(&mf); prefs.parse(xp, "", found_venue, mask); if (!mask.are_prefs_set()) { return ERR_NOT_FOUND; } return 0; } int RPC_CLIENT::set_global_prefs_override_struct(GLOBAL_PREFS& prefs, GLOBAL_PREFS_MASK& mask) { SET_LOCALE sl; char buf[64000]; MIOFILE mf; string s; mf.init_buf_write(buf, sizeof(buf)); prefs.write_subset(mf, mask); s = buf; return set_global_prefs_override(s); } int RPC_CLIENT::read_cc_config() { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc(""); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::get_cc_config(CC_CONFIG& config, LOG_FLAGS& log_flags) { int retval; SET_LOCALE sl; RPC rpc(this); retval = rpc.do_rpc(""); if (retval) return retval; return config.parse(rpc.xp, log_flags); } int RPC_CLIENT::set_cc_config(CC_CONFIG& config, LOG_FLAGS& log_flags) { SET_LOCALE sl; char buf[64000]; MIOFILE mf; int retval; RPC rpc(this); mf.init_buf_write(buf, sizeof(buf)); config.write(mf, log_flags); string x = string("\n")+buf+string("\n"); retval = rpc.do_rpc(x.c_str()); if (retval) return retval; return rpc.parse_reply(); } int RPC_CLIENT::get_app_config(const char* url, APP_CONFIGS& config) { int retval; static LOG_FLAGS log_flags; SET_LOCALE sl; RPC rpc(this); MSG_VEC mv; char buf[1024]; snprintf(buf, sizeof (buf), "\n" " %s\n" "\n", url ); retval = rpc.do_rpc(buf); if (retval) return retval; while (!rpc.xp.get_tag()) { if (rpc.xp.match_tag("app_config")) { return config.parse(rpc.xp, mv, log_flags); } else if (rpc.xp.match_tag("error")) { rpc.xp.element_contents("", buf, sizeof(buf)); printf("get_app_config error: %s\n", buf); return -1; } } return ERR_XML_PARSE; } int RPC_CLIENT::set_app_config(const char* url, APP_CONFIGS& config) { SET_LOCALE sl; char buf[64000]; MIOFILE mf; int retval; RPC rpc(this); mf.init_buf_write(buf, sizeof(buf)); mf.printf("%s\n", url); config.write(mf); string x = string("\n")+buf+string("\n"); retval = rpc.do_rpc(x.c_str()); if (retval) return retval; return rpc.parse_reply(); } static int parse_notices(XML_PARSER& xp, NOTICES& notices) { int retval; while (!xp.get_tag()) { if (!xp.is_tag) continue; if (xp.match_tag("notice")) { NOTICE* np = new NOTICE(); retval = np->parse(xp); if (!retval) { if (np->seqno == -1) { notices.notices.clear(); notices.complete = true; delete np; } else { notices.notices.insert(notices.notices.begin(), np); } } else { delete np; } } } return 0; } int RPC_CLIENT::get_notices(int seqno, NOTICES& notices) { SET_LOCALE sl; char buf[1024]; RPC rpc(this); int retval; snprintf(buf, sizeof(buf), "\n" " %d\n" "\n", seqno ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; notices.received = true; return parse_notices(rpc.xp, notices); } int RPC_CLIENT::get_notices_public(int seqno, NOTICES& notices) { SET_LOCALE sl; char buf[1024]; RPC rpc(this); int retval; snprintf(buf, sizeof(buf), "\n" " %d\n" "\n", seqno ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; notices.received = true; return parse_notices(rpc.xp, notices); } int RPC_CLIENT::get_daily_xfer_history(DAILY_XFER_HISTORY& dxh) { SET_LOCALE sl; RPC rpc(this); int retval; retval = rpc.do_rpc("\n"); if (retval) return retval; return dxh.parse(rpc.xp); } int RPC_CLIENT::set_language(const char* language) { SET_LOCALE sl; RPC rpc(this); int retval; char buf[256]; snprintf(buf, sizeof(buf), "\n" " %s\n" "\n", language ); buf[sizeof(buf)-1] = 0; retval = rpc.do_rpc(buf); if (retval) return retval; return rpc.parse_reply(); }