101
101
# Needed by python
102
102
'/usr/bin/python%s' % PYTHON_VERSION,
103
# Needed by fileservice
104
'/lib/libcom_err.so.2',
105
'/lib/libcrypt.so.1',
106
'/lib/libkeyutils.so.1',
107
'/lib/libresolv.so.2',
110
'/usr/lib/libapr-1.so.0',
111
'/usr/lib/libaprutil-1.so.0',
112
'/usr/lib/libdb-4.4.so',
113
'/usr/lib/libexpat.so.1',
114
'/usr/lib/libgcrypt.so.11',
115
'/usr/lib/libgnutls.so.13',
116
'/usr/lib/libgpg-error.so.0',
117
'/usr/lib/libgssapi_krb5.so.2',
118
'/usr/lib/libk5crypto.so.3',
119
'/usr/lib/libkrb5.so.3',
120
'/usr/lib/libkrb5support.so.0',
121
'/usr/lib/liblber.so.2',
122
'/usr/lib/libldap_r.so.2',
123
'/usr/lib/libneon.so.26',
124
'/usr/lib/libpq.so.5',
125
'/usr/lib/libsasl2.so.2',
126
'/usr/lib/libsqlite3.so.0',
127
'/usr/lib/libsvn_client-1.so.1',
128
'/usr/lib/libsvn_delta-1.so.1',
129
'/usr/lib/libsvn_diff-1.so.1',
130
'/usr/lib/libsvn_fs-1.so.1',
131
'/usr/lib/libsvn_fs_base-1.so.1',
132
'/usr/lib/libsvn_fs_fs-1.so.1',
133
'/usr/lib/libsvn_ra-1.so.1',
134
'/usr/lib/libsvn_ra_dav-1.so.1',
135
'/usr/lib/libsvn_ra_local-1.so.1',
136
'/usr/lib/libsvn_ra_svn-1.so.1',
137
'/usr/lib/libsvn_repos-1.so.1',
138
'/usr/lib/libsvn_subr-1.so.1',
139
'/usr/lib/libsvn_wc-1.so.1',
140
'/usr/lib/libtasn1.so.3',
141
'/usr/lib/libxml2.so.2',
103
142
# Needed by matplotlib
104
143
'/usr/lib/i686/cmov/libssl.so.0.9.8',
105
144
'/usr/lib/i686/cmov/libcrypto.so.0.9.8',
179
218
# In the local file system, where are the per-subject file spaces located.
180
219
# The individual subject directories are expected to be located immediately
181
220
# in subdirectories of this location."""))
221
config_options.append(ConfigOption("problems_base",
222
"/home/informatics/problems",
223
"""Root directory where the problem directories (containing
224
subject-independent problem sheets) are stored (on the local file
227
# In the local file system, where are the subject-independent problem sheet
228
# file spaces located."""))
182
229
config_options.append(ConfigOption("public_host", "public.localhost",
183
230
"""Hostname which will cause the server to go into "public mode",
184
231
providing login-free access to student's published work:""",
220
267
# Database username"""))
221
268
config_options.append(ConfigOption("db_password", "",
222
269
"""Password for DB server login:
223
(Caution: This password is stored in plaintext in www/conf/conf.py)""",
270
(Caution: This password is stored in plaintext in lib/conf/conf.py)""",
225
272
# Database password"""))
273
config_options.append(ConfigOption("svn_conf", "/opt/ivle/svn/svn.conf",
274
"""The location of the subversion configuration file used by apache
275
to host the user repositories:""",
277
# The location of the subversion configuration file used by
278
# apache to host the user repositories."""))
279
config_options.append(ConfigOption("svn_repo_path", "/home/informatics/repositories",
280
"""The root directory for the subversion repositories:""",
282
# The root directory for the subversion repositories."""))
283
config_options.append(ConfigOption("svn_auth_ivle", "/opt/ivle/svn/ivle.auth",
284
"""The location of the password file used to authenticate users
285
of the subversion repository from the ivle server:""",
287
# The location of the password file used to authenticate users
288
# of the subversion repository from the ivle server."""))
289
config_options.append(ConfigOption("svn_auth_local", "/opt/ivle/svn/local.auth",
290
"""The location of the password file used to authenticate local users
291
of the subversion repository:""",
293
# The location of the password file used to authenticate local users
294
# of the subversion repository."""))
295
config_options.append(ConfigOption("usrmgt_host", "localhost",
296
"""The hostname where the usrmgt-server runs:""",
298
# The hostname where the usrmgt-server runs."""))
299
config_options.append(ConfigOption("usrmgt_port", "2178",
300
"""The port where the usrmgt-server runs:""",
302
# The port where the usrmgt-server runs."""))
303
config_options.append(ConfigOption("usrmgt_magic", "",
304
"""The password for the usrmgt-server:""",
306
# The password for the usrmgt-server."""))
227
308
# Try importing existing conf, but if we can't just set up defaults
228
309
# The reason for this is that these settings are used by other phases
229
310
# of setup besides conf, so we need to know them.
230
311
# Also this allows you to hit Return to accept the existing value.
232
confmodule = __import__("www/conf/conf")
313
confmodule = __import__("lib/conf/conf")
233
314
for opt in config_options:
235
316
globals()[opt.option_name] = confmodule.__dict__[opt.option_name]
377
458
Copy subjects/ to subjects directory (unless --nosubjects specified).
379
460
--nojail Do not copy the jail.
380
--nosubjects Do not copy the subjects.
461
--nosubjects Do not copy the subjects and problems directories.
381
462
--dry | -n Print out the actions but don't do anything."""
382
463
elif operation == 'updatejails':
383
464
print """sudo python setup.py updatejails [--dry|-n]
394
475
def listmake(args):
395
476
# We build two separate lists, by walking www and console
396
477
list_www = build_list_py_files('www')
397
list_console = build_list_py_files('console')
478
list_lib = build_list_py_files('lib')
479
list_scripts = build_list_py_files('scripts')
398
480
list_subjects = build_list_py_files('subjects', no_top_level=True)
481
list_problems = build_list_py_files('problems', no_top_level=True)
399
482
# Make sure that the files generated by conf are in the list
400
483
# (since listmake is typically run before conf)
401
if "www/conf/conf.py" not in list_www:
402
list_www.append("www/conf/conf.py")
484
if "lib/conf/conf.py" not in list_lib:
485
list_lib.append("lib/conf/conf.py")
403
486
# Make sure that console/python-console is in the list
404
if "console/python-console" not in list_console:
405
list_console.append("console/python-console")
487
if "scripts/python-console" not in list_scripts:
488
list_scripts.append("scripts/python-console")
489
if "scripts/fileservice" not in list_scripts:
490
list_scripts.append("scripts/fileservice")
406
491
# Write these out to a file
407
492
cwd = os.getcwd()
408
493
# the files that will be created/overwritten
423
508
writelist_pretty(file, list_www)
425
# List of all installable files in console directory.
427
writelist_pretty(file, list_console)
510
# List of all installable files in lib directory.
512
writelist_pretty(file, list_lib)
514
# List of all installable files in scripts directory.
516
writelist_pretty(file, list_scripts)
429
518
# List of all installable files in subjects directory.
430
519
# This is to install sample subjects and material.
431
520
list_subjects = """)
432
521
writelist_pretty(file, list_subjects)
523
# List of all installable files in problems directory.
524
# This is to install sample exercise material.
526
writelist_pretty(file, list_problems)
435
529
except IOError, (errno, strerror):
481
575
cwd = os.getcwd()
482
576
# the files that will be created/overwritten
483
conffile = os.path.join(cwd, "www/conf/conf.py")
577
conffile = os.path.join(cwd, "lib/conf/conf.py")
578
jailconffile = os.path.join(cwd, "lib/conf/jailconf.py")
484
579
conf_hfile = os.path.join(cwd, "trampoline/conf.h")
486
581
# Get command-line arguments to avoid asking questions.
500
595
print """This tool will create the following files:
503
599
prompting you for details about your configuration. The file will be
504
600
overwritten if it already exists. It will *not* install or deploy IVLE.
506
602
Please hit Ctrl+C now if you do not wish to do this.
507
""" % (conffile, conf_hfile)
603
""" % (conffile, jailconffile, conf_hfile)
509
605
# Get information from the administrator
510
606
# If EOF is encountered at any time during the questioning, just exit
556
652
print "IO error(%s): %s" % (errno, strerror)
559
print "Successfully wrote www/conf/conf.py"
655
print "Successfully wrote lib/conf/conf.py"
657
# Write conf/jailconf.py
660
conf = open(jailconffile, "w")
662
# In the "in-jail" version of conf, we don't need MOST of the details
663
# (it would be a security risk to have them here).
664
# So we just write root_dir, and jail_base is "/".
665
# (jail_base being "/" means "jail-relative" paths are relative to "/"
666
# when inside the jail.)
667
conf.write("""# IVLE Configuration File
669
# Miscellaneous application settings
670
# (User jail version)
673
# In URL space, where in the site is IVLE located. (All URLs will be prefixed
675
# eg. "/" or "/ivle".
678
# In the local file system, where are the student/user file spaces located.
679
# The user jails are expected to be located immediately in subdirectories of
683
# The hostname for serving publicly accessible pages
685
""" % (repr(root_dir),repr(public_host)))
688
except IOError, (errno, strerror):
689
print "IO error(%s): %s" % (errno, strerror)
692
print "Successfully wrote lib/conf/jailconf.py"
561
694
# Write trampoline/conf.h
581
714
* (Note that root is an implicit member of this list).
583
716
static const int allowed_uids[] = { %s };
584
""" % (jail_base, repr(allowed_uids_list)[1:-1]))
717
""" % (repr(jail_base)[1:-1], repr(allowed_uids_list)[1:-1]))
718
# Note: The above uses PYTHON reprs, not C reprs
719
# However they should be the same with the exception of the outer
720
# characters, which are stripped off and replaced
587
723
except IOError, (errno, strerror):
619
756
action_mkdir('jail/tmp', dry)
621
758
# Copy all console and operating system files into the jail
622
action_copylist(install_list.list_console, 'jail/opt/ivle', dry)
759
action_copylist(install_list.list_scripts, 'jail/opt/ivle', dry)
623
760
copy_os_files_jail(dry)
624
761
# Chmod the python console
625
action_chmod_x('jail/opt/ivle/console/python-console', dry)
762
action_chmod_x('jail/opt/ivle/scripts/python-console', dry)
763
action_chmod_x('jail/opt/ivle/scripts/fileservice', dry)
765
# Also copy the IVLE lib directory into the jail
766
# This is necessary for running certain scripts
767
action_copylist(install_list.list_lib, 'jail/opt/ivle', dry)
768
# IMPORTANT: The file jail/opt/ivle/lib/conf/conf.py contains details
769
# which could compromise security if left in the jail (such as the DB
771
# The "safe" version is in jailconf.py. Delete conf.py and replace it with
773
action_copyfile('lib/conf/jailconf.py',
774
'jail/opt/ivle/lib/conf/conf.py', dry)
628
776
# Compile .py files into .pyc or .pyo files
629
777
compileall.compile_dir('www', quiet=True)
630
compileall.compile_dir('console', quiet=True)
778
compileall.compile_dir('lib', quiet=True)
779
compileall.compile_dir('scripts', quiet=True)
780
compileall.compile_dir('jail/opt/ivle/lib', quiet=True)
782
# Set up ivle.pth inside the jail
783
# Need to set /opt/ivle/lib to be on the import path
785
"jail/usr/lib/python%s/site-packages/ivle.pth" % PYTHON_VERSION
786
f = open(ivle_pth, 'w')
787
f.write('/opt/ivle/lib\n')
675
833
# chown trampoline to root and set setuid bit
676
834
action_chown_setuid(tramppath, dry)
678
# Copy the www directory using the list
836
# Copy the www and lib directories using the list
679
837
action_copylist(install_list.list_www, ivle_install_dir, dry)
838
action_copylist(install_list.list_lib, ivle_install_dir, dry)
682
841
# Copy the local jail directory built by the build action
684
843
# for all the students' jails).
685
844
action_copytree('jail', os.path.join(jail_base, 'template'), dry)
686
845
if not nosubjects:
687
# Copy the subjects directory across
846
# Copy the subjects and problems directories across
688
847
action_copylist(install_list.list_subjects, subjects_base, dry,
689
848
srcdir="./subjects")
849
action_copylist(install_list.list_problems, problems_base, dry,
691
852
# Append IVLE path to ivle.pth in python site packages
692
853
# (Unless it's already there)
693
854
ivle_pth = os.path.join(sys.prefix,
694
855
"lib/python%s/site-packages/ivle.pth" % PYTHON_VERSION)
695
856
ivle_www = os.path.join(ivle_install_dir, "www")
857
ivle_lib = os.path.join(ivle_install_dir, "lib")
696
858
write_ivle_pth = True
859
write_ivle_lib_pth = True
698
861
file = open(ivle_pth, 'r')
699
862
for line in file:
700
863
if line.strip() == ivle_www:
701
864
write_ivle_pth = False
865
elif line.strip() == ivle_lib:
866
write_ivle_lib_pth = False
703
868
except (IOError, OSError):
705
870
if write_ivle_pth:
706
871
action_append(ivle_pth, ivle_www)
872
if write_ivle_lib_pth:
873
action_append(ivle_pth, ivle_lib)
772
939
raise RunError(prog, ret)
941
def action_remove(path, dry):
942
"""Calls rmtree, deleting the target file if it exists."""
946
shutil.rmtree(path, True)
947
except OSError, (err, msg):
948
if err != errno.EEXIST:
950
# Otherwise, didn't exist, so we don't care
774
952
def action_rename(src, dst, dry):
775
953
"""Calls rename. Deletes the target if it already exists."""
776
if os.access(dst, os.F_OK):
779
shutil.rmtree(dst, True)
954
action_remove(dst, dry)
780
955
print "mv ", src, dst
892
1061
| stat.S_ISUID | stat.S_IRUSR | stat.S_IWUSR)
894
1063
def action_chmod_x(file, dry):
895
"""Chmod +xs a file (sets execute permission)."""
896
print "chmod u+rwx", file
1064
"""Chmod 755 a file (sets permissions to rwxr-xr-x)."""
1065
print "chmod 755", file
898
os.chmod(file, stat.S_IXUSR | stat.S_IRUSR | stat.S_IWUSR)
1067
os.chmod(file, stat.S_IXUSR | stat.S_IRUSR | stat.S_IWUSR
1068
| stat.S_IXGRP | stat.S_IRGRP | stat.S_IXOTH | stat.S_IROTH)
900
1070
def query_user(default, prompt):
901
1071
"""Prompts the user for a string, which is read from a line of stdin.