source: main/waeup.sirp/branches/ulif-fasttables/src/waeup/sirp/utils/converters.py @ 5313

Last change on this file since 5313 was 5313, checked in by uli, 15 years ago

Add a date converter for zope.schema fields.

File size: 7.2 KB
RevLine 
[4805]1"""Converters for zope.schema-based datatypes.
2"""
[5313]3import datetime
[4805]4import grok
[4810]5from zope.component import getMultiAdapter
6from zope.publisher.browser import TestRequest
7try:
8    from zope.app.form.browser.interfaces import ITerms
9except ImportError:
10    from zope.browser.interfaces import ITerms
[5313]11from zope.schema.interfaces import IBool, IText, IInt, IChoice, IDate
[4920]12from waeup.sirp.interfaces import ISchemaTypeConverter
[4805]13
[4834]14# If a string has this value, it is considered as 'missing_value' or None.
15NONE_STRING_VALUE = ''
16
[4810]17class Converter(grok.Adapter):
18    """Base for string-or-none to zope.schema field converters.
19    """
20    grok.baseclass()
21    grok.provides(ISchemaTypeConverter)
22   
23    def __init__(self, context):
24        """Create a converter with context, a zope.schema.Field, as context.
25        """
26        self.context = context
27
28    def _convertValueFromString(self, string):
29        """You must at least override this method to build a working
30           converter.
31        """
32        raise NotImplementedError('method not implemented')
33   
34    def fromString(self, string=None, strict=True):
35        """Convert ``string`` to value according to assigned field type.
36        """
37        result = None
[4834]38        if string is NONE_STRING_VALUE:
39            string = None
[4810]40        if string is None:
41            if self.context.required is True:
42                result = self.context.default
43            else:
44                result = self.context.missing_value
45        else:
46            result = self._convertValueFromString(string)
47        if strict:
48            self.context.validate(result)
49        return result
[4814]50
51    def _convertValueToString(self, value):
52        return str(value)
53
54    def toString(self, value, strict=True):
[5313]55        """Convert given `value` to string according to assigned field type.
56        """
[4814]57        if strict:
58            self.context.validate(value)
59        if value == self.context.missing_value:
60            return None
61        return self._convertValueToString(value)
62
63class BoolConverter(Converter):
[4834]64    """A converter for zope.schema.Bool fields.
65    """
[4814]66    grok.context(IBool)
67    grok.provides(ISchemaTypeConverter)
68
69    def _convertValueFromString(self, string):
[4834]70        if string is NONE_STRING_VALUE:
71            string = None
[4814]72        if string is None:
73            return None
74        if string.lower() in ['1', 'true', 'yes']:
75            return True
76        return False
77
78    def _convertValueToString(self, value):
79        if value is None:
80            return None
81        if value:
82            return '1'
83        return '0'
[4810]84   
85class TextConverter(Converter):
[4834]86    """A converter for zope.schema.interfaces.IText fields.
87    """
[4805]88    grok.context(IText)
89    grok.provides(ISchemaTypeConverter)
90
[4810]91    def _convertValueFromString(self, string):
92        return unicode(string)
93
94class IntConverter(Converter):
[4834]95    """A converter for zope.schema.Int fields.
96    """
[4810]97    grok.context(IInt)
98    grok.provides(ISchemaTypeConverter)
99
100    def _convertValueFromString(self, string):
101        return int(string)
102
103class ChoiceConverter(Converter):
[4834]104    """A converter for zope.schema.Choice fields.
105    """
[4810]106    grok.context(IChoice)
107    grok.provides(ISchemaTypeConverter)
108
[4834]109    tokens = None
110    values = None
111   
[4805]112    def __init__(self, context):
113        self.context = context
[4834]114        if not hasattr(self.context.source, 'factory'):
115            try:
116                self.terms = getMultiAdapter(
117                    (self.context.source, TestRequest()), ITerms)
118            except:
119                self.terms = None
[4810]120
[4834]121            return
122        if not hasattr(self.context.source.factory, 'getToken'):
123            return
124        # For expensive token/key lookups we create a 'cache'
125        # here. This speeds up mass operations with many conversions
126        # by factor 10 or more.
127       
128        # Mapping token -> value
129        self.tokens = dict([(self.context.source.factory.getToken(x), x)
130                            for x in self.context.source.factory.getValues()])
131        # Mapping value -> token
132        self.values = dict([(y,x) for x,y in self.tokens.items()])
133
[4810]134    def _convertValueFromString(self, string):
[4834]135        if self.tokens is not None:
136            return self.tokens[string]
[4814]137        if self.terms is not None:
[4810]138            return self.terms.getValue(string)
[4814]139        return self.context.source.getTermByToken(string).value
[4810]140
[4814]141    def _convertValueToString(self, value):
[4834]142        if self.values is not None:
143            return self.values[value]
[4814]144        if self.terms is not None:
145            return self.terms.getTerm(value).token
146        return str(value)
[4834]147
148    def fromString(self, string=None, strict=False):
149        """Convert ``string`` to value according to assigned field type.
150
151        We change the default for ``strict``: this disables extra
152        validation checks for Choice fields and saves lots of time. If
153        a string/value is out of allowed range we get a value or key
154        error anyway.
155        """
156        return super(ChoiceConverter, self).fromString(string=string,
157                                                       strict=strict)
158
159    def toString(self, value, strict=False):
160        """Convert ``value`` to string according to assigned field type.
161
162        We change the default for ``strict``: this disables extra
163        validation checks for Choice fields and saves lots of time. If
164        a string/value is out of allowed range we get a value or key
165        error anyway.
166        """
167        return super(ChoiceConverter, self).toString(value=value,
168                                                     strict=strict)
[5313]169
170class DateConverter(Converter):
171    """A converter for zope.schema.IDate fields.
172
173    Converts date to string and vice versa. Stringified dates are
174    expected in format ``YYYY-MM-DD``.
175
176    To support at least some other formats, we accept for conversion
177    from string also the following formats:
178
179    * ``YYYY/MM/DD``
180   
181    * ``DD/MM/YYYY``
182   
183    * ``D/M/YYYY``
184
185    * ``DD.MM.YYYY``
186
187    * ``D.M.YYYY``
188
189    When converting to strings, always 'YYYY-MM-DD' is returned.
190
191    For convenience, when converting from strings also string data
192    separated by space is stripped before processing. Therefore
193    strings like '1990-04-01 12:12:01 GMT +1' will also work.
194    """
195    grok.context(IDate)
196    grok.provides(ISchemaTypeConverter)
197
198    #: List of supported date formats in `strftime()` notation.
199    formats = ['%Y-%m-%d', '%Y/%m/%d' , '%d/%m/%Y', '%D/%M/%Y',
200               '%d.%m.%Y', '%D.%M.%Y']
201   
202    def _convertValueFromString(self, string):
203        orig_string = string
204        if string is NONE_STRING_VALUE:
205            string = None
206        if string is None:
207            return None
208        value = None
209        if ' ' in string:
210            string = string.split(' ')[0]
211        for format in self.formats:
212            try:
213                value = datetime.datetime.strptime(string, format)
214                break
215            except ValueError:
216                pass
217        if value is None:
218            raise ValueError(
219                'Cannot convert to date: %s. Use YYYY-MM-DD.' %
220                orig_string)
221        value = value.date()
222        return value
223
224    def _convertValueToString(self, value):
225        return datetime.date.strftime(value, '%Y-%m-%d')
Note: See TracBrowser for help on using the repository browser.