Add functional tests for new add-location API

Related blueprint new-location-apis

Change-Id: If94d7d1fbec07a49222b65aa3e530748a95eaf5b
This commit is contained in:
Pranali Deore 2023-06-26 07:44:37 +00:00
parent 4281558dff
commit b83f38cf25
3 changed files with 635 additions and 15 deletions

View File

@ -99,6 +99,8 @@ class BaseServer(metaclass=abc.ABCMeta):
self.deployment_flavor = '' self.deployment_flavor = ''
self.show_image_direct_url = False self.show_image_direct_url = False
self.show_multiple_locations = False self.show_multiple_locations = False
self.do_secure_hash = True
self.http_retries = '3'
self.property_protection_file = '' self.property_protection_file = ''
self.needs_database = False self.needs_database = False
self.log_file = None self.log_file = None
@ -437,6 +439,8 @@ image_cache_dir = %(image_cache_dir)s
image_cache_driver = %(image_cache_driver)s image_cache_driver = %(image_cache_driver)s
show_image_direct_url = %(show_image_direct_url)s show_image_direct_url = %(show_image_direct_url)s
show_multiple_locations = %(show_multiple_locations)s show_multiple_locations = %(show_multiple_locations)s
do_secure_hash = %(do_secure_hash)s
http_retries = %(http_retries)s
user_storage_quota = %(user_storage_quota)s user_storage_quota = %(user_storage_quota)s
lock_path = %(lock_path)s lock_path = %(lock_path)s
property_protection_file = %(property_protection_file)s property_protection_file = %(property_protection_file)s
@ -623,6 +627,8 @@ image_cache_dir = %(image_cache_dir)s
image_cache_driver = %(image_cache_driver)s image_cache_driver = %(image_cache_driver)s
show_image_direct_url = %(show_image_direct_url)s show_image_direct_url = %(show_image_direct_url)s
show_multiple_locations = %(show_multiple_locations)s show_multiple_locations = %(show_multiple_locations)s
do_secure_hash = %(do_secure_hash)s
http_retries = %(http_retries)s
user_storage_quota = %(user_storage_quota)s user_storage_quota = %(user_storage_quota)s
lock_path = %(lock_path)s lock_path = %(lock_path)s
property_protection_file = %(property_protection_file)s property_protection_file = %(property_protection_file)s

View File

@ -16,10 +16,13 @@
import http.client as http import http.client as http
import time import time
from oslo_log import log as logging
from oslo_serialization import jsonutils from oslo_serialization import jsonutils
from oslo_utils import timeutils from oslo_utils import timeutils
import requests import requests
LOG = logging.getLogger(__name__)
def verify_image_hashes_and_status( def verify_image_hashes_and_status(
test_obj, image_id, checksum=None, os_hash_value=None, status=None, test_obj, image_id, checksum=None, os_hash_value=None, status=None,
@ -48,12 +51,15 @@ def verify_image_hashes_and_status(
test_obj.assertEqual(size, image['size']) test_obj.assertEqual(size, image['size'])
def wait_for_status(request_path, request_headers, status='active', def wait_for_status(test_obj, request_path, request_headers,
max_sec=10, delay_sec=0.2, start_delay_sec=None): status=None, max_sec=10, delay_sec=0.2,
start_delay_sec=None, multistore=False):
""" """
Performs a time-bounded wait for the entity at the request_path to Performs a time-bounded wait for the entity at the request_path to
reach the requested status. reach the requested status.
:param test_obj: The test object; expected to have _url() and
_headers() defined on it
:param request_path: path to use to make the request :param request_path: path to use to make the request
:param request_headers: headers to use when making the request :param request_headers: headers to use when making the request
:param status: the status to wait for (default: 'active') :param status: the status to wait for (default: 'active')
@ -62,6 +68,7 @@ def wait_for_status(request_path, request_headers, status='active',
made (default: 0.2) made (default: 0.2)
:param start_delay_sec: seconds to wait before making the first :param start_delay_sec: seconds to wait before making the first
request (default: None) request (default: None)
:multistore: Optional flag if multiple backends enabled
:raises Exception: if the entity fails to reach the status within :raises Exception: if the entity fails to reach the status within
the requested time or if the server returns something the requested time or if the server returns something
other than a 200 response other than a 200 response
@ -71,11 +78,18 @@ def wait_for_status(request_path, request_headers, status='active',
if start_delay_sec: if start_delay_sec:
time.sleep(start_delay_sec) time.sleep(start_delay_sec)
while time.time() <= done_time: while time.time() <= done_time:
resp = requests.get(request_path, headers=request_headers) if multistore:
resp = test_obj.api_get(request_path, headers=request_headers)
else:
resp = requests.get(request_path, headers=request_headers)
if resp.status_code != http.OK: if resp.status_code != http.OK:
raise Exception("Received {} response from server".format( raise Exception("Received {} response from server".format(
resp.status_code)) resp.status_code))
test_obj.assertEqual(http.OK, resp.status_code)
entity = jsonutils.loads(resp.text) entity = jsonutils.loads(resp.text)
LOG.info('Image status is: %s', entity['status'])
if entity['checksum'] and entity['status'] == 'active':
return
if entity['status'] == status: if entity['status'] == status:
return return
time.sleep(delay_sec) time.sleep(delay_sec)

View File

@ -23,6 +23,9 @@ import urllib
import uuid import uuid
import fixtures import fixtures
import glance_store
from oslo_config import cfg
from oslo_limit import exception as ol_exc from oslo_limit import exception as ol_exc
from oslo_limit import limit from oslo_limit import limit
from oslo_serialization import jsonutils from oslo_serialization import jsonutils
@ -30,12 +33,15 @@ from oslo_utils.secretutils import md5
from oslo_utils import units from oslo_utils import units
import requests import requests
from glance.common import wsgi
from glance.quota import keystone as ks_quota from glance.quota import keystone as ks_quota
from glance.tests import functional from glance.tests import functional
from glance.tests.functional import ft_utils as func_utils from glance.tests.functional import ft_utils as func_utils
from glance.tests import utils as test_utils from glance.tests import utils as test_utils
CONF = cfg.CONF
TENANT1 = str(uuid.uuid4()) TENANT1 = str(uuid.uuid4())
TENANT2 = str(uuid.uuid4()) TENANT2 = str(uuid.uuid4())
TENANT3 = str(uuid.uuid4()) TENANT3 = str(uuid.uuid4())
@ -201,7 +207,7 @@ class TestImages(functional.FunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image_id) path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=10, max_sec=10,
@ -342,7 +348,7 @@ class TestImages(functional.FunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image_id) path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=20, max_sec=20,
@ -982,7 +988,7 @@ class TestImages(functional.FunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image['id']) path = self._url('/v2/images/%s' % image['id'])
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=15, max_sec=15,
@ -3692,6 +3698,286 @@ class TestImages(functional.FunctionalTest):
response = requests.patch(path, headers=headers, data=data) response = requests.patch(path, headers=headers, data=data)
self.assertEqual(http.BAD_REQUEST, response.status_code, response.text) self.assertEqual(http.BAD_REQUEST, response.status_code, response.text)
def test_add_location_with_do_secure_hash_true_negative(self):
self.start_servers(**self.__dict__.copy())
# Create an image
path = self._url('/v2/images')
headers = self._headers({'content-type': 'application/json'})
data = jsonutils.dumps({'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
self.assertIsNone(image['size'])
# Add Location with non image owner
path = self._url('/v2/images/%s/locations' % image_id)
headers = self._headers({'X-Tenant-Id': TENANT2})
url = 'http://127.0.0.1:%s/foo_image' % self.http_port0
data = jsonutils.dumps({'url': url})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.NOT_FOUND, response.status_code, response.text)
# Add location with invalid validation_data
# Invalid os_hash_value
validation_data = {
'os_hash_algo': "sha512",
'os_hash_value': "dbc9e0f80d131e64b94913a7b40bb5"
}
headers = self._headers({'X-Tenant-Id': TENANT1})
data = jsonutils.dumps({'url': url,
'validation_data': validation_data})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.BAD_REQUEST, response.status_code,
response.text)
# Add location with invalid validation_data (without os_hash_algo)
url = 'http://127.0.0.1:%s/foo_image' % self.http_port0
with requests.get(url) as r:
expect_h = str(hashlib.sha512(r.content).hexdigest())
validation_data = {'os_hash_value': expect_h}
data = jsonutils.dumps({'url': url,
'validation_data': validation_data})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.BAD_REQUEST, response.status_code, response.text)
# Add location with invalid validation_data &
# (invalid hash_algo)
validation_data = {
'os_hash_algo': 'sha123',
'os_hash_value': expect_h}
data = jsonutils.dumps({'url': url,
'validation_data': validation_data})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.BAD_REQUEST, response.status_code, response.text)
# Add location with invalid validation_data
# (mismatch hash_value with hash algo)
with requests.get(url) as r:
expect_h = str(hashlib.sha256(r.content).hexdigest())
validation_data = {
'os_hash_algo': 'sha512',
'os_hash_value': expect_h}
data = jsonutils.dumps({'url': url,
'validation_data': validation_data})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.BAD_REQUEST, response.status_code, response.text)
self.stop_servers()
def test_add_location_with_do_secure_hash_true(self):
self.start_servers(**self.__dict__.copy())
# Create an image
path = self._url('/v2/images')
headers = self._headers({'content-type': 'application/json'})
data = jsonutils.dumps({'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
# Add location with os_hash_algo other than sha512
path = self._url('/v2/images/%s/locations' % image_id)
headers = self._headers({'X-Tenant-Id': TENANT1})
url = 'http://127.0.0.1:%s/foo_image' % self.http_port0
with requests.get(url) as r:
expect_c = str(md5(r.content, usedforsecurity=False).hexdigest())
expect_h = str(hashlib.sha256(r.content).hexdigest())
validation_data = {
'os_hash_algo': 'sha256',
'os_hash_value': expect_h}
data = jsonutils.dumps({'url': url,
'validation_data': validation_data})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.ACCEPTED, response.status_code, response.text)
path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(self, request_path=path,
request_headers=headers,
status='active',
max_sec=10,
delay_sec=0.2,
start_delay_sec=1)
# Show Image
path = self._url('/v2/images/%s' % image_id)
resp = requests.get(path, headers=headers)
image = jsonutils.loads(resp.text)
self.assertEqual(expect_c, image['checksum'])
self.assertEqual(expect_h, image['os_hash_value'])
# Add location with valid validation data
# os_hash_algo value sha512
# Create an image 2
path = self._url('/v2/images')
headers = self._headers({'content-type': 'application/json'})
data = jsonutils.dumps({'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
path = self._url('/v2/images/%s/locations' % image_id)
headers = self._headers({'X-Tenant-Id': TENANT1})
url = 'http://127.0.0.1:%s/foo_image' % self.http_port0
with requests.get(url) as r:
expect_c = str(md5(r.content, usedforsecurity=False).hexdigest())
expect_h = str(hashlib.sha512(r.content).hexdigest())
validation_data = {
'os_hash_algo': 'sha512',
'os_hash_value': expect_h}
data = jsonutils.dumps({'url': url,
'validation_data': validation_data})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.ACCEPTED, response.status_code, response.text)
# Show Image
path = self._url('/v2/images/%s' % image_id)
resp = requests.get(path, headers=self._headers())
output = jsonutils.loads(resp.text)
self.assertEqual('queued', output['status'])
func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(),
status='active',
max_sec=10,
delay_sec=0.2,
start_delay_sec=1)
# Show Image
resp = requests.get(path, headers=self._headers())
image = jsonutils.loads(resp.text)
self.assertEqual(expect_c, image['checksum'])
self.assertEqual(expect_h, image['os_hash_value'])
# Add Location with valid URL and do_secure_hash = True
# without validation_data
# Create an image 3
path = self._url('/v2/images')
headers = self._headers({'content-type': 'application/json'})
data = jsonutils.dumps({'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
self.assertIsNone(image['size'])
self.assertIsNone(image['virtual_size'])
path = self._url('/v2/images/%s/locations' % image_id)
headers = self._headers({'X-Tenant-Id': TENANT1})
url = 'http://127.0.0.1:%s/foo_image' % self.http_port0
with requests.get(url) as r:
expect_c = str(md5(r.content, usedforsecurity=False).hexdigest())
expect_h = str(hashlib.sha512(r.content).hexdigest())
data = jsonutils.dumps({'url': url})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.ACCEPTED, response.status_code, response.text)
path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(self, request_path=path,
request_headers=headers,
status='active',
max_sec=10,
delay_sec=0.2,
start_delay_sec=1)
# Show Image
path = self._url('/v2/images/%s' % image_id)
resp = requests.get(path, headers=headers)
image = jsonutils.loads(resp.text)
self.assertEqual(expect_c, image['checksum'])
self.assertEqual(expect_h, image['os_hash_value'])
self.stop_servers()
def test_add_location_with_do_secure_hash_false(self):
self.api_server.do_secure_hash = False
self.start_servers(**self.__dict__.copy())
# Add Location with valid URL and do_secure_hash = False
# with validation_data
# Create an image 1
path = self._url('/v2/images')
headers = self._headers({'content-type': 'application/json'})
data = jsonutils.dumps({'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
url = 'http://127.0.0.1:%s/foo_image' % self.http_port0
with requests.get(url) as r:
expect_h = str(hashlib.sha512(r.content).hexdigest())
validation_data = {
'os_hash_algo': 'sha512',
'os_hash_value': expect_h}
path = self._url('/v2/images/%s/locations' % image_id)
headers = self._headers({'X-Tenant-Id': TENANT1})
data = jsonutils.dumps({'url': url,
'validation_data': validation_data})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.ACCEPTED, response.status_code, response.text)
path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(self, request_path=path,
request_headers=headers,
status='active',
max_sec=2,
delay_sec=0.2,
start_delay_sec=1)
# Add Location with valid URL and do_secure_hash = False
# without validation_data
# Create an image 2
path = self._url('/v2/images')
headers = self._headers({'content-type': 'application/json'})
data = jsonutils.dumps({'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
self.assertIsNone(image['size'])
self.assertIsNone(image['virtual_size'])
url = 'http://127.0.0.1:%s/foo_image' % self.http_port0
path = self._url('/v2/images/%s/locations' % image_id)
headers = self._headers({'X-Tenant-Id': TENANT1})
data = jsonutils.dumps({'url': url})
response = requests.post(path, headers=headers, data=data)
self.assertEqual(http.ACCEPTED, response.status_code, response.text)
path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(self, request_path=path,
request_headers=headers,
status='active',
max_sec=2,
delay_sec=0.2,
start_delay_sec=1)
self.stop_servers()
class TestImagesIPv6(functional.FunctionalTest): class TestImagesIPv6(functional.FunctionalTest):
"""Verify that API and REG servers running IPv6 can communicate""" """Verify that API and REG servers running IPv6 can communicate"""
@ -4640,7 +4926,7 @@ class TestImagesMultipleBackend(functional.MultipleBackendFunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image_id) path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=15, max_sec=15,
@ -4805,7 +5091,7 @@ class TestImagesMultipleBackend(functional.MultipleBackendFunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image_id) path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=15, max_sec=15,
@ -4967,7 +5253,7 @@ class TestImagesMultipleBackend(functional.MultipleBackendFunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image_id) path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=20, max_sec=20,
@ -5130,7 +5416,7 @@ class TestImagesMultipleBackend(functional.MultipleBackendFunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image_id) path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=20, max_sec=20,
@ -5292,7 +5578,7 @@ class TestImagesMultipleBackend(functional.MultipleBackendFunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image_id) path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=40, max_sec=40,
@ -5456,7 +5742,7 @@ class TestImagesMultipleBackend(functional.MultipleBackendFunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image_id) path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=40, max_sec=40,
@ -5680,7 +5966,7 @@ class TestImagesMultipleBackend(functional.MultipleBackendFunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image_id) path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=40, max_sec=40,
@ -5938,7 +6224,7 @@ class TestImagesMultipleBackend(functional.MultipleBackendFunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image_id) path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=40, max_sec=40,
@ -6790,7 +7076,7 @@ class TestCopyImagePermissions(functional.MultipleBackendFunctionalTest):
# NOTE(abhishekk): As import is a async call we need to provide # NOTE(abhishekk): As import is a async call we need to provide
# some timelap to complete the call. # some timelap to complete the call.
path = self._url('/v2/images/%s' % image_id) path = self._url('/v2/images/%s' % image_id)
func_utils.wait_for_status(request_path=path, func_utils.wait_for_status(self, request_path=path,
request_headers=self._headers(), request_headers=self._headers(),
status='active', status='active',
max_sec=40, max_sec=40,
@ -7352,3 +7638,317 @@ class TestStoreWeight(functional.SynchronousAPIBase):
# as store3,store1,store2 # as store3,store1,store2
image = self.api_get('/v2/images/%s' % image_id).json image = self.api_get('/v2/images/%s' % image_id).json
self.assertEqual("store3,store1,store2", image['stores']) self.assertEqual("store3,store1,store2", image['stores'])
class TestMultipleBackendsLocationApi(functional.SynchronousAPIBase):
def setUp(self):
super(TestMultipleBackendsLocationApi, self).setUp()
self.start_server()
for i in range(3):
ret = test_utils.start_http_server("foo_image_id%d" % i,
"foo_image%d" % i)
setattr(self, 'http_server%d' % i, ret[1])
setattr(self, 'http_port%d' % i, ret[2])
def setup_stores(self):
pass
def _headers(self, custom_headers=None):
base_headers = {
'X-Identity-Status': 'Confirmed',
'X-Auth-Token': '932c5c84-02ac-4fe5-a9ba-620af0e2bb96',
'X-User-Id': 'f9a41d13-0c13-47e9-bee2-ce4e8bfe958e',
'X-Tenant-Id': TENANT1,
'X-Roles': 'reader,member',
}
base_headers.update(custom_headers or {})
return base_headers
def _setup_multiple_stores(self):
self.ksa_client = self.useFixture(
fixtures.MockPatch('glance.context.get_ksa_client')).mock
self.config(enabled_backends={'store1': 'http', 'store2': 'http'})
glance_store.register_store_opts(CONF,
reserved_stores=wsgi.RESERVED_STORES)
self.config(default_backend='store1',
group='glance_store')
self.config(filesystem_store_datadir=self._store_dir('staging'),
group='os_glance_staging_store')
self.config(filesystem_store_datadir='/tmp/foo',
group='os_glance_tasks_store')
glance_store.create_multi_stores(CONF,
reserved_stores=wsgi.RESERVED_STORES)
glance_store.verify_store()
def test_add_location_with_do_secure_hash_false(self):
self.config(do_secure_hash=False)
self._setup_multiple_stores()
# Add Location with valid URL and do_secure_hash = False
# with validation_data
# Create an image 1
path = '/v2/images'
headers = self._headers({'content-type': 'application/json'})
data = {'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
self.assertIsNone(image['size'])
self.assertIsNone(image['virtual_size'])
url = 'http://127.0.0.1:%s/store1/foo_image' % self.http_port0
with requests.get(url) as r:
expect_h = str(hashlib.sha512(r.content).hexdigest())
validation_data = {
'os_hash_algo': 'sha512',
'os_hash_value': expect_h}
path = '/v2/images/%s/locations' % image_id
headers = self._headers({'X-Tenant-Id': TENANT1})
data = {'url': url,
'validation_data': validation_data}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.ACCEPTED, response.status_code, response.text)
path = '/v2/images/%s' % image_id
func_utils.wait_for_status(self, request_path=path,
request_headers=headers,
status='active',
max_sec=5,
delay_sec=0.2,
start_delay_sec=1,
multistore=True)
# Add Location with valid URL and do_secure_hash = False
# without validation_data
# Create an image 2
path = '/v2/images'
headers = self._headers({'content-type': 'application/json'})
data = {'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
self.assertIsNone(image['size'])
self.assertIsNone(image['virtual_size'])
url = 'http://127.0.0.1:%s/store1/foo_image' % self.http_port0
path = '/v2/images/%s/locations' % image_id
headers = self._headers({'X-Tenant-Id': TENANT1})
data = {'url': url}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.ACCEPTED, response.status_code, response.text)
path = '/v2/images/%s' % image_id
func_utils.wait_for_status(self, request_path=path,
request_headers=headers,
status='active',
max_sec=5,
delay_sec=0.2,
start_delay_sec=1, multistore=True)
def test_add_location_with_do_secure_hash_true_negative(self):
self._setup_multiple_stores()
# Create an image
path = '/v2/images'
headers = self._headers({'content-type': 'application/json'})
data = {'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
# Add Location with non image owner
path = '/v2/images/%s/locations' % image_id
headers = self._headers({'X-Tenant-Id': TENANT2})
url = 'http://127.0.0.1:%s/store1/foo_image' % self.http_port0
data = {'url': url}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.NOT_FOUND, response.status_code, response.text)
# Add location with invalid validation_data
# Invalid os_hash_value
validation_data = {
'os_hash_algo': "sha512",
'os_hash_value': "dbc9e0f80d131e64b94913a7b40bb5"
}
headers = self._headers({'X-Tenant-Id': TENANT1})
data = {'url': url, 'validation_data': validation_data}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.BAD_REQUEST, response.status_code,
response.text)
# Add location with invalid validation_data (without os_hash_algo)
url = 'http://127.0.0.1:%s/store1/foo_image' % self.http_port0
with requests.get(url) as r:
expect_h = str(hashlib.sha512(r.content).hexdigest())
validation_data = {'os_hash_value': expect_h}
data = {'url': url, 'validation_data': validation_data}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.BAD_REQUEST, response.status_code, response.text)
# Add location with invalid validation_data &
# (invalid hash_algo)
validation_data = {
'os_hash_algo': 'sha123',
'os_hash_value': expect_h}
data = {'url': url, 'validation_data': validation_data}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.BAD_REQUEST, response.status_code, response.text)
# Add location with invalid validation_data
# (mismatch hash_value with hash algo)
with requests.get(url) as r:
expect_h = str(hashlib.sha256(r.content).hexdigest())
validation_data = {
'os_hash_algo': 'sha512',
'os_hash_value': expect_h}
data = {'url': url, 'validation_data': validation_data}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.BAD_REQUEST, response.status_code, response.text)
def test_add_location_with_do_secure_hash_true(self):
self._setup_multiple_stores()
# Create an image
path = '/v2/images'
headers = self._headers({'content-type': 'application/json'})
data = {'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
# Add location with os_hash_algo other than sha512
path = '/v2/images/%s/locations' % image_id
headers = self._headers({'X-Tenant-Id': TENANT1})
url = 'http://127.0.0.1:%s/store1/foo_image' % self.http_port0
with requests.get(url) as r:
expect_c = str(md5(r.content, usedforsecurity=False).hexdigest())
expect_h = str(hashlib.sha256(r.content).hexdigest())
validation_data = {
'os_hash_algo': 'sha256',
'os_hash_value': expect_h}
data = {'url': url, 'validation_data': validation_data}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.ACCEPTED, response.status_code, response.text)
path = '/v2/images/%s' % image_id
func_utils.wait_for_status(self, request_path=path,
request_headers=headers,
status='active',
max_sec=20,
delay_sec=0.2,
start_delay_sec=1, multistore=True)
# Show Image
path = '/v2/images/%s' % image_id
resp = self.api_get(path, headers=self._headers())
image = jsonutils.loads(resp.text)
self.assertEqual(expect_c, image['checksum'])
self.assertEqual(expect_h, image['os_hash_value'])
# Add location with valid validation data
# os_hash_algo value sha512
# Create an image 3
path = '/v2/images'
headers = self._headers({'content-type': 'application/json'})
data = {'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
path = '/v2/images/%s/locations' % image_id
headers = self._headers({'X-Tenant-Id': TENANT1})
url = 'http://127.0.0.1:%s/store2/foo_image' % self.http_port0
with requests.get(url) as r:
expect_c = str(md5(r.content, usedforsecurity=False).hexdigest())
expect_h = str(hashlib.sha512(r.content).hexdigest())
validation_data = {
'os_hash_algo': 'sha512',
'os_hash_value': expect_h}
headers = self._headers({'X-Tenant-Id': TENANT1})
data = {'url': url, 'validation_data': validation_data}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.ACCEPTED, response.status_code, response.text)
# Show Image
path = '/v2/images/%s' % image_id
resp = self.api_get(path, headers=self._headers())
output = jsonutils.loads(resp.text)
self.assertEqual('queued', output['status'])
path = '/v2/images/%s' % image_id
func_utils.wait_for_status(self, request_path=path,
request_headers=headers,
status='active',
max_sec=10,
delay_sec=0.2,
start_delay_sec=1, multistore=True)
# Show Image
path = '/v2/images/%s' % image_id
resp = self.api_get(path, headers=self._headers())
image = jsonutils.loads(resp.text)
self.assertEqual(expect_c, image['checksum'])
self.assertEqual(expect_h, image['os_hash_value'])
# Add Location with valid URL and do_secure_hash = True
# without validation_data
# Create an image 4
path = '/v2/images'
headers = self._headers({'content-type': 'application/json'})
data = {'name': 'image-1', 'disk_format': 'aki',
'container_format': 'aki'}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.CREATED, response.status_code)
# Returned image entity should have a generated id and status
image = jsonutils.loads(response.text)
image_id = image['id']
self.assertEqual('queued', image['status'])
path = '/v2/images/%s/locations' % image_id
headers = self._headers({'X-Tenant-Id': TENANT1})
url = 'http://127.0.0.1:%s/store2/foo_image' % self.http_port0
with requests.get(url) as r:
expect_c = str(md5(r.content, usedforsecurity=False).hexdigest())
expect_h = str(hashlib.sha512(r.content).hexdigest())
data = {'url': url}
response = self.api_post(path, headers=headers, json=data)
self.assertEqual(http.ACCEPTED, response.status_code, response.text)
path = '/v2/images/%s' % image_id
func_utils.wait_for_status(self, request_path=path,
request_headers=headers,
status='active',
max_sec=10,
delay_sec=0.2,
start_delay_sec=1, multistore=True)
# Show Image
path = '/v2/images/%s' % image_id
resp = self.api_get(path, headers=self._headers())
image = jsonutils.loads(resp.text)
self.assertEqual(expect_c, image['checksum'])
self.assertEqual(expect_h, image['os_hash_value'])