ZhiQiang Fan | 39f9722 | 2013-09-20 04:49:44 +0800 | [diff] [blame] | 1 | # Copyright 2012 OpenStack Foundation |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 2 | # All Rights Reserved. |
| 3 | # |
| 4 | # Licensed under the Apache License, Version 2.0 (the "License"); you may |
| 5 | # not use this file except in compliance with the License. You may obtain |
| 6 | # a copy of the License at |
| 7 | # |
| 8 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | # |
| 10 | # Unless required by applicable law or agreed to in writing, software |
| 11 | # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| 12 | # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| 13 | # License for the specific language governing permissions and limitations |
| 14 | # under the License. |
| 15 | |
Matthew Treinish | 7142668 | 2015-04-23 11:19:38 -0400 | [diff] [blame] | 16 | import six |
Matthew Treinish | 6421af8 | 2015-04-23 09:47:50 -0400 | [diff] [blame] | 17 | from six.moves import http_client as httplib |
Matthew Treinish | f077dd2 | 2015-04-23 09:37:41 -0400 | [diff] [blame] | 18 | from six.moves.urllib import parse as urlparse |
Daisuke Morita | 2c87d37 | 2013-10-31 19:39:19 +0900 | [diff] [blame] | 19 | |
Ken'ichi Ohmichi | 19f6881 | 2016-03-02 14:09:17 +0900 | [diff] [blame] | 20 | from tempest.lib.common import rest_client |
Brian Ober | f2deb18 | 2016-04-12 19:28:04 +0000 | [diff] [blame] | 21 | from tempest.lib import exceptions |
Matthew Treinish | 684d899 | 2014-01-30 16:27:40 +0000 | [diff] [blame] | 22 | |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 23 | |
Ken'ichi Ohmichi | 19f6881 | 2016-03-02 14:09:17 +0900 | [diff] [blame] | 24 | class ObjectClient(rest_client.RestClient): |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 25 | |
Daisuke Morita | cf6f695 | 2014-03-19 21:25:50 +0900 | [diff] [blame] | 26 | def create_object(self, container, object_name, data, |
Ken'ichi Ohmichi | 179ea57 | 2015-01-01 14:04:49 +0000 | [diff] [blame] | 27 | params=None, metadata=None, headers=None): |
Sean Dague | f237ccb | 2013-01-04 15:19:14 -0500 | [diff] [blame] | 28 | """Create storage object.""" |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 29 | |
Ken'ichi Ohmichi | 179ea57 | 2015-01-01 14:04:49 +0000 | [diff] [blame] | 30 | if headers is None: |
| 31 | headers = self.get_headers() |
Martina Kollarova | 03720a5 | 2013-06-18 15:08:46 +0200 | [diff] [blame] | 32 | if not data: |
| 33 | headers['content-length'] = '0' |
Daisuke Morita | cf6f695 | 2014-03-19 21:25:50 +0900 | [diff] [blame] | 34 | if metadata: |
| 35 | for key in metadata: |
| 36 | headers[str(key)] = metadata[key] |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 37 | url = "%s/%s" % (str(container), str(object_name)) |
Daisuke Morita | 2c87d37 | 2013-10-31 19:39:19 +0900 | [diff] [blame] | 38 | if params: |
Matthew Treinish | 8912814 | 2015-04-23 10:44:30 -0400 | [diff] [blame] | 39 | url += '?%s' % urlparse.urlencode(params) |
Daisuke Morita | 2c87d37 | 2013-10-31 19:39:19 +0900 | [diff] [blame] | 40 | |
Martina Kollarova | 03720a5 | 2013-06-18 15:08:46 +0200 | [diff] [blame] | 41 | resp, body = self.put(url, data, headers) |
JordanP | a84dde3 | 2014-11-14 15:47:42 +0100 | [diff] [blame] | 42 | self.expected_success(201, resp.status) |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 43 | return resp, body |
| 44 | |
| 45 | def update_object(self, container, object_name, data): |
Sean Dague | f237ccb | 2013-01-04 15:19:14 -0500 | [diff] [blame] | 46 | """Upload data to replace current storage object.""" |
JordanP | a84dde3 | 2014-11-14 15:47:42 +0100 | [diff] [blame] | 47 | resp, body = self.create_object(container, object_name, data) |
| 48 | self.expected_success(201, resp.status) |
| 49 | return resp, body |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 50 | |
Daisuke Morita | 2c87d37 | 2013-10-31 19:39:19 +0900 | [diff] [blame] | 51 | def delete_object(self, container, object_name, params=None): |
Sean Dague | f237ccb | 2013-01-04 15:19:14 -0500 | [diff] [blame] | 52 | """Delete storage object.""" |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 53 | url = "%s/%s" % (str(container), str(object_name)) |
Daisuke Morita | 2c87d37 | 2013-10-31 19:39:19 +0900 | [diff] [blame] | 54 | if params: |
Matthew Treinish | 8912814 | 2015-04-23 10:44:30 -0400 | [diff] [blame] | 55 | url += '?%s' % urlparse.urlencode(params) |
vponomaryov | 67b58fe | 2014-02-06 19:05:41 +0200 | [diff] [blame] | 56 | resp, body = self.delete(url, headers={}) |
JordanP | a84dde3 | 2014-11-14 15:47:42 +0100 | [diff] [blame] | 57 | self.expected_success([200, 204], resp.status) |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 58 | return resp, body |
| 59 | |
| 60 | def update_object_metadata(self, container, object_name, metadata, |
| 61 | metadata_prefix='X-Object-Meta-'): |
Sean Dague | f237ccb | 2013-01-04 15:19:14 -0500 | [diff] [blame] | 62 | """Add, remove, or change X-Object-Meta metadata for storage object.""" |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 63 | |
| 64 | headers = {} |
| 65 | for key in metadata: |
| 66 | headers["%s%s" % (str(metadata_prefix), str(key))] = metadata[key] |
| 67 | |
| 68 | url = "%s/%s" % (str(container), str(object_name)) |
| 69 | resp, body = self.post(url, None, headers=headers) |
JordanP | a84dde3 | 2014-11-14 15:47:42 +0100 | [diff] [blame] | 70 | self.expected_success(202, resp.status) |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 71 | return resp, body |
| 72 | |
| 73 | def list_object_metadata(self, container, object_name): |
Sean Dague | f237ccb | 2013-01-04 15:19:14 -0500 | [diff] [blame] | 74 | """List all storage object X-Object-Meta- metadata.""" |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 75 | |
| 76 | url = "%s/%s" % (str(container), str(object_name)) |
| 77 | resp, body = self.head(url) |
JordanP | a84dde3 | 2014-11-14 15:47:42 +0100 | [diff] [blame] | 78 | self.expected_success(200, resp.status) |
dwalleck | 5d73443 | 2012-10-04 01:11:47 -0500 | [diff] [blame] | 79 | return resp, body |
Larisa Ustalov | 6c3c780 | 2012-11-05 12:25:19 +0200 | [diff] [blame] | 80 | |
Daisuke Morita | 6d50268 | 2014-03-19 21:08:54 +0900 | [diff] [blame] | 81 | def get_object(self, container, object_name, metadata=None): |
Larisa Ustalov | 6c3c780 | 2012-11-05 12:25:19 +0200 | [diff] [blame] | 82 | """Retrieve object's data.""" |
| 83 | |
Daisuke Morita | 6d50268 | 2014-03-19 21:08:54 +0900 | [diff] [blame] | 84 | headers = {} |
| 85 | if metadata: |
| 86 | for key in metadata: |
| 87 | headers[str(key)] = metadata[key] |
| 88 | |
Larisa Ustalov | 6c3c780 | 2012-11-05 12:25:19 +0200 | [diff] [blame] | 89 | url = "{0}/{1}".format(container, object_name) |
Daisuke Morita | 6d50268 | 2014-03-19 21:08:54 +0900 | [diff] [blame] | 90 | resp, body = self.get(url, headers=headers) |
JordanP | a84dde3 | 2014-11-14 15:47:42 +0100 | [diff] [blame] | 91 | self.expected_success([200, 206], resp.status) |
Larisa Ustalov | 6c3c780 | 2012-11-05 12:25:19 +0200 | [diff] [blame] | 92 | return resp, body |
| 93 | |
rajalakshmi-ganesan | 925e239 | 2012-11-30 19:17:24 +0530 | [diff] [blame] | 94 | def copy_object_in_same_container(self, container, src_object_name, |
| 95 | dest_object_name, metadata=None): |
Sean Dague | f237ccb | 2013-01-04 15:19:14 -0500 | [diff] [blame] | 96 | """Copy storage object's data to the new object using PUT.""" |
Larisa Ustalov | 6c3c780 | 2012-11-05 12:25:19 +0200 | [diff] [blame] | 97 | |
| 98 | url = "{0}/{1}".format(container, dest_object_name) |
| 99 | headers = {} |
| 100 | headers['X-Copy-From'] = "%s/%s" % (str(container), |
| 101 | str(src_object_name)) |
| 102 | headers['content-length'] = '0' |
| 103 | if metadata: |
| 104 | for key in metadata: |
| 105 | headers[str(key)] = metadata[key] |
| 106 | |
| 107 | resp, body = self.put(url, None, headers=headers) |
JordanP | a84dde3 | 2014-11-14 15:47:42 +0100 | [diff] [blame] | 108 | self.expected_success(201, resp.status) |
Larisa Ustalov | 6c3c780 | 2012-11-05 12:25:19 +0200 | [diff] [blame] | 109 | return resp, body |
| 110 | |
rajalakshmi-ganesan | 925e239 | 2012-11-30 19:17:24 +0530 | [diff] [blame] | 111 | def copy_object_across_containers(self, src_container, src_object_name, |
| 112 | dst_container, dst_object_name, |
| 113 | metadata=None): |
Sean Dague | f237ccb | 2013-01-04 15:19:14 -0500 | [diff] [blame] | 114 | """Copy storage object's data to the new object using PUT.""" |
rajalakshmi-ganesan | 925e239 | 2012-11-30 19:17:24 +0530 | [diff] [blame] | 115 | |
| 116 | url = "{0}/{1}".format(dst_container, dst_object_name) |
| 117 | headers = {} |
| 118 | headers['X-Copy-From'] = "%s/%s" % (str(src_container), |
| 119 | str(src_object_name)) |
| 120 | headers['content-length'] = '0' |
| 121 | if metadata: |
| 122 | for key in metadata: |
| 123 | headers[str(key)] = metadata[key] |
| 124 | |
| 125 | resp, body = self.put(url, None, headers=headers) |
JordanP | a84dde3 | 2014-11-14 15:47:42 +0100 | [diff] [blame] | 126 | self.expected_success(201, resp.status) |
rajalakshmi-ganesan | 925e239 | 2012-11-30 19:17:24 +0530 | [diff] [blame] | 127 | return resp, body |
| 128 | |
Larisa Ustalov | 6c3c780 | 2012-11-05 12:25:19 +0200 | [diff] [blame] | 129 | def copy_object_2d_way(self, container, src_object_name, dest_object_name, |
| 130 | metadata=None): |
Sean Dague | f237ccb | 2013-01-04 15:19:14 -0500 | [diff] [blame] | 131 | """Copy storage object's data to the new object using COPY.""" |
Larisa Ustalov | 6c3c780 | 2012-11-05 12:25:19 +0200 | [diff] [blame] | 132 | |
| 133 | url = "{0}/{1}".format(container, src_object_name) |
| 134 | headers = {} |
| 135 | headers['Destination'] = "%s/%s" % (str(container), |
| 136 | str(dest_object_name)) |
| 137 | if metadata: |
| 138 | for key in metadata: |
| 139 | headers[str(key)] = metadata[key] |
| 140 | |
| 141 | resp, body = self.copy(url, headers=headers) |
JordanP | a84dde3 | 2014-11-14 15:47:42 +0100 | [diff] [blame] | 142 | self.expected_success(201, resp.status) |
Larisa Ustalov | 6c3c780 | 2012-11-05 12:25:19 +0200 | [diff] [blame] | 143 | return resp, body |
harika-vakadi | 1a9ad61 | 2012-12-14 19:12:08 +0530 | [diff] [blame] | 144 | |
harika-vakadi | 7cfc518 | 2013-01-16 13:59:25 +0530 | [diff] [blame] | 145 | def create_object_segments(self, container, object_name, segment, data): |
| 146 | """Creates object segments.""" |
| 147 | url = "{0}/{1}/{2}".format(container, object_name, segment) |
Valeriy Ponomaryov | 88686d8 | 2014-02-16 12:24:51 +0200 | [diff] [blame] | 148 | resp, body = self.put(url, data) |
JordanP | a84dde3 | 2014-11-14 15:47:42 +0100 | [diff] [blame] | 149 | self.expected_success(201, resp.status) |
ravikumar-venkatesan | e43b77a | 2013-01-09 07:27:13 +0000 | [diff] [blame] | 150 | return resp, body |
| 151 | |
Jordan Pittier | 4408c4a | 2016-04-29 15:05:09 +0200 | [diff] [blame] | 152 | def put_object_with_chunk(self, container, name, contents): |
| 153 | """Put an object with Transfer-Encoding header |
| 154 | |
| 155 | :param container: name of the container |
| 156 | :type container: string |
| 157 | :param name: name of the object |
| 158 | :type name: string |
| 159 | :param contents: object data |
| 160 | :type contents: iterable |
| 161 | """ |
Daisuke Morita | 02f840b | 2014-03-19 20:51:01 +0900 | [diff] [blame] | 162 | headers = {'Transfer-Encoding': 'chunked'} |
| 163 | if self.token: |
| 164 | headers['X-Auth-Token'] = self.token |
| 165 | |
Jordan Pittier | 4408c4a | 2016-04-29 15:05:09 +0200 | [diff] [blame] | 166 | url = "%s/%s" % (container, name) |
| 167 | resp, body = self.put( |
| 168 | url, headers=headers, |
| 169 | body=contents, |
| 170 | chunked=True |
| 171 | ) |
Daisuke Morita | 02f840b | 2014-03-19 20:51:01 +0900 | [diff] [blame] | 172 | |
| 173 | self._error_checker('PUT', None, headers, contents, resp, body) |
JordanP | a84dde3 | 2014-11-14 15:47:42 +0100 | [diff] [blame] | 174 | self.expected_success(201, resp.status) |
Jordan Pittier | 4408c4a | 2016-04-29 15:05:09 +0200 | [diff] [blame] | 175 | return resp.status, resp.reason, resp |
Daisuke Morita | 02f840b | 2014-03-19 20:51:01 +0900 | [diff] [blame] | 176 | |
Daisuke Morita | 02f840b | 2014-03-19 20:51:01 +0900 | [diff] [blame] | 177 | def create_object_continue(self, container, object_name, |
| 178 | data, metadata=None): |
Brian Ober | f2deb18 | 2016-04-12 19:28:04 +0000 | [diff] [blame] | 179 | """Put an object using Expect:100-continue""" |
Daisuke Morita | 02f840b | 2014-03-19 20:51:01 +0900 | [diff] [blame] | 180 | headers = {} |
| 181 | if metadata: |
| 182 | for key in metadata: |
| 183 | headers[str(key)] = metadata[key] |
| 184 | |
Daisuke Morita | 02f840b | 2014-03-19 20:51:01 +0900 | [diff] [blame] | 185 | headers['X-Auth-Token'] = self.token |
Brian Ober | f2deb18 | 2016-04-12 19:28:04 +0000 | [diff] [blame] | 186 | headers['content-length'] = 0 if data is None else len(data) |
| 187 | headers['Expect'] = '100-continue' |
Daisuke Morita | 02f840b | 2014-03-19 20:51:01 +0900 | [diff] [blame] | 188 | |
Brian Ober | f2deb18 | 2016-04-12 19:28:04 +0000 | [diff] [blame] | 189 | parsed = urlparse.urlparse(self.base_url) |
| 190 | path = str(parsed.path) + "/" |
| 191 | path += "%s/%s" % (str(container), str(object_name)) |
Daisuke Morita | 02f840b | 2014-03-19 20:51:01 +0900 | [diff] [blame] | 192 | |
Brian Ober | f2deb18 | 2016-04-12 19:28:04 +0000 | [diff] [blame] | 193 | conn = create_connection(parsed) |
| 194 | |
| 195 | # Send the PUT request and the headers including the "Expect" header |
| 196 | conn.putrequest('PUT', path) |
| 197 | |
| 198 | for header, value in six.iteritems(headers): |
| 199 | conn.putheader(header, value) |
| 200 | conn.endheaders() |
| 201 | |
| 202 | # Read the 100 status prior to sending the data |
Daisuke Morita | 02f840b | 2014-03-19 20:51:01 +0900 | [diff] [blame] | 203 | response = conn.response_class(conn.sock, |
| 204 | strict=conn.strict, |
| 205 | method=conn._method) |
Brian Ober | f2deb18 | 2016-04-12 19:28:04 +0000 | [diff] [blame] | 206 | _, status, _ = response._read_status() |
Daisuke Morita | 02f840b | 2014-03-19 20:51:01 +0900 | [diff] [blame] | 207 | |
Brian Ober | f2deb18 | 2016-04-12 19:28:04 +0000 | [diff] [blame] | 208 | # toss the CRLF at the end of the response |
| 209 | response._safe_read(2) |
| 210 | |
| 211 | # Expecting a 100 here, if not close and throw an exception |
| 212 | if status != 100: |
| 213 | conn.close() |
| 214 | pattern = "%s %s" % ( |
| 215 | """Unexpected http success status code {0}.""", |
| 216 | """The expected status code is {1}""") |
| 217 | details = pattern.format(status, 100) |
| 218 | raise exceptions.UnexpectedResponseCode(details) |
| 219 | |
| 220 | # If a continue was received go ahead and send the data |
| 221 | # and get the final response |
| 222 | conn.send(data) |
| 223 | |
| 224 | resp = conn.getresponse() |
| 225 | |
| 226 | return resp.status, resp.reason |
| 227 | |
| 228 | |
| 229 | def create_connection(parsed_url): |
| 230 | """Helper function to create connection with httplib |
| 231 | |
| 232 | :param parsed_url: parsed url of the remote location |
| 233 | """ |
| 234 | if parsed_url.scheme == 'https': |
| 235 | conn = httplib.HTTPSConnection(parsed_url.netloc) |
| 236 | else: |
| 237 | conn = httplib.HTTPConnection(parsed_url.netloc) |
| 238 | |
| 239 | return conn |