35
35
# Configures IVLE with machine-specific details, most notably, various paths.
36
36
# Either prompts the administrator for these details or accepts them as
37
37
# command-line args.
38
# Creates www/conf/conf.py and trampoline/conf.h.
38
# Creates lib/conf/conf.py and trampoline/conf.h.
41
41
# Compiles all files and sets up a jail template in the source directory.
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',
228
267
# Database username"""))
229
268
config_options.append(ConfigOption("db_password", "",
230
269
"""Password for DB server login:
231
(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)""",
233
272
# Database password"""))
237
276
# of setup besides conf, so we need to know them.
238
277
# Also this allows you to hit Return to accept the existing value.
240
confmodule = __import__("www/conf/conf")
279
confmodule = __import__("lib/conf/conf")
241
280
for opt in config_options:
243
282
globals()[opt.option_name] = confmodule.__dict__[opt.option_name]
352
391
to rebuild/install), just provide ivle_install_dir as the IVLE trunk
353
392
directory, and run build/install one time.
355
Creates www/conf/conf.py and trampoline/conf.h.
394
Creates lib/conf/conf.py and trampoline/conf.h.
358
397
for opt in config_options:
402
441
def listmake(args):
403
442
# We build two separate lists, by walking www and console
404
443
list_www = build_list_py_files('www')
405
list_console = build_list_py_files('console')
444
list_lib = build_list_py_files('lib')
445
list_scripts = build_list_py_files('scripts')
406
446
list_subjects = build_list_py_files('subjects', no_top_level=True)
407
447
list_problems = build_list_py_files('problems', no_top_level=True)
408
448
# Make sure that the files generated by conf are in the list
409
449
# (since listmake is typically run before conf)
410
if "www/conf/conf.py" not in list_www:
411
list_www.append("www/conf/conf.py")
450
if "lib/conf/conf.py" not in list_lib:
451
list_lib.append("lib/conf/conf.py")
412
452
# Make sure that console/python-console is in the list
413
if "console/python-console" not in list_console:
414
list_console.append("console/python-console")
453
if "scripts/python-console" not in list_scripts:
454
list_scripts.append("scripts/python-console")
455
if "scripts/fileservice" not in list_scripts:
456
list_scripts.append("scripts/fileservice")
415
457
# Write these out to a file
416
458
cwd = os.getcwd()
417
459
# the files that will be created/overwritten
432
474
writelist_pretty(file, list_www)
434
# List of all installable files in console directory.
436
writelist_pretty(file, list_console)
476
# List of all installable files in lib directory.
478
writelist_pretty(file, list_lib)
480
# List of all installable files in scripts directory.
482
writelist_pretty(file, list_scripts)
438
484
# List of all installable files in subjects directory.
439
485
# This is to install sample subjects and material.
495
541
cwd = os.getcwd()
496
542
# the files that will be created/overwritten
497
conffile = os.path.join(cwd, "www/conf/conf.py")
543
conffile = os.path.join(cwd, "lib/conf/conf.py")
544
jailconffile = os.path.join(cwd, "lib/conf/jailconf.py")
498
545
conf_hfile = os.path.join(cwd, "trampoline/conf.h")
500
547
# Get command-line arguments to avoid asking questions.
514
561
print """This tool will create the following files:
517
565
prompting you for details about your configuration. The file will be
518
566
overwritten if it already exists. It will *not* install or deploy IVLE.
520
568
Please hit Ctrl+C now if you do not wish to do this.
521
""" % (conffile, conf_hfile)
569
""" % (conffile, jailconffile, conf_hfile)
523
571
# Get information from the administrator
524
572
# If EOF is encountered at any time during the questioning, just exit
570
618
print "IO error(%s): %s" % (errno, strerror)
573
print "Successfully wrote www/conf/conf.py"
621
print "Successfully wrote lib/conf/conf.py"
623
# Write conf/jailconf.py
626
conf = open(jailconffile, "w")
628
# In the "in-jail" version of conf, we don't need MOST of the details
629
# (it would be a security risk to have them here).
630
# So we just write root_dir, and jail_base is "/".
631
# (jail_base being "/" means "jail-relative" paths are relative to "/"
632
# when inside the jail.)
633
conf.write("""# IVLE Configuration File
635
# Miscellaneous application settings
636
# (User jail version)
639
# In URL space, where in the site is IVLE located. (All URLs will be prefixed
641
# eg. "/" or "/ivle".
644
# In the local file system, where are the student/user file spaces located.
645
# The user jails are expected to be located immediately in subdirectories of
649
# The hostname for serving publicly accessible pages
651
""" % (repr(root_dir),repr(public_host)))
654
except IOError, (errno, strerror):
655
print "IO error(%s): %s" % (errno, strerror)
658
print "Successfully wrote lib/conf/jailconf.py"
575
660
# Write trampoline/conf.h
595
680
* (Note that root is an implicit member of this list).
597
682
static const int allowed_uids[] = { %s };
598
""" % (jail_base, repr(allowed_uids_list)[1:-1]))
683
""" % (repr(jail_base)[1:-1], repr(allowed_uids_list)[1:-1]))
684
# Note: The above uses PYTHON reprs, not C reprs
685
# However they should be the same with the exception of the outer
686
# characters, which are stripped off and replaced
601
689
except IOError, (errno, strerror):
633
722
action_mkdir('jail/tmp', dry)
635
724
# Copy all console and operating system files into the jail
636
action_copylist(install_list.list_console, 'jail/opt/ivle', dry)
725
action_copylist(install_list.list_scripts, 'jail/opt/ivle', dry)
637
726
copy_os_files_jail(dry)
638
727
# Chmod the python console
639
action_chmod_x('jail/opt/ivle/console/python-console', dry)
728
action_chmod_x('jail/opt/ivle/scripts/python-console', dry)
729
action_chmod_x('jail/opt/ivle/scripts/fileservice', dry)
731
# Also copy the IVLE lib directory into the jail
732
# This is necessary for running certain scripts
733
action_copylist(install_list.list_lib, 'jail/opt/ivle', dry)
734
# IMPORTANT: The file jail/opt/ivle/lib/conf/conf.py contains details
735
# which could compromise security if left in the jail (such as the DB
737
# The "safe" version is in jailconf.py. Delete conf.py and replace it with
739
action_copyfile('lib/conf/jailconf.py',
740
'jail/opt/ivle/lib/conf/conf.py', dry)
642
742
# Compile .py files into .pyc or .pyo files
643
743
compileall.compile_dir('www', quiet=True)
644
compileall.compile_dir('console', quiet=True)
744
compileall.compile_dir('lib', quiet=True)
745
compileall.compile_dir('scripts', quiet=True)
746
compileall.compile_dir('jail/opt/ivle/lib', quiet=True)
748
# Set up ivle.pth inside the jail
749
# Need to set /opt/ivle/lib to be on the import path
751
"jail/usr/lib/python%s/site-packages/ivle.pth" % PYTHON_VERSION
752
f = open(ivle_pth, 'w')
753
f.write('/opt/ivle/lib\n')
689
799
# chown trampoline to root and set setuid bit
690
800
action_chown_setuid(tramppath, dry)
692
# Copy the www directory using the list
802
# Copy the www and lib directories using the list
693
803
action_copylist(install_list.list_www, ivle_install_dir, dry)
804
action_copylist(install_list.list_lib, ivle_install_dir, dry)
696
807
# Copy the local jail directory built by the build action
709
820
ivle_pth = os.path.join(sys.prefix,
710
821
"lib/python%s/site-packages/ivle.pth" % PYTHON_VERSION)
711
822
ivle_www = os.path.join(ivle_install_dir, "www")
823
ivle_lib = os.path.join(ivle_install_dir, "lib")
712
824
write_ivle_pth = True
825
write_ivle_lib_pth = True
714
827
file = open(ivle_pth, 'r')
715
828
for line in file:
716
829
if line.strip() == ivle_www:
717
830
write_ivle_pth = False
831
elif line.strip() == ivle_lib:
832
write_ivle_lib_pth = False
719
834
except (IOError, OSError):
721
836
if write_ivle_pth:
722
837
action_append(ivle_pth, ivle_www)
838
if write_ivle_lib_pth:
839
action_append(ivle_pth, ivle_lib)
788
905
raise RunError(prog, ret)
907
def action_remove(path, dry):
908
"""Calls rmtree, deleting the target file if it exists."""
912
shutil.rmtree(path, True)
913
except OSError, (err, msg):
914
if err != errno.EEXIST:
916
# Otherwise, didn't exist, so we don't care
790
918
def action_rename(src, dst, dry):
791
919
"""Calls rename. Deletes the target if it already exists."""
792
if os.access(dst, os.F_OK):
795
shutil.rmtree(dst, True)
920
action_remove(dst, dry)
796
921
print "mv ", src, dst
830
952
"""Hard-links an entire directory tree. Same as copytree but the created
831
953
files are hard-links not actual copies. Removes the existing destination.
833
if os.access(dst, os.F_OK):
836
shutil.rmtree(dst, True)
955
action_remove(dst, dry)
837
956
print "<cp with hardlinks> -r", src, dst
839
958
common.makeuser.linktree(src, dst)