1# Copyright 2014 Google Inc. All Rights Reserved. 2# 3# Licensed under the Apache License, Version 2.0 (the "License"); 4# you may not use this file except in compliance with the License. 5# You may obtain a copy of the License at 6# 7# http://www.apache.org/licenses/LICENSE-2.0 8# 9# Unless required by applicable law or agreed to in writing, software 10# distributed under the License is distributed on an "AS IS" BASIS, 11# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12# See the License for the specific language governing permissions and 13# limitations under the License. 14 15"""Model objects for requests and responses. 16 17Each API may support one or more serializations, such 18as JSON, Atom, etc. The model classes are responsible 19for converting between the wire format and the Python 20object representation. 21""" 22from __future__ import absolute_import 23import six 24 25__author__ = 'jcgregorio@google.com (Joe Gregorio)' 26 27import json 28import logging 29 30from six.moves.urllib.parse import urlencode 31 32from googleapiclient import __version__ 33from googleapiclient.errors import HttpError 34 35 36LOGGER = logging.getLogger(__name__) 37 38dump_request_response = False 39 40 41def _abstract(): 42 raise NotImplementedError('You need to override this function') 43 44 45class Model(object): 46 """Model base class. 47 48 All Model classes should implement this interface. 49 The Model serializes and de-serializes between a wire 50 format such as JSON and a Python object representation. 51 """ 52 53 def request(self, headers, path_params, query_params, body_value): 54 """Updates outgoing requests with a serialized body. 55 56 Args: 57 headers: dict, request headers 58 path_params: dict, parameters that appear in the request path 59 query_params: dict, parameters that appear in the query 60 body_value: object, the request body as a Python object, which must be 61 serializable. 62 Returns: 63 A tuple of (headers, path_params, query, body) 64 65 headers: dict, request headers 66 path_params: dict, parameters that appear in the request path 67 query: string, query part of the request URI 68 body: string, the body serialized in the desired wire format. 69 """ 70 _abstract() 71 72 def response(self, resp, content): 73 """Convert the response wire format into a Python object. 74 75 Args: 76 resp: httplib2.Response, the HTTP response headers and status 77 content: string, the body of the HTTP response 78 79 Returns: 80 The body de-serialized as a Python object. 81 82 Raises: 83 googleapiclient.errors.HttpError if a non 2xx response is received. 84 """ 85 _abstract() 86 87 88class BaseModel(Model): 89 """Base model class. 90 91 Subclasses should provide implementations for the "serialize" and 92 "deserialize" methods, as well as values for the following class attributes. 93 94 Attributes: 95 accept: The value to use for the HTTP Accept header. 96 content_type: The value to use for the HTTP Content-type header. 97 no_content_response: The value to return when deserializing a 204 "No 98 Content" response. 99 alt_param: The value to supply as the "alt" query parameter for requests. 100 """ 101 102 accept = None 103 content_type = None 104 no_content_response = None 105 alt_param = None 106 107 def _log_request(self, headers, path_params, query, body): 108 """Logs debugging information about the request if requested.""" 109 if dump_request_response: 110 LOGGER.info('--request-start--') 111 LOGGER.info('-headers-start-') 112 for h, v in six.iteritems(headers): 113 LOGGER.info('%s: %s', h, v) 114 LOGGER.info('-headers-end-') 115 LOGGER.info('-path-parameters-start-') 116 for h, v in six.iteritems(path_params): 117 LOGGER.info('%s: %s', h, v) 118 LOGGER.info('-path-parameters-end-') 119 LOGGER.info('body: %s', body) 120 LOGGER.info('query: %s', query) 121 LOGGER.info('--request-end--') 122 123 def request(self, headers, path_params, query_params, body_value): 124 """Updates outgoing requests with a serialized body. 125 126 Args: 127 headers: dict, request headers 128 path_params: dict, parameters that appear in the request path 129 query_params: dict, parameters that appear in the query 130 body_value: object, the request body as a Python object, which must be 131 serializable by json. 132 Returns: 133 A tuple of (headers, path_params, query, body) 134 135 headers: dict, request headers 136 path_params: dict, parameters that appear in the request path 137 query: string, query part of the request URI 138 body: string, the body serialized as JSON 139 """ 140 query = self._build_query(query_params) 141 headers['accept'] = self.accept 142 headers['accept-encoding'] = 'gzip, deflate' 143 if 'user-agent' in headers: 144 headers['user-agent'] += ' ' 145 else: 146 headers['user-agent'] = '' 147 headers['user-agent'] += 'google-api-python-client/%s (gzip)' % __version__ 148 149 if body_value is not None: 150 headers['content-type'] = self.content_type 151 body_value = self.serialize(body_value) 152 self._log_request(headers, path_params, query, body_value) 153 return (headers, path_params, query, body_value) 154 155 def _build_query(self, params): 156 """Builds a query string. 157 158 Args: 159 params: dict, the query parameters 160 161 Returns: 162 The query parameters properly encoded into an HTTP URI query string. 163 """ 164 if self.alt_param is not None: 165 params.update({'alt': self.alt_param}) 166 astuples = [] 167 for key, value in six.iteritems(params): 168 if type(value) == type([]): 169 for x in value: 170 x = x.encode('utf-8') 171 astuples.append((key, x)) 172 else: 173 if isinstance(value, six.text_type) and callable(value.encode): 174 value = value.encode('utf-8') 175 astuples.append((key, value)) 176 return '?' + urlencode(astuples) 177 178 def _log_response(self, resp, content): 179 """Logs debugging information about the response if requested.""" 180 if dump_request_response: 181 LOGGER.info('--response-start--') 182 for h, v in six.iteritems(resp): 183 LOGGER.info('%s: %s', h, v) 184 if content: 185 LOGGER.info(content) 186 LOGGER.info('--response-end--') 187 188 def response(self, resp, content): 189 """Convert the response wire format into a Python object. 190 191 Args: 192 resp: httplib2.Response, the HTTP response headers and status 193 content: string, the body of the HTTP response 194 195 Returns: 196 The body de-serialized as a Python object. 197 198 Raises: 199 googleapiclient.errors.HttpError if a non 2xx response is received. 200 """ 201 self._log_response(resp, content) 202 # Error handling is TBD, for example, do we retry 203 # for some operation/error combinations? 204 if resp.status < 300: 205 if resp.status == 204: 206 # A 204: No Content response should be treated differently 207 # to all the other success states 208 return self.no_content_response 209 return self.deserialize(content) 210 else: 211 LOGGER.debug('Content from bad request was: %s' % content) 212 raise HttpError(resp, content) 213 214 def serialize(self, body_value): 215 """Perform the actual Python object serialization. 216 217 Args: 218 body_value: object, the request body as a Python object. 219 220 Returns: 221 string, the body in serialized form. 222 """ 223 _abstract() 224 225 def deserialize(self, content): 226 """Perform the actual deserialization from response string to Python 227 object. 228 229 Args: 230 content: string, the body of the HTTP response 231 232 Returns: 233 The body de-serialized as a Python object. 234 """ 235 _abstract() 236 237 238class JsonModel(BaseModel): 239 """Model class for JSON. 240 241 Serializes and de-serializes between JSON and the Python 242 object representation of HTTP request and response bodies. 243 """ 244 accept = 'application/json' 245 content_type = 'application/json' 246 alt_param = 'json' 247 248 def __init__(self, data_wrapper=False): 249 """Construct a JsonModel. 250 251 Args: 252 data_wrapper: boolean, wrap requests and responses in a data wrapper 253 """ 254 self._data_wrapper = data_wrapper 255 256 def serialize(self, body_value): 257 if (isinstance(body_value, dict) and 'data' not in body_value and 258 self._data_wrapper): 259 body_value = {'data': body_value} 260 return json.dumps(body_value) 261 262 def deserialize(self, content): 263 try: 264 content = content.decode('utf-8') 265 except AttributeError: 266 pass 267 body = json.loads(content) 268 if self._data_wrapper and isinstance(body, dict) and 'data' in body: 269 body = body['data'] 270 return body 271 272 @property 273 def no_content_response(self): 274 return {} 275 276 277class RawModel(JsonModel): 278 """Model class for requests that don't return JSON. 279 280 Serializes and de-serializes between JSON and the Python 281 object representation of HTTP request, and returns the raw bytes 282 of the response body. 283 """ 284 accept = '*/*' 285 content_type = 'application/json' 286 alt_param = None 287 288 def deserialize(self, content): 289 return content 290 291 @property 292 def no_content_response(self): 293 return '' 294 295 296class MediaModel(JsonModel): 297 """Model class for requests that return Media. 298 299 Serializes and de-serializes between JSON and the Python 300 object representation of HTTP request, and returns the raw bytes 301 of the response body. 302 """ 303 accept = '*/*' 304 content_type = 'application/json' 305 alt_param = 'media' 306 307 def deserialize(self, content): 308 return content 309 310 @property 311 def no_content_response(self): 312 return '' 313 314 315class ProtocolBufferModel(BaseModel): 316 """Model class for protocol buffers. 317 318 Serializes and de-serializes the binary protocol buffer sent in the HTTP 319 request and response bodies. 320 """ 321 accept = 'application/x-protobuf' 322 content_type = 'application/x-protobuf' 323 alt_param = 'proto' 324 325 def __init__(self, protocol_buffer): 326 """Constructs a ProtocolBufferModel. 327 328 The serialzed protocol buffer returned in an HTTP response will be 329 de-serialized using the given protocol buffer class. 330 331 Args: 332 protocol_buffer: The protocol buffer class used to de-serialize a 333 response from the API. 334 """ 335 self._protocol_buffer = protocol_buffer 336 337 def serialize(self, body_value): 338 return body_value.SerializeToString() 339 340 def deserialize(self, content): 341 return self._protocol_buffer.FromString(content) 342 343 @property 344 def no_content_response(self): 345 return self._protocol_buffer() 346 347 348def makepatch(original, modified): 349 """Create a patch object. 350 351 Some methods support PATCH, an efficient way to send updates to a resource. 352 This method allows the easy construction of patch bodies by looking at the 353 differences between a resource before and after it was modified. 354 355 Args: 356 original: object, the original deserialized resource 357 modified: object, the modified deserialized resource 358 Returns: 359 An object that contains only the changes from original to modified, in a 360 form suitable to pass to a PATCH method. 361 362 Example usage: 363 item = service.activities().get(postid=postid, userid=userid).execute() 364 original = copy.deepcopy(item) 365 item['object']['content'] = 'This is updated.' 366 service.activities.patch(postid=postid, userid=userid, 367 body=makepatch(original, item)).execute() 368 """ 369 patch = {} 370 for key, original_value in six.iteritems(original): 371 modified_value = modified.get(key, None) 372 if modified_value is None: 373 # Use None to signal that the element is deleted 374 patch[key] = None 375 elif original_value != modified_value: 376 if type(original_value) == type({}): 377 # Recursively descend objects 378 patch[key] = makepatch(original_value, modified_value) 379 else: 380 # In the case of simple types or arrays we just replace 381 patch[key] = modified_value 382 else: 383 # Don't add anything to patch if there's no change 384 pass 385 for key in modified: 386 if key not in original: 387 patch[key] = modified[key] 388 389 return patch 390