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

« back to all changes in this revision

Viewing changes to ivle/webapp/admin/subject.py

  • Committer: William Grant
  • Date: 2009-07-05 03:18:14 UTC
  • mto: (1294.4.2 ui-the-third)
  • mto: This revision was merged to the branch mainline in revision 1353.
  • Revision ID: grantw@unimelb.edu.au-20090705031814-yn8oymmawsq0da78
Split out ivle.webapp.admin's routes into annotated functions in ivle.webapp.traversal.

Show diffs side-by-side

added added

removed removed

Lines of Context:
23
23
# A sample / testing application for IVLE.
24
24
 
25
25
import os
 
26
import os.path
26
27
import urllib
 
28
import urlparse
27
29
import cgi
28
30
 
29
 
from common import util
30
 
import common.db
31
 
 
32
 
def handle(req):
33
 
    """Handler for the Subjects application. Links to subject home pages."""
34
 
 
35
 
    req.styles = ["media/subjects/subjects.css"]
36
 
    if req.path == "":
37
 
        handle_toplevel_menu(req)
38
 
    else:
39
 
        handle_subject_page(req, req.path)
40
 
 
41
 
def handle_toplevel_menu(req):
42
 
    # This is represented as a directory. Redirect and add a slash if it is
43
 
    # missing.
44
 
    if req.uri[-1] != '/':
45
 
        req.throw_redirect(req.uri + '/')
46
 
 
47
 
    (enrolled_subjects, unenrolled_subjects) = \
48
 
              common.db.DB().get_subjects_status(req.user.login)
49
 
 
50
 
    def print_subject(subject):
51
 
        if subject['url'] is None:
52
 
            req.write('  <li>%s (no home page)</li>\n'
53
 
                % cgi.escape(subject['subj_name']))
54
 
        else:
55
 
            req.write('  <li><a href="%s">%s</a></li>\n'
56
 
                % (cgi.escape(subject['url']),
57
 
                   cgi.escape(subject['subj_name'])))
58
 
 
59
 
    req.content_type = "text/html"
60
 
    req.write_html_head_foot = True
61
 
    req.write('<div id="ivle_padding">\n')
62
 
    req.write("<h2>IVLE Subject Homepages</h2>\n")
63
 
    req.write("<h2>Subjects</h2>\n<ul>\n")
64
 
    for subject in enrolled_subjects:
65
 
        print_subject(subject)
66
 
    req.write("</ul>\n")
67
 
    if len(unenrolled_subjects) > 0:
68
 
        req.write("<h3>Other Subjects</h3>\n")
69
 
        req.write("<p>You are not currently enrolled in these subjects</p>\n")
70
 
        req.write("<ul>\n")
71
 
        for subject in unenrolled_subjects:
72
 
            print_subject(subject)
73
 
        req.write("</ul>\n")
74
 
    req.write("</div>\n")
75
 
 
76
 
def handle_subject_page(req, path):
77
 
    req.content_type = "text/html"
78
 
    req.write_html_head_foot = True     # Have dispatch print head and foot
79
 
 
80
 
    # Just make the iframe pointing to media/subjects
81
 
    serve_loc = util.make_path(os.path.join('media', 'subjects', path))
82
 
    req.write('<object class="fullscreen" type="text/html" \
83
 
data="%s"></iframe>'% urllib.quote(serve_loc))
 
31
from storm.locals import Desc, Store
 
32
import genshi
 
33
from genshi.filters import HTMLFormFiller
 
34
from genshi.template import Context, TemplateLoader
 
35
import formencode
 
36
 
 
37
from ivle.webapp.base.xhtml import XHTMLView
 
38
from ivle.webapp.base.plugins import ViewPlugin, MediaPlugin
 
39
from ivle.webapp import ApplicationRoot
 
40
 
 
41
from ivle.database import Subject, Semester, Offering, Enrolment, User,\
 
42
                          ProjectSet, Project, ProjectSubmission
 
43
from ivle import util
 
44
import ivle.date
 
45
 
 
46
from ivle.webapp.admin.projectservice import ProjectSetRESTView,\
 
47
                                             ProjectRESTView
 
48
from ivle.webapp.admin.offeringservice import OfferingRESTView
 
49
from ivle.webapp.admin.traversal import (root_to_subject,
 
50
            subject_to_offering, offering_to_projectset, offering_to_project,
 
51
            subject_url, offering_url, projectset_url, project_url)
 
52
 
 
53
class SubjectsView(XHTMLView):
 
54
    '''The view of the list of subjects.'''
 
55
    template = 'templates/subjects.html'
 
56
    tab = 'subjects'
 
57
 
 
58
    def authorize(self, req):
 
59
        return req.user is not None
 
60
 
 
61
    def populate(self, req, ctx):
 
62
        ctx['user'] = req.user
 
63
        ctx['semesters'] = []
 
64
        for semester in req.store.find(Semester).order_by(Desc(Semester.year),
 
65
                                                     Desc(Semester.semester)):
 
66
            enrolments = semester.enrolments.find(user=req.user)
 
67
            if enrolments.count():
 
68
                ctx['semesters'].append((semester, enrolments))
 
69
 
 
70
 
 
71
class UserValidator(formencode.FancyValidator):
 
72
    """A FormEncode validator that turns a username into a user.
 
73
 
 
74
    The state must have a 'store' attribute, which is the Storm store
 
75
    to use."""
 
76
    def _to_python(self, value, state):
 
77
        user = User.get_by_login(state.store, value)
 
78
        if user:
 
79
            return user
 
80
        else:
 
81
            raise formencode.Invalid('User does not exist', value, state)
 
82
 
 
83
 
 
84
class NoEnrolmentValidator(formencode.FancyValidator):
 
85
    """A FormEncode validator that ensures absence of an enrolment.
 
86
 
 
87
    The state must have an 'offering' attribute.
 
88
    """
 
89
    def _to_python(self, value, state):
 
90
        if state.offering.get_enrolment(value):
 
91
            raise formencode.Invalid('User already enrolled', value, state)
 
92
        return value
 
93
 
 
94
 
 
95
class EnrolSchema(formencode.Schema):
 
96
    user = formencode.All(NoEnrolmentValidator(), UserValidator())
 
97
 
 
98
 
 
99
class EnrolView(XHTMLView):
 
100
    """A form to enrol a user in an offering."""
 
101
    template = 'templates/enrol.html'
 
102
    tab = 'subjects'
 
103
    permission = 'edit'
 
104
 
 
105
    def filter(self, stream, ctx):
 
106
        return stream | HTMLFormFiller(data=ctx['data'])
 
107
 
 
108
    def populate(self, req, ctx):
 
109
        if req.method == 'POST':
 
110
            data = dict(req.get_fieldstorage())
 
111
            try:
 
112
                validator = EnrolSchema()
 
113
                req.offering = self.context # XXX: Getting into state.
 
114
                data = validator.to_python(data, state=req)
 
115
                self.context.enrol(data['user'])
 
116
                req.store.commit()
 
117
                req.throw_redirect(req.uri)
 
118
            except formencode.Invalid, e:
 
119
                errors = e.unpack_errors()
 
120
        else:
 
121
            data = {}
 
122
            errors = {}
 
123
 
 
124
        ctx['data'] = data or {}
 
125
        ctx['offering'] = self.context
 
126
        ctx['errors'] = errors
 
127
 
 
128
class OfferingProjectsView(XHTMLView):
 
129
    """View the projects for an offering."""
 
130
    template = 'templates/offering_projects.html'
 
131
    permission = 'edit'
 
132
    tab = 'subjects'
 
133
 
 
134
    def project_url(self, projectset, project):
 
135
        return "/subjects/%s/%s/%s/+projects/%s" % (
 
136
                    self.context.subject.short_name,
 
137
                    self.context.semester.year,
 
138
                    self.context.semester.semester,
 
139
                    project.short_name
 
140
                    )
 
141
 
 
142
    def new_project_url(self, projectset):
 
143
        return "/api/subjects/" + self.context.subject.short_name + "/" +\
 
144
                self.context.semester.year + "/" + \
 
145
                self.context.semester.semester + "/+projectsets/" +\
 
146
                str(projectset.id) + "/+projects/+new"
 
147
    
 
148
    def populate(self, req, ctx):
 
149
        self.plugin_styles[Plugin] = ["project.css"]
 
150
        self.plugin_scripts[Plugin] = ["project.js"]
 
151
        ctx['offering'] = self.context
 
152
        ctx['projectsets'] = []
 
153
 
 
154
        #Open the projectset Fragment, and render it for inclusion
 
155
        #into the ProjectSets page
 
156
        #XXX: This could be a lot cleaner
 
157
        loader = genshi.template.TemplateLoader(".", auto_reload=True)
 
158
 
 
159
        set_fragment = os.path.join(os.path.dirname(__file__),
 
160
                "templates/projectset_fragment.html")
 
161
        project_fragment = os.path.join(os.path.dirname(__file__),
 
162
                "templates/project_fragment.html")
 
163
 
 
164
        for projectset in self.context.project_sets:
 
165
            settmpl = loader.load(set_fragment)
 
166
            setCtx = Context()
 
167
            setCtx['projectset'] = projectset
 
168
            setCtx['new_project_url'] = self.new_project_url(projectset)
 
169
            setCtx['projects'] = []
 
170
 
 
171
            for project in projectset.projects:
 
172
                projecttmpl = loader.load(project_fragment)
 
173
                projectCtx = Context()
 
174
                projectCtx['project'] = project
 
175
                projectCtx['project_url'] = self.project_url(projectset, project)
 
176
 
 
177
                setCtx['projects'].append(
 
178
                        projecttmpl.generate(projectCtx))
 
179
 
 
180
            ctx['projectsets'].append(settmpl.generate(setCtx))
 
181
 
 
182
 
 
183
class ProjectView(XHTMLView):
 
184
    """View the submissions for a ProjectSet"""
 
185
    template = "templates/project.html"
 
186
    permission = "edit"
 
187
    tab = 'subjects'
 
188
 
 
189
    def build_subversion_url(self, svnroot, submission):
 
190
        princ = submission.assessed.principal
 
191
 
 
192
        if isinstance(princ, User):
 
193
            path = 'users/%s' % princ.login
 
194
        else:
 
195
            path = 'groups/%s_%s_%s_%s' % (
 
196
                    princ.project_set.offering.subject.short_name,
 
197
                    princ.project_set.offering.semester.year,
 
198
                    princ.project_set.offering.semester.semester,
 
199
                    princ.name
 
200
                    )
 
201
        return urlparse.urljoin(
 
202
                    svnroot,
 
203
                    os.path.join(path, submission.path[1:] if
 
204
                                       submission.path.startswith(os.sep) else
 
205
                                       submission.path))
 
206
 
 
207
    def populate(self, req, ctx):
 
208
        self.plugin_styles[Plugin] = ["project.css"]
 
209
 
 
210
        ctx['format_datetime_short'] = ivle.date.format_datetime_for_paragraph
 
211
        ctx['build_subversion_url'] = self.build_subversion_url
 
212
        ctx['svn_addr'] = req.config['urls']['svn_addr']
 
213
        ctx['project'] = self.context
 
214
        ctx['user'] = req.user
 
215
 
 
216
class OfferingEnrolmentSet(object):
 
217
    def __init__(self, offering):
 
218
        self.offering = offering
 
219
 
 
220
class Plugin(ViewPlugin, MediaPlugin):
 
221
    forward_routes = (root_to_subject, subject_to_offering,
 
222
                      offering_to_project, offering_to_projectset)
 
223
    reverse_routes = (subject_url, offering_url, projectset_url, project_url)
 
224
 
 
225
    views = [(ApplicationRoot, ('subjects', '+index'), SubjectsView),
 
226
             (Offering, ('+enrolments', '+new'), EnrolView),
 
227
             (Offering, ('+projects', '+index'), OfferingProjectsView),
 
228
             (Project, '+index', ProjectView),
 
229
 
 
230
             (Offering, ('+projectsets', '+new'), OfferingRESTView, 'api'),
 
231
             (ProjectSet, ('+projects', '+new'), ProjectSetRESTView, 'api'),
 
232
             (Project, '+index', ProjectRESTView, 'api'),
 
233
             ]
 
234
 
 
235
    tabs = [
 
236
        ('subjects', 'Subjects',
 
237
         'View subject content and complete worksheets',
 
238
         'subjects.png', 'subjects', 5)
 
239
    ]
 
240
 
 
241
    media = 'subject-media'