~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-12-08 03:50:24 UTC
  • mfrom: (1294.2.143 ui-the-third)
  • Revision ID: grantw@unimelb.edu.au-20091208035024-wjx8zp54gth15ph8
Merge ui-the-third. This is another UI revamp.

The header is now thin! Thin! The yellow bar is gone. The tabs are gone.
Breadcrumbs are here. Routes is replaced (with an object publishing
mechanism). Views are less repetitive. etc.

Show diffs side-by-side

added added

removed removed

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