~azzar1/unity/add-show-desktop-key

« back to all changes in this revision

Viewing changes to ivle/fileservice_lib/action.py

  • Committer: drtomc
  • Date: 2007-12-11 03:26:29 UTC
  • Revision ID: svn-v3-trunk0:2b9c9e99-6f39-0410-b283-7f802c844ae2:trunk:25
A bit more work on the userdb stuff.

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# IVLE
2
 
# Copyright (C) 2007-2008 The University of Melbourne
3
 
#
4
 
# This program is free software; you can redistribute it and/or modify
5
 
# it under the terms of the GNU General Public License as published by
6
 
# the Free Software Foundation; either version 2 of the License, or
7
 
# (at your option) any later version.
8
 
#
9
 
# This program is distributed in the hope that it will be useful,
10
 
# but WITHOUT ANY WARRANTY; without even the implied warranty of
11
 
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12
 
# GNU General Public License for more details.
13
 
#
14
 
# You should have received a copy of the GNU General Public License
15
 
# along with this program; if not, write to the Free Software
16
 
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
17
 
 
18
 
# Module: File Service / Action
19
 
# Author: Matt Giuca
20
 
# Date: 10/1/2008
21
 
 
22
 
# Handles actions requested by the client as part of the 2-stage process of
23
 
# fileservice (the second part being the return listing).
24
 
 
25
 
### Actions ###
26
 
 
27
 
# The most important argument is "action". This determines which action is
28
 
# taken. Note that action, and all other arguments, are ignored unless the
29
 
# request is a POST request. The other arguments depend upon the action.
30
 
# Note that paths are often specified as arguments. Paths that begin with a
31
 
# slash are taken relative to the user's home directory (the top-level
32
 
# directory seen when fileservice has no arguments or path). Paths without a
33
 
# slash are taken relative to the specified path.
34
 
 
35
 
# action=remove: Delete a file(s) or directory(s) (recursively).
36
 
#       path:   The path to the file or directory to delete. Can be specified
37
 
#               multiple times.
38
 
#
39
 
# action=move: Move or rename a file or directory.
40
 
#       from:   The path to the file or directory to be renamed.
41
 
#       to:     The path of the target filename. Error if the file already
42
 
#               exists.
43
 
#
44
 
# action=putfile: Upload a file to the student workspace.
45
 
#       path:   The path to the file to be written. Error if the target
46
 
#               file is a directory.
47
 
#       data:   Bytes to be written to the file verbatim. May either be
48
 
#               a string variable or a file upload.
49
 
#       overwrite: Optional. If supplied, the file will be overwritten.
50
 
#               Otherwise, error if path already exists.
51
 
#
52
 
# action=putfiles: Upload multiple files to the student workspace, and
53
 
#                 optionally accept zip files which will be unpacked.
54
 
#       path:   The path to the DIRECTORY to place files in. Must not be a
55
 
#               file.
56
 
#       data:   A file upload (may not be a simple string). The filename
57
 
#               will be used to determine the target filename within
58
 
#               the given path.
59
 
#       unpack: Optional. If supplied, if any data is a valid ZIP file,
60
 
#               will create a directory instead and unpack the ZIP file
61
 
#               into it.
62
 
#
63
 
# action=mkdir: Create a directory. The parent dir must exist.
64
 
#       path:   The path to a file which does not exist, but whose parent
65
 
#               does. The dir will be made with this name.
66
 
#
67
 
# The differences between putfile and putfiles are:
68
 
# * putfile can only accept a single file, and can't unpack zipfiles.
69
 
# * putfile can accept string data, doesn't have to be a file upload.
70
 
# * putfile ignores the upload filename, the entire filename is specified on
71
 
#       path. putfiles calls files after the name on the user's machine.
72
 
#
73
 
# action=paste: Copy or move the files to a specified dir.
74
 
#       src:    The path to the DIRECTORY to get the files from (relative).
75
 
#       dst:    The path to the DIRECTORY to paste the files to. Must not
76
 
#               be a file.
77
 
#       mode:   'copy' or 'move'
78
 
#       file:   File to be copied or moved, relative to src, to a destination
79
 
#               relative to dst. Can be specified multiple times.
80
 
#
81
 
# Subversion actions.
82
 
# action=svnadd: Add an existing file(s) to version control.
83
 
#       path:   The path to the file to be added. Can be specified multiple
84
 
#               times.
85
 
#
86
 
# action=svnrevert: Revert a file(s) to its state as of the current revision
87
 
#               / undo local edits.
88
 
#       path:   The path to the file to be reverted. Can be specified multiple
89
 
#               times.
90
 
#
91
 
# action=svnupdate: Bring a file up to date with the head revision.
92
 
#       path:   The path to the file to be updated. Only one file may be
93
 
#               specified.
94
 
#       revision: The revision number to update to. If not provided this
95
 
#               defaults to HEAD.
96
 
#
97
 
# action=svnpublish: Set the "published" flag on a file to True.
98
 
#       path:   The path to the file to be published. Can be specified
99
 
#               multiple times.
100
 
#
101
 
# action=svnunpublish: Set the "published" flag on a file to False.
102
 
#       path:   The path to the file to be unpublished. Can be specified
103
 
#               multiple times.
104
 
#
105
 
# action=svncommit: Commit a file(s) or directory(s) to the repository.
106
 
#       path:   The path to the file or directory to be committed. Can be
107
 
#               specified multiple times. Directories are committed
108
 
#               recursively.
109
 
#       logmsg: Text of the log message. Optional. There is a default log
110
 
#               message if unspecified.
111
 
# action=svncheckout: Checkout a file/directory from the repository.
112
 
#       path:   The [repository] path to the file or directory to be
113
 
#               checked out.
114
 
115
 
# action=svnrepomkdir: Create a directory in a repository (not WC).
116
 
#       path:   The path to the directory to be created (under the IVLE
117
 
#               repository base).
118
 
#       logmsg: Text of the log message.
119
 
120
 
# action=svnrepostat: Check if a path exists in a repository (not WC).
121
 
#       path:   The path to the directory to be checked (under the IVLE
122
 
#               repository base).
123
 
#
124
 
# action=svncleanup: Recursively clean up the working copy, removing locks,
125
 
#   resuming unfinished operations, etc.
126
 
#       path:   The path to the directory to be cleaned
127
 
#
128
 
# TODO: Implement the following actions:
129
 
#   svnupdate (done?)
130
 
# TODO: Implement ZIP unpacking in putfiles (done?).
131
 
# TODO: svnupdate needs a digest to tell the user the files that were updated.
132
 
#   This can be implemented by some message passing between action and
133
 
#   listing, and having the digest included in the listing. (Problem if
134
 
#   the listing is not a directory, but we could make it an error to do an
135
 
#   update if the path is not a directory).
136
 
 
137
 
import os
138
 
import cStringIO
139
 
import shutil
140
 
import urllib
141
 
 
142
 
import pysvn
143
 
 
144
 
from ivle import (util, studpath, zip)
145
 
from ivle.fileservice_lib.exceptions import WillNotOverwrite
146
 
import ivle.conf
147
 
import ivle.svn
148
 
 
149
 
# Make a Subversion client object (which will log in with this user's
150
 
# credentials, upon request)
151
 
svnclient = ivle.svn.create_auth_svn_client(username=ivle.conf.login,
152
 
                                            password=ivle.conf.svn_pass)
153
 
svnclient.exception_style = 0               # Simple (string) exceptions
154
 
 
155
 
DEFAULT_LOGMESSAGE = "No log message supplied."
156
 
 
157
 
# Mime types
158
 
# application/json is the "best" content type but is not good for
159
 
# debugging because Firefox just tries to download it
160
 
mime_dirlisting = "text/html"
161
 
#mime_dirlisting = "application/json"
162
 
 
163
 
class ActionError(Exception):
164
 
    """Represents an error processing an action. This can be
165
 
    raised by any of the action functions, and will be caught
166
 
    by the top-level handler, put into the HTTP response field,
167
 
    and continue.
168
 
 
169
 
    Important Security Consideration: The message passed to this
170
 
    exception will be relayed to the client.
171
 
    """
172
 
    pass
173
 
 
174
 
def handle_action(req, action, fields):
175
 
    """Perform the "action" part of the response.
176
 
    This function should only be called if the response is a POST.
177
 
    This performs the action's side-effect on the server. If unsuccessful,
178
 
    writes the X-IVLE-Action-Error header to the request object. Otherwise,
179
 
    does not touch the request object. Does NOT write any bytes in response.
180
 
 
181
 
    May throw an ActionError. The caller should put this string into the
182
 
    X-IVLE-Action-Error header, and then continue normally.
183
 
 
184
 
    action: String, the action requested. Not sanitised.
185
 
    fields: FieldStorage object containing all arguments passed.
186
 
    """
187
 
    global actions_table        # Table of function objects
188
 
    try:
189
 
        action = actions_table[action]
190
 
    except KeyError:
191
 
        # Default, just send an error but then continue
192
 
        raise ActionError("Unknown action")
193
 
    return action(req, fields)
194
 
 
195
 
def actionpath_to_urlpath(req, path):
196
 
    """Determines the URL path (relative to the student home) upon which the
197
 
    action is intended to act. See actionpath_to_local.
198
 
    """
199
 
    if path is None:
200
 
        return req.path
201
 
    elif len(path) > 0 and path[0] == os.sep:
202
 
        # Relative to student home
203
 
        return path[1:]
204
 
    else:
205
 
        # Relative to req.path
206
 
        return os.path.join(req.path, path)
207
 
 
208
 
def actionpath_to_local(req, path):
209
 
    """Determines the local path upon which an action is intended to act.
210
 
    Note that fileservice actions accept two paths: the request path,
211
 
    and the "path" argument given to the action.
212
 
    According to the rules, if the "path" argument begins with a '/' it is
213
 
    relative to the user's home; if it does not, it is relative to the
214
 
    supplied path.
215
 
 
216
 
    This resolves the path, given the request and path argument.
217
 
 
218
 
    May raise an ActionError("Invalid path"). The caller is expected to
219
 
    let this fall through to the top-level handler, where it will be
220
 
    put into the HTTP response field. Never returns None.
221
 
 
222
 
    Does not mutate req.
223
 
    """
224
 
    r = studpath.to_home_path(actionpath_to_urlpath(req, path))
225
 
    if r is None:
226
 
        raise ActionError("Invalid path")
227
 
    return r
228
 
 
229
 
def movefile(req, frompath, topath, copy=False):
230
 
    """Performs a file move, resolving filenames, checking for any errors,
231
 
    and throwing ActionErrors if necessary. Can also be used to do a copy
232
 
    operation instead.
233
 
 
234
 
    frompath and topath are straight paths from the client. Will be checked.
235
 
    """
236
 
    # TODO: Do an SVN mv if the file is versioned.
237
 
    # TODO: Disallow tampering with student's home directory
238
 
    if frompath is None or topath is None:
239
 
        raise ActionError("Required field missing")
240
 
    frompath = actionpath_to_local(req, frompath)
241
 
    topath = actionpath_to_local(req, topath)
242
 
    if not os.path.exists(frompath):
243
 
        raise ActionError("The source file does not exist")
244
 
    if os.path.exists(topath):
245
 
        if frompath == topath:
246
 
            raise ActionError("Source and destination are the same")
247
 
        raise ActionError("A file already exists with that name")
248
 
 
249
 
    try:
250
 
        if copy:
251
 
            if os.path.isdir(frompath):
252
 
                shutil.copytree(frompath, topath)
253
 
            else:
254
 
                shutil.copy2(frompath, topath)
255
 
        else:
256
 
            shutil.move(frompath, topath)
257
 
    except OSError:
258
 
        raise ActionError("Could not move the file specified")
259
 
    except shutil.Error:
260
 
        raise ActionError("Could not move the file specified")
261
 
 
262
 
def svn_movefile(req, frompath, topath, copy=False):
263
 
    """Performs an svn move, resolving filenames, checking for any errors,
264
 
    and throwing ActionErrors if necessary. Can also be used to do a copy
265
 
    operation instead.
266
 
 
267
 
    frompath and topath are straight paths from the client. Will be checked.
268
 
    """
269
 
    if frompath is None or topath is None:
270
 
        raise ActionError("Required field missing")
271
 
    frompath = actionpath_to_local(req, frompath)
272
 
    topath = actionpath_to_local(req, topath)
273
 
    if not os.path.exists(frompath):
274
 
        raise ActionError("The source file does not exist")
275
 
    if os.path.exists(topath):
276
 
        if frompath == topath:
277
 
            raise ActionError("Source and destination are the same")
278
 
        raise ActionError("A file already exists with that name")
279
 
 
280
 
    try:
281
 
        if copy:
282
 
            svnclient.copy(frompath, topath)
283
 
        else:
284
 
            svnclient.move(frompath, topath)
285
 
    except OSError:
286
 
        raise ActionError("Could not move the file specified")
287
 
    except pysvn.ClientError:
288
 
        raise ActionError("Could not move the file specified")  
289
 
 
290
 
 
291
 
### ACTIONS ###
292
 
 
293
 
def action_delete(req, fields):
294
 
    # TODO: Disallow removal of student's home directory
295
 
    """Removes a list of files or directories.
296
 
 
297
 
    Reads fields: 'path' (multiple)
298
 
    """
299
 
    paths = fields.getlist('path')
300
 
    goterror = False
301
 
    for path in paths:
302
 
        path = actionpath_to_local(req, path)
303
 
        try:
304
 
            if os.path.isdir(path):
305
 
                shutil.rmtree(path)
306
 
            else:
307
 
                os.remove(path)
308
 
        except OSError:
309
 
            goterror = True
310
 
        except shutil.Error:
311
 
            goterror = True
312
 
    if goterror:
313
 
        if len(paths) == 1:
314
 
            raise ActionError("Could not delete the file specified")
315
 
        else:
316
 
            raise ActionError(
317
 
                "Could not delete one or more of the files specified")
318
 
 
319
 
def action_move(req, fields):
320
 
    # TODO: Do an SVN mv if the file is versioned.
321
 
    # TODO: Disallow tampering with student's home directory
322
 
    """Removes a list of files or directories.
323
 
 
324
 
    Reads fields: 'from', 'to'
325
 
    """
326
 
    frompath = fields.getfirst('from')
327
 
    topath = fields.getfirst('to')
328
 
    movefile(req, frompath, topath)
329
 
 
330
 
def action_mkdir(req, fields):
331
 
    """Creates a directory with the given path.
332
 
    Reads fields: 'path'
333
 
    """
334
 
    path = fields.getfirst('path')
335
 
    if path is None:
336
 
        raise ActionError("Required field missing")
337
 
    path = actionpath_to_local(req, path)
338
 
 
339
 
    if os.path.exists(path):
340
 
        raise ActionError("A file already exists with that name")
341
 
 
342
 
    # Create the directory
343
 
    try:
344
 
        os.mkdir(path)
345
 
    except OSError:
346
 
        raise ActionError("Could not create directory")
347
 
 
348
 
def action_putfile(req, fields):
349
 
    """Writes data to a file, overwriting it if it exists and creating it if
350
 
    it doesn't.
351
 
 
352
 
    Reads fields: 'path', 'data' (file upload), 'overwrite'
353
 
    """
354
 
    # TODO: Read field "unpack".
355
 
    # Important: Data is "None" if the file submitted is empty.
356
 
    path = fields.getfirst('path')
357
 
    data = fields.getfirst('data')
358
 
    if path is None:
359
 
        raise ActionError("Required field missing")
360
 
    if data is None:
361
 
        # Workaround - field reader treats "" as None, so this is the only
362
 
        # way to allow blank file uploads
363
 
        data = ""
364
 
    path = actionpath_to_local(req, path)
365
 
 
366
 
    if data is not None:
367
 
        data = cStringIO.StringIO(data)
368
 
 
369
 
    overwrite = fields.getfirst('overwrite')
370
 
    if overwrite is None:
371
 
        overwrite = False
372
 
    else:
373
 
        overwrite = True
374
 
 
375
 
    if overwrite:
376
 
        # Overwrite files; but can't if it's a directory
377
 
        if os.path.isdir(path):
378
 
            raise ActionError("A directory already exists "
379
 
                    + "with that name")
380
 
    else:
381
 
        if os.path.exists(path):
382
 
            raise ActionError("A file already exists with that name")
383
 
 
384
 
    # Copy the contents of file object 'data' to the path 'path'
385
 
    try:
386
 
        dest = open(path, 'wb')
387
 
        if data is not None:
388
 
            shutil.copyfileobj(data, dest)
389
 
    except (IOError, OSError), e:
390
 
        raise ActionError("Could not write to target file: %s" % e.strerror)
391
 
 
392
 
def action_putfiles(req, fields):
393
 
    """Writes data to one or more files in a directory, overwriting them if
394
 
    it they exist.
395
 
 
396
 
    Reads fields: 'path', 'data' (file upload, multiple), 'unpack'
397
 
    """
398
 
    # Important: Data is "None" if the file submitted is empty.
399
 
    path = fields.getfirst('path')
400
 
    data = fields['data']
401
 
    if type(data) != type([]):
402
 
        data = [data]
403
 
    unpack = fields.getfirst('unpack')
404
 
    if unpack is None:
405
 
        unpack = False
406
 
    else:
407
 
        unpack = True
408
 
    if path is None:
409
 
        raise ActionError("Required field missing")
410
 
    path = actionpath_to_urlpath(req, path)
411
 
    goterror = False
412
 
 
413
 
    for datum in data:
414
 
        # Each of the uploaded files
415
 
        filepath = os.path.join(path, datum.filename)
416
 
        filepath_local = studpath.to_home_path(filepath)
417
 
        if os.path.isdir(filepath_local):
418
 
            raise ActionError("A directory already exists "
419
 
                    + "with that name")
420
 
        else:
421
 
            if os.path.exists(filepath_local):
422
 
                raise ActionError("A file already exists with that name")
423
 
        filedata = datum.file
424
 
 
425
 
        if unpack and datum.filename.lower().endswith(".zip"):
426
 
            # A zip file - unpack it instead of just copying
427
 
            # TODO: Use the magic number instead of file extension
428
 
            # Note: Just unzip into the current directory (ignore the
429
 
            # filename)
430
 
            try:
431
 
                # First get the entire path (within jail)
432
 
                abspath = studpath.to_home_path(path)
433
 
                abspath = os.path.join(os.sep, abspath)
434
 
                zip.unzip(abspath, filedata)
435
 
            except (OSError, IOError):
436
 
                goterror = True
437
 
            except WillNotOverwrite, e:
438
 
                raise ActionError("File '" + e.filename + "' already exists.")
439
 
        else:
440
 
            # Not a zip file
441
 
            filepath_local = studpath.to_home_path(filepath)
442
 
            if filepath_local is None:
443
 
                raise ActionError("Invalid path")
444
 
 
445
 
            # Copy the contents of file object 'data' to the path 'path'
446
 
            try:
447
 
                dest = open(filepath_local, 'wb')
448
 
                if data is not None:
449
 
                    shutil.copyfileobj(filedata, dest)
450
 
            except (OSError, IOError):
451
 
                # TODO: Be more descriptive.
452
 
                goterror = True
453
 
 
454
 
    if goterror:
455
 
        if len(data) == 1:
456
 
            raise ActionError("Could not write to target file")
457
 
        else:
458
 
            raise ActionError(
459
 
                "Could not write to one or more of the target files")
460
 
 
461
 
def action_paste(req, fields):
462
 
    """Performs the copy or move action with the files specified.
463
 
    Copies/moves the files to the specified directory.
464
 
 
465
 
    Reads fields: 'src', 'dst', 'mode', 'file' (multiple).
466
 
    src: Base path that all the files are relative to (source).
467
 
    dst: Destination path to paste into.
468
 
    mode: 'copy' or 'move'.
469
 
    file: (Multiple) Files relative to base, which will be copied
470
 
        or moved to new locations relative to path.
471
 
    """
472
 
    errormsg = None
473
 
 
474
 
    dst = fields.getfirst('dst')
475
 
    src = fields.getfirst('src')
476
 
    mode = fields.getfirst('mode')
477
 
    files = fields.getlist('file')
478
 
    if dst is None or src is None or mode is None:
479
 
        raise ActionError("Required field missing")
480
 
 
481
 
    dst_local = actionpath_to_local(req, dst)
482
 
    if not os.path.isdir(dst_local):
483
 
        raise ActionError("dst is not a directory")
484
 
 
485
 
    errorfiles = []
486
 
    for file in files:
487
 
        # The source must not be interpreted as relative to req.path
488
 
        # Add a slash (relative to top-level)
489
 
        if src[:1] != '/':
490
 
            src = '/' + src
491
 
        frompath = os.path.join(src, file)
492
 
        # The destination is found by taking just the basename of the file
493
 
        topath = os.path.join(dst, os.path.basename(file))
494
 
        try:
495
 
            if mode == "copy":
496
 
                movefile(req, frompath, topath, True)
497
 
            elif mode == "move":
498
 
                movefile(req, frompath, topath, False)
499
 
            elif mode == "svncopy":
500
 
                svn_movefile(req, frompath, topath, True)
501
 
            elif mode == "svnmove":
502
 
                svn_movefile(req, frompath, topath, False)
503
 
            else:
504
 
                raise ActionError("Invalid mode (must be '(svn)copy' or '(svn)move')")
505
 
        except ActionError, message:
506
 
            # Store the error for later; we want to copy as many as possible
507
 
            if errormsg is None:
508
 
                errormsg = message
509
 
            else:
510
 
                # Multiple errors; generic message
511
 
                errormsg = "One or more files could not be pasted"
512
 
            # Add this file to errorfiles; it will be put back on the
513
 
            # clipboard for possible future pasting.
514
 
            errorfiles.append(file)
515
 
    if errormsg is not None:
516
 
        raise ActionError(errormsg)
517
 
 
518
 
    # XXX errorfiles contains a list of files that couldn't be pasted.
519
 
    # we currently do nothing with this.
520
 
 
521
 
def action_publish(req,fields):
522
 
    """Marks the folder as published by adding a '.published' file to the 
523
 
    directory and ensuring that the parent directory permissions are correct
524
 
 
525
 
    Reads fields: 'path'
526
 
    """
527
 
    paths = fields.getlist('path')
528
 
    user = util.split_path(req.path)[0]
529
 
    homedir = "/home/%s" % user
530
 
    if len(paths):
531
 
        paths = map(lambda path: actionpath_to_local(req, path), paths)
532
 
    else:
533
 
        paths = [studpath.to_home_path(req.path)]
534
 
 
535
 
    # Set all the dirs in home dir world browsable (o+r,o+x)
536
 
    #FIXME: Should really only do those in the direct path not all of the 
537
 
    # folders in a students home directory
538
 
    for root,dirs,files in os.walk(homedir):
539
 
        os.chmod(root, os.stat(root).st_mode|0005)
540
 
 
541
 
    try:
542
 
        for path in paths:
543
 
            if os.path.isdir(path):
544
 
                pubfile = open(os.path.join(path,'.published'),'w')
545
 
                pubfile.write("This directory is published\n")
546
 
                pubfile.close()
547
 
            else:
548
 
                raise ActionError("Can only publish directories")
549
 
    except OSError, e:
550
 
        raise ActionError("Directory could not be published")
551
 
 
552
 
def action_unpublish(req,fields):
553
 
    """Marks the folder as unpublished by removing a '.published' file in the 
554
 
    directory (if it exits). It does not change the permissions of the parent 
555
 
    directories.
556
 
 
557
 
    Reads fields: 'path'
558
 
    """
559
 
    paths = fields.getlist('path')
560
 
    if len(paths):
561
 
        paths = map(lambda path: actionpath_to_local(req, path), paths)
562
 
    else:
563
 
        paths = [studpath.to_home_path(req.path)]
564
 
 
565
 
    try:
566
 
        for path in paths:
567
 
            if os.path.isdir(path):
568
 
                pubfile = os.path.join(path,'.published')
569
 
                if os.path.isfile(pubfile):
570
 
                    os.remove(pubfile)
571
 
            else:
572
 
                raise ActionError("Can only unpublish directories")
573
 
    except OSError, e:
574
 
        raise ActionError("Directory could not be unpublished")
575
 
 
576
 
 
577
 
def action_svnadd(req, fields):
578
 
    """Performs a "svn add" to each file specified.
579
 
 
580
 
    Reads fields: 'path' (multiple)
581
 
    """
582
 
    paths = fields.getlist('path')
583
 
    paths = map(lambda path: actionpath_to_local(req, path), paths)
584
 
 
585
 
    try:
586
 
        svnclient.add(paths, recurse=True, force=True)
587
 
    except pysvn.ClientError, e:
588
 
        raise ActionError(str(e))
589
 
 
590
 
def action_svnremove(req, fields):
591
 
    """Performs a "svn remove" on each file specified.
592
 
 
593
 
    Reads fields: 'path' (multiple)
594
 
    """
595
 
    paths = fields.getlist('path')
596
 
    paths = map(lambda path: actionpath_to_local(req, path), paths)
597
 
 
598
 
    try:
599
 
        svnclient.remove(paths, force=True)
600
 
    except pysvn.ClientError, e:
601
 
        raise ActionError(str(e))
602
 
 
603
 
def action_svnupdate(req, fields):
604
 
    """Performs a "svn update" to each file specified.
605
 
 
606
 
    Reads fields: 'path' and 'revision'
607
 
    """
608
 
    path = fields.getfirst('path')
609
 
    revision = fields.getfirst('revision')
610
 
    if path is None:
611
 
        raise ActionError("Required field missing")
612
 
    if revision is None:
613
 
        revision = pysvn.Revision( pysvn.opt_revision_kind.head )
614
 
    else:
615
 
        try:
616
 
            revision = pysvn.Revision(pysvn.opt_revision_kind.number,
617
 
                    int(revision))
618
 
        except ValueError, e:
619
 
            raise ActionError("Bad revision number: '%s'"%revision,)
620
 
    path = actionpath_to_local(req, path)
621
 
 
622
 
    try:
623
 
        svnclient.update(path, recurse=True, revision=revision)
624
 
    except pysvn.ClientError, e:
625
 
        raise ActionError(str(e))
626
 
 
627
 
def action_svnresolved(req, fields):
628
 
    """Performs a "svn resolved" to each file specified.
629
 
 
630
 
    Reads fields: 'path'
631
 
    """
632
 
    path = fields.getfirst('path')
633
 
    if path is None:
634
 
        raise ActionError("Required field missing")
635
 
    path = actionpath_to_local(req, path)
636
 
 
637
 
    try:
638
 
        svnclient.resolved(path, recurse=True)
639
 
    except pysvn.ClientError, e:
640
 
        raise ActionError(str(e))
641
 
 
642
 
def action_svnrevert(req, fields):
643
 
    """Performs a "svn revert" to each file specified.
644
 
 
645
 
    Reads fields: 'path' (multiple)
646
 
    """
647
 
    paths = fields.getlist('path')
648
 
    paths = map(lambda path: actionpath_to_local(req, path), paths)
649
 
 
650
 
    try:
651
 
        svnclient.revert(paths, recurse=True)
652
 
    except pysvn.ClientError, e:
653
 
        raise ActionError(str(e))
654
 
 
655
 
def action_svnpublish(req, fields):
656
 
    """Sets svn property "ivle:published" on each file specified.
657
 
    Should only be called on directories (only effective on directories
658
 
    anyway).
659
 
 
660
 
    Reads fields: 'path'
661
 
 
662
 
    XXX Currently unused by the client (calls action_publish instead, which
663
 
    has a completely different publishing model).
664
 
    """
665
 
    paths = fields.getlist('path')
666
 
    if len(paths):
667
 
        paths = map(lambda path: actionpath_to_local(req, path), paths)
668
 
    else:
669
 
        paths = [studpath.to_home_path(req.path)]
670
 
 
671
 
    try:
672
 
        for path in paths:
673
 
            # Note: Property value doesn't matter
674
 
            svnclient.propset("ivle:published", "", path, recurse=False)
675
 
    except pysvn.ClientError, e:
676
 
        raise ActionError("Directory could not be published")
677
 
 
678
 
def action_svnunpublish(req, fields):
679
 
    """Deletes svn property "ivle:published" on each file specified.
680
 
 
681
 
    Reads fields: 'path'
682
 
 
683
 
    XXX Currently unused by the client (calls action_unpublish instead, which
684
 
    has a completely different publishing model).
685
 
    """
686
 
    paths = fields.getlist('path')
687
 
    paths = map(lambda path: actionpath_to_local(req, path), paths)
688
 
 
689
 
    try:
690
 
        for path in paths:
691
 
            svnclient.propdel("ivle:published", path, recurse=False)
692
 
    except pysvn.ClientError, e:
693
 
        raise ActionError("Directory could not be unpublished")
694
 
 
695
 
def action_svncommit(req, fields):
696
 
    """Performs a "svn commit" to each file specified.
697
 
 
698
 
    Reads fields: 'path' (multiple), 'logmsg' (optional)
699
 
    """
700
 
    paths = fields.getlist('path')
701
 
    paths = map(lambda path: actionpath_to_local(req, str(path)), paths)
702
 
    logmsg = str(fields.getfirst('logmsg', DEFAULT_LOGMESSAGE))
703
 
    if logmsg == '': logmsg = DEFAULT_LOGMESSAGE
704
 
 
705
 
    try:
706
 
        svnclient.checkin(paths, logmsg, recurse=True)
707
 
    except pysvn.ClientError, e:
708
 
        raise ActionError(str(e))
709
 
 
710
 
def action_svncheckout(req, fields):
711
 
    """Performs a "svn checkout" of the first path into the second path.
712
 
 
713
 
    Reads fields: 'path'    (multiple)
714
 
    """
715
 
    paths = fields.getlist('path')
716
 
    if len(paths) != 2:
717
 
        raise ActionError("usage: svncheckout url local-path")
718
 
    url = ivle.conf.svn_addr + "/" + urllib.quote(paths[0])
719
 
    local_path = actionpath_to_local(req, str(paths[1]))
720
 
    try:
721
 
        svnclient.checkout(url, local_path, recurse=True)
722
 
    except pysvn.ClientError, e:
723
 
        raise ActionError(str(e))
724
 
 
725
 
def action_svnrepomkdir(req, fields):
726
 
    """Performs a "svn mkdir" on a path under the IVLE SVN root.
727
 
 
728
 
    Reads fields: 'path'
729
 
    """
730
 
    path = fields.getfirst('path')
731
 
    logmsg = fields.getfirst('logmsg')
732
 
    url = ivle.conf.svn_addr + "/" + urllib.quote(path)
733
 
    try:
734
 
        svnclient.mkdir(url, log_message=logmsg)
735
 
    except pysvn.ClientError, e:
736
 
        raise ActionError(str(e))
737
 
 
738
 
def action_svnrepostat(req, fields):
739
 
    """Discovers whether a path exists in a repo under the IVLE SVN root.
740
 
 
741
 
    If it does exist, returns a dict containing its metadata.
742
 
 
743
 
    Reads fields: 'path'
744
 
    """
745
 
    path = fields.getfirst('path')
746
 
    url = ivle.conf.svn_addr + "/" + urllib.quote(path)
747
 
    svnclient.exception_style = 1
748
 
 
749
 
    try:
750
 
        info = svnclient.info2(url,
751
 
            revision=pysvn.Revision(pysvn.opt_revision_kind.head))[0][1]
752
 
        return {'svnrevision': info['rev'].number
753
 
                  if info['rev'] and
754
 
                     info['rev'].kind == pysvn.opt_revision_kind.number
755
 
                  else None}
756
 
    except pysvn.ClientError, e:
757
 
        # Error code 170000 means ENOENT in this revision.
758
 
        if e[1][0][1] == 170000:
759
 
            raise util.IVLEError(404, 'The specified repository path does not exist')
760
 
        else:
761
 
            raise ActionError(str(e[0]))
762
 
 
763
 
 
764
 
def action_svncleanup(req, fields):
765
 
    """Recursively clean up the working copy, removing locks, resuming 
766
 
    unfinished operations, etc.
767
 
        path:   The path to be cleaned"""
768
 
 
769
 
    path = fields.getfirst('path')
770
 
    if path is None:
771
 
        raise ActionError("Required field missing")
772
 
    path = actionpath_to_local(req, path)
773
 
 
774
 
    try:
775
 
        svnclient.cleanup(path)
776
 
    except pysvn.ClientError, e:
777
 
        raise ActionError(str(e))
778
 
 
779
 
 
780
 
# Table of all action functions #
781
 
# Each function has the interface f(req, fields).
782
 
 
783
 
actions_table = {
784
 
    "delete" : action_delete,
785
 
    "move" : action_move,
786
 
    "mkdir" : action_mkdir,
787
 
    "putfile" : action_putfile,
788
 
    "putfiles" : action_putfiles,
789
 
    "paste" : action_paste,
790
 
    "publish" : action_publish,
791
 
    "unpublish" : action_unpublish,
792
 
 
793
 
    "svnadd" : action_svnadd,
794
 
    "svnremove" : action_svnremove,
795
 
    "svnupdate" : action_svnupdate,
796
 
    "svnresolved" : action_svnresolved,
797
 
    "svnrevert" : action_svnrevert,
798
 
    "svnpublish" : action_svnpublish,
799
 
    "svnunpublish" : action_svnunpublish,
800
 
    "svncommit" : action_svncommit,
801
 
    "svncheckout" : action_svncheckout,
802
 
    "svnrepomkdir" : action_svnrepomkdir,
803
 
    "svnrepostat" : action_svnrepostat,
804
 
    "svncleanup" : action_svncleanup,
805
 
}