183 lines
6.6 KiB
Python
183 lines
6.6 KiB
Python
|
#!/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.12' % self.source
|
||
|
else:
|
||
|
http_request.headers['User-Agent'] = 'gdata-py/2.0.12'
|
||
|
|
||
|
return http_request
|
||
|
|
||
|
ModifyRequest = modify_request
|