# Copyright (c) 2011 OpenStack, LLC. # All Rights Reserved. # # 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. import time import urlparse try: import json except ImportError: import simplejson as json from novaclient.client import HTTPClient from novaclient.v1_1.client import Client from novaclient import exceptions as nova_exceptions from reddwarfclient import exceptions class ReddwarfHTTPClient(HTTPClient): """ Class for overriding the HTTP authenticate call and making it specific to reddwarf """ def __init__(self, user, apikey, tenant, auth_url, service_name, service_url=None, auth_strategy=None, **kwargs): super(ReddwarfHTTPClient, self).__init__(user, apikey, tenant, auth_url, **kwargs) self.api_key = apikey self.tenant = tenant self.service = service_name self.management_url = service_url if auth_strategy == "basic": self.auth_strategy = self.basic_auth else: self.auth_strategy = super(ReddwarfHTTPClient, self).authenticate def authenticate(self): self.auth_strategy() def _authenticate_without_tokens(self, url, body): """Authenticate and extract the service catalog.""" #TODO(tim.simpson): Copy pasta from Nova client's "_authenticate" but # does not append "tokens" to the url. # Make sure we follow redirects when trying to reach Keystone tmp_follow_all_redirects = self.follow_all_redirects self.follow_all_redirects = True try: resp, body = self.request(url, "POST", body=body) finally: self.follow_all_redirects = tmp_follow_all_redirects return resp, body def basic_auth(self): """Authenticate against a v2.0 auth service.""" auth_url = self.auth_url body = {"credentials": {"username": self.user, "key": self.password}} resp, resp_body = self._authenticate_without_tokens(auth_url, body) try: self.auth_token = resp_body['auth']['token']['id'] except KeyError: raise nova_exceptions.AuthorizationFailure() catalog = resp_body['auth']['serviceCatalog'] if 'cloudDatabases' not in catalog: raise nova_exceptions.EndpointNotFound() endpoints = catalog['cloudDatabases'] for endpoint in endpoints: if self.region_name is None or \ endpoint['region'] == self.region_name: self.management_url = endpoint['publicURL'] return raise nova_exceptions.EndpointNotFound() def _get_token(self, path, req_body): """Set the management url and auth token""" token_url = urlparse.urljoin(self.auth_url, path) resp, body = self.request(token_url, "POST", body=req_body) if 'access' in body: if not self.management_url: # Assume the new Keystone lite: catalog = body['access']['serviceCatalog'] for service in catalog: if service['name'] == self.service: self.management_url = service['adminURL'] self.auth_token = body['access']['token']['id'] else: # Assume pre-Keystone Light: try: if not self.management_url: keys = ['auth', 'serviceCatalog', self.service, 0, 'publicURL'] url = body for key in keys: url = url[key] self.management_url = url self.auth_token = body['auth']['token']['id'] except KeyError: raise NotImplementedError("Service: %s is not available" % self.service) def request(self, *args, **kwargs): #TODO(tim.simpson): Copy and pasted from novaclient, since we raise # extra exception subclasses not raised there. kwargs.setdefault('headers', kwargs.get('headers', {})) kwargs['headers']['User-Agent'] = self.USER_AGENT kwargs['headers']['Accept'] = 'application/json' if 'body' in kwargs: kwargs['headers']['Content-Type'] = 'application/json' kwargs['body'] = json.dumps(kwargs['body']) resp, body = super(HTTPClient, self).request(*args, **kwargs) self.http_log(args, kwargs, resp, body) if body: try: body = json.loads(body) except ValueError: pass else: body = None if resp.status in (400, 401, 403, 404, 408, 409, 413, 500, 501): raise exceptions.from_response(resp, body) return resp, body class Dbaas(Client): """ Top-level object to access the Rackspace Database as a Service API. Create an instance with your creds:: >>> red = Dbaas(USERNAME, API_KEY, TENANT, AUTH_URL, SERVICE_NAME, SERVICE_URL) Then call methods on its managers:: >>> red.instances.list() ... >>> red.flavors.list() ... &c. """ def __init__(self, username, api_key, tenant=None, auth_url=None, service_type='reddwarf', service_name='Reddwarf Service', service_url=None, insecure=False, auth_strategy=None, region_name=None): from reddwarfclient.versions import Versions from reddwarfclient.databases import Databases from reddwarfclient.flavors import Flavors from reddwarfclient.instances import Instances from reddwarfclient.users import Users from reddwarfclient.root import Root from reddwarfclient.hosts import Hosts from reddwarfclient.storage import StorageInfo from reddwarfclient.management import Management from reddwarfclient.accounts import Accounts from reddwarfclient.config import Configs from reddwarfclient.diagnostics import Interrogator super(Dbaas, self).__init__(username, api_key, tenant, auth_url) self.client = ReddwarfHTTPClient(username, api_key, tenant, auth_url, service_type=service_type, service_name=service_name, service_url=service_url, insecure=insecure, auth_strategy=auth_strategy, region_name=region_name) self.versions = Versions(self) self.databases = Databases(self) self.flavors = Flavors(self) self.instances = Instances(self) self.users = Users(self) self.root = Root(self) self.hosts = Hosts(self) self.storage = StorageInfo(self) self.management = Management(self) self.accounts = Accounts(self) self.configs = Configs(self) self.diagnostics = Interrogator(self)