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

« back to all changes in this revision

Viewing changes to ivle/webapp/base/forms.py

[Uber-commit of holiday work because I lacked a local copy of the branch.]

 ivle.makeuser: Don't use jailconf.py as a header for the in-jail conf.py;
     generate the whole thing using string formatting operators and include
     the template inline.

 ivle.makeuser.make_conf_py: XXX the inclusion of ivle.conf.jail_base in
     the jail. It is simply there to placate ivle.studpath, and needs
     to go before we can entirely remove the in-jail config.

 ivle-buildjail:
   - Add. Converted from setup.buildjail.
   - Build the jail in __base_build__ and rsync it to __base__ when
     done, rather than operating only in ./jail
   - Rename --rebuildjail/-j to --recreate/-r, as the whole script
     is now for jail rebuilding. Also add a warning to the usage string about
     the large volume likely to be downloaded.
   - Check existence before removing trees.
   - Don't copy jailconf.py over conf.py in the jail. Also make
     sure that we remove conf.pyc.

 setup.configure:
   - Stop generating jailconf.py at all.
   - Add a jail_system_build setting, defaulting to __base_build__ next to
     the existing __base__.
   - Don't use an OptionParser before calling the real function, as that
     adds options dynamically.

 setup.install:
   - Add an option (-R) to avoid writing out svn revision info to
     $PREFIX/share/ivle/revision.txt.
   - Remove jail-copying things.
   - Install all services to the host, rather than just usrmgt-server. We do
     this so we can build the jail from the host without the source tree.
   - Shuffle some things, and don't install phpBB3 twice.
   - Add a --root argument, to take an alternate root directory to install
     into (as given to autotools in $DESTDIR).

 setup.build:
   - Allow running as non-root.
   - Take a --no-compile option to not byte-compile Python files.

 setup.util:
   - Include usrmgt-server in the list of services.
   - Add make_install_path(), a wrapper around os.path.join() that ensures
     the second path is relative.
   - Install ivle-buildjail with the other binaries.

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# IVLE
2
 
# Copyright (C) 2010 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
 
import re
19
 
import datetime
20
 
 
21
 
import formencode
22
 
import formencode.validators
23
 
from genshi.filters import HTMLFormFiller
24
 
 
25
 
from ivle.webapp.base.xhtml import XHTMLView
26
 
 
27
 
 
28
 
class BaseFormView(XHTMLView):
29
 
    """A base form view."""
30
 
 
31
 
    @property
32
 
    def validator(self):
33
 
        """The FormEncode validator to use.
34
 
 
35
 
        The request will be passed in as state, after potentially being
36
 
        modified by populate_state().
37
 
        """
38
 
        raise NotImplementedError()
39
 
 
40
 
    def populate_state(self, state):
41
 
        """Populate the state given to the FormEncode validator.
42
 
 
43
 
        Subclasses can override this and set additional attributes.
44
 
        """
45
 
        pass
46
 
 
47
 
    def get_return_url(self, obj):
48
 
        """Return the URL to which the completed form should redirect.
49
 
 
50
 
        By default this will redirect to the saved object.
51
 
        """
52
 
        return self.req.publisher.generate(obj)
53
 
 
54
 
    def get_default_data(self, req):
55
 
        """Return a dict mapping field names to default form values.
56
 
 
57
 
        For an edit form, this should return the object's existing data.
58
 
        For a creation form, this should probably return an empty dict.
59
 
 
60
 
        This must be overridden by subclasses.
61
 
        """
62
 
        raise NotImplementedError()
63
 
 
64
 
    def save_object(self, req, data):
65
 
        """Take the validated form data and turn it into an object.
66
 
 
67
 
        The object must then be returned.
68
 
 
69
 
        For an edit form, this should just overwrite data on an existing
70
 
        object.
71
 
        For a creation form, this should create a new object with the given
72
 
        data and add it to the request's store.
73
 
        """
74
 
        raise NotImplementedError()
75
 
 
76
 
    def filter(self, stream, ctx):
77
 
        return stream | HTMLFormFiller(data=ctx['data'])
78
 
 
79
 
    def populate(self, req, ctx):
80
 
        if req.method == 'POST':
81
 
            data = dict(req.get_fieldstorage())
82
 
            try:
83
 
                self.populate_state(req)
84
 
                data = self.validator.to_python(data, state=req)
85
 
 
86
 
                obj = self.save_object(req, data)
87
 
 
88
 
                req.store.commit()
89
 
                req.throw_redirect(self.get_return_url(obj))
90
 
            except formencode.Invalid, e:
91
 
                error_value = e.msg
92
 
                errors = e.unpack_errors()
93
 
        else:
94
 
            data = self.get_default_data(req)
95
 
            error_value = None
96
 
            errors = {}
97
 
 
98
 
        if errors:
99
 
            req.store.rollback()
100
 
 
101
 
        ctx['req'] = req
102
 
        ctx['context'] = self.context
103
 
        ctx['data'] = data or {}
104
 
        ctx['errors'] = errors
105
 
        # If all of the fields validated, set the global form error.
106
 
        if isinstance(errors, basestring):
107
 
            ctx['error_value'] = errors
108
 
 
109
 
 
110
 
VALID_URL_NAME = re.compile(r'^[a-z0-9][a-z0-9_\+\.\-]*$')
111
 
 
112
 
 
113
 
class URLNameValidator(formencode.validators.UnicodeString):
114
 
    def validate_python(self, value, state):
115
 
        super(URLNameValidator, self).validate_python(value, state)
116
 
        if not VALID_URL_NAME.match(value):
117
 
            raise formencode.Invalid(
118
 
                'Must consist of a lowercase alphanumeric character followed '
119
 
                'by any number of lowercase alphanumerics, ., +, - or _.',
120
 
                value, state)
121
 
 
122
 
class DateTimeValidator(formencode.validators.FancyValidator):
123
 
    """Accepts a date/time in YYYY-MM-DD HH:MM:SS format. Converts to a
124
 
    datetime.datetime object."""
125
 
    def _to_python(self, value, state):
126
 
        """Validate and convert."""
127
 
        try:
128
 
            return datetime.datetime.strptime(value, "%Y-%m-%d %H:%M:%S")
129
 
        except ValueError, e:
130
 
            raise formencode.Invalid("Must be a timestamp in "
131
 
                "YYYY-MM-DD HH:MM:SS format", value, state)
132
 
    def _from_python(self, value, state):
133
 
        try:
134
 
            return value.strftime("%Y-%m-%d %H:%M:%S")
135
 
        except AttributeError:
136
 
            raise formencode.Invalid("Must be a datetime.datetime object")