217
213
warnings.simplefilter('ignore')
218
214
homebackup = os.tempnam(tempdir)
219
215
warnings.resetwarnings()
220
# Note: shutil.move does not behave like "mv" - it does not put a file
221
# into a directory if it already exists, just fails. Therefore it is
222
# not susceptible to tmpnam symlink attack.
216
# Back up the /home directory, delete the entire jail, recreate the
217
# jail directory tree, then copy the /home back
218
# NOTE that shutil.move changed in Python 2.6, it now moves a
219
# directory INTO the target (like `mv`), which it didn't use to do.
220
# This code works regardless.
223
221
shutil.move(homedir, homebackup)
224
222
shutil.rmtree(userdir)
226
224
shutil.move(homebackup, homedir)
227
225
# Change the ownership of all the files to the right unixid
228
226
logging.debug("chown %s's home directory files to uid %d"
229
227
%(user.login, user.unixid))
230
os.chown(userhomedir, user.unixid, user.unixid)
231
for root, dirs, files in os.walk(userhomedir):
232
for fsobj in dirs + files:
233
os.chown(os.path.join(root, fsobj), user.unixid, user.unixid)
228
os.spawnvp(os.P_WAIT, 'chown', ['chown', '-R', '%d:%d' % (user.unixid,
229
user.unixid), userhomedir])
235
231
# No user jail exists
236
232
# Set up the user's home directory
240
236
# Chmod to rwxr-xr-x (755)
241
237
os.chmod(userhomedir, 0755)
243
make_conf_py(user.login, userdir, user.svn_pass)
239
make_ivle_conf(user.login, userdir, user.svn_pass)
244
240
make_etc_passwd(user.login, userdir, ivle.conf.jail_system, user.unixid)
246
242
return userhomedir
248
def make_conf_py(username, user_jail_dir, svn_pass):
244
def make_ivle_conf(username, user_jail_dir, svn_pass):
250
246
Creates (overwriting any existing file, and creating directories) a
251
file ${python_site_packages}/ivle/conf/conf.py in a given user's jail.
247
file /etc/ivle/ivle.conf in a given user's jail.
252
248
username: Username.
253
249
user_jail_dir: User's jail dir, ie. ivle.conf.jail_base + username
254
250
svn_pass: User's SVN password.
256
conf_path = os.path.join(user_jail_dir,
257
ivle.conf.python_site_packages[1:], "ivle/conf/conf.py")
252
conf_path = os.path.join(user_jail_dir, "etc/ivle/ivle.conf")
258
253
os.makedirs(os.path.dirname(conf_path))
260
255
# In the "in-jail" version of conf, we don't need MOST of the details
261
256
# (it would be a security risk to have them here).
262
# So we just write root_dir, and jail_base is "/".
263
# (jail_base being "/" means "jail-relative" paths are relative to "/"
264
# when inside the jail.)
266
# XXX: jail_base is wrong and shouldn't be here. Unfortunately, jail code
267
# uses ivle.studpath.url_to_{local,jailpaths}, both of which use
268
# jail_base. Note that they don't use the bits of the return value
269
# that depend on jail_base, so it can be any string.
270
conf_file = open(conf_path, "w")
271
conf_file.write("""# IVLE jail configuration
273
# In URL space, where in the site is IVLE located. (All URLs will be prefixed
275
# eg. "/" or "/ivle".
276
root_dir = %(root_dir)r
278
# This value is not relevant inside the jail, but must remain for now. See
279
# the XXX in ivle.makeuser.make_conf_py.
282
# The hostname for serving publicly accessible pages
283
public_host = %(public_host)r
285
# The URL under which the Subversion repositories are located.
286
svn_addr = %(svn_addr)r
288
# The login name for the owner of the jail
291
# The subversion-only password for the owner of the jail
292
svn_pass = %(svn_pass)r
293
""" % {'root_dir': ivle.conf.root_dir,
294
'public_host': ivle.conf.public_host,
295
'svn_addr': ivle.conf.svn_addr,
296
'username': username,
297
'svn_pass': svn_pass,
257
# So we just write root_dir.
258
conf_obj = ivle.config.Config(blank=True)
259
conf_obj.filename = conf_path
260
conf_obj['urls']['root'] = ivle.conf.root_dir
261
conf_obj['urls']['public_host'] = ivle.conf.public_host
262
conf_obj['urls']['svn_addr'] = ivle.conf.svn_addr
263
conf_obj['user_info']['login'] = username
264
conf_obj['user_info']['svn_pass'] = svn_pass
301
267
# Make this file world-readable
302
268
# (chmod 644 conf_path)