// 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):
//
#ifdef _WIN32
#include "boinc_win.h"
#endif
#include "parse.h"
#include "error_numbers.h"
#include "filesys.h"
#include "file_names.h"
#include "client_msgs.h"
#include "client_state.h"
void CLIENT_STATE::set_client_state_dirty(char* source) {
log_messages.printf(CLIENT_MSG_LOG::DEBUG_STATE, "set dirty: %s\n", source);
client_state_dirty = true;
}
// Parse the client_state.xml file
//
int CLIENT_STATE::parse_state_file() {
char buf[256];
PROJECT *project=NULL;
int retval=0;
int failnum;
char *fname;
SCOPE_MSG_LOG scope_messages(log_messages, CLIENT_MSG_LOG::DEBUG_STATE);
// Look for a valid state file:
// First the regular one, then the "next" one.
//
if (boinc_file_exists(STATE_FILE_NAME)) {
fname = STATE_FILE_NAME;
} else if (boinc_file_exists(STATE_FILE_NEXT)) {
fname = STATE_FILE_NEXT;
} else {
scope_messages.printf("CLIENT_STATE::parse_state_file(): No state file; will create one\n");
// avoid warning messages about version
//
old_major_version = BOINC_MAJOR_VERSION;
old_minor_version = BOINC_MINOR_VERSION;
return ERR_FOPEN;
}
FILE* f = fopen(fname, "r");
MIOFILE mf;
mf.init_file(f);
fgets(buf, 256, f);
if (!match_tag(buf, "")) {
// if file is invalid (probably empty), try the previous statefile
//
fclose(f);
f = fopen(STATE_FILE_PREV, "r");
mf.init_file(f);
fgets(buf, 256, f);
if (!match_tag(buf, "")) {
msg_printf(NULL, MSG_ERROR, "Missing open tag in state file.\n");
retval = ERR_XML_PARSE;
goto done;
}
}
while (fgets(buf, 256, f)) {
if (match_tag(buf, "")) {
retval = 0;
break;
} else if (match_tag(buf, "")) {
PROJECT temp_project;
retval = temp_project.parse_state(mf);
if (retval) {
msg_printf(NULL, MSG_ERROR, "Can't parse project in state file");
} else {
project = lookup_project(temp_project.master_url);
if (project) {
project->copy_state_fields(temp_project);
} else {
msg_printf(&temp_project, MSG_ERROR,
"Project %s found in state file but not prefs",
temp_project.get_project_name()
);
}
}
} else if (match_tag(buf, "")) {
APP* app = new APP;
retval = app->parse(mf);
if (project && project->anonymous_platform) continue;
if (retval) {
msg_printf(NULL, MSG_ERROR, "Can't parse app in state file");
delete app;
} else {
if (project) {
retval = link_app(project, app);
if (retval) {
msg_printf(project, MSG_ERROR,
"Can't link app %s in state file", app->name
);
delete app;
} else {
apps.push_back(app);
}
} else {
msg_printf(NULL, MSG_ERROR,
"App %s outside project in state file", app->name
);
delete app;
}
}
} else if (match_tag(buf, "")) {
FILE_INFO* fip = new FILE_INFO;
retval = fip->parse(mf, false);
if (retval) {
msg_printf(NULL, MSG_ERROR, "Can't parse file info in state file");
delete fip;
} else {
if (project) {
retval = link_file_info(project, fip, false);
if (project->anonymous_platform && retval == ERR_NOT_UNIQUE) {
continue;
}
if (retval) {
msg_printf(project, MSG_ERROR,
"Can't link file info %s in state file", fip->name
);
delete fip;
} else {
file_infos.push_back(fip);
// If the file had a failure before,
// don't start another file transfer
if (fip->had_failure(failnum)) {
if (fip->pers_file_xfer) {
delete fip->pers_file_xfer;
fip->pers_file_xfer = NULL;
}
}
if (fip->pers_file_xfer) {
retval = fip->pers_file_xfer->init(fip, fip->upload_when_present);
if (retval) {
msg_printf(project, MSG_ERROR,
"Can't initialize pers file xfer for %s", fip->name
);
}
retval = pers_file_xfers->insert(fip->pers_file_xfer);
if (retval) {
msg_printf(project, MSG_ERROR,
"Can't insert pers file xfer for %s", fip->name
);
}
}
}
} else {
msg_printf(NULL, MSG_ERROR, "File info outside project in state file");
delete fip;
}
}
} else if (match_tag(buf, "")) {
APP_VERSION* avp = new APP_VERSION;
retval = avp->parse(mf);
if (project && project->anonymous_platform) continue;
if (retval) {
msg_printf(NULL, MSG_ERROR, "Can't parse app version in state file");
delete avp;
} else {
if (project) {
retval = link_app_version(project, avp);
if (retval) {
msg_printf(project, MSG_ERROR, "Can't link app version in state file");
delete avp;
} else {
app_versions.push_back(avp);
}
} else {
msg_printf(NULL, MSG_ERROR, "App version outside project in state file");
delete avp;
}
}
} else if (match_tag(buf, "")) {
WORKUNIT* wup = new WORKUNIT;
retval = wup->parse(mf);
if (retval) {
msg_printf(NULL, MSG_ERROR, "Can't parse workunit in state file");
delete wup;
} else {
if (project) {
retval = link_workunit(project, wup);
if (retval) {
msg_printf(project, MSG_ERROR, "Can't link workunit in state file");
delete wup;
} else {
workunits.push_back(wup);
}
} else {
msg_printf(NULL, MSG_ERROR, "Workunit outside project in state file");
delete wup;
}
}
} else if (match_tag(buf, "")) {
RESULT* rp = new RESULT;
retval = rp->parse_state(mf);
if (retval) {
msg_printf(NULL, MSG_ERROR, "Can't parse result in state file");
delete rp;
} else {
if (project) {
retval = link_result(project, rp);
if (retval) {
msg_printf(project, MSG_ERROR,
"Can't link result %s in state file",
rp->name
);
delete rp;
} else {
results.push_back(rp);
}
} else {
msg_printf(NULL, MSG_ERROR,
"Result %s outside project in state file",
rp->name
);
delete rp;
}
}
} else if (match_tag(buf, "")) {
retval = host_info.parse(mf);
if (retval) {
msg_printf(NULL, MSG_ERROR, "Can't parse host info in state file\n");
}
} else if (match_tag(buf, "")) {
retval = time_stats.parse(mf);
if (retval) {
msg_printf(NULL, MSG_ERROR, "Can't parse time stats in state file\n");
}
} else if (match_tag(buf, "")) {
retval = net_stats.parse(mf);
if (retval) {
msg_printf(NULL, MSG_ERROR, "Can't parse net stats in state file\n");
}
} else if (match_tag(buf, "")) {
retval = active_tasks.parse(mf);
if (retval) {
msg_printf(NULL, MSG_ERROR, "Can't parse active tasks in state file\n");
}
} else if (match_tag(buf, "")) {
// should match our current platform name
} else if (match_tag(buf, "")) {
// could put logic here to detect incompatible state files
// after core client update
} else if (parse_int(buf, "", old_major_version)) {
} else if (parse_int(buf, "", old_minor_version)) {
} else if (parse_int(buf, "", cpu_sched_period)) {
} else if (parse_double(buf, "", cpu_sched_work_done_this_period)) {
} else if (match_tag(buf, "")) {
retval = pi.parse(mf);
if (retval) {
msg_printf(NULL, MSG_ERROR, "Can't parse proxy info in state file\n");
}
// } else if (parse_int(buf, "")) {
} else if (parse_str(buf, "", host_venue, sizeof(host_venue))) {
} else scope_messages.printf("CLIENT_STATE::parse_state_file: unrecognized: %s\n", buf);
}
done:
fclose(f);
return retval;
}
// read just the venue from the state file.
//
int CLIENT_STATE::parse_venue() {
char buf[256];
if (!boinc_file_exists(STATE_FILE_NAME)) return 0;
FILE* f = fopen(STATE_FILE_NAME, "r");
while (fgets(buf, 256, f)) {
if (match_tag(buf, "")) {
break;
}
if (parse_str(buf, "", host_venue, sizeof(host_venue))) {
break;
}
}
fclose(f);
return 0;
}
// Write the client_state.xml file
//
int CLIENT_STATE::write_state_file() {
FILE* f = boinc_fopen(STATE_FILE_NEXT, "w");
SCOPE_MSG_LOG scope_messages(log_messages, CLIENT_MSG_LOG::DEBUG_STATE);
scope_messages.printf("CLIENT_STATE::write_state_file(): Writing state file\n");
if (!f) {
msg_printf(0, MSG_ERROR, "Can't open temp state file: %s\n", STATE_FILE_NEXT);
return ERR_FOPEN;
}
MIOFILE mf;
mf.init_file(f);
int retval = write_state(mf);
fclose(f);
if (retval) return retval;
// the following fails if no current file, so don't check
//
retval = boinc_rename(STATE_FILE_NAME, STATE_FILE_PREV);
retval = boinc_rename(STATE_FILE_NEXT, STATE_FILE_NAME);
scope_messages.printf("CLIENT_STATE::write_state_file(): Done writing state file\n");
if (retval) return ERR_RENAME;
return 0;
}
int CLIENT_STATE::write_state(MIOFILE& f) {
unsigned int i, j;
int retval;
f.printf("\n");
retval = host_info.write(f);
if (retval) return retval;
retval = time_stats.write(f, false);
if (retval) return retval;
retval = net_stats.write(f, false);
if (retval) return retval;
for (j=0; jwrite_state(f);
if (retval) return retval;
for (i=0; iproject == p) {
retval = apps[i]->write(f);
if (retval) return retval;
}
}
for (i=0; iproject == p) {
retval = file_infos[i]->write(f, false);
if (retval) return retval;
}
}
for (i=0; iproject == p) app_versions[i]->write(f);
}
for (i=0; iproject == p) workunits[i]->write(f);
}
for (i=0; iproject == p) results[i]->write(f, false);
}
}
active_tasks.write(f);
f.printf(
"%s\n"
"%d\n"
"%d\n",
platform_name,
core_client_major_version,
core_client_minor_version
);
// save CPU scheduling state
//
f.printf(
"%d\n"
"%f\n",
cpu_sched_period,
cpu_sched_work_done_this_period
);
// save proxy info
//
pi.write(f);
if (strlen(host_venue)) {
f.printf("%s\n", host_venue);
}
f.printf("\n");
return 0;
}
// Write the client_state.xml file if necessary
// TODO: write no more often than X seconds
//
int CLIENT_STATE::write_state_file_if_needed() {
int retval;
if (client_state_dirty) {
client_state_dirty = false;
retval = write_state_file();
if (retval) return retval;
}
return 0;
}
// look for app_versions.xml file in project dir.
// If find, get app versions from there,
// and use "anonymous platform" mechanism for this project
//
void CLIENT_STATE::check_anonymous() {
unsigned int i;
char dir[256], path[256];
FILE* f;
int retval;
for (i=0; ianonymous_platform = true;
// flag as anonymous even if can't parse file
retval = parse_app_info(p, f);
fclose(f);
}
}
int CLIENT_STATE::parse_app_info(PROJECT* p, FILE* in) {
char buf[256];
MIOFILE mf;
mf.init_file(in);
while (fgets(buf, 256, in)) {
if (match_tag(buf, "")) continue;
if (match_tag(buf, "")) return 0;
if (match_tag(buf, "")) {
FILE_INFO* fip = new FILE_INFO;
if (fip->parse(mf, false)) {
delete fip;
continue;
}
if (link_file_info(p, fip, true)) {
delete fip;
continue;
}
fip->status = FILE_PRESENT;
file_infos.push_back(fip);
continue;
}
if (match_tag(buf, "")) {
APP* app = new APP;
if (app->parse(mf)) {
delete app;
continue;
}
if (lookup_app(p, app->name)) {
delete app;
continue;
}
link_app(p, app);
apps.push_back(app);
continue;
}
if (match_tag(buf, "")) {
APP_VERSION* avp = new APP_VERSION;
if (avp->parse(mf)) {
delete avp;
continue;
}
if (gstate.link_app_version(p, avp)) {
delete avp;
continue;
}
link_app_version(p, avp);
app_versions.push_back(avp);
continue;
}
}
return ERR_XML_PARSE;
}
int CLIENT_STATE::write_state_gui(MIOFILE& f) {
unsigned int i, j;
int retval;
f.printf("\n");
retval = host_info.write(f);
if (retval) return retval;
retval = time_stats.write(f, false);
if (retval) return retval;
retval = net_stats.write(f, false);
if (retval) return retval;
for (j=0; jwrite_state(f, true);
if (retval) return retval;
for (i=0; iproject == p) {
retval = apps[i]->write(f);
if (retval) return retval;
}
}
for (i=0; iproject == p) app_versions[i]->write(f);
}
for (i=0; iproject == p) workunits[i]->write(f);
}
for (i=0; iproject == p) results[i]->write_gui(f);
}
}
f.printf(
"%s\n"
"%d\n"
"%d\n",
platform_name,
core_client_major_version,
core_client_minor_version
);
// save CPU scheduling state
//
f.printf(
"%d\n"
"%f\n",
cpu_sched_period,
cpu_sched_work_done_this_period
);
// save proxy info
//
pi.write(f);
if (strlen(host_venue)) {
f.printf("%s\n", host_venue);
}
f.printf("\n");
return 0;
}
int CLIENT_STATE::write_tasks_gui(MIOFILE& f) {
unsigned int i;
f.printf("\n");
for(i=0; iwrite_gui(f);
}
f.printf("\n");
return 0;
}
int CLIENT_STATE::write_file_transfers_gui(MIOFILE& f) {
unsigned int i;
f.printf("\n");
for (i=0; ipers_file_xfer) {
fip->write_gui(f);
}
}
f.printf("\n");
return 0;
}