Adds more test to cover Swift tempURL middleware

The patch moves away the single test existing from
test_object_services.py to its own file.

It then adds more tests to validate other HTTP
verbs like PUT and HEAD.

blueprint test-swift-tempurl-middleware

Change-Id: I1d7b021e8e3749b58a9fa657c709fb7154bd104e
This commit is contained in:
Joe H. Rahme
2013-08-29 15:50:35 +02:00
parent 900f02b34f
commit 2eccc33f19
4 changed files with 167 additions and 62 deletions

View File

@@ -16,7 +16,6 @@
# under the License.
import hashlib
import time
from tempest.api.object_storage import base
from tempest.common.utils.data_utils import arbitrary_string
@@ -224,44 +223,6 @@ class ObjectTest(base.BaseObjectTest):
self.assertTrue(actual_meta_key in resp)
self.assertEqual(resp[actual_meta_key], meta_value)
@attr(type='gate')
def test_get_object_using_temp_url(self):
# access object using temporary URL within expiration time
try:
# update account metadata
# flag to check if account metadata got updated
flag = False
key = 'Meta'
metadata = {'Temp-URL-Key': key}
resp, _ = self.account_client.create_account_metadata(
metadata=metadata)
self.assertIn(int(resp['status']), HTTP_SUCCESS)
flag = True
resp, _ = self.account_client.list_account_metadata()
self.assertIn('x-account-meta-temp-url-key', resp)
self.assertEqual(resp['x-account-meta-temp-url-key'], key)
# create object
object_name = rand_name(name='ObjectTemp')
data = arbitrary_string(size=len(object_name),
base_text=object_name)
self.object_client.create_object(self.container_name,
object_name, data)
expires = int(time.time() + 10)
# trying to get object using temp url with in expiry time
_, body = self.object_client.get_object_using_temp_url(
self.container_name, object_name,
expires, key)
self.assertEqual(body, data)
finally:
if flag:
resp, _ = self.account_client.delete_account_metadata(
metadata=metadata)
resp, _ = self.account_client.list_account_metadata()
self.assertNotIn('x-account-meta-temp-url-key', resp)
@attr(type='gate')
def test_object_upload_in_segments(self):
# create object

View File

@@ -0,0 +1,157 @@
# vim: tabstop=4 shiftwidth=4 softtabstop=4
# Copyright (C) 2013 eNovance SAS <licensing@enovance.com>
#
# Author: Joe H. Rahme <joe.hakim.rahme@enovance.com>
#
# 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 hmac
import time
import urlparse
from tempest.api.object_storage import base
from tempest.common.utils.data_utils import arbitrary_string
from tempest.common.utils.data_utils import rand_name
from tempest import exceptions
from tempest.test import attr
from tempest.test import HTTP_SUCCESS
class ObjectTempUrlTest(base.BaseObjectTest):
@classmethod
def setUpClass(cls):
super(ObjectTempUrlTest, cls).setUpClass()
cls.container_name = rand_name(name='TestContainer')
cls.container_client.create_container(cls.container_name)
cls.containers = [cls.container_name]
# update account metadata
cls.key = 'Meta'
cls.metadata = {'Temp-URL-Key': cls.key}
cls.account_client.create_account_metadata(metadata=cls.metadata)
cls.account_client_metadata, _ = \
cls.account_client.list_account_metadata()
@classmethod
def tearDownClass(cls):
resp, _ = cls.account_client.delete_account_metadata(
metadata=cls.metadata)
resp, _ = cls.account_client.list_account_metadata()
cls.delete_containers(cls.containers)
# delete the user setup created
cls.data.teardown_all()
super(ObjectTempUrlTest, cls).tearDownClass()
def setUp(self):
super(ObjectTempUrlTest, self).setUp()
# make sure the metadata has been set
self.assertIn('x-account-meta-temp-url-key',
self.account_client_metadata)
self.assertEqual(
self.account_client_metadata['x-account-meta-temp-url-key'],
self.key)
# create object
self.object_name = rand_name(name='ObjectTemp')
self.data = arbitrary_string(size=len(self.object_name),
base_text=self.object_name)
self.object_client.create_object(self.container_name,
self.object_name, self.data)
def get_temp_url(self, container, object_name, method, expires,
key):
"""Create the temporary URL."""
path = "%s/%s/%s" % (
urlparse.urlparse(self.object_client.base_url).path,
container, object_name)
hmac_body = '%s\n%s\n%s' % (method, expires, path)
sig = hmac.new(key, hmac_body, hashlib.sha1).hexdigest()
url = "%s/%s?temp_url_sig=%s&temp_url_expires=%s" % (container,
object_name,
sig, expires)
return url
@attr(type='gate')
def test_get_object_using_temp_url(self):
EXPIRATION_TIME = 10000 # high to ensure the test finishes.
expires = int(time.time() + EXPIRATION_TIME)
# get a temp URL for the created object
url = self.get_temp_url(self.container_name,
self.object_name, "GET",
expires, self.key)
# trying to get object using temp url within expiry time
_, body = self.object_client.get_object_using_temp_url(url)
self.assertEqual(body, self.data)
# Testing a HEAD on this Temp URL
resp, body = self.object_client.head(url)
self.assertIn(int(resp['status']), HTTP_SUCCESS)
@attr(type='gate')
def test_put_object_using_temp_url(self):
# make sure the metadata has been set
new_data = arbitrary_string(size=len(self.object_name),
base_text=rand_name(name="random"))
EXPIRATION_TIME = 10000 # high to ensure the test finishes.
expires = int(time.time() + EXPIRATION_TIME)
url = self.get_temp_url(self.container_name,
self.object_name, "PUT",
expires, self.key)
# trying to put random data in the object using temp url
resp, body = self.object_client.put_object_using_temp_url(
url, new_data)
self.assertIn(int(resp['status']), HTTP_SUCCESS)
# Testing a HEAD on this Temp URL
resp, body = self.object_client.head(url)
self.assertIn(int(resp['status']), HTTP_SUCCESS)
# Validate that the content of the object has been modified
url = self.get_temp_url(self.container_name,
self.object_name, "GET",
expires, self.key)
_, body = self.object_client.get_object_using_temp_url(url)
self.assertEqual(body, new_data)
@attr(type=['gate', 'negative'])
def test_get_object_after_expiration_time(self):
EXPIRATION_TIME = 1
expires = int(time.time() + EXPIRATION_TIME)
# get a temp URL for the created object
url = self.get_temp_url(self.container_name,
self.object_name, "GET",
expires, self.key)
# temp URL is valid for 1 seconds, let's wait 3
time.sleep(EXPIRATION_TIME + 2)
self.assertRaises(exceptions.Unauthorized,
self.object_client.get_object_using_temp_url,
url)

View File

@@ -448,8 +448,8 @@ class RestClient(object):
# NOTE(mtreinish): This is for compatibility with Glance and swift
# APIs. These are the return content types that Glance api v1
# (and occasionally swift) are using.
TXT_ENC = ['text/plain; charset=UTF-8', 'text/html; charset=UTF-8',
'text/plain; charset=utf-8']
TXT_ENC = ['text/plain', 'text/plain; charset=UTF-8',
'text/html; charset=UTF-8', 'text/plain; charset=utf-8']
XML_ENC = ['application/xml', 'application/xml; charset=UTF-8']
if ctype in JSON_ENC or ctype in XML_ENC:

View File

@@ -15,10 +15,6 @@
# License for the specific language governing permissions and limitations
# under the License.
import hashlib
import hmac
import urlparse
from tempest.common import http
from tempest.common.rest_client import RestClient
from tempest import exceptions
@@ -125,29 +121,20 @@ class ObjectClient(RestClient):
resp, body = self.copy(url, headers=headers)
return resp, body
def get_object_using_temp_url(self, container, object_name, expires, key):
"""Retrieve object's data using temporary URL."""
self._set_auth()
method = 'GET'
path = "%s/%s/%s" % (urlparse.urlparse(self.base_url).path, container,
object_name)
hmac_body = '%s\n%s\n%s' % (method, expires, path)
sig = hmac.new(key, hmac_body, hashlib.sha1).hexdigest()
url = "%s/%s?temp_url_sig=%s&temp_url_expires=%s" % (container,
object_name,
sig, expires)
resp, body = self.get(url)
return resp, body
def create_object_segments(self, container, object_name, segment, data):
"""Creates object segments."""
url = "{0}/{1}/{2}".format(container, object_name, segment)
resp, body = self.put(url, data, self.headers)
return resp, body
def get_object_using_temp_url(self, url):
"""Retrieve object's data using temp URL."""
return self.get(url)
def put_object_using_temp_url(self, url, data):
"""Put data in an object using temp URL."""
return self.put(url, data, None)
class ObjectClientCustomizedHeader(RestClient):