71
# Import modules from the website is tricky since they're in the www
73
sys.path.append(os.path.join(os.getcwd(), 'lib'))
75
import common.makeuser
77
# Determine which Python version (2.4 or 2.5, for example) we are running,
78
# and use that as the filename to the Python directory.
79
# Just get the first 3 characters of sys.version.
80
PYTHON_VERSION = sys.version[0:3]
82
# Operating system files to copy over into the jail.
83
# These will be copied from the given place on the OS file system into the
84
# same place within the jail.
87
'/lib/tls/i686/cmov/libc.so.6',
88
'/lib/tls/i686/cmov/libdl.so.2',
89
'/lib/tls/i686/cmov/libm.so.6',
90
'/lib/tls/i686/cmov/libpthread.so.0',
91
'/lib/tls/i686/cmov/libutil.so.1',
94
# These 2 files do not exist in Ubuntu
95
#'/etc/ld.so.preload',
96
#'/etc/ld.so.nohwcap',
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',
142
# Needed by matplotlib
143
'/usr/lib/i686/cmov/libssl.so.0.9.8',
144
'/usr/lib/i686/cmov/libcrypto.so.0.9.8',
145
'/lib/tls/i686/cmov/libnsl.so.1',
146
'/usr/lib/libz.so.1',
147
'/usr/lib/atlas/liblapack.so.3',
148
'/usr/lib/atlas/libblas.so.3',
149
'/usr/lib/libg2c.so.0',
150
'/usr/lib/libstdc++.so.6',
151
'/usr/lib/libfreetype.so.6',
152
'/usr/lib/libpng12.so.0',
153
'/usr/lib/libBLT.2.4.so.8.4',
154
'/usr/lib/libtk8.4.so.0',
155
'/usr/lib/libtcl8.4.so.0',
156
'/usr/lib/tcl8.4/init.tcl',
157
'/usr/lib/libX11.so.6',
158
'/usr/lib/libXau.so.6',
159
'/usr/lib/libXdmcp.so.6',
160
'/lib/libgcc_s.so.1',
163
# Symlinks to make within the jail. Src mapped to dst.
165
'python%s' % PYTHON_VERSION: 'jail/usr/bin/python',
167
# Trees to copy. Src mapped to dst (these will be passed to action_copytree).
169
'/usr/lib/python%s' % PYTHON_VERSION:
170
'jail/usr/lib/python%s' % PYTHON_VERSION,
171
'/usr/share/matplotlib': 'jail/usr/share/matplotlib',
172
'/etc/ld.so.conf.d': 'jail/etc/ld.so.conf.d',
176
"""A configuration option; one of the things written to conf.py."""
177
def __init__(self, option_name, default, prompt, comment):
178
"""Creates a configuration option.
179
option_name: Name of the variable in conf.py. Also name of the
180
command-line argument to setup.py conf.
181
default: Default value for this variable.
182
prompt: (Short) string presented during the interactive prompt in
184
comment: (Long) comment string stored in conf.py. Each line of this
185
string should begin with a '#'.
187
self.option_name = option_name
188
self.default = default
190
self.comment = comment
192
# Configuration options, defaults and descriptions
194
config_options.append(ConfigOption("root_dir", "/",
195
"""Root directory where IVLE is located (in URL space):""",
197
# In URL space, where in the site is IVLE located. (All URLs will be prefixed
199
# eg. "/" or "/ivle"."""))
200
config_options.append(ConfigOption("ivle_install_dir", "/opt/ivle",
201
'Root directory where IVLE will be installed (on the local file '
204
# In the local file system, where IVLE is actually installed.
205
# This directory should contain the "www" and "bin" directories."""))
206
config_options.append(ConfigOption("jail_base", "/home/informatics/jails",
207
"""Root directory where the jails (containing user files) are stored
208
(on the local file system):""",
210
# In the local file system, where are the student/user file spaces located.
211
# The user jails are expected to be located immediately in subdirectories of
212
# this location."""))
213
config_options.append(ConfigOption("subjects_base",
214
"/home/informatics/subjects",
215
"""Root directory where the subject directories (containing worksheets
216
and other per-subject files) are stored (on the local file system):""",
218
# In the local file system, where are the per-subject file spaces located.
219
# The individual subject directories are expected to be located immediately
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."""))
229
config_options.append(ConfigOption("public_host", "public.localhost",
230
"""Hostname which will cause the server to go into "public mode",
231
providing login-free access to student's published work:""",
233
# The server goes into "public mode" if the browser sends a request with this
234
# host. This is for security reasons - we only serve public student files on a
235
# separate domain to the main IVLE site.
236
# Public mode does not use cookies, and serves only public content.
237
# Private mode (normal mode) requires login, and only serves files relevant to
238
# the logged-in user."""))
239
config_options.append(ConfigOption("allowed_uids", "33",
240
"""UID of the web server process which will run IVLE.
241
Only this user may execute the trampoline. May specify multiple users as
242
a comma-separated list.
245
# The User-ID of the web server process which will run IVLE, and any other
246
# users who are allowed to run the trampoline. This is stores as a string of
247
# comma-separated integers, simply because it is not used within Python, only
248
# used by the setup program to write to conf.h (see setup.py config)."""))
249
config_options.append(ConfigOption("db_host", "localhost",
250
"""PostgreSQL Database config
251
==========================
252
Hostname of the DB server:""",
254
### PostgreSQL Database config ###
255
# Database server hostname"""))
256
config_options.append(ConfigOption("db_port", "5432",
257
"""Port of the DB server:""",
259
# Database server port"""))
260
config_options.append(ConfigOption("db_dbname", "ivle",
261
"""Database name:""",
264
config_options.append(ConfigOption("db_user", "postgres",
265
"""Username for DB server login:""",
267
# Database username"""))
268
config_options.append(ConfigOption("db_password", "",
269
"""Password for DB server login:
270
(Caution: This password is stored in plaintext in lib/conf/conf.py)""",
272
# Database password"""))
274
config_options.append(ConfigOption("ldap_url", "ldaps://www.example.com",
275
"""URL for LDAP authentication server:""",
277
# URL for LDAP authentication server"""))
278
config_options.append(ConfigOption("ldap_format_string",
279
"uid=%s,ou=users,o=example",
280
"""Format string for LDAP auth request:
281
(Must contain a single "%s" for the user's login name)""",
283
# Format string for LDAP auth request
284
# (Must contain a single "%s" for the user's login name)"""))
286
config_options.append(ConfigOption("svn_conf", "/opt/ivle/svn/svn.conf",
287
"""The location of the subversion configuration file used by apache
288
to host the user repositories:""",
290
# The location of the subversion configuration file used by
291
# apache to host the user repositories."""))
292
config_options.append(ConfigOption("svn_repo_path", "/home/informatics/repositories",
293
"""The root directory for the subversion repositories:""",
295
# The root directory for the subversion repositories."""))
296
config_options.append(ConfigOption("svn_auth_ivle", "/opt/ivle/svn/ivle.auth",
297
"""The location of the password file used to authenticate users
298
of the subversion repository from the ivle server:""",
300
# The location of the password file used to authenticate users
301
# of the subversion repository from the ivle server."""))
302
config_options.append(ConfigOption("svn_auth_local", "/opt/ivle/svn/local.auth",
303
"""The location of the password file used to authenticate local users
304
of the subversion repository:""",
306
# The location of the password file used to authenticate local users
307
# of the subversion repository."""))
308
config_options.append(ConfigOption("usrmgt_host", "localhost",
309
"""The hostname where the usrmgt-server runs:""",
311
# The hostname where the usrmgt-server runs."""))
312
config_options.append(ConfigOption("usrmgt_port", "2178",
313
"""The port where the usrmgt-server runs:""",
315
# The port where the usrmgt-server runs."""))
316
config_options.append(ConfigOption("usrmgt_magic", "",
317
"""The password for the usrmgt-server:""",
319
# The password for the usrmgt-server."""))
71
321
# Try importing existing conf, but if we can't just set up defaults
72
322
# The reason for this is that these settings are used by other phases
73
323
# of setup besides conf, so we need to know them.
74
324
# Also this allows you to hit Return to accept the existing value.
76
confmodule = __import__("www/conf/conf")
77
root_dir = confmodule.root_dir
78
ivle_install_dir = confmodule.ivle_install_dir
79
jail_base = confmodule.jail_base
326
confmodule = __import__("lib/conf/conf")
327
for opt in config_options:
329
globals()[opt.option_name] = confmodule.__dict__[opt.option_name]
331
globals()[opt.option_name] = opt.default
80
332
except ImportError:
81
333
# Just set reasonable defaults
83
ivle_install_dir = "/opt/ivle"
84
jail_base = "/home/informatics/jails"
334
for opt in config_options:
335
globals()[opt.option_name] = opt.default
88
337
# Try importing install_list, but don't fail if we can't, because listmake can
89
338
# function without it.
326
607
print """This tool will create the following files:
329
611
prompting you for details about your configuration. The file will be
330
612
overwritten if it already exists. It will *not* install or deploy IVLE.
332
614
Please hit Ctrl+C now if you do not wish to do this.
333
""" % (conffile, conf_hfile)
615
""" % (conffile, jailconffile, conf_hfile)
335
617
# Get information from the administrator
336
618
# If EOF is encountered at any time during the questioning, just exit
339
root_dir = query_user(root_dir,
340
"""Root directory where IVLE is located (in URL space):""")
341
ivle_install_dir = query_user(ivle_install_dir,
342
'Root directory where IVLE will be installed (on the local file '
344
jail_base = query_user(jail_base,
345
"""Root directory where the jails (containing user files) are stored
346
(on the local file system):""")
347
allowed_uids = query_user(allowed_uids,
348
"""UID of the web server process which will run IVLE.
349
Only this user may execute the trampoline. May specify multiple users as
350
a comma-separated list.
621
for opt in config_options:
622
globals()[opt.option_name] = \
623
query_user(globals()[opt.option_name], opt.prompt)
354
625
opts = dict(opts)
355
626
# Non-interactive mode. Parse the options.
356
if '--root_dir' in opts:
357
root_dir = opts['--root_dir']
358
if '--ivle_install_dir' in opts:
359
ivle_install_dir = opts['--ivle_install_dir']
360
if '--jail_base' in opts:
361
jail_base = opts['--jail_base']
362
if '--allowed_uids' in opts:
363
allowed_uids = opts['--allowed_uids']
627
for opt in config_options:
628
if '--' + opt.option_name in opts:
629
globals()[opt.option_name] = opts['--' + opt.option_name]
365
631
# Error handling on input values
367
allowed_uids = map(int, allowed_uids.split(','))
633
allowed_uids_list = map(int, allowed_uids.split(','))
368
634
except ValueError:
369
635
print >>sys.stderr, (
370
636
"Invalid UID list (%s).\n"
371
637
"Must be a comma-separated list of integers." % allowed_uids)
640
db_port = int(db_port)
641
if db_port < 0 or db_port >= 65536: raise ValueError()
643
print >>sys.stderr, (
644
"Invalid DB port (%s).\n"
645
"Must be an integer between 0 and 65535." % repr(db_port))
648
usrmgt_port = int(usrmgt_port)
649
if usrmgt_port < 0 or usrmgt_port >= 65536: raise ValueError()
651
print >>sys.stderr, (
652
"Invalid user management port (%s).\n"
653
"Must be an integer between 0 and 65535." % repr(usrmgt_port))
374
# Write www/conf/conf.py
656
# Write lib/conf/conf.py
377
659
conf = open(conffile, "w")
522
853
# chown trampoline to root and set setuid bit
523
854
action_chown_setuid(tramppath, dry)
525
# Copy the www directory using the list
856
# Copy the www and lib directories using the list
526
857
action_copylist(install_list.list_www, ivle_install_dir, dry)
858
action_copylist(install_list.list_lib, ivle_install_dir, dry)
529
861
# Copy the local jail directory built by the build action
530
862
# to the jails template directory (it will be used as a template
531
863
# for all the students' jails).
532
864
action_copytree('jail', os.path.join(jail_base, 'template'), dry)
866
# Copy the subjects and problems directories across
867
action_copylist(install_list.list_subjects, subjects_base, dry,
869
action_copylist(install_list.list_problems, problems_base, dry,
872
# Append IVLE path to ivle.pth in python site packages
873
# (Unless it's already there)
874
ivle_pth = os.path.join(sys.prefix,
875
"lib/python%s/site-packages/ivle.pth" % PYTHON_VERSION)
876
ivle_www = os.path.join(ivle_install_dir, "www")
877
ivle_lib = os.path.join(ivle_install_dir, "lib")
878
write_ivle_pth = True
879
write_ivle_lib_pth = True
881
file = open(ivle_pth, 'r')
883
if line.strip() == ivle_www:
884
write_ivle_pth = False
885
elif line.strip() == ivle_lib:
886
write_ivle_lib_pth = False
888
except (IOError, OSError):
891
action_append(ivle_pth, ivle_www)
892
if write_ivle_lib_pth:
893
action_append(ivle_pth, ivle_lib)
897
def updatejails(args):
898
# Get "dry" variable from command line
899
(opts, args) = getopt.gnu_getopt(args, "n", ['dry'])
901
dry = '-n' in opts or '--dry' in opts
904
print "Dry run (no actions will be executed\n"
906
if not dry and os.geteuid() != 0:
907
print >>sys.stderr, "Must be root to run install"
908
print >>sys.stderr, "(I need to chown some files)."
911
# Update the template jail directory in case it hasn't been installed
913
action_copytree('jail', os.path.join(jail_base, 'template'), dry)
915
# Re-link all the files in all students jails.
916
for dir in os.listdir(jail_base):
917
if dir == 'template': continue
918
# First back up the student's home directory
919
temp_home = os.tmpnam()
920
action_rename(os.path.join(jail_base, dir, 'home'), temp_home, dry)
921
# Delete the student's jail and relink the jail files
922
action_linktree(os.path.join(jail_base, 'template'),
923
os.path.join(jail_base, dir), dry)
924
# Restore the student's home directory
925
action_rename(temp_home, os.path.join(jail_base, dir, 'home'), dry)
926
# Set up the user's home directory just in case they don't have a
927
# directory for this yet
928
action_mkdir(os.path.join(jail_base, dir, 'home', dir), dry)