URL encode swift objects endpoints

Swift containers and objects can contain any kind of characters.
If a container or name contains special characters, shade was failing to
access them.
We need to url encode containers and objects when creating the url to access
them.

Story: 2005828
Task: 33585
Change-Id: I170277293eb4ed1a12cf62bcf863707cb3d918eb
This commit is contained in:
Yves-Gwenael Bourhis 2019-06-07 14:30:06 +02:00
parent b47fb2af55
commit 8e183ca4d2

View File

@ -30,7 +30,7 @@ import warnings
import dogpile.cache import dogpile.cache
import munch import munch
import requestsexceptions import requestsexceptions
from six.moves import urllib from six.moves import urllib_parse
import keystoneauth1.exceptions import keystoneauth1.exceptions
import keystoneauth1.session import keystoneauth1.session
@ -646,7 +646,7 @@ class OpenStackCloud(
if not endpoint.rstrip().rsplit('/')[1] == 'v2.0': if not endpoint.rstrip().rsplit('/')[1] == 'v2.0':
if not endpoint.endswith('/'): if not endpoint.endswith('/'):
endpoint += '/' endpoint += '/'
endpoint = urllib.parse.urljoin( endpoint = urllib_parse.urljoin(
endpoint, 'v2.0') endpoint, 'v2.0')
client.endpoint_override = endpoint client.endpoint_override = endpoint
self._raw_clients['network'] = client self._raw_clients['network'] = client
@ -2233,10 +2233,10 @@ class OpenStackCloud(
self._expand_server(server, detailed, bare) self._expand_server(server, detailed, bare)
for server in servers for server in servers
] ]
parse_result = urllib.parse.urlparse( parse_result = urllib_parse.urlparse(
data['servers_links'][0]['href']) data['servers_links'][0]['href'])
pagination_params = dict( pagination_params = dict(
urllib.parse.parse_qsl(parse_result.query)) urllib_parse.parse_qsl(parse_result.query))
params.update(pagination_params) params.update(pagination_params)
data = self._compute_client.get( data = self._compute_client.get(
'/servers/detail', params=params, error_message=error_msg) '/servers/detail', params=params, error_message=error_msg)
@ -7375,7 +7375,9 @@ class OpenStackCloud(
def get_container(self, name, skip_cache=False): def get_container(self, name, skip_cache=False):
if skip_cache or name not in self._container_cache: if skip_cache or name not in self._container_cache:
try: try:
container = self._object_store_client.head(name) container = self._object_store_client.head(
self._get_object_endpoint(name)
)
self._container_cache[name] = container.headers self._container_cache[name] = container.headers
except exc.OpenStackCloudHTTPError as e: except exc.OpenStackCloudHTTPError as e:
if e.response.status_code == 404: if e.response.status_code == 404:
@ -7387,14 +7389,16 @@ class OpenStackCloud(
container = self.get_container(name) container = self.get_container(name)
if container: if container:
return container return container
self._object_store_client.put(name) self._object_store_client.put(self._get_object_endpoint(name))
if public: if public:
self.set_container_access(name, 'public') self.set_container_access(name, 'public')
return self.get_container(name, skip_cache=True) return self.get_container(name, skip_cache=True)
def delete_container(self, name): def delete_container(self, name):
try: try:
self._object_store_client.delete(name) self._object_store_client.delete(
self._get_object_endpoint(name)
)
return True return True
except exc.OpenStackCloudHTTPError as e: except exc.OpenStackCloudHTTPError as e:
if e.response.status_code == 404: if e.response.status_code == 404:
@ -7408,7 +7412,9 @@ class OpenStackCloud(
raise raise
def update_container(self, name, headers): def update_container(self, name, headers):
self._object_store_client.post(name, headers=headers) self._object_store_client.post(
self._get_object_endpoint(name), headers=headers
)
def set_container_access(self, name, access): def set_container_access(self, name, access):
if access not in OBJECT_CONTAINER_ACLS: if access not in OBJECT_CONTAINER_ACLS:
@ -7460,7 +7466,7 @@ class OpenStackCloud(
# The endpoint in the catalog has version and project-id in it # The endpoint in the catalog has version and project-id in it
# To get capabilities, we have to disassemble and reassemble the URL # To get capabilities, we have to disassemble and reassemble the URL
# This logic is taken from swiftclient # This logic is taken from swiftclient
endpoint = urllib.parse.urlparse( endpoint = urllib_parse.urlparse(
self._object_store_client.get_endpoint()) self._object_store_client.get_endpoint())
url = "{scheme}://{netloc}/info".format( url = "{scheme}://{netloc}/info".format(
scheme=endpoint.scheme, netloc=endpoint.netloc) scheme=endpoint.scheme, netloc=endpoint.netloc)
@ -7582,8 +7588,7 @@ class OpenStackCloud(
if self.is_object_stale(container, name, filename, md5, sha256): if self.is_object_stale(container, name, filename, md5, sha256):
endpoint = '{container}/{name}'.format( endpoint = self._get_object_endpoint(container, name)
container=container, name=name)
self.log.debug( self.log.debug(
"swift uploading %(filename)s to %(endpoint)s", "swift uploading %(filename)s to %(endpoint)s",
{'filename': filename, 'endpoint': endpoint}) {'filename': filename, 'endpoint': endpoint})
@ -7656,7 +7661,8 @@ class OpenStackCloud(
for name, segment in segments.items(): for name, segment in segments.items():
# Async call to put - schedules execution and returns a future # Async call to put - schedules execution and returns a future
segment_future = self._object_store_client.put( segment_future = self._object_store_client.put(
name, headers=headers, data=segment, run_async=True) self._get_object_endpoint(name),
headers=headers, data=segment, run_async=True)
segment_futures.append(segment_future) segment_futures.append(segment_future)
# TODO(mordred) Collect etags from results to add to this manifest # TODO(mordred) Collect etags from results to add to this manifest
# dict. Then sort the list of dicts by path. # dict. Then sort the list of dicts by path.
@ -7707,6 +7713,18 @@ class OpenStackCloud(
headers['X-Object-Manifest'] = endpoint headers['X-Object-Manifest'] = endpoint
return self._object_store_client.put(endpoint, headers=headers) return self._object_store_client.put(endpoint, headers=headers)
def _get_object_endpoint(self, container, obj=None, query_string=None):
endpoint = urllib_parse.quote(container)
if obj:
endpoint = '{endpoint}/{object}'.format(
endpoint=endpoint,
object=urllib_parse.quote(obj)
)
if query_string:
endpoint = '{endpoint}?{query_string}'.format(
endpoint=endpoint, query_string=query_string)
return endpoint
def update_object(self, container, name, metadata=None, **headers): def update_object(self, container, name, metadata=None, **headers):
"""Update the metadata of an object """Update the metadata of an object
@ -7730,8 +7748,7 @@ class OpenStackCloud(
headers = dict(headers, **metadata_headers) headers = dict(headers, **metadata_headers)
return self._object_store_client.post( return self._object_store_client.post(
'{container}/{object}'.format( self._get_object_endpoint(container, name),
container=container, object=name),
headers=headers) headers=headers)
def list_objects(self, container, full_listing=True): def list_objects(self, container, full_listing=True):
@ -7745,7 +7762,7 @@ class OpenStackCloud(
:raises: OpenStackCloudException on operation error. :raises: OpenStackCloudException on operation error.
""" """
return self._object_store_client.get( return self._object_store_client.get(
container, params=dict(format='json')) self._get_object_endpoint(container), params=dict(format='json'))
def delete_object(self, container, name, meta=None): def delete_object(self, container, name, meta=None):
"""Delete an object from a container. """Delete an object from a container.
@ -7776,8 +7793,7 @@ class OpenStackCloud(
if meta.get('X-Static-Large-Object', None) == 'True': if meta.get('X-Static-Large-Object', None) == 'True':
params['multipart-manifest'] = 'delete' params['multipart-manifest'] = 'delete'
self._object_store_client.delete( self._object_store_client.delete(
'{container}/{object}'.format( self._get_object_endpoint(container, name),
container=container, object=name),
params=params) params=params)
return True return True
except exc.OpenStackCloudHTTPError: except exc.OpenStackCloudHTTPError:
@ -7808,8 +7824,7 @@ class OpenStackCloud(
def get_object_metadata(self, container, name): def get_object_metadata(self, container, name):
try: try:
return self._object_store_client.head( return self._object_store_client.head(
'{container}/{object}'.format( self._get_object_endpoint(container, name)).headers
container=container, object=name)).headers
except exc.OpenStackCloudException as e: except exc.OpenStackCloudException as e:
if e.response.status_code == 404: if e.response.status_code == 404:
return None return None
@ -7838,11 +7853,7 @@ class OpenStackCloud(
""" """
# TODO(mordred) implement resp_chunk_size # TODO(mordred) implement resp_chunk_size
try: try:
endpoint = '{container}/{object}'.format( endpoint = self._get_object_endpoint(container, obj, query_string)
container=container, object=obj)
if query_string:
endpoint = '{endpoint}?{query_string}'.format(
endpoint=endpoint, query_string=query_string)
response = self._object_store_client.get( response = self._object_store_client.get(
endpoint, stream=True) endpoint, stream=True)
response_headers = { response_headers = {