// 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): // // add.C - add items to the DB // // usages: // add project -project_short_name x -project_long_name x // add project // add app -app_name x // add application // create DB record // add platform -platform_name x -user_friendly_name y // create DB record // add app_version // -app_name x -platform_name y -version a // -download_dir d -download_url e // -exec_dir b // [ -exec_files file1 file2 ... ] // [ -signed_exec_files file1 sign1 file2 sign2 ... ] // create DB record // copy exec to data directory // add core_version // -platform_name x -version n // [ -message 'foo' ] [ -message_priority 'foo'] // -download_dir d -download_url e // -exec_dir b // -exec_files file1 // add user -email_addr x -name y -authenticator a // [ -global_prefs_file y ] #include #include #include #include "boinc_db.h" #include "sched_config.h" #include "backend_lib.h" #include "md5_file.h" #include "crypt.h" //APP app; //PLATFORM platform; //APP_VERSION app_version; //USER user; //PROJECT project; int version, retval, nexec_files; double nbytes; bool signed_exec_files; char buf[256], md5_cksum[64]; char *db_name=0, *db_passwd=0, *app_name=0, *platform_name=0; char *project_short_name=0, *project_long_name=0; char* user_friendly_name=0; char* exec_dir=0, *exec_files[10], *signature_files[10]; char *email_addr=0, *user_name=0, *authenticator=0; char *global_prefs_file=0, *download_dir, *download_url; char* code_sign_keyfile=0; char *message=0, *message_priority=0; void add_project() { int retval; DB_PROJECT project; if (!project_short_name || !project_long_name) { fprintf( stderr, "Project name (long or short) not specified.\n" ); exit(1); } project.clear(); strcpy(project.short_name, project_short_name); strcpy(project.long_name, project_long_name); retval = project.insert(); if (retval) { boinc_db_print_error("project.insert()"); } } void add_app() { int retval; DB_APP app; if (!app_name) { fprintf( stderr, "Application name not specified.\n" ); exit(1); } app.clear(); strcpy(app.name, app_name); app.create_time = time(0); app.min_version = version; retval = app.insert(); if (retval) { boinc_db_print_error("app.insert()"); } } void add_platform() { int retval; DB_PLATFORM platform; if (!user_friendly_name) { fprintf( stderr, "User friendly name not specified.\n" ); exit(1); } if (!platform_name) { fprintf( stderr, "Platform name not specified.\n" ); exit(1); } platform.clear(); strcpy(platform.name, platform_name); strcpy(platform.user_friendly_name, user_friendly_name); platform.create_time = time(0); retval = platform.insert(); if (retval) { boinc_db_print_error("platform.insert()"); } } int sign_executable(char* path, char* signature_text) { DATA_BLOCK signature; unsigned char signature_buf[SIGNATURE_SIZE_BINARY]; R_RSA_PRIVATE_KEY code_sign_key; retval = read_key_file(code_sign_keyfile, code_sign_key); if (retval) { fprintf(stderr, "add: can't read key\n"); exit(1); } signature.data = signature_buf; sign_file(path, code_sign_key, signature); sprint_hex_data(signature_text, signature); return 0; } // copy executable file to the download dir, generate XML // static int process_executable_file( char* filename, char* signature_text, char* xml_doc ) { char longbuf[MAX_BLOB_SIZE]; char path[256]; sprintf(path, "%s/%s", exec_dir, filename); sprintf( buf, "cp %s %s/%s", path, download_dir, filename ); retval = system(buf); if (retval) { printf("failed: %s\n", buf); return retval; } retval = md5_file(path, md5_cksum, nbytes); if (retval) return retval; // generate the XML doc directly. // TODO: use a template, as in create_work (??) // sprintf(longbuf, "\n" " %s\n" " %s/%s\n" " \n", filename, download_url, filename ); strcat(xml_doc, longbuf); if (signature_text) { sprintf(longbuf, " \n%s" " \n", signature_text ); } else { sprintf(longbuf, " %s\n", md5_cksum ); } strcat(xml_doc, longbuf); sprintf(longbuf, " %f\n" "\n", nbytes ); strcat(xml_doc, longbuf); return 0; } void add_core_version() { DB_CORE_VERSION core_version; DB_PLATFORM platform; core_version.clear(); sprintf(buf, "where name='%s'", platform_name); retval = platform.lookup(buf); if (retval) { fprintf(stderr, "add_core_version(): can't find platform %s\n", platform_name); boinc_db_print_error("platform.lookup()"); return; } core_version.platformid = platform.id; core_version.version_num = version; if (message) strcpy(core_version.message, message); if (message_priority) strcpy(core_version.message, message_priority); if (nexec_files != 1) { fprintf(stderr, "add_core_version(): multiple files not allowed\n"); return; } strcpy(core_version.xml_doc, ""); process_executable_file(exec_files[0], NULL, core_version.xml_doc); core_version.create_time = time(0); retval = core_version.insert(); if (retval) { boinc_db_print_error("core_version.insert()"); } } void add_app_version() { char path[256]; char longbuf[MAX_BLOB_SIZE]; char signature_text[1024]; int i; DB_APP app; DB_APP_VERSION app_version; DB_PLATFORM platform; app_version.clear(); if (!app_name) { fprintf( stderr, "Application name not specified.\n" ); exit(1); } sprintf(buf, "where name='%s'", app_name); retval = app.lookup(buf); if (retval) { fprintf(stderr, "add_app_version(): can't find app %s\n", app_name); boinc_db_print_error("app.lookup()"); return; } app_version.appid = app.id; sprintf(buf, "where name='%s'", platform_name); retval = platform.lookup(buf); if (retval) { fprintf(stderr, "add_app_version(): can't find platform %s\n", platform_name); boinc_db_print_error("platform.lookup()"); return; } app_version.platformid = platform.id; app_version.version_num = version; strcpy(app_version.xml_doc, ""); // copy executables to download directory and sign them // for (i=0; i\n" " %s\n" " %d\n", app_name, version ); strcat(app_version.xml_doc, longbuf); for (i=0; i\n" " %s\n" "%s" " \n", exec_files[i], i?"":" \n" ); strcat(app_version.xml_doc, longbuf); } strcat(app_version.xml_doc, "\n"); app_version.create_time = time(0); retval = app_version.insert(); if (retval) { boinc_db_print_error("app_version.insert()"); return; } } void add_user() { DB_USER user; user.clear(); user.create_time = time(0); strcpy(user.email_addr, email_addr); strcpy(user.name, user_name); strcpy(user.authenticator, authenticator); strcpy(user.country, "United States"); strcpy(user.postal_code, "94703"); if (global_prefs_file) { retval = read_filename(global_prefs_file, user.global_prefs); if (retval) { printf("read_file: %s", global_prefs_file); return; } } retval = user.insert(); if (retval) { boinc_db_print_error("user.insert()"); return; } } int main(int argc, char** argv) { int i, retval; SCHED_CONFIG config; retval = config.parse_file(); if (retval) { printf("no config file found\n"); } else { strcpy(db_name, config.db_name); strcpy(db_passwd, config.db_passwd); strcpy(download_url, config.download_url); strcpy(download_dir, config.download_dir); sprintf(code_sign_keyfile, "%s/code_sign_private", config.key_dir); } for (i=2; i