/atom/client.py
http://radioappz.googlecode.com/ · Python · 182 lines · 81 code · 29 blank · 72 comment · 9 complexity · 34d7b6735437f73be022c26882ca3653 MD5 · raw file
- #!/usr/bin/env python
- #
- # Copyright (C) 2009 Google Inc.
- #
- # Licensed under the Apache License, Version 2.0 (the "License");
- # you may not use this file except in compliance with the License.
- # You may obtain a copy of the License at
- #
- # http://www.apache.org/licenses/LICENSE-2.0
- #
- # Unless required by applicable law or agreed to in writing, software
- # distributed under the License is distributed on an "AS IS" BASIS,
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- # See the License for the specific language governing permissions and
- # limitations under the License.
- """AtomPubClient provides CRUD ops. in line with the Atom Publishing Protocol.
- """
- __author__ = 'j.s@google.com (Jeff Scudder)'
- import atom.http_core
- class Error(Exception):
- pass
- class MissingHost(Error):
- pass
- class AtomPubClient(object):
- host = None
- auth_token = None
- ssl = False # Whether to force all requests over https
- def __init__(self, http_client=None, host=None,
- auth_token=None, source=None, **kwargs):
- """Creates a new AtomPubClient instance.
- Args:
- source: The name of your application.
- http_client: An object capable of performing HTTP requests through a
- request method. This object is used to perform the request
- when the AtomPubClient's request method is called. Used to
- allow HTTP requests to be directed to a mock server, or use
- an alternate library instead of the default of httplib to
- make HTTP requests.
- host: str The default host name to use if a host is not specified in the
- requested URI.
- auth_token: An object which sets the HTTP Authorization header when its
- modify_request method is called.
- """
- self.http_client = http_client or atom.http_core.ProxiedHttpClient()
- if host is not None:
- self.host = host
- if auth_token is not None:
- self.auth_token = auth_token
- self.source = source
- def request(self, method=None, uri=None, auth_token=None,
- http_request=None, **kwargs):
- """Performs an HTTP request to the server indicated.
- Uses the http_client instance to make the request.
- Args:
- method: The HTTP method as a string, usually one of 'GET', 'POST',
- 'PUT', or 'DELETE'
- uri: The URI desired as a string or atom.http_core.Uri.
- http_request:
- auth_token: An authorization token object whose modify_request method
- sets the HTTP Authorization header.
- Returns:
- The results of calling self.http_client.request. With the default
- http_client, this is an HTTP response object.
- """
- # Modify the request based on the AtomPubClient settings and parameters
- # passed in to the request.
- http_request = self.modify_request(http_request)
- if isinstance(uri, (str, unicode)):
- uri = atom.http_core.Uri.parse_uri(uri)
- if uri is not None:
- uri.modify_request(http_request)
- if isinstance(method, (str, unicode)):
- http_request.method = method
- # Any unrecognized arguments are assumed to be capable of modifying the
- # HTTP request.
- for name, value in kwargs.iteritems():
- if value is not None:
- value.modify_request(http_request)
- # Default to an http request if the protocol scheme is not set.
- if http_request.uri.scheme is None:
- http_request.uri.scheme = 'http'
- # Override scheme. Force requests over https.
- if self.ssl:
- http_request.uri.scheme = 'https'
- if http_request.uri.path is None:
- http_request.uri.path = '/'
- # Add the Authorization header at the very end. The Authorization header
- # value may need to be calculated using information in the request.
- if auth_token:
- auth_token.modify_request(http_request)
- elif self.auth_token:
- self.auth_token.modify_request(http_request)
- # Check to make sure there is a host in the http_request.
- if http_request.uri.host is None:
- raise MissingHost('No host provided in request %s %s' % (
- http_request.method, str(http_request.uri)))
- # Perform the fully specified request using the http_client instance.
- # Sends the request to the server and returns the server's response.
- return self.http_client.request(http_request)
- Request = request
- def get(self, uri=None, auth_token=None, http_request=None, **kwargs):
- """Performs a request using the GET method, returns an HTTP response."""
- return self.request(method='GET', uri=uri, auth_token=auth_token,
- http_request=http_request, **kwargs)
- Get = get
- def post(self, uri=None, data=None, auth_token=None, http_request=None,
- **kwargs):
- """Sends data using the POST method, returns an HTTP response."""
- return self.request(method='POST', uri=uri, auth_token=auth_token,
- http_request=http_request, data=data, **kwargs)
- Post = post
- def put(self, uri=None, data=None, auth_token=None, http_request=None,
- **kwargs):
- """Sends data using the PUT method, returns an HTTP response."""
- return self.request(method='PUT', uri=uri, auth_token=auth_token,
- http_request=http_request, data=data, **kwargs)
- Put = put
- def delete(self, uri=None, auth_token=None, http_request=None, **kwargs):
- """Performs a request using the DELETE method, returns an HTTP response."""
- return self.request(method='DELETE', uri=uri, auth_token=auth_token,
- http_request=http_request, **kwargs)
- Delete = delete
- def modify_request(self, http_request):
- """Changes the HTTP request before sending it to the server.
-
- Sets the User-Agent HTTP header and fills in the HTTP host portion
- of the URL if one was not included in the request (for this it uses
- the self.host member if one is set). This method is called in
- self.request.
- Args:
- http_request: An atom.http_core.HttpRequest() (optional) If one is
- not provided, a new HttpRequest is instantiated.
- Returns:
- An atom.http_core.HttpRequest() with the User-Agent header set and
- if this client has a value in its host member, the host in the request
- URL is set.
- """
- if http_request is None:
- http_request = atom.http_core.HttpRequest()
- if self.host is not None and http_request.uri.host is None:
- http_request.uri.host = self.host
- # Set the user agent header for logging purposes.
- if self.source:
- http_request.headers['User-Agent'] = '%s gdata-py/2.0.9' % self.source
- else:
- http_request.headers['User-Agent'] = 'gdata-py/2.0.9'
- return http_request
- ModifyRequest = modify_request