2011-10-25 16:50:08 -07:00
|
|
|
# Copyright 2010 Jacob Kaplan-Moss
|
|
|
|
# Copyright 2011 OpenStack LLC.
|
|
|
|
# Copyright 2011 Piston Cloud Computing, Inc.
|
|
|
|
# Copyright 2011 Nebula, Inc.
|
|
|
|
|
|
|
|
# All Rights Reserved.
|
|
|
|
"""
|
|
|
|
OpenStack Client interface. Handles the REST calls and responses.
|
|
|
|
"""
|
|
|
|
|
|
|
|
import copy
|
|
|
|
import logging
|
|
|
|
import urlparse
|
|
|
|
|
2012-11-16 17:43:05 -06:00
|
|
|
import requests
|
2011-10-25 16:50:08 -07:00
|
|
|
|
2013-05-22 17:00:53 -04:00
|
|
|
try:
|
|
|
|
import keyring
|
|
|
|
import pickle
|
|
|
|
except ImportError:
|
|
|
|
keyring = None
|
|
|
|
pickle = None
|
|
|
|
|
2011-10-25 16:50:08 -07:00
|
|
|
try:
|
|
|
|
import json
|
|
|
|
except ImportError:
|
|
|
|
import simplejson as json
|
|
|
|
|
|
|
|
# Python 2.5 compat fix
|
|
|
|
if not hasattr(urlparse, 'parse_qsl'):
|
|
|
|
import cgi
|
|
|
|
urlparse.parse_qsl = cgi.parse_qsl
|
|
|
|
|
|
|
|
|
2012-10-13 00:15:39 +00:00
|
|
|
from keystoneclient import access
|
2011-10-25 16:50:08 -07:00
|
|
|
from keystoneclient import exceptions
|
|
|
|
|
|
|
|
|
|
|
|
_logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2012-11-16 17:43:05 -06:00
|
|
|
class HTTPClient(object):
|
2011-10-25 16:50:08 -07:00
|
|
|
|
|
|
|
USER_AGENT = 'python-keystoneclient'
|
|
|
|
|
2011-12-17 22:36:59 -08:00
|
|
|
def __init__(self, username=None, tenant_id=None, tenant_name=None,
|
2011-12-19 10:00:39 -08:00
|
|
|
password=None, auth_url=None, region_name=None, timeout=None,
|
2012-05-23 18:16:50 +00:00
|
|
|
endpoint=None, token=None, cacert=None, key=None,
|
2012-10-13 00:15:39 +00:00
|
|
|
cert=None, insecure=False, original_ip=None, debug=False,
|
2012-12-06 15:26:46 -05:00
|
|
|
auth_ref=None, use_keyring=False, force_new_token=False,
|
2013-02-13 22:52:05 -06:00
|
|
|
stale_duration=None, user_id=None, user_domain_id=None,
|
|
|
|
user_domain_name=None, domain_id=None, domain_name=None,
|
|
|
|
project_id=None, project_name=None, project_domain_id=None,
|
|
|
|
project_domain_name=None):
|
2013-01-11 21:56:24 -08:00
|
|
|
"""Construct a new http client
|
|
|
|
|
2013-02-13 22:52:05 -06:00
|
|
|
:param string user_id: User ID for authentication. (optional)
|
|
|
|
:param string username: Username for authentication. (optional)
|
|
|
|
:param string user_domain_id: User's domain ID for authentication.
|
|
|
|
(optional)
|
|
|
|
:param string user_domain_name: User's domain name for authentication.
|
|
|
|
(optional)
|
|
|
|
:param string password: Password for authentication. (optional)
|
|
|
|
:param string domain_id: Domain ID for domain scoping. (optional)
|
|
|
|
:param string domain_name: Domain name for domain scoping. (optional)
|
|
|
|
:param string project_id: Project ID for project scoping. (optional)
|
|
|
|
:param string project_name: Project name for project scoping.
|
|
|
|
(optional)
|
|
|
|
:param string project_domain_id: Project's domain ID for project
|
|
|
|
scoping. (optional)
|
|
|
|
:param string project_domain_name: Project's domain name for project
|
|
|
|
scoping. (optional)
|
|
|
|
:param string auth_url: Identity service endpoint for authorization.
|
|
|
|
:param string region_name: Name of a region to select when choosing an
|
|
|
|
endpoint from the service catalog.
|
|
|
|
:param integer timeout: Allows customization of the timeout for client
|
|
|
|
http requests. (optional)
|
|
|
|
:param string endpoint: A user-supplied endpoint URL for the identity
|
|
|
|
service. Lazy-authentication is possible for
|
|
|
|
API service calls if endpoint is set at
|
|
|
|
instantiation. (optional)
|
|
|
|
:param string token: Token for authentication. (optional)
|
|
|
|
:param string cacert: Path to the Privacy Enhanced Mail (PEM) file
|
|
|
|
which contains the trusted authority X.509
|
|
|
|
certificates needed to established SSL connection
|
|
|
|
with the identity service. (optional)
|
|
|
|
:param string key: Path to the Privacy Enhanced Mail (PEM) file which
|
|
|
|
contains the unencrypted client private key needed
|
|
|
|
to established two-way SSL connection with the
|
|
|
|
identity service. (optional)
|
|
|
|
:param string cert: Path to the Privacy Enhanced Mail (PEM) file which
|
|
|
|
contains the corresponding X.509 client certificate
|
|
|
|
needed to established two-way SSL connection with
|
|
|
|
the identity service. (optional)
|
|
|
|
:param boolean insecure: Does not perform X.509 certificate validation
|
|
|
|
when establishing SSL connection with identity
|
|
|
|
service. default: False (optional)
|
|
|
|
:param string original_ip: The original IP of the requesting user
|
|
|
|
which will be sent to identity service in a
|
|
|
|
'Forwarded' header. (optional)
|
|
|
|
:param boolean debug: Enables debug logging of all request and
|
|
|
|
responses to identity service.
|
|
|
|
default False (optional)
|
|
|
|
:param dict auth_ref: To allow for consumers of the client to manage
|
|
|
|
their own caching strategy, you may initialize a
|
|
|
|
client with a previously captured auth_reference
|
|
|
|
(token). If there are keyword arguments passed
|
|
|
|
that also exist in auth_ref, the value from the
|
|
|
|
argument will take precedence.
|
|
|
|
:param boolean use_keyring: Enables caching auth_ref into keyring.
|
|
|
|
default: False (optional)
|
|
|
|
:param boolean force_new_token: Keyring related parameter, forces
|
|
|
|
request for new token.
|
|
|
|
default: False (optional)
|
|
|
|
:param integer stale_duration: Gap in seconds to determine if token
|
|
|
|
from keyring is about to expire.
|
|
|
|
default: 30 (optional)
|
|
|
|
:param string tenant_name: Tenant name. (optional)
|
|
|
|
The tenant_name keyword argument is
|
|
|
|
deprecated, use project_name instead.
|
|
|
|
:param string tenant_id: Tenant id. (optional)
|
|
|
|
The tenant_id keyword argument is
|
|
|
|
deprecated, use project_id instead.
|
2013-01-11 21:56:24 -08:00
|
|
|
|
|
|
|
"""
|
2012-11-14 05:38:16 +00:00
|
|
|
# set baseline defaults
|
2013-02-13 22:52:05 -06:00
|
|
|
|
|
|
|
self.user_id = None
|
2012-11-14 05:38:16 +00:00
|
|
|
self.username = None
|
2013-02-13 22:52:05 -06:00
|
|
|
self.user_domain_id = None
|
|
|
|
self.user_domain_name = None
|
|
|
|
|
|
|
|
self.domain_id = None
|
|
|
|
self.domain_name = None
|
|
|
|
|
|
|
|
self.project_id = None
|
|
|
|
self.project_name = None
|
|
|
|
self.project_domain_id = None
|
|
|
|
self.project_domain_name = None
|
|
|
|
|
2012-11-14 05:38:16 +00:00
|
|
|
self.auth_url = None
|
|
|
|
self.management_url = None
|
2013-02-13 22:52:05 -06:00
|
|
|
self.timeout = float(timeout) if timeout is not None else None
|
|
|
|
|
2012-11-14 05:38:16 +00:00
|
|
|
# if loading from a dictionary passed in via auth_ref,
|
|
|
|
# load values from AccessInfo parsing that dictionary
|
2013-02-13 22:52:05 -06:00
|
|
|
if auth_ref:
|
|
|
|
self.auth_ref = access.AccessInfo.factory(**auth_ref)
|
|
|
|
self.version = self.auth_ref.version
|
|
|
|
self.user_id = self.auth_ref.user_id
|
2012-10-13 00:15:39 +00:00
|
|
|
self.username = self.auth_ref.username
|
2013-02-13 22:52:05 -06:00
|
|
|
self.user_domain_id = self.auth_ref.user_domain_id
|
|
|
|
self.domain_id = self.auth_ref.domain_id
|
|
|
|
self.domain_name = self.auth_ref.domain_name
|
|
|
|
self.project_id = self.auth_ref.project_id
|
|
|
|
self.project_name = self.auth_ref.project_name
|
|
|
|
self.project_domain_id = self.auth_ref.project_domain_id
|
2012-11-14 05:38:16 +00:00
|
|
|
self.auth_url = self.auth_ref.auth_url[0]
|
|
|
|
self.management_url = self.auth_ref.management_url[0]
|
2013-02-13 22:52:05 -06:00
|
|
|
self.auth_token = self.auth_ref.auth_token
|
|
|
|
else:
|
|
|
|
self.auth_ref = None
|
|
|
|
|
2012-11-14 05:38:16 +00:00
|
|
|
# allow override of the auth_ref defaults from explicit
|
2012-10-13 00:15:39 +00:00
|
|
|
# values provided to the client
|
2013-02-13 22:52:05 -06:00
|
|
|
|
|
|
|
# apply deprecated variables first, so modern variables override them
|
2012-11-14 05:38:16 +00:00
|
|
|
if tenant_id:
|
2013-02-13 22:52:05 -06:00
|
|
|
self.project_id = tenant_id
|
2012-11-14 05:38:16 +00:00
|
|
|
if tenant_name:
|
2013-02-13 22:52:05 -06:00
|
|
|
self.project_name = tenant_name
|
|
|
|
|
|
|
|
# user-related attributes
|
|
|
|
self.password = password
|
|
|
|
if user_id:
|
|
|
|
self.user_id = user_id
|
|
|
|
if username:
|
|
|
|
self.username = username
|
|
|
|
if user_domain_id:
|
|
|
|
self.user_domain_id = user_domain_id
|
|
|
|
elif not (user_id or user_domain_name):
|
|
|
|
self.user_domain_id = 'default'
|
|
|
|
if user_domain_name:
|
|
|
|
self.user_domain_name = user_domain_name
|
|
|
|
|
|
|
|
# domain-related attributes
|
|
|
|
if domain_id:
|
|
|
|
self.domain_id = domain_id
|
|
|
|
if domain_name:
|
|
|
|
self.domain_name = domain_name
|
|
|
|
|
|
|
|
# project-related attributes
|
|
|
|
if project_id:
|
|
|
|
self.project_id = project_id
|
|
|
|
if project_name:
|
|
|
|
self.project_name = project_name
|
|
|
|
if project_domain_id:
|
|
|
|
self.project_domain_id = project_domain_id
|
|
|
|
elif not (project_id or project_domain_name):
|
|
|
|
self.project_domain_id = 'default'
|
|
|
|
if project_domain_name:
|
|
|
|
self.project_domain_name = project_domain_name
|
|
|
|
|
|
|
|
# endpoint selection
|
2012-11-14 05:38:16 +00:00
|
|
|
if auth_url:
|
|
|
|
self.auth_url = auth_url.rstrip('/')
|
|
|
|
if token:
|
2013-01-30 18:07:29 +01:00
|
|
|
self.auth_token_from_user = token
|
|
|
|
else:
|
|
|
|
self.auth_token_from_user = None
|
2012-11-14 05:38:16 +00:00
|
|
|
if endpoint:
|
|
|
|
self.management_url = endpoint.rstrip('/')
|
2012-10-13 00:15:39 +00:00
|
|
|
self.region_name = region_name
|
2013-02-13 22:52:05 -06:00
|
|
|
|
|
|
|
self.original_ip = original_ip
|
2012-11-16 17:43:05 -06:00
|
|
|
if cacert:
|
|
|
|
self.verify_cert = cacert
|
|
|
|
else:
|
|
|
|
self.verify_cert = True
|
|
|
|
if insecure:
|
|
|
|
self.verify_cert = False
|
|
|
|
self.cert = cert
|
|
|
|
if cert and key:
|
|
|
|
self.cert = (cert, key,)
|
|
|
|
self.domain = ''
|
2011-10-25 16:50:08 -07:00
|
|
|
|
2012-08-16 18:18:22 -07:00
|
|
|
# logging setup
|
2012-10-13 00:15:39 +00:00
|
|
|
self.debug_log = debug
|
2013-05-21 16:25:11 -07:00
|
|
|
if self.debug_log and not _logger.handlers:
|
2012-08-16 18:18:22 -07:00
|
|
|
ch = logging.StreamHandler()
|
|
|
|
_logger.setLevel(logging.DEBUG)
|
|
|
|
_logger.addHandler(ch)
|
2013-02-09 03:22:00 +01:00
|
|
|
if hasattr(requests, 'logging'):
|
2013-02-06 09:36:51 -06:00
|
|
|
requests.logging.getLogger(requests.__name__).addHandler(ch)
|
2012-08-16 18:18:22 -07:00
|
|
|
|
2012-11-08 16:32:17 -08:00
|
|
|
# keyring setup
|
2013-05-22 17:00:53 -04:00
|
|
|
if use_keyring and keyring is None:
|
|
|
|
_logger.warning('Failed to load keyring modules.')
|
|
|
|
self.use_keyring = use_keyring and keyring is not None
|
2012-11-08 16:32:17 -08:00
|
|
|
self.force_new_token = force_new_token
|
|
|
|
self.stale_duration = stale_duration or access.STALE_TOKEN_DURATION
|
|
|
|
self.stale_duration = int(self.stale_duration)
|
|
|
|
|
2013-01-30 18:07:29 +01:00
|
|
|
@property
|
|
|
|
def auth_token(self):
|
|
|
|
if self.auth_token_from_user:
|
|
|
|
return self.auth_token_from_user
|
|
|
|
if self.auth_ref:
|
|
|
|
if self.auth_ref.will_expire_soon(self.stale_duration):
|
|
|
|
self.authenticate()
|
|
|
|
return self.auth_ref.auth_token
|
|
|
|
|
|
|
|
@auth_token.setter
|
|
|
|
def auth_token(self, value):
|
|
|
|
self.auth_token_from_user = value
|
|
|
|
|
|
|
|
@auth_token.deleter
|
2013-02-20 08:48:14 +01:00
|
|
|
def auth_token(self):
|
2013-01-30 18:07:29 +01:00
|
|
|
del self.auth_token_from_user
|
|
|
|
|
2013-02-13 22:52:05 -06:00
|
|
|
@property
|
|
|
|
def service_catalog(self):
|
|
|
|
"""Returns this client's service catalog."""
|
|
|
|
return self.auth_ref.service_catalog
|
|
|
|
|
|
|
|
def has_service_catalog(self):
|
|
|
|
"""Returns True if this client provides a service catalog."""
|
|
|
|
return self.auth_ref.has_service_catalog()
|
|
|
|
|
|
|
|
@property
|
|
|
|
def tenant_id(self):
|
|
|
|
"""Provide read-only backwards compatibility for tenant_id.
|
|
|
|
This is deprecated, use project_id instead.
|
|
|
|
"""
|
|
|
|
return self.project_id
|
|
|
|
|
|
|
|
@property
|
|
|
|
def tenant_name(self):
|
|
|
|
"""Provide read-only backwards compatibility for tenant_name.
|
|
|
|
This is deprecated, use project_name instead.
|
|
|
|
"""
|
|
|
|
return self.project_name
|
|
|
|
|
2012-11-08 16:32:17 -08:00
|
|
|
def authenticate(self, username=None, password=None, tenant_name=None,
|
2013-02-13 22:52:05 -06:00
|
|
|
tenant_id=None, auth_url=None, token=None,
|
|
|
|
user_id=None, domain_name=None, domain_id=None,
|
|
|
|
project_name=None, project_id=None, user_domain_id=None,
|
|
|
|
user_domain_name=None, project_domain_id=None,
|
|
|
|
project_domain_name=None):
|
|
|
|
"""Authenticate user.
|
2012-11-08 16:32:17 -08:00
|
|
|
|
|
|
|
Uses the data provided at instantiation to authenticate against
|
2013-02-13 22:52:05 -06:00
|
|
|
the Identity server. This may use either a username and password
|
2012-11-08 16:32:17 -08:00
|
|
|
or token for authentication. If a tenant name or id was provided
|
|
|
|
then the resulting authenticated client will be scoped to that
|
|
|
|
tenant and contain a service catalog of available endpoints.
|
|
|
|
|
|
|
|
With the v2.0 API, if a tenant name or ID is not provided, the
|
2013-02-13 22:52:05 -06:00
|
|
|
authentication token returned will be 'unscoped' and limited in
|
2012-11-08 16:32:17 -08:00
|
|
|
capabilities until a fully-scoped token is acquired.
|
|
|
|
|
2013-02-13 22:52:05 -06:00
|
|
|
With the v3 API, if a domain name or id was provided then the resulting
|
|
|
|
authenticated client will be scoped to that domain. If a project name
|
|
|
|
or ID is not provided, and the authenticating user has a default
|
|
|
|
project configured, the authentication token returned will be 'scoped'
|
|
|
|
to the default project. Otherwise, the authentication token returned
|
|
|
|
will be 'unscoped' and limited in capabilities until a fully-scoped
|
|
|
|
token is acquired.
|
|
|
|
|
2012-11-08 16:32:17 -08:00
|
|
|
If successful, sets the self.auth_ref and self.auth_token with
|
|
|
|
the returned token. If not already set, will also set
|
|
|
|
self.management_url from the details provided in the token.
|
|
|
|
|
|
|
|
:returns: ``True`` if authentication was successful.
|
|
|
|
:raises: AuthorizationFailure if unable to authenticate or validate
|
|
|
|
the existing authorization token
|
|
|
|
:raises: ValueError if insufficient parameters are used.
|
|
|
|
|
|
|
|
If keyring is used, token is retrieved from keyring instead.
|
|
|
|
Authentication will only be necessary if any of the following
|
|
|
|
conditions are met:
|
|
|
|
|
|
|
|
* keyring is not used
|
|
|
|
* if token is not found in keyring
|
|
|
|
* if token retrieved from keyring is expired or about to
|
|
|
|
expired (as determined by stale_duration)
|
|
|
|
* if force_new_token is true
|
|
|
|
|
|
|
|
"""
|
2013-04-30 17:34:14 +00:00
|
|
|
auth_url = auth_url or self.auth_url
|
2013-02-13 22:52:05 -06:00
|
|
|
user_id = user_id or self.user_id
|
2012-11-08 16:32:17 -08:00
|
|
|
username = username or self.username
|
|
|
|
password = password or self.password
|
2013-02-13 22:52:05 -06:00
|
|
|
|
|
|
|
user_domain_id = user_domain_id or self.user_domain_id
|
|
|
|
user_domain_name = user_domain_name or self.user_domain_name
|
|
|
|
domain_id = domain_id or self.domain_id
|
|
|
|
domain_name = domain_name or self.domain_name
|
|
|
|
project_id = project_id or tenant_id or self.project_id
|
|
|
|
project_name = project_name or tenant_name or self.project_name
|
|
|
|
project_domain_id = project_domain_id or self.project_domain_id
|
|
|
|
project_domain_name = project_domain_name or self.project_domain_name
|
2013-01-30 18:07:29 +01:00
|
|
|
|
|
|
|
if not token:
|
|
|
|
token = self.auth_token_from_user
|
2013-05-28 08:48:44 -05:00
|
|
|
if (not token and self.auth_ref and not
|
|
|
|
self.auth_ref.will_expire_soon(self.stale_duration)):
|
2013-01-30 18:07:29 +01:00
|
|
|
token = self.auth_ref.auth_token
|
2012-11-08 16:32:17 -08:00
|
|
|
|
2013-02-13 22:52:05 -06:00
|
|
|
kwargs = {
|
|
|
|
'auth_url': auth_url,
|
|
|
|
'user_id': user_id,
|
|
|
|
'username': username,
|
|
|
|
'user_domain_id': user_domain_id,
|
|
|
|
'user_domain_name': user_domain_name,
|
|
|
|
'domain_id': domain_id,
|
|
|
|
'domain_name': domain_name,
|
|
|
|
'project_id': project_id,
|
|
|
|
'project_name': project_name,
|
|
|
|
'project_domain_id': project_domain_id,
|
|
|
|
'project_domain_name': project_domain_name,
|
|
|
|
'token': token
|
|
|
|
}
|
|
|
|
(keyring_key, auth_ref) = self.get_auth_ref_from_keyring(**kwargs)
|
2012-11-08 16:32:17 -08:00
|
|
|
new_token_needed = False
|
|
|
|
if auth_ref is None or self.force_new_token:
|
|
|
|
new_token_needed = True
|
2013-02-13 22:52:05 -06:00
|
|
|
kwargs['password'] = password
|
|
|
|
resp, body = self.get_raw_token_from_identity_service(**kwargs)
|
|
|
|
self.auth_ref = access.AccessInfo.factory(resp, body)
|
2012-11-08 16:32:17 -08:00
|
|
|
else:
|
|
|
|
self.auth_ref = auth_ref
|
|
|
|
self.process_token()
|
|
|
|
if new_token_needed:
|
|
|
|
self.store_auth_ref_into_keyring(keyring_key)
|
|
|
|
return True
|
|
|
|
|
2013-02-13 22:52:05 -06:00
|
|
|
def _build_keyring_key(self, **kwargs):
|
|
|
|
"""Create a unique key for keyring.
|
2012-11-08 16:32:17 -08:00
|
|
|
|
|
|
|
Used to store and retrieve auth_ref from keyring.
|
|
|
|
|
2013-02-13 22:52:05 -06:00
|
|
|
Returns a slash-separated string of values ordered by key name.
|
|
|
|
|
2012-11-08 16:32:17 -08:00
|
|
|
"""
|
2013-02-13 22:52:05 -06:00
|
|
|
return '/'.join([kwargs[k] or '?' for k in sorted(kwargs.keys())])
|
2012-11-08 16:32:17 -08:00
|
|
|
|
2013-02-13 22:52:05 -06:00
|
|
|
def get_auth_ref_from_keyring(self, **kwargs):
|
|
|
|
"""Retrieve auth_ref from keyring.
|
2012-11-08 16:32:17 -08:00
|
|
|
|
|
|
|
If auth_ref is found in keyring, (keyring_key, auth_ref) is returned.
|
|
|
|
Otherwise, (keyring_key, None) is returned.
|
|
|
|
|
|
|
|
:returns: (keyring_key, auth_ref) or (keyring_key, None)
|
2013-05-22 17:00:53 -04:00
|
|
|
:returns: or (None, None) if use_keyring is not set in the object
|
2012-11-08 16:32:17 -08:00
|
|
|
|
|
|
|
"""
|
|
|
|
keyring_key = None
|
|
|
|
auth_ref = None
|
|
|
|
if self.use_keyring:
|
2013-02-13 22:52:05 -06:00
|
|
|
keyring_key = self._build_keyring_key(**kwargs)
|
2012-11-08 16:32:17 -08:00
|
|
|
try:
|
|
|
|
auth_ref = keyring.get_password("keystoneclient_auth",
|
|
|
|
keyring_key)
|
|
|
|
if auth_ref:
|
|
|
|
auth_ref = pickle.loads(auth_ref)
|
2013-05-22 17:00:53 -04:00
|
|
|
if auth_ref.will_expire_soon(self.stale_duration):
|
2012-11-08 16:32:17 -08:00
|
|
|
# token has expired, don't use it
|
|
|
|
auth_ref = None
|
|
|
|
except Exception as e:
|
|
|
|
auth_ref = None
|
|
|
|
_logger.warning('Unable to retrieve token from keyring %s' % (
|
|
|
|
e))
|
|
|
|
return (keyring_key, auth_ref)
|
|
|
|
|
|
|
|
def store_auth_ref_into_keyring(self, keyring_key):
|
2013-02-13 22:52:05 -06:00
|
|
|
"""Store auth_ref into keyring.
|
2012-11-08 16:32:17 -08:00
|
|
|
|
|
|
|
"""
|
|
|
|
if self.use_keyring:
|
|
|
|
try:
|
|
|
|
keyring.set_password("keystoneclient_auth",
|
|
|
|
keyring_key,
|
|
|
|
pickle.dumps(self.auth_ref))
|
|
|
|
except Exception as e:
|
|
|
|
_logger.warning("Failed to store token into keyring %s" % (e))
|
|
|
|
|
|
|
|
def process_token(self):
|
2013-02-13 22:52:05 -06:00
|
|
|
"""Extract and process information from the new auth_ref.
|
2012-11-08 16:32:17 -08:00
|
|
|
|
2013-07-18 15:54:41 +10:00
|
|
|
And set the relevant authentication information.
|
2012-11-08 16:32:17 -08:00
|
|
|
"""
|
2013-07-18 15:54:41 +10:00
|
|
|
# if we got a response without a service catalog, set the local
|
|
|
|
# list of tenants for introspection, and leave to client user
|
|
|
|
# to determine what to do. Otherwise, load up the service catalog
|
|
|
|
if self.auth_ref.project_scoped:
|
|
|
|
if not self.auth_ref.tenant_id:
|
|
|
|
raise exceptions.AuthorizationFailure(
|
|
|
|
"Token didn't provide tenant_id")
|
|
|
|
if self.management_url is None and self.auth_ref.management_url:
|
|
|
|
self.management_url = self.auth_ref.management_url[0]
|
|
|
|
self.project_name = self.auth_ref.tenant_name
|
|
|
|
self.project_id = self.auth_ref.tenant_id
|
|
|
|
|
|
|
|
if not self.auth_ref.user_id:
|
|
|
|
raise exceptions.AuthorizationFailure(
|
|
|
|
"Token didn't provide user_id")
|
|
|
|
|
|
|
|
self.user_id = self.auth_ref.user_id
|
|
|
|
|
|
|
|
self.auth_domain_id = self.auth_ref.domain_id
|
|
|
|
self.auth_tenant_id = self.auth_ref.tenant_id
|
|
|
|
self.auth_user_id = self.auth_ref.user_id
|
2012-11-08 16:32:17 -08:00
|
|
|
|
|
|
|
def get_raw_token_from_identity_service(self, auth_url, username=None,
|
|
|
|
password=None, tenant_name=None,
|
2013-02-13 22:52:05 -06:00
|
|
|
tenant_id=None, token=None,
|
|
|
|
user_id=None, user_domain_id=None,
|
|
|
|
user_domain_name=None,
|
|
|
|
domain_id=None, domain_name=None,
|
|
|
|
project_id=None, project_name=None,
|
|
|
|
project_domain_id=None,
|
|
|
|
project_domain_name=None):
|
|
|
|
"""Authenticate against the Identity API and get a token.
|
2011-10-25 16:50:08 -07:00
|
|
|
|
|
|
|
Not implemented here because auth protocols should be API
|
|
|
|
version-specific.
|
2012-10-13 00:15:39 +00:00
|
|
|
|
|
|
|
Expected to authenticate or validate an existing authentication
|
|
|
|
reference already associated with the client. Invoking this call
|
2013-02-13 22:52:05 -06:00
|
|
|
*always* makes a call to the Identity service.
|
2012-11-08 16:32:17 -08:00
|
|
|
|
2013-02-13 22:52:05 -06:00
|
|
|
:returns: (``resp``, ``body``)
|
2012-11-08 16:32:17 -08:00
|
|
|
|
2011-10-25 16:50:08 -07:00
|
|
|
"""
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
def _extract_service_catalog(self, url, body):
|
2013-02-13 22:52:05 -06:00
|
|
|
"""Set the client's service catalog from the response data.
|
2011-10-25 16:50:08 -07:00
|
|
|
|
|
|
|
Not implemented here because data returned may be API
|
|
|
|
version-specific.
|
|
|
|
"""
|
|
|
|
raise NotImplementedError
|
|
|
|
|
2012-08-16 18:18:22 -07:00
|
|
|
def http_log_req(self, args, kwargs):
|
|
|
|
if not self.debug_log:
|
2011-10-25 16:50:08 -07:00
|
|
|
return
|
|
|
|
|
|
|
|
string_parts = ['curl -i']
|
|
|
|
for element in args:
|
|
|
|
if element in ('GET', 'POST'):
|
|
|
|
string_parts.append(' -X %s' % element)
|
|
|
|
else:
|
|
|
|
string_parts.append(' %s' % element)
|
|
|
|
|
|
|
|
for element in kwargs['headers']:
|
|
|
|
header = ' -H "%s: %s"' % (element, kwargs['headers'][element])
|
|
|
|
string_parts.append(header)
|
|
|
|
|
2012-11-16 17:43:05 -06:00
|
|
|
_logger.debug("REQ: %s" % "".join(string_parts))
|
2013-01-24 21:42:14 +00:00
|
|
|
if 'data' in kwargs:
|
|
|
|
_logger.debug("REQ BODY: %s\n" % (kwargs['data']))
|
2012-08-16 18:18:22 -07:00
|
|
|
|
2012-11-16 17:43:05 -06:00
|
|
|
def http_log_resp(self, resp):
|
2012-08-16 18:18:22 -07:00
|
|
|
if self.debug_log:
|
2012-11-16 17:43:05 -06:00
|
|
|
_logger.debug(
|
|
|
|
"RESP: [%s] %s\nRESP BODY: %s\n",
|
|
|
|
resp.status_code,
|
|
|
|
resp.headers,
|
|
|
|
resp.text)
|
2011-10-25 16:50:08 -07:00
|
|
|
|
2012-09-11 11:10:40 -05:00
|
|
|
def serialize(self, entity):
|
|
|
|
return json.dumps(entity)
|
|
|
|
|
2013-01-24 17:46:29 +01:00
|
|
|
@property
|
|
|
|
def service_catalog(self):
|
|
|
|
"""Returns this client's service catalog."""
|
|
|
|
return self.auth_ref.service_catalog
|
|
|
|
|
|
|
|
def has_service_catalog(self):
|
|
|
|
"""Returns True if this client provides a service catalog."""
|
|
|
|
return self.auth_ref.has_service_catalog()
|
|
|
|
|
2011-10-25 16:50:08 -07:00
|
|
|
def request(self, url, method, **kwargs):
|
2013-02-13 22:52:05 -06:00
|
|
|
"""Send an http request with the specified characteristics.
|
2011-10-25 16:50:08 -07:00
|
|
|
|
2012-11-16 17:43:05 -06:00
|
|
|
Wrapper around requests.request to handle tasks such as
|
2011-10-25 16:50:08 -07:00
|
|
|
setting headers, JSON encoding/decoding, and error handling.
|
|
|
|
"""
|
|
|
|
# Copy the kwargs so we can reuse the original in case of redirects
|
|
|
|
request_kwargs = copy.copy(kwargs)
|
|
|
|
request_kwargs.setdefault('headers', kwargs.get('headers', {}))
|
|
|
|
request_kwargs['headers']['User-Agent'] = self.USER_AGENT
|
2012-09-13 15:45:40 +02:00
|
|
|
if self.original_ip:
|
|
|
|
request_kwargs['headers']['Forwarded'] = "for=%s;by=%s" % (
|
|
|
|
self.original_ip, self.USER_AGENT)
|
2011-10-25 16:50:08 -07:00
|
|
|
if 'body' in kwargs:
|
|
|
|
request_kwargs['headers']['Content-Type'] = 'application/json'
|
2012-11-16 17:43:05 -06:00
|
|
|
request_kwargs['data'] = self.serialize(kwargs['body'])
|
|
|
|
del request_kwargs['body']
|
|
|
|
if self.cert:
|
|
|
|
request_kwargs['cert'] = self.cert
|
2013-01-11 21:56:24 -08:00
|
|
|
if self.timeout is not None:
|
|
|
|
request_kwargs.setdefault('timeout', self.timeout)
|
2011-10-25 16:50:08 -07:00
|
|
|
|
2012-08-16 18:18:22 -07:00
|
|
|
self.http_log_req((url, method,), request_kwargs)
|
2013-04-02 21:25:30 -07:00
|
|
|
|
|
|
|
try:
|
|
|
|
resp = requests.request(
|
|
|
|
method,
|
|
|
|
url,
|
|
|
|
verify=self.verify_cert,
|
|
|
|
**request_kwargs)
|
|
|
|
except requests.ConnectionError:
|
|
|
|
msg = 'Unable to establish connection to %s' % url
|
|
|
|
raise exceptions.ClientException(msg)
|
2012-11-16 17:43:05 -06:00
|
|
|
|
|
|
|
self.http_log_resp(resp)
|
|
|
|
|
|
|
|
if resp.text:
|
2011-10-25 16:50:08 -07:00
|
|
|
try:
|
2012-11-16 17:43:05 -06:00
|
|
|
body = json.loads(resp.text)
|
2013-03-13 20:59:03 -07:00
|
|
|
except (ValueError, TypeError):
|
2012-11-16 17:43:05 -06:00
|
|
|
body = None
|
|
|
|
_logger.debug("Could not decode JSON from body: %s"
|
|
|
|
% resp.text)
|
2011-10-25 16:50:08 -07:00
|
|
|
else:
|
|
|
|
_logger.debug("No body was returned.")
|
|
|
|
body = None
|
|
|
|
|
2013-03-13 20:59:03 -07:00
|
|
|
if resp.status_code >= 400:
|
|
|
|
_logger.debug(
|
|
|
|
"Request returned failure status: %s",
|
|
|
|
resp.status_code)
|
|
|
|
raise exceptions.from_response(resp, body or resp.text)
|
|
|
|
elif resp.status_code in (301, 302, 305):
|
|
|
|
# Redirected. Reissue the request to the new location.
|
|
|
|
return self.request(resp.headers['location'], method, **kwargs)
|
|
|
|
|
2011-10-25 16:50:08 -07:00
|
|
|
return resp, body
|
|
|
|
|
|
|
|
def _cs_request(self, url, method, **kwargs):
|
2013-02-13 22:52:05 -06:00
|
|
|
"""Makes an authenticated request to keystone endpoint by
|
2012-10-13 00:15:39 +00:00
|
|
|
concatenating self.management_url and url and passing in method and
|
2013-02-13 22:52:05 -06:00
|
|
|
any associated kwargs.
|
|
|
|
"""
|
2011-10-25 16:50:08 -07:00
|
|
|
|
2012-11-23 09:17:24 +00:00
|
|
|
is_management = kwargs.pop('management', True)
|
|
|
|
|
|
|
|
if is_management and self.management_url is None:
|
2012-10-13 00:15:39 +00:00
|
|
|
raise exceptions.AuthorizationFailure(
|
|
|
|
'Current authorization does not have a known management url')
|
2012-11-23 09:17:24 +00:00
|
|
|
|
|
|
|
url_to_use = self.auth_url
|
|
|
|
if is_management:
|
|
|
|
url_to_use = self.management_url
|
|
|
|
|
2011-10-25 16:50:08 -07:00
|
|
|
kwargs.setdefault('headers', {})
|
2013-04-30 17:34:14 +00:00
|
|
|
if self.auth_token:
|
|
|
|
kwargs['headers']['X-Auth-Token'] = self.auth_token
|
2011-10-25 16:50:08 -07:00
|
|
|
|
2012-11-23 09:17:24 +00:00
|
|
|
resp, body = self.request(url_to_use + url, method,
|
2012-10-13 00:15:39 +00:00
|
|
|
**kwargs)
|
|
|
|
return resp, body
|
2011-10-25 16:50:08 -07:00
|
|
|
|
|
|
|
def get(self, url, **kwargs):
|
|
|
|
return self._cs_request(url, 'GET', **kwargs)
|
|
|
|
|
2012-09-11 11:06:54 -05:00
|
|
|
def head(self, url, **kwargs):
|
|
|
|
return self._cs_request(url, 'HEAD', **kwargs)
|
|
|
|
|
2011-10-25 16:50:08 -07:00
|
|
|
def post(self, url, **kwargs):
|
|
|
|
return self._cs_request(url, 'POST', **kwargs)
|
|
|
|
|
|
|
|
def put(self, url, **kwargs):
|
|
|
|
return self._cs_request(url, 'PUT', **kwargs)
|
|
|
|
|
2012-09-11 11:06:54 -05:00
|
|
|
def patch(self, url, **kwargs):
|
|
|
|
return self._cs_request(url, 'PATCH', **kwargs)
|
|
|
|
|
2011-10-25 16:50:08 -07:00
|
|
|
def delete(self, url, **kwargs):
|
|
|
|
return self._cs_request(url, 'DELETE', **kwargs)
|