boinc/tools/updater.cpp

203 lines
5.9 KiB
C++
Raw Normal View History

// This file is part of BOINC.
// http://boinc.berkeley.edu
// Copyright (C) 2008 University of California
//
// BOINC is free software; you can redistribute it and/or modify it
// under the terms of the GNU Lesser General Public License
// as published by the Free Software Foundation,
// either version 3 of the License, or (at your option) any later version.
//
// BOINC is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
// See the GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with BOINC. If not, see <http://www.gnu.org/licenses/>.
// Program to install files as part of auto-update.
// Run in a directory that contains the new files.
// Arguments:
//
// --install_dir X copy files to X (required)
// --run_manager when done, run Manager
// --run_as_service when done, run core client as service
//
// What it does:
// 1) wait for a mutex, to ensure that the core client has exited
// 2) create or empty out a "previous-version" dir
// 3) copy files from install dir to previous-version dir
// 4) copy files from current dir to install dir
// 5) run the new core client and/or manager
//
// If we get an error in 2) or 3):
// 6) run the (old) core client and/or manager
// We pass it a "--run_from_updater" option; this causes it to
// mark the update as failed, so it won't try again.
// If we get an error in 4) or 5):
// 7) copy files from previous-version dir back to install dir
// 8) run the old core client and/or manager
#include <stdio.h>
#ifdef _WIN32
#else
#include <unistd.h>
- Added checks for net/*.h, arpa/*.h, netinet/*.h and code to figure out which of those files to include - Modified MAC address check to work on some non-Linux unixes. (mac_address.cpp) - Added suggested change to "already attached to project" checking. (ProjectInfoPage.cpp) - changed includes of standard c header files to their c++ equivalents (i.e. replaced <stdio.h> with <cstdio>) for namespace protection. - replaced "using namespace std;" with more explicit "using std::function" in several files. - Fixed bug in checking whether the os is OS/2 and added conditional OS_OS2 to the build environment. (boinc_platform.m4,configure.ac) - Changed build environment to not use -nostandardlibs unless we are using G++ and static linkage is specified. (configure.ac) - Added makefiles and package building files for solaris CSW package manager. - Fixed bug with attempting to find login name using logname. (configure.ac) - Added ifdef HAVE_* protection around some include files commonly found in sys. - Added support for unified binary for x86_64/i686-pc-solaris. (cs_platforms.cpp) - generate_host_cpid() now uses MAC address on non-linux unix. (hostinfo_network.cpp) - Macro BOINC_SET_COMPILE_FLAGS now doesn't check gcc only flags on non-gcc compilers. (boinc_set_compile_flags.m4) - Library compiles no longer depend upon the library extension or require the library to be prefixed with lib. - More fixes for fcgi builds. - Added declaration of "struct ether_addr" and ether_ntoa(). Have not yet implemented ether_ntoa() for machines that don't have it, or where it is buggy. (unix_util.h) - Added FCGI::perror() which calls FCGI_perror(). (boinc_fcgi.{h,cpp}) - Fixed library Makefiles so that all required headers get installed. svn path=/trunk/boinc/; revision=17388
2009-02-26 00:23:23 +00:00
#include <cerrno>
#endif
#include <vector>
#include "filesys.h"
#include "util.h"
#include "sandbox.h"
using std::vector;
#ifdef _WIN32
#define CORE_NAME "boinc.exe"
#define MANAGER_NAME "boincmgr.exe"
#else
#define CORE_NAME "boinc_client"
#define MANAGER_NAME "boinc_mgr"
#endif
char* install_dir;
char prev_dir[1024];
int prepare_prev_dir() {
int retval;
if (is_dir(prev_dir)) {
retval = clean_out_dir(prev_dir);
if (retval) return retval;
} else {
retval = boinc_mkdir(prev_dir);
if (retval) return retval;
}
return 0;
}
int move_file(const char* file, char* old_dir, char* new_dir) {
char old_path[MAXPATHLEN], new_path[MAXPATHLEN];
sprintf(old_path, "%s/%s", old_dir, file);
sprintf(new_path, "%s/%s", new_dir, file);
int retval = boinc_rename(old_path, new_path);
fprintf(stderr, "rename %s to %s\n", old_path, new_path);
if (retval) {
fprintf(stderr, "couldn't rename %s to %s\n", old_path, new_path);
}
return retval;
}
// try to move all; return 0 if moved all
//
int move_files(vector<const char*> files, char* old_dir, char* new_dir) {
int retval = 0;
for (unsigned int i=0; i<files.size(); i++) {
int ret = move_file(files[i], old_dir, new_dir);
if (ret) retval = ret;
}
return retval;
}
// NOTE: this program must always (re)start the core client before exiting
int main(int argc, char** argv) {
int i, retval, argc2;
char filepath[MAXPATHLEN];
#ifdef _WIN32
HANDLE core_pid, mgr_pid;
#else
int core_pid, mgr_pid;
#endif
bool run_as_service = false;
bool run_manager = false;
char* argv2[10];
char cur_dir[1024];
bool new_version_installed = false;
vector<const char*> files;
install_dir = 0;
for (i=1; i<argc; i++) {
printf("updater: argv[%d] is %s\n", i, argv[i]);
if (!strcmp(argv[i], "--run_as_service")) {
run_as_service = true;
} else if (!strcmp(argv[i], "--run_manager")) {
run_manager = true;
} else if (!strcmp(argv[i], "--install_dir")) {
install_dir = argv[++i];
}
}
if (!install_dir) {
fprintf(stderr, "updater: install dir not specified\n");
retval = 1;
goto restart;
}
sprintf(prev_dir, "%s/prev_version", install_dir);
boinc_getcwd(cur_dir);
printf("%s\n", cur_dir);
wait_client_mutex(install_dir, 30);
retval = prepare_prev_dir();
if (retval) {
fprintf(stderr, "couldn't prepare prev_dir %s: %d\n", prev_dir, retval);
goto restart;
}
files.push_back(CORE_NAME);
files.push_back(MANAGER_NAME);
// save existing files to "previous version" dir
//
retval = move_files(files, install_dir, prev_dir);
if (retval) {
move_files(files, prev_dir, install_dir);
goto restart;
}
retval = move_files(files, cur_dir, install_dir);
if (retval) {
move_files(files, prev_dir, install_dir);
goto restart;
}
new_version_installed = true;
retval = 0;
restart:
argv2[0] = CORE_NAME;
argv2[1] = "--run_by_updater";
argv2[2] = 0;
argc2 = 2;
sprintf(filepath, "%s/%s", install_dir, CORE_NAME);
retval = run_program(install_dir, filepath, argc2, argv2, 5, core_pid);
if (retval) {
fprintf(stderr, "failed to run core client (%d); backing out\n", retval);
if (new_version_installed) {
retval = move_files(files, prev_dir, install_dir);
if (retval) exit(retval); // abandon ship
new_version_installed = false;
goto restart;
}
exit(retval);
}
if (run_manager) {
argv2[0] = MANAGER_NAME;
argv2[1] = 0;
argc2 = 1;
sprintf(filepath, "%s/%s", install_dir, MANAGER_NAME);
retval = run_program(install_dir, filepath, argc2, argv2, 5, mgr_pid);
if (retval) {
fprintf(stderr, "failed to run manager (%d); backing out\n", retval);
if (new_version_installed) {
kill_program(core_pid);
retval = move_files(files, prev_dir, install_dir);
if (retval) exit(retval);
new_version_installed = false;
goto restart;
}
exit(retval);
}
}
}