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

« back to all changes in this revision

Viewing changes to ivle/webapp/security/views.py

  • Committer: William Grant
  • Date: 2009-02-27 03:24:23 UTC
  • Revision ID: grantw@unimelb.edu.au-20090227032423-4d2x815i6adzkskg
Unlock the session everywhere as soon as we are done with it, and add a warning
to the Request.get_sessions()'s docstring to do so.

Otherwise, all other requests block.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
# IVLE
 
2
# Copyright (C) 2007-2009 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
# Author: Will Grant, Nick Chadwick
 
19
 
 
20
import urllib
 
21
import datetime
 
22
try:
 
23
    import mod_python.Cookie
 
24
except ImportError:
 
25
    # This needs to be importable from outside Apache.
 
26
    pass
 
27
 
 
28
import ivle.util
 
29
import ivle.webapp.security
 
30
from ivle.auth import authenticate, AuthError
 
31
from ivle.webapp.base.xhtml import XHTMLView
 
32
from ivle.webapp.base.plugins import CookiePlugin
 
33
 
 
34
class LoginView(XHTMLView):
 
35
    '''A view to allow a user to log in.'''
 
36
    template = 'login.html'
 
37
    allow_overlays = False
 
38
 
 
39
    def authorize(self, req):
 
40
        return True
 
41
 
 
42
    def populate(self, req, ctx):
 
43
        fields = req.get_fieldstorage()
 
44
        nexturl = fields.getfirst('url')
 
45
 
 
46
        if nexturl is None:
 
47
            nexturl = '/'
 
48
 
 
49
        # We are already logged in. Don't bother logging in again.
 
50
        # Note that req.user is None even if we are 'logged in', if the user is
 
51
        # invalid.
 
52
        if req.user is not None:
 
53
            req.throw_redirect(nexturl)
 
54
 
 
55
        # Don't give any URL if we want /.
 
56
        if nexturl == '/':
 
57
            query_string = ''
 
58
        else:
 
59
            query_string = '?url=' + urllib.quote(nexturl, safe="/~")
 
60
 
 
61
        ctx['path'] = ivle.util.make_path('+login') + query_string
 
62
 
 
63
        # If this succeeds, the user is invalid.
 
64
        user = ivle.webapp.security.get_user_details(req)
 
65
        if user is not None:
 
66
            if user.state == "no_agreement":
 
67
                # Authenticated, but need to accept the ToS. Send them there.
 
68
                # IMPORTANT NOTE FOR HACKERS: You can't simply disable this
 
69
                # if you are not planning to display a ToS page - the ToS
 
70
                # acceptance process actually calls usrmgt to create the user
 
71
                # jails and related stuff.
 
72
                req.throw_redirect(ivle.util.make_path('+tos') + query_string)
 
73
            elif user.state == "pending":
 
74
                # FIXME: this isn't quite the right answer, but it
 
75
                # should be more robust in the short term.
 
76
                session = req.get_session()
 
77
                session.invalidate()
 
78
                session.delete()
 
79
                user.state = u'no_agreement'
 
80
                req.store.commit()
 
81
                req.throw_redirect(nexturl)
 
82
 
 
83
        if req.method == "POST":
 
84
            # While req.user is normally set to get_user_details, it won't set
 
85
            # it if the account isn't valid. So we get it ourselves.
 
86
            user = ivle.webapp.security.get_user_details(req)
 
87
 
 
88
            badlogin = None
 
89
 
 
90
            username = fields.getfirst('user')
 
91
            password = fields.getfirst('pass')
 
92
            if username is not None:
 
93
                # From this point onwards, we will be showing an error message
 
94
                # if unsuccessful.
 
95
                # Authenticate
 
96
                if password is None:
 
97
                    badlogin = "No password supplied."
 
98
                else:
 
99
                    user = None
 
100
                    try:
 
101
                        user = authenticate.authenticate(req.store,
 
102
                                    username.value, password.value)
 
103
                    except AuthError, msg:
 
104
                        badlogin = msg
 
105
                    if user is None:
 
106
                        # Must have got an error. Do not authenticate.
 
107
                        # The except: above will have set a message.
 
108
                        pass
 
109
                    else:
 
110
                        # Success - Set the session and redirect to the URL.
 
111
                        session = req.get_session()
 
112
                        session['login'] = user.login
 
113
                        session.save()
 
114
                        session.unlock()
 
115
                        user.last_login = datetime.datetime.now()
 
116
                        req.store.commit()
 
117
 
 
118
                        # Create cookies for plugins that might request them.
 
119
                        for plugin in req.config.plugin_index[CookiePlugin]:
 
120
                            for cookie in plugin.cookies:
 
121
                                # The function can be None if they just need to be
 
122
                                # deleted at logout.
 
123
                                if plugin.cookies[cookie] is not None:
 
124
                                    req.add_cookie(mod_python.Cookie.Cookie(cookie,
 
125
                                          plugin.cookies[cookie](user), path='/'))
 
126
 
 
127
                        req.throw_redirect(nexturl)
 
128
 
 
129
                # We didn't succeed.
 
130
                # Render the login form with the error message.
 
131
                ctx['error'] = badlogin
 
132
 
 
133
 
 
134
class LogoutView(XHTMLView):
 
135
    '''A view to log the current session out.'''
 
136
    template = 'logout.html'
 
137
    allow_overlays = False
 
138
 
 
139
    def authorize(self, req):
 
140
        # This can be used by any authenticated user, even if they haven't
 
141
        # accepted the ToS yet.
 
142
        return ivle.webapp.security.get_user_details(req) is not None
 
143
 
 
144
    def populate(self, req, ctx):
 
145
        if req.method == "POST":
 
146
            req.logout()
 
147
        else:
 
148
            ctx['path'] =  ivle.util.make_path('+logout')