All HTTP response headers for each Swift API are checked their existence and style. For checking, create custom_matchers.py file in common directory and implement specific matchers so that they can be reused. Change-Id: I7dcd87c814cf657999796bda75482505112b4edb Closes-Bug: #1240856
427 lines
20 KiB
Python
427 lines
20 KiB
Python
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
|
|
|
# Copyright 2012 OpenStack Foundation
|
|
# 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 hashlib
|
|
import re
|
|
|
|
from tempest.api.object_storage import base
|
|
from tempest.common import custom_matchers
|
|
from tempest.common.utils import data_utils
|
|
from tempest.test import attr
|
|
from tempest.test import HTTP_SUCCESS
|
|
|
|
|
|
class ObjectTest(base.BaseObjectTest):
|
|
@classmethod
|
|
def setUpClass(cls):
|
|
super(ObjectTest, cls).setUpClass()
|
|
cls.container_name = data_utils.rand_name(name='TestContainer')
|
|
cls.container_client.create_container(cls.container_name)
|
|
cls.containers = [cls.container_name]
|
|
|
|
cls.data.setup_test_user()
|
|
resp, body = cls.token_client.auth(cls.data.test_user,
|
|
cls.data.test_password,
|
|
cls.data.test_tenant)
|
|
cls.new_token = cls.token_client.get_token(cls.data.test_user,
|
|
cls.data.test_password,
|
|
cls.data.test_tenant)
|
|
cls.custom_headers = {'X-Auth-Token': cls.new_token}
|
|
|
|
@classmethod
|
|
def tearDownClass(cls):
|
|
cls.delete_containers(cls.containers)
|
|
# delete the user setup created
|
|
cls.data.teardown_all()
|
|
super(ObjectTest, cls).tearDownClass()
|
|
|
|
@attr(type='smoke')
|
|
def test_create_object(self):
|
|
# create object
|
|
object_name = data_utils.rand_name(name='TestObject')
|
|
data = data_utils.arbitrary_string()
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
object_name, data)
|
|
# create another object
|
|
object_name = data_utils.rand_name(name='TestObject')
|
|
data = data_utils.arbitrary_string()
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
object_name, data)
|
|
self.assertEqual(resp['status'], '201')
|
|
self.assertHeaders(resp, 'Object', 'PUT')
|
|
|
|
@attr(type='smoke')
|
|
def test_delete_object(self):
|
|
# create object
|
|
object_name = data_utils.rand_name(name='TestObject')
|
|
data = data_utils.arbitrary_string()
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
object_name, data)
|
|
# delete object
|
|
resp, _ = self.object_client.delete_object(self.container_name,
|
|
object_name)
|
|
self.assertIn(int(resp['status']), HTTP_SUCCESS)
|
|
self.assertHeaders(resp, 'Object', 'DELETE')
|
|
|
|
@attr(type='smoke')
|
|
def test_object_metadata(self):
|
|
# add metadata to storage object, test if metadata is retrievable
|
|
|
|
# create Object
|
|
object_name = data_utils.rand_name(name='TestObject')
|
|
data = data_utils.arbitrary_string()
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
object_name, data)
|
|
# set object metadata
|
|
meta_key = data_utils.rand_name(name='test-')
|
|
meta_value = data_utils.rand_name(name='MetaValue-')
|
|
orig_metadata = {meta_key: meta_value}
|
|
resp, _ = self.object_client.update_object_metadata(
|
|
self.container_name, object_name, orig_metadata)
|
|
self.assertIn(int(resp['status']), HTTP_SUCCESS)
|
|
self.assertHeaders(resp, 'Object', 'POST')
|
|
|
|
# get object metadata
|
|
resp, resp_metadata = self.object_client.list_object_metadata(
|
|
self.container_name, object_name)
|
|
self.assertIn(int(resp['status']), HTTP_SUCCESS)
|
|
self.assertHeaders(resp, 'Object', 'HEAD')
|
|
|
|
actual_meta_key = 'x-object-meta-' + meta_key
|
|
self.assertIn(actual_meta_key, resp)
|
|
self.assertEqual(resp[actual_meta_key], meta_value)
|
|
|
|
@attr(type='smoke')
|
|
def test_get_object(self):
|
|
# retrieve object's data (in response body)
|
|
|
|
# create object
|
|
object_name = data_utils.rand_name(name='TestObject')
|
|
data = data_utils.arbitrary_string()
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
object_name, data)
|
|
# get object
|
|
resp, body = self.object_client.get_object(self.container_name,
|
|
object_name)
|
|
self.assertIn(int(resp['status']), HTTP_SUCCESS)
|
|
self.assertHeaders(resp, 'Object', 'GET')
|
|
|
|
self.assertEqual(body, data)
|
|
|
|
@attr(type='smoke')
|
|
def test_copy_object_in_same_container(self):
|
|
# create source object
|
|
src_object_name = data_utils.rand_name(name='SrcObject')
|
|
src_data = data_utils.arbitrary_string(size=len(src_object_name) * 2,
|
|
base_text=src_object_name)
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
src_object_name,
|
|
src_data)
|
|
# create destination object
|
|
dst_object_name = data_utils.rand_name(name='DstObject')
|
|
dst_data = data_utils.arbitrary_string(size=len(dst_object_name) * 3,
|
|
base_text=dst_object_name)
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
dst_object_name,
|
|
dst_data)
|
|
# copy source object to destination
|
|
resp, _ = self.object_client.copy_object_in_same_container(
|
|
self.container_name, src_object_name, dst_object_name)
|
|
self.assertEqual(resp['status'], '201')
|
|
self.assertHeaders(resp, 'Object', 'PUT')
|
|
|
|
# check data
|
|
resp, body = self.object_client.get_object(self.container_name,
|
|
dst_object_name)
|
|
self.assertEqual(body, src_data)
|
|
|
|
@attr(type='smoke')
|
|
def test_copy_object_to_itself(self):
|
|
# change the content type of an existing object
|
|
|
|
# create object
|
|
object_name = data_utils.rand_name(name='TestObject')
|
|
data = data_utils.arbitrary_string()
|
|
self.object_client.create_object(self.container_name,
|
|
object_name, data)
|
|
# get the old content type
|
|
resp_tmp, _ = self.object_client.list_object_metadata(
|
|
self.container_name, object_name)
|
|
# change the content type of the object
|
|
metadata = {'content-type': 'text/plain; charset=UTF-8'}
|
|
self.assertNotEqual(resp_tmp['content-type'], metadata['content-type'])
|
|
resp, _ = self.object_client.copy_object_in_same_container(
|
|
self.container_name, object_name, object_name, metadata)
|
|
self.assertEqual(resp['status'], '201')
|
|
self.assertHeaders(resp, 'Object', 'PUT')
|
|
|
|
# check the content type
|
|
resp, _ = self.object_client.list_object_metadata(self.container_name,
|
|
object_name)
|
|
self.assertEqual(resp['content-type'], metadata['content-type'])
|
|
|
|
@attr(type='smoke')
|
|
def test_copy_object_2d_way(self):
|
|
# create source object
|
|
src_object_name = data_utils.rand_name(name='SrcObject')
|
|
src_data = data_utils.arbitrary_string(size=len(src_object_name) * 2,
|
|
base_text=src_object_name)
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
src_object_name, src_data)
|
|
# create destination object
|
|
dst_object_name = data_utils.rand_name(name='DstObject')
|
|
dst_data = data_utils.arbitrary_string(size=len(dst_object_name) * 3,
|
|
base_text=dst_object_name)
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
dst_object_name, dst_data)
|
|
# copy source object to destination
|
|
resp, _ = self.object_client.copy_object_2d_way(self.container_name,
|
|
src_object_name,
|
|
dst_object_name)
|
|
self.assertEqual(resp['status'], '201')
|
|
self.assertHeaders(resp, 'Object', 'COPY')
|
|
|
|
self.assertIn('last-modified', resp)
|
|
self.assertIn('x-copied-from', resp)
|
|
self.assertIn('x-copied-from-last-modified', resp)
|
|
self.assertNotEqual(len(resp['last-modified']), 0)
|
|
self.assertEqual(
|
|
resp['x-copied-from'],
|
|
self.container_name + "/" + src_object_name)
|
|
self.assertNotEqual(len(resp['x-copied-from-last-modified']), 0)
|
|
|
|
# check data
|
|
resp, body = self.object_client.get_object(self.container_name,
|
|
dst_object_name)
|
|
self.assertEqual(body, src_data)
|
|
|
|
@attr(type='smoke')
|
|
def test_copy_object_across_containers(self):
|
|
# create a container to use as asource container
|
|
src_container_name = data_utils.rand_name(name='TestSourceContainer')
|
|
self.container_client.create_container(src_container_name)
|
|
self.containers.append(src_container_name)
|
|
# create a container to use as a destination container
|
|
dst_container_name = data_utils.rand_name(
|
|
name='TestDestinationContainer')
|
|
self.container_client.create_container(dst_container_name)
|
|
self.containers.append(dst_container_name)
|
|
# create object in source container
|
|
object_name = data_utils.rand_name(name='Object')
|
|
data = data_utils.arbitrary_string(size=len(object_name) * 2,
|
|
base_text=object_name)
|
|
resp, _ = self.object_client.create_object(src_container_name,
|
|
object_name, data)
|
|
# set object metadata
|
|
meta_key = data_utils.rand_name(name='test-')
|
|
meta_value = data_utils.rand_name(name='MetaValue-')
|
|
orig_metadata = {meta_key: meta_value}
|
|
resp, _ = self.object_client.update_object_metadata(src_container_name,
|
|
object_name,
|
|
orig_metadata)
|
|
self.assertIn(int(resp['status']), HTTP_SUCCESS)
|
|
self.assertHeaders(resp, 'Object', 'POST')
|
|
|
|
# copy object from source container to destination container
|
|
resp, _ = self.object_client.copy_object_across_containers(
|
|
src_container_name, object_name, dst_container_name,
|
|
object_name)
|
|
self.assertEqual(resp['status'], '201')
|
|
self.assertHeaders(resp, 'Object', 'PUT')
|
|
|
|
# check if object is present in destination container
|
|
resp, body = self.object_client.get_object(dst_container_name,
|
|
object_name)
|
|
self.assertEqual(body, data)
|
|
actual_meta_key = 'x-object-meta-' + meta_key
|
|
self.assertIn(actual_meta_key, resp)
|
|
self.assertEqual(resp[actual_meta_key], meta_value)
|
|
|
|
@attr(type='gate')
|
|
def test_object_upload_in_segments(self):
|
|
# create object
|
|
object_name = data_utils.rand_name(name='LObject')
|
|
data = data_utils.arbitrary_string()
|
|
segments = 10
|
|
data_segments = [data + str(i) for i in xrange(segments)]
|
|
# uploading segments
|
|
for i in xrange(segments):
|
|
resp, _ = self.object_client.create_object_segments(
|
|
self.container_name, object_name, i, data_segments[i])
|
|
self.assertEqual(resp['status'], '201')
|
|
# creating a manifest file
|
|
metadata = {'X-Object-Manifest': '%s/%s/'
|
|
% (self.container_name, object_name)}
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
object_name, data='')
|
|
self.assertHeaders(resp, 'Object', 'PUT')
|
|
|
|
resp, _ = self.object_client.update_object_metadata(
|
|
self.container_name, object_name, metadata, metadata_prefix='')
|
|
self.assertHeaders(resp, 'Object', 'POST')
|
|
|
|
resp, _ = self.object_client.list_object_metadata(
|
|
self.container_name, object_name)
|
|
|
|
# Check only the existence of common headers with custom matcher
|
|
self.assertThat(resp, custom_matchers.ExistsAllResponseHeaders(
|
|
'Object', 'HEAD'))
|
|
self.assertIn('x-object-manifest', resp)
|
|
|
|
# Etag value of a large object is enclosed in double-quotations.
|
|
# This is a special case, therefore the formats of response headers
|
|
# are checked without a custom matcher.
|
|
self.assertTrue(resp['etag'].startswith('\"'))
|
|
self.assertTrue(resp['etag'].endswith('\"'))
|
|
self.assertTrue(resp['etag'].strip('\"').isalnum())
|
|
self.assertTrue(re.match("^\d+\.?\d*\Z", resp['x-timestamp']))
|
|
self.assertNotEqual(len(resp['content-type']), 0)
|
|
self.assertTrue(re.match("^tx[0-9a-f]*-[0-9a-f]*$",
|
|
resp['x-trans-id']))
|
|
self.assertNotEqual(len(resp['date']), 0)
|
|
self.assertEqual(resp['accept-ranges'], 'bytes')
|
|
self.assertEqual(resp['x-object-manifest'],
|
|
'%s/%s/' % (self.container_name, object_name))
|
|
|
|
# downloading the object
|
|
resp, body = self.object_client.get_object(
|
|
self.container_name, object_name)
|
|
self.assertEqual(''.join(data_segments), body)
|
|
|
|
@attr(type='gate')
|
|
def test_get_object_if_different(self):
|
|
# http://en.wikipedia.org/wiki/HTTP_ETag
|
|
# Make a conditional request for an object using the If-None-Match
|
|
# header, it should get downloaded only if the local file is different,
|
|
# otherwise the response code should be 304 Not Modified
|
|
object_name = data_utils.rand_name(name='TestObject')
|
|
data = data_utils.arbitrary_string()
|
|
self.object_client.create_object(self.container_name,
|
|
object_name, data)
|
|
# local copy is identical, no download
|
|
md5 = hashlib.md5(data).hexdigest()
|
|
headers = {'If-None-Match': md5}
|
|
url = "%s/%s" % (self.container_name, object_name)
|
|
resp, _ = self.object_client.get(url, headers=headers)
|
|
self.assertEqual(resp['status'], '304')
|
|
|
|
# When the file is not downloaded from Swift server, response does
|
|
# not contain 'X-Timestamp' header. This is the special case, therefore
|
|
# the existence of response headers is checked without custom matcher.
|
|
self.assertIn('content-type', resp)
|
|
self.assertIn('x-trans-id', resp)
|
|
self.assertIn('date', resp)
|
|
self.assertIn('accept-ranges', resp)
|
|
# Check only the format of common headers with custom matcher
|
|
self.assertThat(resp, custom_matchers.AreAllWellFormatted())
|
|
|
|
# local copy is different, download
|
|
local_data = "something different"
|
|
md5 = hashlib.md5(local_data).hexdigest()
|
|
headers = {'If-None-Match': md5}
|
|
resp, body = self.object_client.get(url, headers=headers)
|
|
self.assertIn(int(resp['status']), HTTP_SUCCESS)
|
|
self.assertHeaders(resp, 'Object', 'GET')
|
|
|
|
|
|
class PublicObjectTest(base.BaseObjectTest):
|
|
def setUp(self):
|
|
super(PublicObjectTest, self).setUp()
|
|
self.container_name = data_utils.rand_name(name='TestContainer')
|
|
self.container_client.create_container(self.container_name)
|
|
|
|
def tearDown(self):
|
|
self.delete_containers([self.container_name])
|
|
super(PublicObjectTest, self).tearDown()
|
|
|
|
@attr(type='smoke')
|
|
def test_access_public_container_object_without_using_creds(self):
|
|
# make container public-readable and access an object in it object
|
|
# anonymously, without using credentials
|
|
|
|
# update container metadata to make it publicly readable
|
|
cont_headers = {'X-Container-Read': '.r:*,.rlistings'}
|
|
resp_meta, body = self.container_client.update_container_metadata(
|
|
self.container_name, metadata=cont_headers, metadata_prefix='')
|
|
self.assertIn(int(resp_meta['status']), HTTP_SUCCESS)
|
|
self.assertHeaders(resp_meta, 'Container', 'POST')
|
|
|
|
# create object
|
|
object_name = data_utils.rand_name(name='Object')
|
|
data = data_utils.arbitrary_string(size=len(object_name),
|
|
base_text=object_name)
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
object_name, data)
|
|
self.assertEqual(resp['status'], '201')
|
|
self.assertHeaders(resp, 'Object', 'PUT')
|
|
|
|
# list container metadata
|
|
resp_meta, _ = self.container_client.list_container_metadata(
|
|
self.container_name)
|
|
self.assertIn(int(resp_meta['status']), HTTP_SUCCESS)
|
|
self.assertHeaders(resp_meta, 'Container', 'HEAD')
|
|
|
|
self.assertIn('x-container-read', resp_meta)
|
|
self.assertEqual(resp_meta['x-container-read'], '.r:*,.rlistings')
|
|
|
|
# trying to get object with empty headers as it is public readable
|
|
resp, body = self.custom_object_client.get_object(
|
|
self.container_name, object_name, metadata={})
|
|
self.assertHeaders(resp, 'Object', 'GET')
|
|
|
|
self.assertEqual(body, data)
|
|
|
|
@attr(type='smoke')
|
|
def test_access_public_object_with_another_user_creds(self):
|
|
# make container public-readable and access an object in it using
|
|
# another user's credentials
|
|
cont_headers = {'X-Container-Read': '.r:*,.rlistings'}
|
|
resp_meta, body = self.container_client.update_container_metadata(
|
|
self.container_name, metadata=cont_headers,
|
|
metadata_prefix='')
|
|
self.assertIn(int(resp_meta['status']), HTTP_SUCCESS)
|
|
self.assertHeaders(resp_meta, 'Container', 'POST')
|
|
|
|
# create object
|
|
object_name = data_utils.rand_name(name='Object')
|
|
data = data_utils.arbitrary_string(size=len(object_name) * 1,
|
|
base_text=object_name)
|
|
resp, _ = self.object_client.create_object(self.container_name,
|
|
object_name, data)
|
|
self.assertEqual(resp['status'], '201')
|
|
self.assertHeaders(resp, 'Object', 'PUT')
|
|
|
|
# list container metadata
|
|
resp, _ = self.container_client.list_container_metadata(
|
|
self.container_name)
|
|
self.assertIn(int(resp['status']), HTTP_SUCCESS)
|
|
self.assertHeaders(resp, 'Container', 'HEAD')
|
|
|
|
self.assertIn('x-container-read', resp)
|
|
self.assertEqual(resp['x-container-read'], '.r:*,.rlistings')
|
|
|
|
# get auth token of alternative user
|
|
token = self.identity_client_alt.get_auth()
|
|
headers = {'X-Auth-Token': token}
|
|
# access object using alternate user creds
|
|
resp, body = self.custom_object_client.get_object(
|
|
self.container_name, object_name,
|
|
metadata=headers)
|
|
self.assertHeaders(resp, 'Object', 'GET')
|
|
|
|
self.assertEqual(body, data)
|