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

« back to all changes in this revision

Viewing changes to ivle/webapp/filesystem/diff/__init__.py

Merged from new-dispatch branch.
This branch is now a child of new-dispatch (until that branch is merged with
    trunk).

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# IVLE
2
 
# Copyright (C) 2007-2008 The University of Melbourne
 
1
# IVLE - Informatics Virtual Learning Environment
 
2
# Copyright (C) 2007-2009 The University of Melbourne
3
3
#
4
4
# This program is free software; you can redistribute it and/or modify
5
5
# it under the terms of the GNU General Public License as published by
15
15
# along with this program; if not, write to the Free Software
16
16
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
17
17
 
18
 
# App: Diff Service
19
 
# Author: David Coles
20
 
# Date: 21/2/2008
21
 
 
22
 
# This app is a wrapper around the diff script run through the trampoline
23
 
 
24
 
import os.path
 
18
# Author: David Coles, Will Grant
 
19
 
 
20
'''Components of the webapp for diffing user files.'''
 
21
 
 
22
import os
 
23
import re
 
24
import cgi
 
25
 
 
26
import cjson
 
27
import genshi
25
28
 
26
29
import ivle.conf
27
30
import ivle.interpret
28
 
 
29
 
 
30
 
# handle_with_trampoline controls the way in which fileservice_lib is invoked.
31
 
# If False, it will simply be called directly by this handler.
32
 
# If True, the request will get marshalled into a CGI environment and the
33
 
# trampoline will invoke services/fileservices within the user's jail (SetUID'd
34
 
# to them). This script will then wrap the CGI environment in a replica of the
35
 
# original environment and handle it that way.
36
 
# This is a lot of overhead but it's the only way to properly ensure we are
37
 
# acting "as" that user and therefore we don't run into permissions problems.
38
 
# If set to True, it will be a lot more efficient, but there will be
39
 
# permissions issues unless all user's files are owned by the web server user.
40
 
HANDLE_WITH_TRAMPOLINE = True
41
 
 
42
 
diffservice_path = os.path.join(ivle.conf.share_path, 'services/diffservice')
43
 
 
44
 
def handle(req):
45
 
    """Handler for the File Services application."""
46
 
    req.styles = ["media/diff/diff.css"] # CSS styles
47
 
    req.write_html_head_foot = True     # Have dispatch print head and foot
48
 
 
49
 
    # Handle
50
 
    if not HANDLE_WITH_TRAMPOLINE:
51
 
        pass
52
 
    else:
53
 
        if req.path == "":
54
 
            req.throw_redirect(os.path.join(req.uri,req.user.login));
55
 
        interp_object = ivle.interpret.interpreter_objects["cgi-python"]
56
 
        user_jail_dir = os.path.join(ivle.conf.jail_base, req.user.login)
57
 
        ivle.interpret.interpret_file(req, req.user, user_jail_dir,
58
 
            diffservice_path, interp_object)
59
 
 
 
31
from ivle.webapp.base.xhtml import XHTMLView
 
32
from ivle.webapp.base.plugins import ViewPlugin, MediaPlugin
 
33
from ivle.webapp.errors import NotFound, BadRequest
 
34
 
 
35
class DiffView(XHTMLView):
 
36
    '''A view to present a nice XHTML Subversion diff from a user's jail.'''
 
37
    template = 'template.html'
 
38
 
 
39
    def __init__(self, req, path):
 
40
        self.path = path
 
41
 
 
42
    def populate(self, req, ctx):
 
43
        self.plugin_styles[Plugin] = ['diff.css']
 
44
 
 
45
        revfields = req.get_fieldstorage().getlist("r")
 
46
        if len(revfields) > 2:
 
47
            raise BadRequest('A maximum of two revisions can be given.')
 
48
 
 
49
        revs = [revfield.value for revfield in revfields]
 
50
 
 
51
        jail_dir = os.path.join(ivle.conf.jail_base, req.user.login)
 
52
        (out, err) = ivle.interpret.execute_raw(req.user, jail_dir, '/home',
 
53
                    os.path.join(ivle.conf.share_path, 'services/diffservice'),
 
54
                    [self.path] + revs)
 
55
        assert not err
 
56
 
 
57
        response = cjson.decode(out)
 
58
        if 'error' in response:
 
59
            if response['error'] == 'notfound':
 
60
                raise NotFound()
 
61
            else:
 
62
                raise AssertionError('Unknown error from diffservice: %s' %
 
63
                                     response['error'])
 
64
 
 
65
        # No error. We must be safe.
 
66
        diff = response['diff']
 
67
 
 
68
        # Split up the udiff into individual files
 
69
        diff_matcher = re.compile(
 
70
            r'^Index: (.*)\n\=+\n((?:[^I].*\n)*)',re.MULTILINE
 
71
        )
 
72
 
 
73
        # Create a dict with (name, HTMLdiff) pairs for each non-empty diff.
 
74
        ctx['files'] = dict([(fd[0], genshi.XML(htmlfy_diff(fd[1])))
 
75
                             for fd in diff_matcher.findall(diff)
 
76
                             if fd[1]])
 
77
 
 
78
 
 
79
def htmlfy_diff(difftext):
 
80
    """Adds HTML markup to a udiff string"""
 
81
    output = cgi.escape(difftext)
 
82
    subs = {
 
83
     r'^([\+\-]{3})\s(\S+)\s\((.+)\)$':
 
84
         r'<span class="diff-files">\1 \2 <em>(\3)</em></span>',
 
85
     r'^\@\@ (.*) \@\@$':
 
86
         r'<span class="diff-range">@@ \1 @@</span>',
 
87
     r'^\+(.*)$':
 
88
         r'<span class="diff-add">+\1</span>',
 
89
     r'^\-(.*)$':
 
90
         r'<span class="diff-sub">-\1</span>',
 
91
     r'^\\(.*)$':
 
92
         r'<span class="diff-special">\\\1</span>'
 
93
    }
 
94
 
 
95
    for match in subs:
 
96
        output = re.compile(match, re.MULTILINE).sub(subs[match], output)
 
97
 
 
98
    return '<pre class="diff">%s</pre>' % output
 
99
 
 
100
class Plugin(ViewPlugin, MediaPlugin):
 
101
    '''Registration class for diff components.'''
 
102
    urls = [
 
103
        ('diff/', DiffView, {'path': ''}),
 
104
        ('diff/*(path)', DiffView),
 
105
    ]
 
106
 
 
107
    media = 'media'