Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 1 | import json |
Jay Pipes | 7f75763 | 2011-12-02 15:53:32 -0500 | [diff] [blame] | 2 | import httplib2 |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 3 | import logging |
| 4 | import sys |
Eoghan Glynn | a559897 | 2012-03-01 09:27:17 -0500 | [diff] [blame] | 5 | import time |
Daryl Walleck | ed8bef3 | 2011-12-05 23:02:08 -0600 | [diff] [blame] | 6 | from tempest import exceptions |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 7 | |
| 8 | |
Eoghan Glynn | a559897 | 2012-03-01 09:27:17 -0500 | [diff] [blame] | 9 | # redrive rate limited calls at most twice |
| 10 | MAX_RECURSION_DEPTH = 2 |
| 11 | |
| 12 | |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 13 | class RestClient(object): |
| 14 | |
Daryl Walleck | 587385b | 2012-03-03 13:00:26 -0600 | [diff] [blame] | 15 | def __init__(self, config, user, password, auth_url, service, |
| 16 | tenant_name=None): |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 17 | self.log = logging.getLogger(__name__) |
| 18 | self.log.setLevel(logging.ERROR) |
Jay Pipes | 7f75763 | 2011-12-02 15:53:32 -0500 | [diff] [blame] | 19 | self.config = config |
Daryl Walleck | 587385b | 2012-03-03 13:00:26 -0600 | [diff] [blame] | 20 | if self.config.identity.strategy == 'keystone': |
| 21 | self.token, self.base_url = self.keystone_auth(user, |
| 22 | password, |
| 23 | auth_url, |
| 24 | service, |
| 25 | tenant_name) |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 26 | else: |
| 27 | self.token, self.base_url = self.basic_auth(user, |
Daryl Walleck | 587385b | 2012-03-03 13:00:26 -0600 | [diff] [blame] | 28 | password, |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 29 | auth_url) |
| 30 | |
Daryl Walleck | 587385b | 2012-03-03 13:00:26 -0600 | [diff] [blame] | 31 | def basic_auth(self, user, password, auth_url): |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 32 | """ |
| 33 | Provides authentication for the target API |
| 34 | """ |
| 35 | |
| 36 | params = {} |
| 37 | params['headers'] = {'User-Agent': 'Test-Client', 'X-Auth-User': user, |
Daryl Walleck | 587385b | 2012-03-03 13:00:26 -0600 | [diff] [blame] | 38 | 'X-Auth-Key': password} |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 39 | |
| 40 | self.http_obj = httplib2.Http() |
| 41 | resp, body = self.http_obj.request(auth_url, 'GET', **params) |
| 42 | try: |
| 43 | return resp['x-auth-token'], resp['x-server-management-url'] |
| 44 | except: |
| 45 | raise |
| 46 | |
Daryl Walleck | 587385b | 2012-03-03 13:00:26 -0600 | [diff] [blame] | 47 | def keystone_auth(self, user, password, auth_url, service, tenant_name): |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 48 | """ |
Daryl Walleck | 587385b | 2012-03-03 13:00:26 -0600 | [diff] [blame] | 49 | Provides authentication via Keystone |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 50 | """ |
| 51 | |
| 52 | creds = {'auth': { |
| 53 | 'passwordCredentials': { |
| 54 | 'username': user, |
Daryl Walleck | 587385b | 2012-03-03 13:00:26 -0600 | [diff] [blame] | 55 | 'password': password, |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 56 | }, |
| 57 | 'tenantName': tenant_name |
| 58 | } |
| 59 | } |
| 60 | |
| 61 | self.http_obj = httplib2.Http() |
| 62 | headers = {'Content-Type': 'application/json'} |
| 63 | body = json.dumps(creds) |
| 64 | resp, body = self.http_obj.request(auth_url, 'POST', |
| 65 | headers=headers, body=body) |
| 66 | |
Jay Pipes | 7f75763 | 2011-12-02 15:53:32 -0500 | [diff] [blame] | 67 | if resp.status == 200: |
| 68 | try: |
| 69 | auth_data = json.loads(body)['access'] |
| 70 | token = auth_data['token']['id'] |
Jay Pipes | 7f75763 | 2011-12-02 15:53:32 -0500 | [diff] [blame] | 71 | except Exception, e: |
Adam Gandelman | e2d46b4 | 2012-01-03 17:40:44 -0800 | [diff] [blame] | 72 | print "Failed to obtain token for user: %s" % e |
Jay Pipes | 7f75763 | 2011-12-02 15:53:32 -0500 | [diff] [blame] | 73 | raise |
Adam Gandelman | e2d46b4 | 2012-01-03 17:40:44 -0800 | [diff] [blame] | 74 | |
| 75 | mgmt_url = None |
| 76 | for ep in auth_data['serviceCatalog']: |
Daryl Walleck | b90a1a6 | 2012-02-27 11:23:10 -0600 | [diff] [blame] | 77 | if ep["type"] == service: |
Adam Gandelman | e2d46b4 | 2012-01-03 17:40:44 -0800 | [diff] [blame] | 78 | mgmt_url = ep['endpoints'][0]['publicURL'] |
Jay Pipes | 745259c | 2012-01-23 23:24:54 -0500 | [diff] [blame] | 79 | # See LP#920817. The tenantId is *supposed* |
| 80 | # to be returned for each endpoint accorsing to the |
| 81 | # Keystone spec. But... it isn't, so we have to parse |
| 82 | # the tenant ID out of hte public URL :( |
| 83 | tenant_id = mgmt_url.split('/')[-1] |
Adam Gandelman | e2d46b4 | 2012-01-03 17:40:44 -0800 | [diff] [blame] | 84 | break |
| 85 | |
| 86 | if mgmt_url == None: |
| 87 | raise exceptions.EndpointNotFound(service) |
| 88 | |
| 89 | #TODO (dwalleck): This is a horrible stopgap. |
| 90 | #Need to join strings more cleanly |
| 91 | temp = mgmt_url.rsplit('/') |
| 92 | service_url = temp[0] + '//' + temp[2] + '/' + temp[3] + '/' |
Jay Pipes | 745259c | 2012-01-23 23:24:54 -0500 | [diff] [blame] | 93 | management_url = service_url + tenant_id |
Adam Gandelman | e2d46b4 | 2012-01-03 17:40:44 -0800 | [diff] [blame] | 94 | return token, management_url |
Jay Pipes | 7f75763 | 2011-12-02 15:53:32 -0500 | [diff] [blame] | 95 | elif resp.status == 401: |
Daryl Walleck | a22f57b | 2012-03-20 16:52:07 -0500 | [diff] [blame^] | 96 | raise exceptions.AuthenticationFailure(user=user, |
| 97 | password=password) |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 98 | |
| 99 | def post(self, url, body, headers): |
| 100 | return self.request('POST', url, headers, body) |
| 101 | |
| 102 | def get(self, url): |
| 103 | return self.request('GET', url) |
| 104 | |
| 105 | def delete(self, url): |
| 106 | return self.request('DELETE', url) |
| 107 | |
| 108 | def put(self, url, body, headers): |
| 109 | return self.request('PUT', url, headers, body) |
| 110 | |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 111 | def _log(self, req_url, body, resp, resp_body): |
| 112 | self.log.error('Request URL: ' + req_url) |
| 113 | self.log.error('Request Body: ' + str(body)) |
| 114 | self.log.error('Response Headers: ' + str(resp)) |
| 115 | self.log.error('Response Body: ' + str(resp_body)) |
| 116 | |
Eoghan Glynn | a559897 | 2012-03-01 09:27:17 -0500 | [diff] [blame] | 117 | def request(self, method, url, headers=None, body=None, depth=0): |
Daryl Walleck | e5b83d4 | 2011-11-10 14:39:02 -0600 | [diff] [blame] | 118 | """A simple HTTP request interface.""" |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 119 | |
| 120 | self.http_obj = httplib2.Http() |
| 121 | if headers == None: |
| 122 | headers = {} |
| 123 | headers['X-Auth-Token'] = self.token |
| 124 | |
| 125 | req_url = "%s/%s" % (self.base_url, url) |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 126 | resp, resp_body = self.http_obj.request(req_url, method, |
Daryl Walleck | 1465d61 | 2011-11-02 02:22:15 -0500 | [diff] [blame] | 127 | headers=headers, body=body) |
Jay Pipes | 5135bfc | 2012-01-05 15:46:49 -0500 | [diff] [blame] | 128 | |
| 129 | if resp.status == 404: |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 130 | self._log(req_url, body, resp, resp_body) |
| 131 | raise exceptions.NotFound(resp_body) |
Jay Pipes | 5135bfc | 2012-01-05 15:46:49 -0500 | [diff] [blame] | 132 | |
Daryl Walleck | adea1fa | 2011-11-15 18:36:39 -0600 | [diff] [blame] | 133 | if resp.status == 400: |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 134 | resp_body = json.loads(resp_body) |
| 135 | self._log(req_url, body, resp, resp_body) |
| 136 | raise exceptions.BadRequest(resp_body['badRequest']['message']) |
Daryl Walleck | adea1fa | 2011-11-15 18:36:39 -0600 | [diff] [blame] | 137 | |
David Kranz | 5a23d86 | 2012-02-14 09:48:55 -0500 | [diff] [blame] | 138 | if resp.status == 409: |
| 139 | resp_body = json.loads(resp_body) |
| 140 | self._log(req_url, body, resp, resp_body) |
| 141 | raise exceptions.Duplicate(resp_body) |
| 142 | |
Daryl Walleck | ed8bef3 | 2011-12-05 23:02:08 -0600 | [diff] [blame] | 143 | if resp.status == 413: |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 144 | resp_body = json.loads(resp_body) |
| 145 | self._log(req_url, body, resp, resp_body) |
| 146 | if 'overLimit' in resp_body: |
| 147 | raise exceptions.OverLimit(resp_body['overLimit']['message']) |
Eoghan Glynn | a559897 | 2012-03-01 09:27:17 -0500 | [diff] [blame] | 148 | elif depth < MAX_RECURSION_DEPTH: |
| 149 | delay = resp['Retry-After'] if 'Retry-After' in resp else 60 |
| 150 | time.sleep(int(delay)) |
| 151 | return self.request(method, url, headers, body, depth + 1) |
Jay Pipes | 9b04384 | 2012-01-23 23:34:26 -0500 | [diff] [blame] | 152 | else: |
| 153 | raise exceptions.RateLimitExceeded( |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 154 | message=resp_body['overLimitFault']['message'], |
| 155 | details=resp_body['overLimitFault']['details']) |
Brian Lamar | 12d9b29 | 2011-12-08 12:41:21 -0500 | [diff] [blame] | 156 | |
Daryl Walleck | ed8bef3 | 2011-12-05 23:02:08 -0600 | [diff] [blame] | 157 | if resp.status in (500, 501): |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 158 | resp_body = json.loads(resp_body) |
| 159 | self._log(req_url, body, resp, resp_body) |
Daryl Walleck | f008703 | 2011-12-18 13:37:05 -0600 | [diff] [blame] | 160 | #I'm seeing both computeFault and cloudServersFault come back. |
| 161 | #Will file a bug to fix, but leave as is for now. |
| 162 | |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 163 | if 'cloudServersFault' in resp_body: |
| 164 | message = resp_body['cloudServersFault']['message'] |
Daryl Walleck | f008703 | 2011-12-18 13:37:05 -0600 | [diff] [blame] | 165 | else: |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 166 | message = resp_body['computeFault']['message'] |
Daryl Walleck | f008703 | 2011-12-18 13:37:05 -0600 | [diff] [blame] | 167 | raise exceptions.ComputeFault(message) |
Daryl Walleck | ed8bef3 | 2011-12-05 23:02:08 -0600 | [diff] [blame] | 168 | |
David Kranz | 5a23d86 | 2012-02-14 09:48:55 -0500 | [diff] [blame] | 169 | if resp.status >= 400: |
| 170 | resp_body = json.loads(resp_body) |
| 171 | self._log(req_url, body, resp, resp_body) |
| 172 | raise exceptions.TempestException(str(resp.status)) |
| 173 | |
Daryl Walleck | 8a707db | 2012-01-25 00:46:24 -0600 | [diff] [blame] | 174 | return resp, resp_body |