#!/usr/bin/env python
# $Id$
# Creates a new BOINC project.
import boinc_path_config
from Boinc.setup_project import *
from Boinc import database, db_mid, configxml, tools, projectxml
import sys, os, getopt, re, socket
def getfqdn():
try:
return socket.getfqdn()
except:
return 'localhost'
def isurl(s):
return s.startswith('http://') or s.startswith('https://')
argv0 = sys.argv[0]
HOME = os.path.expanduser('~')
USER = os.environ['USER']
NODENAME = getfqdn()
HELP = """
syntax: %(argv0)s [options] project ['Project Long Name']
Creates a new project with given name with everything running on a single
server.
Misc options:
--no_query accept all directories without querying
--user_name default: $USER (%(USER)s)
--delete_prev_inst delete project-root first (from prev installation)
--drop_db_first drop database first (from prev installation)
--test_app install example application
--web_only install web files, no executables (for Bossa, Bolt)
--no_db don't create the database
--srcdir where to find the source files (default: current directory)
Dir-options:
--project_root default: HOME/projects/PROJECT
--key_dir default: PROJECT_ROOT/keys
--project_host default: $HOSTNAME (%(NODENAME)s)
--url_base default: http://PROJECT_HOST/
--html_user_url default: URL_BASE/PROJECT/
--html_ops_url default: URL_BASE/PROJECT_ops/
Other:
--db_name default: PROJECT
--db_user default: USER_NAME
--db_passwd default: None
--db_host default: None
Example command line:
./make_project --project_root $HOME/boinc/projects --url_base http://tah.org/ tah 'Test@home'
Then upload_dir = $HOME/boinc/projects/tah/upload
and cgi_url = http://tah.org/tah_cgi/
By default, directory options will be queried if they do not exist yet.
""" %locals()
def syntax_error(str):
raise SystemExit('%s; See "%s --help" for help\n' % (str,sys.argv[0]))
def usage():
print HELP
raise SystemExit
try:
opts, args = getopt.getopt(sys.argv[1:],
'hv', [
'base=',
'db_host=',
'db_name=',
'db_passwd=',
'db_user=',
'delete_prev_inst',
'drop_db_first',
'help',
'html_ops_url=',
'html_user_url=',
'key_dir=',
'no_db',
'no_query',
'project_host=',
'project_root=',
'srcdir=',
'test_app',
'url_base=',
'user_name=',
'verbose=',
'web_only',
]
)
except getopt.GetoptError, e:
syntax_error(e)
options.url_base = None
options.no_query = False
options.test_app = False
options.web_only = False
options.no_db = False
options.delete_prev_inst = False
options.srcdir = None
for o,a in opts:
if o == '-h' or o == '--help': usage()
elif o == '--db_host': options.db_host = a
elif o == '--db_name': options.db_name = a
elif o == '--db_passwd': options.db_passwd = a
elif o == '--db_user': options.db_user = a
elif o == '--delete_prev_inst': options.delete_prev_inst = True
elif o == '--drop_db_first': options.drop_db_first = True
elif o == '--html_ops_url': options.html_ops_url = a
elif o == '--html_user_url': options.html_user_url = a
elif o == '--key_dir': options.key_dir = a
elif o == '--no_db': options.no_db = True
elif o == '--no_query': options.no_query = True
elif o == '--project_host': options.project_host = a
elif o == '--project_root': options.project_root = a
elif o == '--srcdir': options.srcdir = a
elif o == '--test_app': options.test_app = True
elif o == '--url_base': options.url_base = a
elif o == '--user_name': options.user_name = a
elif o == '--verbose': options.echo_verbose = int(a)
elif o == '--web_only': options.web_only = True
elif o == '-v': options.echo_verbose = 2
else:
raise SystemExit('internal error o=%s'%o)
if len(args) == 2:
(project_shortname, project_longname) = args
elif len(args) == 1:
(project_shortname, project_longname) = args[0], args[0]
else:
syntax_error('Need one or two arguments')
if not options.srcdir:
for dir in ('.', '..'):
if os.path.exists(os.path.join(dir, 'html', 'project.sample', 'project.inc')):
options.srcdir = dir
if not options.srcdir:
syntax_error('Not running in the source directory and --srcdir was not specified')
opt_repls = {'PROJECT':project_shortname,
'PROJECT_ops':project_shortname+'_ops',
'PROJECT_cgi':project_shortname+'_cgi'}
def delete_slash(str):
return os.path.join(str,'')[:-1]
def add_slash(str, action=True):
if action:
return os.path.join(str,'')
else:
return str
def replopt(str):
for key in opt_repls:
str = str.replace(key, delete_slash(opt_repls[key]))
return str
def defopt(name, v, isdir=True):
options.__dict__[name] = opt_repls[name.upper()] = add_slash(replopt(options.__dict__.get(name) or v), isdir)
defopt('project_host' , NODENAME, isdir=False)
defopt('url_base' , 'http://%s/'%options.project_host)
# strip project_host to get short hostname
options.project_host=options.project_host.split('.')[0]
if not isurl(options.url_base):
syntax_error('url_base needs to be an URL')
defopt('html_user_url' , 'URL_BASE/PROJECT')
defopt('html_ops_url' , 'URL_BASE/PROJECT_ops')
defopt('user_name' , USER, isdir=False)
defopt('project_root' , HOME+'/projects/PROJECT')
defopt('key_dir' , options.project_root+'keys')
defopt('db_name' , 'PROJECT', isdir=False)
defopt('db_user' , 'USER_NAME', isdir=False)
defopt('db_passwd' , '', isdir=False)
defopt('db_host' , '', isdir=False)
print "Creating project '%s' (short name '%s'):" %(project_longname, project_shortname)
for k in [
'project_root',
'project_host',
'url_base',
'html_user_url',
'html_ops_url',
'key_dir',
'db_name',
'db_host',
]:
print k.upper().rjust(15), "=", options.__dict__[k]
project_root_parent = os.path.realpath(os.path.join(options.project_root,'..'))
if not os.path.exists(project_root_parent):
os.makedirs(project_root_parent)
if os.path.exists(options.project_root):
if options.delete_prev_inst:
if not options.no_query:
if not query_noyes('Delete %s?'%options.project_root):
raise SystemExit('Aborted')
print "Deleting", options.project_root
rmtree(options.project_root)
else:
raise SystemExit('Project root already exists! Specify --delete_prev_inst --drop_db_first to clobber')
if not options.no_query:
if not query_yesno("Continue?"):
raise SystemExit('Aborted')
options.install_method = 'copy'
init()
project = Project(
project_shortname,
project_longname,
options.url_base + project_shortname + '_cgi',
project_dir = options.project_root,
master_url = options.html_user_url,
key_dir = options.key_dir,
db_name = options.db_name,
host = options.project_host,
web_only = options.web_only,
no_db = options.no_db,
production = True
)
project.install_project()
proot = delete_slash(options.project_root)
# set up default daemons
project.sched_install('feeder')
project.sched_install('transitioner')
project.sched_install('file_deleter')
if options.test_app:
project.config.daemons.make_node_and_append("daemon").cmd = 'sample_work_generator -d 3'
project.config.daemons.make_node_and_append("daemon").cmd = 'sample_bitwise_validator -d 3 --app example_app'
project.config.daemons.make_node_and_append("daemon").cmd = 'sample_assimilator -d 3 --app example_app'
# set up default tasks
t = project.config.tasks.make_node_and_append("task")
t.period = '24 hours'
t.output = 'antique_file_deleter.out'
t.cmd = 'antique_file_deleter -d 2'
t.disabled = 0
t = project.config.tasks.make_node_and_append("task")
t.period = '24 hours'
t.output = 'db_dump.out'
t.cmd = 'db_dump -d 2 --dump_spec ../db_dump_spec.xml'
t.disabled = 1
t = project.config.tasks.make_node_and_append("task")
t.period = '1 days'
t.output = 'update_uotd.out'
t.cmd = 'run_in_ops ./update_uotd.php'
t.disabled = 0
t = project.config.tasks.make_node_and_append("task")
t.period = '1 hour'
t.output = 'update_forum_activities.out'
t.cmd = 'run_in_ops ./update_forum_activities.php'
t.disabled = 0
t = project.config.tasks.make_node_and_append("task")
t.period = '1 days'
t.output = 'update_stats.out'
t.cmd = 'update_stats'
t.disabled = 0
t = project.config.tasks.make_node_and_append("task")
t.period = '24 hours'
t.output = 'update_profile_pages.out'
t.cmd = 'run_in_ops ./update_profile_pages.php'
t.disabled = 0
t = project.config.tasks.make_node_and_append("task")
t.period = '24 hours'
t.output = 'team_import.out'
t.cmd = 'run_in_ops ./team_import.php'
t.disabled = 1
t = project.config.tasks.make_node_and_append("task")
t.period = '24 hours'
t.output = 'notify.out'
t.cmd = 'run_in_ops ./notify.php'
t.disabled = 0
t = project.config.tasks.make_node_and_append("task")
t.period = '24 hours'
t.output = 'badge_assign.out'
t.cmd = 'run_in_ops ./badge_assign.php'
t.disabled = 0
project.config.write()
if not options.no_db:
try:
os.system('cd '+proot+'/html/ops; ./db_schemaversion.php > '+proot+'/db_revision')
except:
print '''Couldn't set db schema version number'''
try:
os.system('cd '+proot+'/html/ops; ./update_translations.php -d 1')
except:
print '''Couldn't install translation files'''
print '''Done installing default daemons.'''
# copy the test app if needed
if options.test_app:
app_dir = proot+'/apps/example_app/'
os.mkdir(app_dir)
os.system('cp -r ../samples/example_app/bin/* ' + app_dir);
shutil.copy('example_app_in', proot+'/templates/')
shutil.copy('example_app_out', proot+'/templates/')
shutil.copy('../tools/create_work_example', proot+'/bin/')
#add app to project.xml
pf = projectxml.ProjectFile(os.path.join(proot,'project.xml')).read()
a = pf.elements.make_node_and_append('app')
a.name = 'example_app'
a.user_friendly_name = 'Example Application'
pf.write()
httpd_conf_template_filename = os.path.join(
options.project_root,
project_shortname+'.httpd.conf'
)
print >>open(httpd_conf_template_filename,'w'), '''
## Settings for BOINC project %(project_longname)s
Alias /%(project_shortname)s/download %(proot)s/download
Alias /%(project_shortname)s/stats %(proot)s/html/stats
Alias /%(project_shortname)s/user_profile %(proot)s/html/user_profile
Alias /%(project_shortname)s %(proot)s/html/user
Alias /%(project_shortname)s_ops %(proot)s/html/ops
ScriptAlias /%(project_shortname)s_cgi %(proot)s/cgi-bin
# in the following, the "Order" and "Allow" lines are for Apache 2.2;
# for Apache 2.4, replace them with the single line
# Require all granted
# or
# Require all denied
# NOTE: Turn off access to certain default directories
Order deny,allow
Deny from all
Order deny,allow
Deny from all
# NOTE: Allow access but disable PHP script execution
RemoveType .php .phtml
Order allow,deny
Allow from all
RemoveType .php .phtml
Order allow,deny
Allow from all
RemoveType .php .phtml
Order allow,deny
Allow from all
# NOTE: Allow access and allow PHP script execution
Options Indexes MultiViews
AllowOverride AuthConfig
Order allow,deny
Allow from all
# NOTE: Allow access and allow CGI execution
Options ExecCGI
AllowOverride AuthConfig
Order allow,deny
Allow from all
''' %locals()
htaccess_filename = options.project_root+'/html/ops/.htaccess'
htpasswd_filename = options.project_root+'/html/ops/.htpasswd'
print >>open(htaccess_filename, 'w'), '''
AuthName "%(project_shortname)s"
AuthType Basic
AuthUserFile %(htpasswd_filename)s
require valid-user
'''%locals()
cronjob_filename = os.path.join(
options.project_root,
project_shortname+'.cronjob'
)
print >>open(cronjob_filename,'w'), '''0,5,10,15,20,25,30,35,40,45,50,55 * * * * cd %(proot)s ; %(proot)s/bin/start --cron
'''%locals()
readme_filename = os.path.join(
options.project_root,
project_shortname+'.readme'
)
test_app_msg = ''
if options.test_app:
test_app_msg = '''
- The project is configured with a test application.
To install this application (recommended) run:
cd %(proot)s
bin/update_versions
'''%locals()
xadd_msg = ''
if not options.web_only:
xadd_msg = '''
- In the project home directory (%(proot)s) run
bin/xadd
'''%locals()
compute_msg = ''
if not options.web_only:
compute_msg = '''
----------------------------
To use this project for your own computations, you'll need to
- Port your application(s) and add them
- Develop programs to submit and handle jobs
See the online documentation at http://boinc.berkeley.edu/
'''
html_user_url = options.html_user_url
html_ops_url = options.html_ops_url
print >>open(readme_filename,'w'), '''Steps to complete installation:
- Change Apache configuration (as root):
If you are using Apache 2.4, edit the %(httpd_conf_template_filename)s
(see file for specific instructions).
cat %(httpd_conf_template_filename)s >> /etc/apache/httpd.conf
(path to httpd.conf varies; try /etc/httpd/ or /etc/apache2)
Then restart the web server:
/usr/sbin/apache2ctl restart
(or /usr/sbin/apachectl restart)
- Add to crontab (as %(USER)s)
crontab -e
and add the line:
0,5,10,15,20,25,30,35,40,45,50,55 * * * * cd %(proot)s ; %(proot)s/bin/start --cron
%(xadd_msg)s%(test_app_msg)s
- Add a .htpasswd file in the project's html/ops directory:
(You'll need this to access the project's Administrative page)
htpasswd -c .htpasswd username
- Add the project name and copyright holder for the boinc Web site:
edit html/project/project.inc
change PROJECT and COPYRIGHT_HOLDER
----------------------------
To start, show status, and stop the project, run:
bin/start
bin/status
bin/stop
The project's URLs are:
Home page (and master URL): %(html_user_url)s
Administrative page: %(html_ops_url)s
%(compute_msg)s
'''%locals()
print '''
Done creating project. Please view
%(readme_filename)s
for important additional instructions.
'''%locals()