2014-01-30 10:21:22 +00:00
|
|
|
# Copyright 2011 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.
|
|
|
|
|
|
|
|
"""Tests the filesystem backend store"""
|
|
|
|
|
|
|
|
import __builtin__
|
|
|
|
import errno
|
|
|
|
import hashlib
|
|
|
|
import json
|
2014-02-10 13:51:47 +00:00
|
|
|
import mock
|
2014-01-30 10:21:22 +00:00
|
|
|
import os
|
2014-09-06 03:57:37 +00:00
|
|
|
import stat
|
2014-01-30 10:21:22 +00:00
|
|
|
import StringIO
|
|
|
|
import uuid
|
|
|
|
|
2014-07-04 14:05:33 +00:00
|
|
|
import fixtures
|
2015-01-09 19:20:39 +00:00
|
|
|
from oslo_utils import units
|
2014-07-04 14:05:33 +00:00
|
|
|
import six
|
2015-02-25 12:12:43 +00:00
|
|
|
# NOTE(jokke): simplified transition to py3, behaves like py2 xrange
|
|
|
|
from six.moves import range
|
2014-07-04 14:05:33 +00:00
|
|
|
|
2014-08-28 08:09:36 +00:00
|
|
|
from glance_store._drivers.filesystem import ChunkedFile
|
|
|
|
from glance_store._drivers.filesystem import Store
|
|
|
|
from glance_store import exceptions
|
2014-09-17 12:56:35 +00:00
|
|
|
from glance_store import location
|
2014-08-28 08:09:36 +00:00
|
|
|
from glance_store.tests import base
|
2014-11-21 14:05:49 +00:00
|
|
|
from tests.unit import test_store_capabilities
|
2014-01-30 10:21:22 +00:00
|
|
|
|
2014-07-09 13:58:38 +00:00
|
|
|
|
2014-02-10 13:51:47 +00:00
|
|
|
KB = 1024
|
2014-01-30 10:21:22 +00:00
|
|
|
|
2014-07-09 13:58:38 +00:00
|
|
|
|
2014-11-21 14:05:49 +00:00
|
|
|
class TestStore(base.StoreBaseTest,
|
|
|
|
test_store_capabilities.TestStoreCapabilitiesChecking):
|
2014-01-30 10:21:22 +00:00
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
"""Establish a clean test environment."""
|
|
|
|
super(TestStore, self).setUp()
|
2014-07-29 14:40:06 +00:00
|
|
|
self.orig_chunksize = Store.READ_CHUNKSIZE
|
|
|
|
Store.READ_CHUNKSIZE = 10
|
2014-02-10 13:51:47 +00:00
|
|
|
self.store = Store(self.conf)
|
2014-04-07 13:40:31 +00:00
|
|
|
self.config(filesystem_store_datadir=self.test_dir,
|
|
|
|
group="glance_store")
|
|
|
|
self.store.configure()
|
2014-01-30 10:21:22 +00:00
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
"""Clear the test environment."""
|
|
|
|
super(TestStore, self).tearDown()
|
|
|
|
ChunkedFile.CHUNKSIZE = self.orig_chunksize
|
|
|
|
|
2014-12-08 13:50:33 +00:00
|
|
|
def _create_metadata_json_file(self, metadata):
|
|
|
|
expected_image_id = str(uuid.uuid4())
|
|
|
|
jsonfilename = os.path.join(self.test_dir,
|
|
|
|
"storage_metadata.%s" % expected_image_id)
|
|
|
|
|
|
|
|
self.config(filesystem_store_metadata_file=jsonfilename,
|
|
|
|
group="glance_store")
|
|
|
|
with open(jsonfilename, 'w') as fptr:
|
|
|
|
json.dump(metadata, fptr)
|
|
|
|
|
|
|
|
def _store_image(self, in_metadata):
|
|
|
|
expected_image_id = str(uuid.uuid4())
|
|
|
|
expected_file_size = 10
|
|
|
|
expected_file_contents = "*" * expected_file_size
|
|
|
|
image_file = StringIO.StringIO(expected_file_contents)
|
|
|
|
self.store.FILESYSTEM_STORE_METADATA = in_metadata
|
|
|
|
return self.store.add(expected_image_id, image_file,
|
|
|
|
expected_file_size)
|
|
|
|
|
2014-01-30 10:21:22 +00:00
|
|
|
def test_get(self):
|
|
|
|
"""Test a "normal" retrieval of an image in chunks."""
|
|
|
|
# First add an image...
|
|
|
|
image_id = str(uuid.uuid4())
|
|
|
|
file_contents = "chunk00000remainder"
|
|
|
|
image_file = StringIO.StringIO(file_contents)
|
|
|
|
|
2014-09-17 12:56:35 +00:00
|
|
|
loc, size, checksum, _ = self.store.add(image_id,
|
|
|
|
image_file,
|
|
|
|
len(file_contents))
|
2014-01-30 10:21:22 +00:00
|
|
|
|
|
|
|
# Now read it back...
|
|
|
|
uri = "file:///%s/%s" % (self.test_dir, image_id)
|
2014-09-17 12:56:35 +00:00
|
|
|
loc = location.get_location_from_uri(uri, conf=self.conf)
|
2014-01-30 10:21:22 +00:00
|
|
|
(image_file, image_size) = self.store.get(loc)
|
|
|
|
|
|
|
|
expected_data = "chunk00000remainder"
|
|
|
|
expected_num_chunks = 2
|
|
|
|
data = ""
|
|
|
|
num_chunks = 0
|
|
|
|
|
|
|
|
for chunk in image_file:
|
|
|
|
num_chunks += 1
|
|
|
|
data += chunk
|
|
|
|
self.assertEqual(expected_data, data)
|
|
|
|
self.assertEqual(expected_num_chunks, num_chunks)
|
|
|
|
|
2014-09-04 08:45:04 +00:00
|
|
|
def test_get_random_access(self):
|
|
|
|
"""Test a "normal" retrieval of an image in chunks."""
|
|
|
|
# First add an image...
|
|
|
|
image_id = str(uuid.uuid4())
|
|
|
|
file_contents = "chunk00000remainder"
|
|
|
|
image_file = StringIO.StringIO(file_contents)
|
|
|
|
|
2014-09-17 12:56:35 +00:00
|
|
|
loc, size, checksum, _ = self.store.add(image_id,
|
|
|
|
image_file,
|
|
|
|
len(file_contents))
|
2014-09-04 08:45:04 +00:00
|
|
|
|
|
|
|
# Now read it back...
|
|
|
|
uri = "file:///%s/%s" % (self.test_dir, image_id)
|
2014-09-17 12:56:35 +00:00
|
|
|
loc = location.get_location_from_uri(uri, conf=self.conf)
|
2014-09-04 08:45:04 +00:00
|
|
|
|
|
|
|
data = ""
|
|
|
|
for offset in range(len(file_contents)):
|
|
|
|
(image_file, image_size) = self.store.get(loc,
|
|
|
|
offset=offset,
|
|
|
|
chunk_size=1)
|
|
|
|
for chunk in image_file:
|
|
|
|
data += chunk
|
|
|
|
|
|
|
|
self.assertEqual(data, file_contents)
|
|
|
|
|
|
|
|
data = ""
|
2014-09-08 13:44:29 +00:00
|
|
|
chunk_size = 5
|
2014-09-04 08:45:04 +00:00
|
|
|
(image_file, image_size) = self.store.get(loc,
|
2014-09-08 13:44:29 +00:00
|
|
|
offset=chunk_size,
|
|
|
|
chunk_size=chunk_size)
|
2014-09-04 08:45:04 +00:00
|
|
|
for chunk in image_file:
|
|
|
|
data += chunk
|
|
|
|
|
|
|
|
self.assertEqual(data, '00000')
|
2014-09-08 13:44:29 +00:00
|
|
|
self.assertEqual(image_size, chunk_size)
|
2014-09-04 08:45:04 +00:00
|
|
|
|
2014-01-30 10:21:22 +00:00
|
|
|
def test_get_non_existing(self):
|
|
|
|
"""
|
|
|
|
Test that trying to retrieve a file that doesn't exist
|
|
|
|
raises an error
|
|
|
|
"""
|
2014-09-17 12:56:35 +00:00
|
|
|
loc = location.get_location_from_uri(
|
|
|
|
"file:///%s/non-existing" % self.test_dir, conf=self.conf)
|
2014-06-12 09:42:45 +00:00
|
|
|
self.assertRaises(exceptions.NotFound,
|
2014-01-30 10:21:22 +00:00
|
|
|
self.store.get,
|
|
|
|
loc)
|
|
|
|
|
|
|
|
def test_add(self):
|
2014-09-17 09:34:28 +00:00
|
|
|
"""Test that we can add an image via the filesystem backend."""
|
2014-01-30 10:21:22 +00:00
|
|
|
ChunkedFile.CHUNKSIZE = 1024
|
|
|
|
expected_image_id = str(uuid.uuid4())
|
2014-02-10 13:51:47 +00:00
|
|
|
expected_file_size = 5 * KB # 5K
|
2014-01-30 10:21:22 +00:00
|
|
|
expected_file_contents = "*" * expected_file_size
|
|
|
|
expected_checksum = hashlib.md5(expected_file_contents).hexdigest()
|
|
|
|
expected_location = "file://%s/%s" % (self.test_dir,
|
|
|
|
expected_image_id)
|
|
|
|
image_file = StringIO.StringIO(expected_file_contents)
|
|
|
|
|
2014-09-17 12:56:35 +00:00
|
|
|
loc, size, checksum, _ = self.store.add(expected_image_id,
|
|
|
|
image_file,
|
|
|
|
expected_file_size)
|
2014-01-30 10:21:22 +00:00
|
|
|
|
2014-09-17 12:56:35 +00:00
|
|
|
self.assertEqual(expected_location, loc)
|
2014-01-30 10:21:22 +00:00
|
|
|
self.assertEqual(expected_file_size, size)
|
|
|
|
self.assertEqual(expected_checksum, checksum)
|
|
|
|
|
|
|
|
uri = "file:///%s/%s" % (self.test_dir, expected_image_id)
|
2014-09-17 12:56:35 +00:00
|
|
|
loc = location.get_location_from_uri(uri, conf=self.conf)
|
2014-01-30 10:21:22 +00:00
|
|
|
(new_image_file, new_image_size) = self.store.get(loc)
|
|
|
|
new_image_contents = ""
|
|
|
|
new_image_file_size = 0
|
|
|
|
|
|
|
|
for chunk in new_image_file:
|
|
|
|
new_image_file_size += len(chunk)
|
|
|
|
new_image_contents += chunk
|
|
|
|
|
|
|
|
self.assertEqual(expected_file_contents, new_image_contents)
|
|
|
|
self.assertEqual(expected_file_size, new_image_file_size)
|
|
|
|
|
2014-12-08 13:50:33 +00:00
|
|
|
def test_add_check_metadata_with_invalid_mountpoint_location(self):
|
|
|
|
in_metadata = [{'id': 'abcdefg',
|
|
|
|
'mountpoint': '/xyz/images'}]
|
|
|
|
location, size, checksum, metadata = self._store_image(in_metadata)
|
|
|
|
self.assertEqual({}, metadata)
|
2014-01-30 10:21:22 +00:00
|
|
|
|
2014-12-08 13:50:33 +00:00
|
|
|
def test_add_check_metadata_list_with_invalid_mountpoint_locations(self):
|
|
|
|
in_metadata = [{'id': 'abcdefg', 'mountpoint': '/xyz/images'},
|
|
|
|
{'id': 'xyz1234', 'mountpoint': '/pqr/images'}]
|
|
|
|
location, size, checksum, metadata = self._store_image(in_metadata)
|
|
|
|
self.assertEqual({}, metadata)
|
2014-01-30 10:21:22 +00:00
|
|
|
|
2014-12-08 13:50:33 +00:00
|
|
|
def test_add_check_metadata_list_with_valid_mountpoint_locations(self):
|
|
|
|
in_metadata = [{'id': 'abcdefg', 'mountpoint': '/tmp'},
|
|
|
|
{'id': 'xyz1234', 'mountpoint': '/xyz'}]
|
|
|
|
location, size, checksum, metadata = self._store_image(in_metadata)
|
|
|
|
self.assertEqual(in_metadata[0], metadata)
|
2014-01-30 10:21:22 +00:00
|
|
|
|
|
|
|
def test_add_check_metadata_bad_nosuch_file(self):
|
|
|
|
expected_image_id = str(uuid.uuid4())
|
|
|
|
jsonfilename = os.path.join(self.test_dir,
|
|
|
|
"storage_metadata.%s" % expected_image_id)
|
|
|
|
|
2014-04-07 13:40:31 +00:00
|
|
|
self.config(filesystem_store_metadata_file=jsonfilename,
|
|
|
|
group="glance_store")
|
2014-01-30 10:21:22 +00:00
|
|
|
expected_file_size = 10
|
|
|
|
expected_file_contents = "*" * expected_file_size
|
|
|
|
image_file = StringIO.StringIO(expected_file_contents)
|
|
|
|
|
|
|
|
location, size, checksum, metadata = self.store.add(expected_image_id,
|
|
|
|
image_file,
|
|
|
|
expected_file_size)
|
|
|
|
|
|
|
|
self.assertEqual(metadata, {})
|
|
|
|
|
|
|
|
def test_add_already_existing(self):
|
|
|
|
"""
|
|
|
|
Tests that adding an image with an existing identifier
|
|
|
|
raises an appropriate exception
|
|
|
|
"""
|
|
|
|
ChunkedFile.CHUNKSIZE = 1024
|
|
|
|
image_id = str(uuid.uuid4())
|
2014-02-10 13:51:47 +00:00
|
|
|
file_size = 5 * KB # 5K
|
2014-01-30 10:21:22 +00:00
|
|
|
file_contents = "*" * file_size
|
|
|
|
image_file = StringIO.StringIO(file_contents)
|
|
|
|
|
|
|
|
location, size, checksum, _ = self.store.add(image_id,
|
|
|
|
image_file,
|
|
|
|
file_size)
|
|
|
|
image_file = StringIO.StringIO("nevergonnamakeit")
|
2014-06-12 09:42:45 +00:00
|
|
|
self.assertRaises(exceptions.Duplicate,
|
2014-01-30 10:21:22 +00:00
|
|
|
self.store.add,
|
|
|
|
image_id, image_file, 0)
|
|
|
|
|
|
|
|
def _do_test_add_write_failure(self, errno, exception):
|
|
|
|
ChunkedFile.CHUNKSIZE = 1024
|
|
|
|
image_id = str(uuid.uuid4())
|
2014-02-10 13:51:47 +00:00
|
|
|
file_size = 5 * KB # 5K
|
2014-01-30 10:21:22 +00:00
|
|
|
file_contents = "*" * file_size
|
|
|
|
path = os.path.join(self.test_dir, image_id)
|
|
|
|
image_file = StringIO.StringIO(file_contents)
|
|
|
|
|
2014-02-10 13:51:47 +00:00
|
|
|
with mock.patch.object(__builtin__, 'open') as popen:
|
|
|
|
e = IOError()
|
|
|
|
e.errno = errno
|
|
|
|
popen.side_effect = e
|
2014-01-30 10:21:22 +00:00
|
|
|
|
|
|
|
self.assertRaises(exception,
|
|
|
|
self.store.add,
|
|
|
|
image_id, image_file, 0)
|
|
|
|
self.assertFalse(os.path.exists(path))
|
2014-02-10 13:51:47 +00:00
|
|
|
|
2014-01-30 10:21:22 +00:00
|
|
|
def test_add_storage_full(self):
|
|
|
|
"""
|
|
|
|
Tests that adding an image without enough space on disk
|
|
|
|
raises an appropriate exception
|
|
|
|
"""
|
2014-06-12 09:42:45 +00:00
|
|
|
self._do_test_add_write_failure(errno.ENOSPC, exceptions.StorageFull)
|
2014-01-30 10:21:22 +00:00
|
|
|
|
|
|
|
def test_add_file_too_big(self):
|
|
|
|
"""
|
|
|
|
Tests that adding an excessively large image file
|
|
|
|
raises an appropriate exception
|
|
|
|
"""
|
2014-06-12 09:42:45 +00:00
|
|
|
self._do_test_add_write_failure(errno.EFBIG, exceptions.StorageFull)
|
2014-01-30 10:21:22 +00:00
|
|
|
|
|
|
|
def test_add_storage_write_denied(self):
|
|
|
|
"""
|
|
|
|
Tests that adding an image with insufficient filestore permissions
|
|
|
|
raises an appropriate exception
|
|
|
|
"""
|
|
|
|
self._do_test_add_write_failure(errno.EACCES,
|
2014-06-12 09:42:45 +00:00
|
|
|
exceptions.StorageWriteDenied)
|
2014-01-30 10:21:22 +00:00
|
|
|
|
|
|
|
def test_add_other_failure(self):
|
|
|
|
"""
|
|
|
|
Tests that a non-space-related IOError does not raise a
|
2014-06-12 09:42:45 +00:00
|
|
|
StorageFull exceptions.
|
2014-01-30 10:21:22 +00:00
|
|
|
"""
|
|
|
|
self._do_test_add_write_failure(errno.ENOTDIR, IOError)
|
|
|
|
|
|
|
|
def test_add_cleanup_on_read_failure(self):
|
|
|
|
"""
|
|
|
|
Tests the partial image file is cleaned up after a read
|
|
|
|
failure.
|
|
|
|
"""
|
|
|
|
ChunkedFile.CHUNKSIZE = 1024
|
|
|
|
image_id = str(uuid.uuid4())
|
2014-02-10 13:51:47 +00:00
|
|
|
file_size = 5 * KB # 5K
|
2014-01-30 10:21:22 +00:00
|
|
|
file_contents = "*" * file_size
|
|
|
|
path = os.path.join(self.test_dir, image_id)
|
|
|
|
image_file = StringIO.StringIO(file_contents)
|
|
|
|
|
|
|
|
def fake_Error(size):
|
|
|
|
raise AttributeError()
|
|
|
|
|
2014-02-10 13:51:47 +00:00
|
|
|
with mock.patch.object(image_file, 'read') as mock_read:
|
|
|
|
mock_read.side_effect = fake_Error
|
2014-01-30 10:21:22 +00:00
|
|
|
|
2014-02-10 13:51:47 +00:00
|
|
|
self.assertRaises(AttributeError,
|
|
|
|
self.store.add,
|
|
|
|
image_id, image_file, 0)
|
|
|
|
self.assertFalse(os.path.exists(path))
|
2014-01-30 10:21:22 +00:00
|
|
|
|
|
|
|
def test_delete(self):
|
|
|
|
"""
|
|
|
|
Test we can delete an existing image in the filesystem store
|
|
|
|
"""
|
|
|
|
# First add an image
|
|
|
|
image_id = str(uuid.uuid4())
|
2014-02-10 13:51:47 +00:00
|
|
|
file_size = 5 * KB # 5K
|
2014-01-30 10:21:22 +00:00
|
|
|
file_contents = "*" * file_size
|
|
|
|
image_file = StringIO.StringIO(file_contents)
|
|
|
|
|
2014-09-17 12:56:35 +00:00
|
|
|
loc, size, checksum, _ = self.store.add(image_id,
|
|
|
|
image_file,
|
|
|
|
file_size)
|
2014-01-30 10:21:22 +00:00
|
|
|
|
|
|
|
# Now check that we can delete it
|
|
|
|
uri = "file:///%s/%s" % (self.test_dir, image_id)
|
2014-09-17 12:56:35 +00:00
|
|
|
loc = location.get_location_from_uri(uri, conf=self.conf)
|
2014-01-30 10:21:22 +00:00
|
|
|
self.store.delete(loc)
|
|
|
|
|
2014-06-12 09:42:45 +00:00
|
|
|
self.assertRaises(exceptions.NotFound, self.store.get, loc)
|
2014-01-30 10:21:22 +00:00
|
|
|
|
|
|
|
def test_delete_non_existing(self):
|
|
|
|
"""
|
|
|
|
Test that trying to delete a file that doesn't exist
|
|
|
|
raises an error
|
|
|
|
"""
|
2014-09-17 12:56:35 +00:00
|
|
|
loc = location.get_location_from_uri(
|
|
|
|
"file:///tmp/glance-tests/non-existing", conf=self.conf)
|
2014-06-12 09:42:45 +00:00
|
|
|
self.assertRaises(exceptions.NotFound,
|
2014-01-30 10:21:22 +00:00
|
|
|
self.store.delete,
|
|
|
|
loc)
|
2014-07-04 14:05:33 +00:00
|
|
|
|
|
|
|
def test_configure_add_with_multi_datadirs(self):
|
|
|
|
"""
|
|
|
|
Tests multiple filesystem specified by filesystem_store_datadirs
|
|
|
|
are parsed correctly.
|
|
|
|
"""
|
|
|
|
store_map = [self.useFixture(fixtures.TempDir()).path,
|
|
|
|
self.useFixture(fixtures.TempDir()).path]
|
|
|
|
self.conf.clear_override('filesystem_store_datadir',
|
|
|
|
group='glance_store')
|
|
|
|
self.conf.set_override('filesystem_store_datadirs',
|
|
|
|
[store_map[0] + ":100",
|
|
|
|
store_map[1] + ":200"],
|
|
|
|
group='glance_store')
|
|
|
|
self.store.configure_add()
|
|
|
|
|
|
|
|
expected_priority_map = {100: [store_map[0]], 200: [store_map[1]]}
|
|
|
|
expected_priority_list = [200, 100]
|
|
|
|
self.assertEqual(self.store.priority_data_map, expected_priority_map)
|
|
|
|
self.assertEqual(self.store.priority_list, expected_priority_list)
|
|
|
|
|
2014-12-08 13:50:33 +00:00
|
|
|
def test_configure_add_with_metadata_file_success(self):
|
|
|
|
metadata = {'id': 'asdf1234',
|
|
|
|
'mountpoint': '/tmp'}
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.store.configure_add()
|
|
|
|
self.assertEqual([metadata], self.store.FILESYSTEM_STORE_METADATA)
|
|
|
|
|
|
|
|
def test_configure_add_check_metadata_list_of_dicts_success(self):
|
|
|
|
metadata = [{'id': 'abcdefg', 'mountpoint': '/xyz/images'},
|
|
|
|
{'id': 'xyz1234', 'mountpoint': '/tmp/'}]
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.store.configure_add()
|
|
|
|
self.assertEqual(metadata, self.store.FILESYSTEM_STORE_METADATA)
|
|
|
|
|
|
|
|
def test_configure_add_check_metadata_success_list_val_for_some_key(self):
|
|
|
|
metadata = {'akey': ['value1', 'value2'], 'id': 'asdf1234',
|
|
|
|
'mountpoint': '/tmp'}
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.store.configure_add()
|
|
|
|
self.assertEqual([metadata], self.store.FILESYSTEM_STORE_METADATA)
|
|
|
|
|
|
|
|
def test_configure_add_check_metadata_bad_data(self):
|
|
|
|
metadata = {'akey': 10, 'id': 'asdf1234',
|
|
|
|
'mountpoint': '/tmp'} # only unicode is allowed
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.assertRaises(exceptions.BadStoreConfiguration,
|
|
|
|
self.store.configure_add)
|
|
|
|
|
|
|
|
def test_configure_add_check_metadata_with_no_id_or_mountpoint(self):
|
|
|
|
metadata = {'mountpoint': '/tmp'}
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.assertRaises(exceptions.BadStoreConfiguration,
|
|
|
|
self.store.configure_add)
|
|
|
|
|
|
|
|
metadata = {'id': 'asdfg1234'}
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.assertRaises(exceptions.BadStoreConfiguration,
|
|
|
|
self.store.configure_add)
|
|
|
|
|
|
|
|
def test_configure_add_check_metadata_id_or_mountpoint_is_not_string(self):
|
|
|
|
metadata = {'id': 10, 'mountpoint': '/tmp'}
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.assertRaises(exceptions.BadStoreConfiguration,
|
|
|
|
self.store.configure_add)
|
|
|
|
|
|
|
|
metadata = {'id': 'asdf1234', 'mountpoint': 12345}
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.assertRaises(exceptions.BadStoreConfiguration,
|
|
|
|
self.store.configure_add)
|
|
|
|
|
|
|
|
def test_configure_add_check_metadata_list_with_no_id_or_mountpoint(self):
|
|
|
|
metadata = [{'id': 'abcdefg', 'mountpoint': '/xyz/images'},
|
|
|
|
{'mountpoint': '/pqr/images'}]
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.assertRaises(exceptions.BadStoreConfiguration,
|
|
|
|
self.store.configure_add)
|
|
|
|
|
|
|
|
metadata = [{'id': 'abcdefg'},
|
|
|
|
{'id': 'xyz1234', 'mountpoint': '/pqr/images'}]
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.assertRaises(exceptions.BadStoreConfiguration,
|
|
|
|
self.store.configure_add)
|
|
|
|
|
|
|
|
def test_add_check_metadata_list_id_or_mountpoint_is_not_string(self):
|
|
|
|
metadata = [{'id': 'abcdefg', 'mountpoint': '/xyz/images'},
|
|
|
|
{'id': 1234, 'mountpoint': '/pqr/images'}]
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.assertRaises(exceptions.BadStoreConfiguration,
|
|
|
|
self.store.configure_add)
|
|
|
|
|
|
|
|
metadata = [{'id': 'abcdefg', 'mountpoint': 1234},
|
|
|
|
{'id': 'xyz1234', 'mountpoint': '/pqr/images'}]
|
|
|
|
self._create_metadata_json_file(metadata)
|
|
|
|
self.assertRaises(exceptions.BadStoreConfiguration,
|
|
|
|
self.store.configure_add)
|
|
|
|
|
2014-07-04 14:05:33 +00:00
|
|
|
def test_configure_add_same_dir_multiple_times(self):
|
|
|
|
"""
|
|
|
|
Tests BadStoreConfiguration exception is raised if same directory
|
|
|
|
is specified multiple times in filesystem_store_datadirs.
|
|
|
|
"""
|
|
|
|
store_map = [self.useFixture(fixtures.TempDir()).path,
|
|
|
|
self.useFixture(fixtures.TempDir()).path]
|
|
|
|
self.conf.clear_override('filesystem_store_datadir',
|
|
|
|
group='glance_store')
|
|
|
|
self.conf.set_override('filesystem_store_datadirs',
|
|
|
|
[store_map[0] + ":100",
|
|
|
|
store_map[1] + ":200",
|
|
|
|
store_map[0] + ":300"],
|
|
|
|
group='glance_store')
|
|
|
|
self.assertRaises(exceptions.BadStoreConfiguration,
|
|
|
|
self.store.configure_add)
|
|
|
|
|
|
|
|
def test_add_with_multiple_dirs(self):
|
|
|
|
"""Test adding multiple filesystem directories."""
|
|
|
|
store_map = [self.useFixture(fixtures.TempDir()).path,
|
|
|
|
self.useFixture(fixtures.TempDir()).path]
|
|
|
|
self.conf.clear_override('filesystem_store_datadir',
|
|
|
|
group='glance_store')
|
|
|
|
self.conf.set_override('filesystem_store_datadirs',
|
|
|
|
[store_map[0] + ":100",
|
|
|
|
store_map[1] + ":200"],
|
|
|
|
group='glance_store')
|
2014-07-18 13:07:33 +00:00
|
|
|
|
|
|
|
self.store.configure()
|
2014-07-04 14:05:33 +00:00
|
|
|
|
|
|
|
"""Test that we can add an image via the filesystem backend"""
|
|
|
|
ChunkedFile.CHUNKSIZE = 1024
|
|
|
|
expected_image_id = str(uuid.uuid4())
|
|
|
|
expected_file_size = 5 * units.Ki # 5K
|
|
|
|
expected_file_contents = "*" * expected_file_size
|
|
|
|
expected_checksum = hashlib.md5(expected_file_contents).hexdigest()
|
|
|
|
expected_location = "file://%s/%s" % (store_map[1],
|
|
|
|
expected_image_id)
|
|
|
|
image_file = six.StringIO(expected_file_contents)
|
|
|
|
|
2014-09-17 12:56:35 +00:00
|
|
|
loc, size, checksum, _ = self.store.add(expected_image_id,
|
|
|
|
image_file,
|
|
|
|
expected_file_size)
|
2014-07-04 14:05:33 +00:00
|
|
|
|
2014-09-17 12:56:35 +00:00
|
|
|
self.assertEqual(expected_location, loc)
|
2014-07-04 14:05:33 +00:00
|
|
|
self.assertEqual(expected_file_size, size)
|
|
|
|
self.assertEqual(expected_checksum, checksum)
|
|
|
|
|
2014-09-17 12:56:35 +00:00
|
|
|
loc = location.get_location_from_uri(expected_location,
|
|
|
|
conf=self.conf)
|
2014-07-04 14:05:33 +00:00
|
|
|
(new_image_file, new_image_size) = self.store.get(loc)
|
|
|
|
new_image_contents = ""
|
|
|
|
new_image_file_size = 0
|
|
|
|
|
|
|
|
for chunk in new_image_file:
|
|
|
|
new_image_file_size += len(chunk)
|
|
|
|
new_image_contents += chunk
|
|
|
|
|
|
|
|
self.assertEqual(expected_file_contents, new_image_contents)
|
|
|
|
self.assertEqual(expected_file_size, new_image_file_size)
|
|
|
|
|
|
|
|
def test_add_with_multiple_dirs_storage_full(self):
|
|
|
|
"""
|
|
|
|
Test StorageFull exception is raised if no filesystem directory
|
|
|
|
is found that can store an image.
|
|
|
|
"""
|
|
|
|
store_map = [self.useFixture(fixtures.TempDir()).path,
|
|
|
|
self.useFixture(fixtures.TempDir()).path]
|
|
|
|
self.conf.clear_override('filesystem_store_datadir',
|
|
|
|
group='glance_store')
|
|
|
|
self.conf.set_override('filesystem_store_datadirs',
|
|
|
|
[store_map[0] + ":100",
|
|
|
|
store_map[1] + ":200"],
|
|
|
|
group='glance_store')
|
|
|
|
|
|
|
|
self.store.configure_add()
|
|
|
|
|
|
|
|
def fake_get_capacity_info(mount_point):
|
|
|
|
return 0
|
|
|
|
|
|
|
|
with mock.patch.object(self.store, '_get_capacity_info') as capacity:
|
|
|
|
capacity.return_value = 0
|
|
|
|
|
|
|
|
ChunkedFile.CHUNKSIZE = 1024
|
|
|
|
expected_image_id = str(uuid.uuid4())
|
|
|
|
expected_file_size = 5 * units.Ki # 5K
|
|
|
|
expected_file_contents = "*" * expected_file_size
|
|
|
|
image_file = six.StringIO(expected_file_contents)
|
|
|
|
|
|
|
|
self.assertRaises(exceptions.StorageFull, self.store.add,
|
2014-07-09 13:58:38 +00:00
|
|
|
expected_image_id, image_file,
|
|
|
|
expected_file_size)
|
2014-09-06 03:57:37 +00:00
|
|
|
|
|
|
|
def test_configure_add_with_file_perm(self):
|
|
|
|
"""
|
|
|
|
Tests filesystem specified by filesystem_store_file_perm
|
|
|
|
are parsed correctly.
|
|
|
|
"""
|
|
|
|
store = self.useFixture(fixtures.TempDir()).path
|
|
|
|
self.conf.set_override('filesystem_store_datadir', store,
|
|
|
|
group='glance_store')
|
|
|
|
self.conf.set_override('filesystem_store_file_perm', 700, # -rwx------
|
|
|
|
group='glance_store')
|
|
|
|
self.store.configure_add()
|
|
|
|
self.assertEqual(self.store.datadir, store)
|
|
|
|
|
|
|
|
def test_configure_add_with_unaccessible_file_perm(self):
|
|
|
|
"""
|
|
|
|
Tests BadStoreConfiguration exception is raised if an invalid
|
|
|
|
file permission specified in filesystem_store_file_perm.
|
|
|
|
"""
|
|
|
|
store = self.useFixture(fixtures.TempDir()).path
|
|
|
|
self.conf.set_override('filesystem_store_datadir', store,
|
|
|
|
group='glance_store')
|
|
|
|
self.conf.set_override('filesystem_store_file_perm', 7, # -------rwx
|
|
|
|
group='glance_store')
|
|
|
|
self.assertRaises(exceptions.BadStoreConfiguration,
|
|
|
|
self.store.configure_add)
|
|
|
|
|
|
|
|
def test_add_with_file_perm_for_group_other_users_access(self):
|
|
|
|
"""
|
|
|
|
Test that we can add an image via the filesystem backend with a
|
|
|
|
required image file permission.
|
|
|
|
"""
|
|
|
|
store = self.useFixture(fixtures.TempDir()).path
|
|
|
|
self.conf.set_override('filesystem_store_datadir', store,
|
|
|
|
group='glance_store')
|
|
|
|
self.conf.set_override('filesystem_store_file_perm', 744, # -rwxr--r--
|
|
|
|
group='glance_store')
|
|
|
|
|
|
|
|
# -rwx------
|
|
|
|
os.chmod(store, 0o700)
|
|
|
|
self.assertEqual(0o700, stat.S_IMODE(os.stat(store)[stat.ST_MODE]))
|
|
|
|
|
|
|
|
self.store.configure_add()
|
|
|
|
|
|
|
|
Store.WRITE_CHUNKSIZE = 1024
|
|
|
|
expected_image_id = str(uuid.uuid4())
|
|
|
|
expected_file_size = 5 * units.Ki # 5K
|
|
|
|
expected_file_contents = "*" * expected_file_size
|
|
|
|
expected_checksum = hashlib.md5(expected_file_contents).hexdigest()
|
|
|
|
expected_location = "file://%s/%s" % (store,
|
|
|
|
expected_image_id)
|
|
|
|
image_file = six.StringIO(expected_file_contents)
|
|
|
|
|
|
|
|
location, size, checksum, _ = self.store.add(expected_image_id,
|
|
|
|
image_file,
|
|
|
|
expected_file_size)
|
|
|
|
|
|
|
|
self.assertEqual(expected_location, location)
|
|
|
|
self.assertEqual(expected_file_size, size)
|
|
|
|
self.assertEqual(expected_checksum, checksum)
|
|
|
|
|
|
|
|
# -rwx--x--x for store directory
|
|
|
|
self.assertEqual(0o711, stat.S_IMODE(os.stat(store)[stat.ST_MODE]))
|
|
|
|
# -rwxr--r-- for image file
|
|
|
|
mode = os.stat(expected_location[len('file:/'):])[stat.ST_MODE]
|
|
|
|
perm = int(str(self.conf.glance_store.filesystem_store_file_perm), 8)
|
|
|
|
self.assertEqual(perm, stat.S_IMODE(mode))
|
|
|
|
|
|
|
|
def test_add_with_file_perm_for_owner_users_access(self):
|
|
|
|
"""
|
|
|
|
Test that we can add an image via the filesystem backend with a
|
|
|
|
required image file permission.
|
|
|
|
"""
|
|
|
|
store = self.useFixture(fixtures.TempDir()).path
|
|
|
|
self.conf.set_override('filesystem_store_datadir', store,
|
|
|
|
group='glance_store')
|
|
|
|
self.conf.set_override('filesystem_store_file_perm', 600, # -rw-------
|
|
|
|
group='glance_store')
|
|
|
|
|
|
|
|
# -rwx------
|
|
|
|
os.chmod(store, 0o700)
|
|
|
|
self.assertEqual(0o700, stat.S_IMODE(os.stat(store)[stat.ST_MODE]))
|
|
|
|
|
|
|
|
self.store.configure_add()
|
|
|
|
|
|
|
|
Store.WRITE_CHUNKSIZE = 1024
|
|
|
|
expected_image_id = str(uuid.uuid4())
|
|
|
|
expected_file_size = 5 * units.Ki # 5K
|
|
|
|
expected_file_contents = "*" * expected_file_size
|
|
|
|
expected_checksum = hashlib.md5(expected_file_contents).hexdigest()
|
|
|
|
expected_location = "file://%s/%s" % (store,
|
|
|
|
expected_image_id)
|
|
|
|
image_file = six.StringIO(expected_file_contents)
|
|
|
|
|
|
|
|
location, size, checksum, _ = self.store.add(expected_image_id,
|
|
|
|
image_file,
|
|
|
|
expected_file_size)
|
|
|
|
|
|
|
|
self.assertEqual(expected_location, location)
|
|
|
|
self.assertEqual(expected_file_size, size)
|
|
|
|
self.assertEqual(expected_checksum, checksum)
|
|
|
|
|
|
|
|
# -rwx------ for store directory
|
|
|
|
self.assertEqual(0o700, stat.S_IMODE(os.stat(store)[stat.ST_MODE]))
|
|
|
|
# -rw------- for image file
|
|
|
|
mode = os.stat(expected_location[len('file:/'):])[stat.ST_MODE]
|
|
|
|
perm = int(str(self.conf.glance_store.filesystem_store_file_perm), 8)
|
|
|
|
self.assertEqual(perm, stat.S_IMODE(mode))
|