## $Id$ import boinc_path_config import configxml import os, md5, shutil def md5_file(path): """Return a 16-digit MD5 hex digest of a file's contents""" return md5.new(open(path).read()).hexdigest() def file_size(path): """Return the size of a file""" f = open(path) f.seek(0,2) return f.tell() def sign_executable(executable_path, quiet=False): '''Returns signed text for executable''' config = configxml.default_config() if not quiet: print 'Signing', executable_path code_sign_key = os.path.join(config.config.key_dir, 'code_sign_private') sign_executable_path = os.path.join(boinc_path_config.TOP_BUILD_DIR, 'tools','sign_executable') if not os.path.exists(sign_executable_path): raise SystemExit("tools/sign_executable not found! did you `make' it?") signature_text = os.popen('%s %s %s'%(sign_executable_path, executable_path,code_sign_key)).read() if not signature_text: raise SystemExit("Couldn't sign executable %s"%executable_path) return signature_text def process_executable_file(file, signature_text=None, quiet=False): '''Handle a new executable file to be added to the database. 1. Copy file to download_dir if necessary. 2. Return XML. - if signature_text specified, include it; else generate md5sum. ''' config = configxml.default_config() file_dir, file_base = os.path.split(file) target_path = os.path.join(config.config.download_dir, file_base) if file_dir != config.config.download_dir: if not quiet: print "Copying %s to %s"%(file_base, config.config.download_dir) shutil.copy(file, target_path) xml = ''' %s %s ''' %(file_base, os.path.join(config.config.download_url, file_base)) if signature_text: xml += ' \n%s \n'%signature_text else: xml += ' %s\n' % md5_file(target_path) xml += ' %f\n\n' % file_size(target_path) return xml def process_app_version(app, version_num, exec_files, signature_files={}, quiet=False): """Return xml for application version app is an instance of database.App version_num is an integer such as 102 for version 1.02 exec_files is a list of full-path executables signature_files is a dictionary of exec_file -> signature file mappings. process_app_version() will generate a new signature for any exec_files that don't have one given already. """ assert(exec_files) xml_doc = '' for exec_file in exec_files: signature_file = signature_files.get(exec_file) if signature_file: signature_text = open(signature_file).read() else: signature_text = sign_executable(exec_file, quiet=quiet) xml_doc += process_executable_file(exec_file, signature_text, quiet=quiet) xml_doc += ('\n'+ ' %s\n'+ ' %d\n') %( app.name, version_num) first = True for exec_file in exec_files: xml_doc += (' \n'+ ' %s\n') %( os.path.basename(exec_file)) if first: xml_doc += ' \n' xml_doc += ' \n' first = False xml_doc += '\n' return xml_doc def query_yesno(str): '''Query user; default Yes''' print str, "[Y/n] ", return not raw_input().strip().lower().startswith('n') def query_noyes(str): '''Query user; default No''' print str, "[y/N] ", return raw_input().strip().lower().startswith('y')