2
# IVLE - Informatics Virtual Learning Environment
3
# Copyright (C) 2009 The University of Melbourne
5
# This program is free software; you can redistribute it and/or modify
6
# it under the terms of the GNU General Public License as published by
7
# the Free Software Foundation; either version 2 of the License, or
8
# (at your option) any later version.
10
# This program is distributed in the hope that it will be useful,
11
# but WITHOUT ANY WARRANTY; without even the implied warranty of
12
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13
# GNU General Public License for more details.
15
# You should have received a copy of the GNU General Public License
16
# along with this program; if not, write to the Free Software
17
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
26
import ivle.jailbuilder.debian
28
class UnsafeJail(Exception):
31
usage = """usage: %prog [options]
33
Builds or updates the base IVLE jail."""
37
print >> sys.stderr, "This script requires root privlages to run"
40
conf = ivle.config.Config()
41
build_path = conf['paths']['jails']['template_build']
44
parser = optparse.OptionParser(usage)
45
parser.add_option("-r", "--recreate",
46
action="store_true", dest="recreate",
47
help='''Completely recreate the jail - don't just update its IVLE code.
48
Be warned, this may download hundreds of megabytes!''')
49
parser.add_option("-u", "--upgrade",
50
action="store_true", dest="upgrade",
51
help='''Apply any package updates in the jail.''')
52
parser.add_option("-m", "--mirror",
53
action="store", dest="apt_mirror",
54
help="Sets the apt mirror.", default=conf['jail']['mirror'])
55
parser.add_option("--python-site-packages",
56
action="store", dest="python_site_packages",
57
help="Path to Python site packages directory inside the jail.",
59
(options, args) = parser.parse_args(sys.argv)
62
print >> sys.stderr, "Must be root to run buildjail."
65
if not options.recreate and not os.path.exists(build_path):
66
print >> sys.stderr, "No jail exists -- please rerun with -r."
69
if (options.python_site_packages is not None and
70
options.python_site_packages[:1] not in (os.path.sep, os.path.altsep)):
71
print >> sys.stderr, "python-site-packages must be an absolute path."
75
options.upgrade = True
77
# Create the jail and its subdirectories
78
# Note: Other subdirs will be made by copying files
79
if options.apt_mirror is not None:
80
os.environ['MIRROR'] = options.apt_mirror
82
os.system('rm -rf --one-file-system ' + build_path)
83
ivle.jailbuilder.debian.debootstrap_create_jail(conf['jail']['suite'],
84
build_path, mirror=options.apt_mirror)
86
ivle.jailbuilder.debian.apt_update_cache(build_path)
87
ivle.jailbuilder.debian.apt_install(build_path,
88
['python2.5', 'python-cjson', 'python-svn'])
90
ivle.jailbuilder.debian.apt_clean(build_path)
93
# Run apt-get update, apt-get upgrade and apt-get clean.
94
ivle.jailbuilder.debian.mangle_sources_list(build_path, clobber=True)
95
ivle.jailbuilder.debian.mangle_sources_list(build_path, lines=[
96
'deb %s %s%s %s' % (options.apt_mirror, conf['jail']['suite'],
97
pocket, ' '.join(['main', 'universe']))
98
for pocket in ('', '-updates', '-security')])
100
# Add any extra site apt sources.
101
if conf['jail']['extra_sources']:
102
ivle.jailbuilder.debian.mangle_sources_list(build_path,
103
conf['jail']['extra_sources'])
105
# Add any extra site apt keys.
106
if conf['jail']['extra_keys']:
107
ivle.jailbuilder.debian.apt_add_key(build_path,
108
conf['jail']['extra_keys'])
110
ivle.jailbuilder.debian.apt_update_cache(build_path)
111
ivle.jailbuilder.debian.apt_upgrade(build_path)
113
# Install any extra site packages.
114
if conf['jail']['extra_packages']:
115
ivle.jailbuilder.debian.apt_install(build_path,
116
conf['jail']['extra_packages'])
118
ivle.jailbuilder.debian.apt_clean(build_path)
120
if conf['jail']['devmode']:
121
# Copy all console and operating system files into the jail
122
services_path = os.path.join(conf['paths']['share'], 'services')
123
jail_services_path = os.path.join(build_path, services_path[1:])
124
if os.path.exists(jail_services_path):
125
shutil.rmtree(jail_services_path)
126
shutil.copytree(services_path, jail_services_path)
128
# Also copy the IVLE lib directory into the jail
129
# This is necessary for running certain services
131
# ivle_site_packages is the IVLE install location outside the jail
132
ivle_site_packages = os.path.dirname(ivle.__file__)
134
if options.python_site_packages is None:
135
# Get the site packages from the IVLE install location *OUTSIDE* the
136
# jail. Warning! This only works if you have the same Python site
137
# packages directory inside and out (ie. same Python version).
138
# If not, you should use --python-site-packages.
139
jail_site_packages = os.path.join(build_path, ivle_site_packages[1:])
141
# User-specified site packages
142
jail_site_packages = os.path.join(build_path,
143
options.python_site_packages[1:], "ivle")
144
if os.path.exists(jail_site_packages):
145
shutil.rmtree(jail_site_packages)
146
shutil.copytree(ivle_site_packages, jail_site_packages)
148
# Make /tmp and /var/lock un-world-writable. /tmp will be mounted over,
149
# and /var/{lock,tmp} should die.
150
for path in ('tmp', 'var/lock', 'var/tmp'):
151
path = os.path.join(build_path, path)
152
os.chmod(path, os.stat(path).st_mode & ~stat.S_IWOTH)
154
# Verify that nothing in the jail is world-writable.
155
# We don't want students to write into places that others can see.
157
for path, dirs, files in os.walk(build_path):
159
d = os.path.join(path, dname)
160
if os.path.islink(d):
162
if os.stat(d).st_mode & stat.S_IWOTH:
166
f = os.path.join(path, fname)
167
if os.path.islink(f):
169
if os.stat(f).st_mode & stat.S_IWOTH:
170
if (os.path.dirname(f) == os.path.join(build_path, 'dev') and
171
os.path.basename(f) in ('ptmx', 'null', 'tty', 'full', 'zero',
176
except UnsafeJail, e:
177
print >> sys.stderr,"""Error: Jail contains world writable path: '%s'.
178
This is a security vulnerability as jail template contents are shared between
179
users. Please either make this path world unwriteable or remove it from the
183
# Copy jail template build to actual jail template
184
template_path = conf['paths']['jails']['template']
185
if os.spawnvp(os.P_WAIT, 'rsync', ['rsync', '-a', '--delete',
186
build_path + '/', template_path]) != 0:
187
print >> sys.stderr, "Jail copying failed."
190
# Now mangle things a bit, so we can bind-mount the user bits in.
191
# /etc/passwd and /etc/ivle/ivle.conf need to be symlinks to somewhere in /home
193
os.rename(os.path.join(template_path, 'etc/passwd'),
194
os.path.join(template_path, 'home/.passwd')
196
os.symlink('../home/.passwd', os.path.join(template_path, 'etc/passwd'))
198
os.makedirs(os.path.join(template_path, "etc/ivle"))
199
os.symlink('../../home/.ivle.conf',
200
os.path.join(template_path, "etc/ivle/ivle.conf"))