1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
|
# Copyright (C) 2010-2017 by the Free Software Foundation, Inc.
#
# This file is part of GNU Mailman.
#
# GNU Mailman is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free
# Software Foundation, either version 3 of the License, or (at your option)
# any later version.
#
# GNU Mailman is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
# more details.
#
# You should have received a copy of the GNU General Public License along with
# GNU Mailman. If not, see <http://www.gnu.org/licenses/>.
"""Web service helpers."""
import json
import falcon
import hashlib
from contextlib import suppress
from datetime import datetime, timedelta
from email.header import Header
from email.message import Message
from enum import Enum
from lazr.config import as_boolean
from mailman.config import config
from pprint import pformat
from public import public
class ExtendedEncoder(json.JSONEncoder):
"""An extended JSON encoder which knows about other data types."""
def default(self, obj):
if isinstance(obj, datetime):
return obj.isoformat()
elif isinstance(obj, timedelta):
# as_timedelta() does not recognize microseconds, so convert these
# to floating seconds, but only if there are any seconds.
if obj.seconds > 0 or obj.microseconds > 0:
seconds = obj.seconds + obj.microseconds / 1000000.0
return '{}d{}s'.format(obj.days, seconds)
return '{}d'.format(obj.days)
elif isinstance(obj, Enum):
# It's up to the decoding validator to associate this name with
# the right Enum class.
return obj.name
elif isinstance(obj, bytes):
return bytes_to_str(obj)
elif isinstance(obj, Message):
return obj.as_string()
elif isinstance(obj, Header):
return str(obj)
return super().default(obj)
def bytes_to_str(value):
# Convert a string to unicode when the encoding is not declared.
if not isinstance(value, bytes):
return value
for encoding in ('ascii', 'utf-8', 'raw_unicode_escape'):
with suppress(UnicodeDecodeError):
return value.decode(encoding)
@public
def etag(resource):
"""Calculate the etag and return a JSON representation.
The input is a dictionary representing the resource. This
dictionary must not contain an `http_etag` key. This function
calculates the etag by using the sha1 hexdigest of the
pretty-printed (and thus key-sorted and predictable) representation
of the dictionary. It then inserts this value under the `http_etag`
key, and returns the JSON representation of the modified dictionary.
:param resource: The original resource representation.
:type resource: dictionary
:return: JSON representation of the modified dictionary.
:rtype string
"""
assert 'http_etag' not in resource, 'Resource already etagged'
# Calculate the tag from a predictable (i.e. sorted) representation of the
# dictionary. The actual details aren't so important. pformat() is
# guaranteed to sort the keys, however it returns a str and the hash
# library requires a bytes. Use the safest possible encoding.
hashfood = pformat(resource).encode('raw-unicode-escape')
etag = hashlib.sha1(hashfood).hexdigest()
resource['http_etag'] = '"{}"'.format(etag)
return json.dumps(resource, cls=ExtendedEncoder,
sort_keys=as_boolean(config.devmode.enabled))
@public
class CollectionMixin:
"""Mixin class for common collection-ish things."""
def _resource_as_dict(self, resource):
"""Return the dictionary representation of a resource.
This must be implemented by subclasses.
:param resource: The resource object.
:type resource: object
:return: The representation of the resource.
:rtype: dict
"""
raise NotImplementedError
def _resource_as_json(self, resource):
"""Return the JSON formatted representation of the resource."""
resource = self._resource_as_dict(resource)
assert resource is not None, resource
return etag(resource)
def _get_collection(self, request):
"""Return the collection as a sequence.
The returned value must support the collections.abc.Sequence
API. This method must be implemented by subclasses.
:param request: An http request.
:return: The collection
:rtype: collections.abc.Sequence
"""
raise NotImplementedError
def _paginate(self, request, collection):
"""Method to paginate through collection result lists.
Use this to return only a slice of a collection, specified in
the request itself. The request should use query parameters
`count` and `page` to specify the slice they want. The slice
will start at index ``(page - 1) * count`` and end (exclusive)
at ``(page * count)``.
"""
# Allow falcon's HTTPBadRequest exceptions to percolate up. They'll
# get turned into HTTP 400 errors.
count = request.get_param_as_int('count', min=0)
page = request.get_param_as_int('page', min=1)
total_size = len(collection)
if count is None and page is None:
return 0, total_size, collection
list_start = (page - 1) * count
list_end = page * count
return list_start, total_size, collection[list_start:list_end]
def _make_collection(self, request):
"""Provide the collection to the REST layer."""
start, total_size, collection = self._paginate(
request, self._get_collection(request))
result = dict(start=start, total_size=total_size)
if len(collection) != 0:
entries = [self._resource_as_dict(resource)
for resource in collection]
assert None not in entries, entries
# Tag the resources but use the dictionaries.
[etag(resource) for resource in entries]
# Create the collection resource
result['entries'] = entries
return result
@public
class GetterSetter:
"""Get and set attributes on an object.
Most attributes are fairly simple - a getattr() or setattr() on the object
does the trick, with the appropriate encoding or decoding on the way in
and out. Encoding doesn't happen here though; the standard JSON library
handles most types, but see ExtendedEncoder for additional support.
Others are more complicated since they aren't kept in the model as direct
columns in the database. These will use subclasses of this base class.
Read-only attributes will have a decoder which always raises ValueError.
"""
def __init__(self, decoder=None):
"""Create a getter/setter for a specific attribute.
:param decoder: The callable for decoding a web request value string
into the specific data type needed by the object's attribute. Use
None to indicate a read-only attribute. The callable should raise
ValueError when the web request value cannot be converted.
:type decoder: callable
"""
self.decoder = decoder
def get(self, obj, attribute):
"""Return the named object attribute value.
:param obj: The object to access.
:type obj: object
:param attribute: The attribute name.
:type attribute: string
:return: The attribute value, ready for JSON encoding.
:rtype: object
"""
return getattr(obj, attribute)
def put(self, obj, attribute, value):
"""Set the named object attribute value.
:param obj: The object to change.
:type obj: object
:param attribute: The attribute name.
:type attribute: string
:param value: The new value for the attribute.
"""
setattr(obj, attribute, value)
def __call__(self, value):
"""Convert the value to its internal format.
:param value: The web request value to convert.
:type value: string
:return: The converted value.
:rtype: object
"""
if self.decoder is None:
return value
return self.decoder(value)
# Falcon REST framework add-ons.
@public
def child(matcher=None):
def decorator(func):
if matcher is None:
func.__matcher__ = func.__name__
else:
func.__matcher__ = matcher
return func
return decorator
@public
class ChildError:
def __init__(self, status):
self._status = status
def _oops(self, request, response):
raise falcon.HTTPError(self._status, None)
on_get = _oops
on_post = _oops
on_put = _oops
on_patch = _oops
on_delete = _oops
@public
class BadRequest(ChildError):
def __init__(self):
super().__init__(falcon.HTTP_400)
@public
class NotFound(ChildError):
def __init__(self):
super().__init__(falcon.HTTP_404)
@public
def okay(response, body=None):
response.status = falcon.HTTP_200
if body is not None:
response.body = body
@public
def no_content(response):
response.status = falcon.HTTP_204
@public
def not_found(response, body=b'404 Not Found'):
response.status = falcon.HTTP_404
if body is not None:
response.body = body
@public
def accepted(response, body=None):
response.status = falcon.HTTP_202
if body is not None:
response.body = body
@public
def bad_request(response, body='400 Bad Request'):
response.status = falcon.HTTP_400
if body is not None:
response.body = body
@public
def created(response, location):
response.status = falcon.HTTP_201
response.location = location
@public
def conflict(response, body=b'409 Conflict'):
response.status = falcon.HTTP_409
if body is not None:
response.body = body
@public
def forbidden(response, body=b'403 Forbidden'):
response.status = falcon.HTTP_403
if body is not None:
response.body = body
|