Merge "Update cached images based on update time"
This commit is contained in:
		| @@ -1750,10 +1750,12 @@ For iLO drivers, fields that should be provided are: | ||||
| * ``ilo_boot_iso``, ``image_source``, ``root_gb`` under ``instance_info``. | ||||
|  | ||||
| .. note:: | ||||
|    There is one limitation in this method - Ironic is not tracking changes of | ||||
|    content under hrefs that are specified. I.e., if the content under | ||||
|    "http://my.server.net/images/deploy.ramdisk" changes, Ironic does not know | ||||
|    about that and does not redownload the content. | ||||
|    Before Liberty release Ironic was not able to track non-Glance images' | ||||
|    content changes. Starting with Liberty, it is possible to do so using image | ||||
|    modification date. For example, for HTTP image, if 'Last-Modified' header | ||||
|    value of HEAD request to "http://my.server.net/images/deploy.ramdisk" is | ||||
|    greater than cached image modification time, Ironic will re-download the | ||||
|    content. For "file://" images, the file system modification time is used. | ||||
|  | ||||
|  | ||||
| Other references | ||||
|   | ||||
| @@ -77,10 +77,15 @@ def _extract_attributes(image): | ||||
|  | ||||
|  | ||||
| def _convert_timestamps_to_datetimes(image_meta): | ||||
|     """Returns image with timestamp fields converted to datetime objects.""" | ||||
|     """Convert timestamps to datetime objects | ||||
|  | ||||
|     Returns image metadata with timestamp fields converted to naive UTC | ||||
|     datetime objects. | ||||
|     """ | ||||
|     for attr in ['created_at', 'updated_at', 'deleted_at']: | ||||
|         if image_meta.get(attr): | ||||
|             image_meta[attr] = timeutils.parse_isotime(image_meta[attr]) | ||||
|             image_meta[attr] = timeutils.normalize_time( | ||||
|                 timeutils.parse_isotime(image_meta[attr])) | ||||
|     return image_meta | ||||
|  | ||||
| _CONVERT_PROPS = ('block_device_mapping', 'mappings') | ||||
|   | ||||
| @@ -16,6 +16,7 @@ | ||||
|  | ||||
|  | ||||
| import abc | ||||
| import datetime | ||||
| import os | ||||
| import shutil | ||||
|  | ||||
| @@ -30,6 +31,7 @@ import six.moves.urllib.parse as urlparse | ||||
| from ironic.common import exception | ||||
| from ironic.common.i18n import _ | ||||
| from ironic.common import keystone | ||||
| from ironic.common import utils | ||||
|  | ||||
| LOG = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -119,7 +121,9 @@ class BaseImageService(object): | ||||
|  | ||||
|         :param image_href: Image reference. | ||||
|         :raises: exception.ImageRefValidationFailed. | ||||
|         :returns: dictionary of image properties. | ||||
|         :returns: dictionary of image properties. It has three of them: 'size', | ||||
|             'updated_at' and 'properties'. 'updated_at' attribute is a naive | ||||
|             UTC datetime object. | ||||
|         """ | ||||
|  | ||||
|  | ||||
| @@ -178,7 +182,9 @@ class HttpImageService(BaseImageService): | ||||
|             * HEAD request failed; | ||||
|             * HEAD request returned response code not equal to 200; | ||||
|             * Content-Length header not found in response to HEAD request. | ||||
|         :returns: dictionary of image properties. | ||||
|         :returns: dictionary of image properties. It has three of them: 'size', | ||||
|             'updated_at' and 'properties'. 'updated_at' attribute is a naive | ||||
|             UTC datetime object. | ||||
|         """ | ||||
|         response = self.validate_href(image_href) | ||||
|         image_size = response.headers.get('Content-Length') | ||||
| @@ -188,8 +194,26 @@ class HttpImageService(BaseImageService): | ||||
|                 reason=_("Cannot determine image size as there is no " | ||||
|                          "Content-Length header specified in response " | ||||
|                          "to HEAD request.")) | ||||
|  | ||||
|         # Parse last-modified header to return naive datetime object | ||||
|         str_date = response.headers.get('Last-Modified') | ||||
|         date = None | ||||
|         if str_date: | ||||
|             http_date_format_strings = [ | ||||
|                 '%a, %d %b %Y %H:%M:%S GMT',  # RFC 822 | ||||
|                 '%A, %d-%b-%y %H:%M:%S GMT',  # RFC 850 | ||||
|                 '%a %b %d %H:%M:%S %Y'        # ANSI C | ||||
|             ] | ||||
|             for fmt in http_date_format_strings: | ||||
|                 try: | ||||
|                     date = datetime.datetime.strptime(str_date, fmt) | ||||
|                     break | ||||
|                 except ValueError: | ||||
|                     continue | ||||
|  | ||||
|         return { | ||||
|             'size': int(image_size), | ||||
|             'updated_at': date, | ||||
|             'properties': {} | ||||
|         } | ||||
|  | ||||
| @@ -248,11 +272,15 @@ class FileImageService(BaseImageService): | ||||
|         :param image_href: Image reference. | ||||
|         :raises: exception.ImageRefValidationFailed if image file specified | ||||
|             doesn't exist. | ||||
|         :returns: dictionary of image properties. | ||||
|         :returns: dictionary of image properties. It has three of them: 'size', | ||||
|             'updated_at' and 'properties'. 'updated_at' attribute is a naive | ||||
|             UTC datetime object. | ||||
|         """ | ||||
|         source_image_path = self.validate_href(image_href) | ||||
|         return { | ||||
|             'size': os.path.getsize(source_image_path), | ||||
|             'updated_at': utils.unix_file_modification_datetime( | ||||
|                 source_image_path), | ||||
|             'properties': {} | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -19,6 +19,7 @@ | ||||
| """Utilities and helper functions.""" | ||||
|  | ||||
| import contextlib | ||||
| import datetime | ||||
| import errno | ||||
| import hashlib | ||||
| import os | ||||
| @@ -32,7 +33,9 @@ from oslo_concurrency import processutils | ||||
| from oslo_config import cfg | ||||
| from oslo_log import log as logging | ||||
| from oslo_utils import excutils | ||||
| from oslo_utils import timeutils | ||||
| import paramiko | ||||
| import pytz | ||||
| import six | ||||
|  | ||||
| from ironic.common import exception | ||||
| @@ -658,3 +661,13 @@ def get_updated_capabilities(current_capabilities, new_capabilities): | ||||
| def is_regex_string_in_file(path, string): | ||||
|     with open(path, 'r') as inf: | ||||
|         return any(re.search(string, line) for line in inf.readlines()) | ||||
|  | ||||
|  | ||||
| def unix_file_modification_datetime(file_name): | ||||
|     return timeutils.normalize_time( | ||||
|         # normalize time to be UTC without timezone | ||||
|         datetime.datetime.fromtimestamp( | ||||
|             # fromtimestamp will return local time by default, make it UTC | ||||
|             os.path.getmtime(file_name), tz=pytz.utc | ||||
|         ) | ||||
|     ) | ||||
|   | ||||
| @@ -34,6 +34,7 @@ from ironic.common.glance_service import service_utils | ||||
| from ironic.common.i18n import _ | ||||
| from ironic.common.i18n import _LI | ||||
| from ironic.common.i18n import _LW | ||||
| from ironic.common import image_service | ||||
| from ironic.common import images | ||||
| from ironic.common import utils | ||||
|  | ||||
| @@ -75,10 +76,12 @@ class ImageCache(object): | ||||
|     def fetch_image(self, href, dest_path, ctx=None, force_raw=True): | ||||
|         """Fetch image by given href to the destination path. | ||||
|  | ||||
|         Does nothing if destination path exists and corresponds to a file that | ||||
|         exists. | ||||
|         Only creates a link if master image for this UUID is already in cache. | ||||
|         Otherwise downloads an image and also stores it in cache. | ||||
|         Does nothing if destination path exists and is up to date with cache | ||||
|         and href contents. | ||||
|         Only creates a hard link (dest_path) to cached image if requested | ||||
|         image is already in cache and up to date with href contents. | ||||
|         Otherwise downloads an image, stores it in cache and creates a hard | ||||
|         link (dest_path) to it. | ||||
|  | ||||
|         :param href: image UUID or href to fetch | ||||
|         :param dest_path: destination file path | ||||
| @@ -115,33 +118,31 @@ class ImageCache(object): | ||||
|  | ||||
|         # TODO(dtantsur): lock expiration time | ||||
|         with lockutils.lock(img_download_lock_name, 'ironic-'): | ||||
|             if os.path.exists(dest_path): | ||||
|                 # NOTE(vdrok): After rebuild requested image can change, so we | ||||
|                 # should ensure that dest_path and master_path (if exists) are | ||||
|                 # pointing to the same file | ||||
|                 if (os.path.exists(master_path) and | ||||
|                         (os.stat(dest_path).st_ino == | ||||
|                          os.stat(master_path).st_ino)): | ||||
|                     LOG.debug("Destination %(dest)s already exists for " | ||||
|                               "image %(uuid)s" % | ||||
|                               {'uuid': href, | ||||
|                                'dest': dest_path}) | ||||
|                     return | ||||
|                 os.unlink(dest_path) | ||||
|             # NOTE(vdrok): After rebuild requested image can change, so we | ||||
|             # should ensure that dest_path and master_path (if exists) are | ||||
|             # pointing to the same file and their content is up to date | ||||
|             cache_up_to_date = _delete_master_path_if_stale(master_path, href, | ||||
|                                                             ctx) | ||||
|             dest_up_to_date = _delete_dest_path_if_stale(master_path, | ||||
|                                                          dest_path) | ||||
|  | ||||
|             try: | ||||
|             if cache_up_to_date and dest_up_to_date: | ||||
|                 LOG.debug("Destination %(dest)s already exists " | ||||
|                           "for image %(href)s", | ||||
|                           {'href': href, 'dest': dest_path}) | ||||
|                 return | ||||
|  | ||||
|             if cache_up_to_date: | ||||
|                 # NOTE(dtantsur): ensure we're not in the middle of clean up | ||||
|                 with lockutils.lock('master_image', 'ironic-'): | ||||
|                     os.link(master_path, dest_path) | ||||
|             except OSError: | ||||
|                 LOG.info(_LI("Master cache miss for image %(uuid)s, " | ||||
|                              "starting download"), | ||||
|                          {'uuid': href}) | ||||
|             else: | ||||
|                 LOG.debug("Master cache hit for image %(uuid)s", | ||||
|                           {'uuid': href}) | ||||
|                 LOG.debug("Master cache hit for image %(href)s", | ||||
|                           {'href': href}) | ||||
|                 return | ||||
|  | ||||
|             LOG.info(_LI("Master cache miss for image %(href)s, " | ||||
|                          "starting download"), | ||||
|                      {'href': href}) | ||||
|             self._download_image( | ||||
|                 href, master_path, dest_path, ctx=ctx, force_raw=force_raw) | ||||
|  | ||||
| @@ -381,3 +382,59 @@ def cleanup(priority): | ||||
|         return cls | ||||
|  | ||||
|     return _add_property_to_class_func | ||||
|  | ||||
|  | ||||
| def _delete_master_path_if_stale(master_path, href, ctx): | ||||
|     """Delete image from cache if it is not up to date with href contents. | ||||
|  | ||||
|     :param master_path: path to an image in master cache | ||||
|     :param href: image href | ||||
|     :param ctx: context to use | ||||
|     :returns: True if master_path is up to date with href contents, | ||||
|         False if master_path was stale and was deleted or it didn't exist | ||||
|     """ | ||||
|     if service_utils.is_glance_image(href): | ||||
|         # Glance image contents cannot be updated without its UUID change | ||||
|         return os.path.exists(master_path) | ||||
|     if os.path.exists(master_path): | ||||
|         img_service = image_service.get_image_service(href, context=ctx) | ||||
|         img_mtime = img_service.show(href).get('updated_at') | ||||
|         if not img_mtime: | ||||
|             # This means that href is not a glance image and doesn't have an | ||||
|             # updated_at attribute | ||||
|             LOG.warn(_LW("Image service couldn't determine last " | ||||
|                          "modification time of %(href)s, considering " | ||||
|                          "cached image up to date."), {'href': href}) | ||||
|             return True | ||||
|         master_mtime = utils.unix_file_modification_datetime(master_path) | ||||
|         if img_mtime < master_mtime: | ||||
|             return True | ||||
|         # Delete image from cache as it is outdated | ||||
|         LOG.info(_LI('Image %(href)s was last modified at %(remote_time)s. ' | ||||
|                      'Deleting the cached copy since it was cached at ' | ||||
|                      '%(local_time)s and may be outdated.'), | ||||
|                  {'href': href, 'remote_time': img_mtime, | ||||
|                   'local_time': master_mtime}) | ||||
|         os.unlink(master_path) | ||||
|     return False | ||||
|  | ||||
|  | ||||
| def _delete_dest_path_if_stale(master_path, dest_path): | ||||
|     """Delete dest_path if it does not point to cached image. | ||||
|  | ||||
|     :param master_path: path to an image in master cache | ||||
|     :param dest_path: hard link to an image | ||||
|     :returns: True if dest_path points to master_path, False if dest_path was | ||||
|         stale and was deleted or it didn't exist | ||||
|     """ | ||||
|     dest_path_exists = os.path.exists(dest_path) | ||||
|     if not dest_path_exists: | ||||
|         # Image not cached, re-download | ||||
|         return False | ||||
|     master_path_exists = os.path.exists(master_path) | ||||
|     if (not master_path_exists or | ||||
|             os.stat(master_path).st_ino != os.stat(dest_path).st_ino): | ||||
|         # Image exists in cache, but dest_path out of date | ||||
|         os.unlink(dest_path) | ||||
|         return False | ||||
|     return True | ||||
|   | ||||
| @@ -16,6 +16,7 @@ | ||||
|  | ||||
| """Tests for ImageCache class and helper functions.""" | ||||
|  | ||||
| import datetime | ||||
| import os | ||||
| import tempfile | ||||
| import time | ||||
| @@ -37,7 +38,6 @@ def touch(filename): | ||||
|     open(filename, 'w').close() | ||||
|  | ||||
|  | ||||
| @mock.patch.object(image_cache, '_fetch', autospec=True) | ||||
| class TestImageCacheFetch(base.TestCase): | ||||
|  | ||||
|     def setUp(self): | ||||
| @@ -49,6 +49,7 @@ class TestImageCacheFetch(base.TestCase): | ||||
|         self.uuid = uuidutils.generate_uuid() | ||||
|         self.master_path = os.path.join(self.master_dir, self.uuid) | ||||
|  | ||||
|     @mock.patch.object(image_cache, '_fetch', autospec=True) | ||||
|     @mock.patch.object(image_cache.ImageCache, 'clean_up', autospec=True) | ||||
|     @mock.patch.object(image_cache.ImageCache, '_download_image', | ||||
|                        autospec=True) | ||||
| @@ -61,94 +62,101 @@ class TestImageCacheFetch(base.TestCase): | ||||
|             None, self.uuid, self.dest_path, True) | ||||
|         self.assertFalse(mock_clean_up.called) | ||||
|  | ||||
|     @mock.patch.object(os, 'unlink', autospec=True) | ||||
|     @mock.patch.object(image_cache.ImageCache, 'clean_up', autospec=True) | ||||
|     @mock.patch.object(image_cache.ImageCache, '_download_image', | ||||
|                        autospec=True) | ||||
|     def test_fetch_image_dest_and_master_exist_uptodate( | ||||
|             self, mock_download, mock_clean_up, mock_unlink, mock_fetch): | ||||
|         touch(self.master_path) | ||||
|         os.link(self.master_path, self.dest_path) | ||||
|     @mock.patch.object(os, 'link', autospec=True) | ||||
|     @mock.patch.object(image_cache, '_delete_dest_path_if_stale', | ||||
|                        return_value=True, autospec=True) | ||||
|     @mock.patch.object(image_cache, '_delete_master_path_if_stale', | ||||
|                        return_value=True, autospec=True) | ||||
|     def test_fetch_image_dest_and_master_uptodate( | ||||
|             self, mock_cache_upd, mock_dest_upd, mock_link, mock_download, | ||||
|             mock_clean_up): | ||||
|         self.cache.fetch_image(self.uuid, self.dest_path) | ||||
|         self.assertFalse(mock_unlink.called) | ||||
|         mock_cache_upd.assert_called_once_with(self.master_path, self.uuid, | ||||
|                                                None) | ||||
|         mock_dest_upd.assert_called_once_with(self.master_path, self.dest_path) | ||||
|         self.assertFalse(mock_link.called) | ||||
|         self.assertFalse(mock_download.called) | ||||
|         self.assertFalse(mock_fetch.called) | ||||
|         self.assertFalse(mock_clean_up.called) | ||||
|  | ||||
|     @mock.patch.object(image_cache.ImageCache, 'clean_up', autospec=True) | ||||
|     @mock.patch.object(image_cache.ImageCache, '_download_image', | ||||
|                        autospec=True) | ||||
|     def test_fetch_image_dest_and_master_exist_outdated( | ||||
|             self, mock_download, mock_clean_up, mock_fetch): | ||||
|         touch(self.master_path) | ||||
|         touch(self.dest_path) | ||||
|         self.assertNotEqual(os.stat(self.dest_path).st_ino, | ||||
|                             os.stat(self.master_path).st_ino) | ||||
|     @mock.patch.object(os, 'link', autospec=True) | ||||
|     @mock.patch.object(image_cache, '_delete_dest_path_if_stale', | ||||
|                        return_value=False, autospec=True) | ||||
|     @mock.patch.object(image_cache, '_delete_master_path_if_stale', | ||||
|                        return_value=True, autospec=True) | ||||
|     def test_fetch_image_dest_out_of_date( | ||||
|             self, mock_cache_upd, mock_dest_upd, mock_link, mock_download, | ||||
|             mock_clean_up): | ||||
|         self.cache.fetch_image(self.uuid, self.dest_path) | ||||
|         mock_cache_upd.assert_called_once_with(self.master_path, self.uuid, | ||||
|                                                None) | ||||
|         mock_dest_upd.assert_called_once_with(self.master_path, self.dest_path) | ||||
|         mock_link.assert_called_once_with(self.master_path, self.dest_path) | ||||
|         self.assertFalse(mock_download.called) | ||||
|         self.assertFalse(mock_fetch.called) | ||||
|         self.assertTrue(os.path.isfile(self.dest_path)) | ||||
|         self.assertEqual(os.stat(self.dest_path).st_ino, | ||||
|                          os.stat(self.master_path).st_ino) | ||||
|         self.assertFalse(mock_clean_up.called) | ||||
|  | ||||
|     @mock.patch.object(os, 'unlink', autospec=True) | ||||
|     @mock.patch.object(image_cache.ImageCache, 'clean_up', autospec=True) | ||||
|     @mock.patch.object(image_cache.ImageCache, '_download_image', | ||||
|                        autospec=True) | ||||
|     def test_fetch_image_only_dest_exists( | ||||
|             self, mock_download, mock_clean_up, mock_unlink, mock_fetch): | ||||
|         touch(self.dest_path) | ||||
|     @mock.patch.object(os, 'link', autospec=True) | ||||
|     @mock.patch.object(image_cache, '_delete_dest_path_if_stale', | ||||
|                        return_value=True, autospec=True) | ||||
|     @mock.patch.object(image_cache, '_delete_master_path_if_stale', | ||||
|                        return_value=False, autospec=True) | ||||
|     def test_fetch_image_master_out_of_date( | ||||
|             self, mock_cache_upd, mock_dest_upd, mock_link, mock_download, | ||||
|             mock_clean_up): | ||||
|         self.cache.fetch_image(self.uuid, self.dest_path) | ||||
|         mock_unlink.assert_called_once_with(self.dest_path) | ||||
|         self.assertFalse(mock_fetch.called) | ||||
|         mock_cache_upd.assert_called_once_with(self.master_path, self.uuid, | ||||
|                                                None) | ||||
|         mock_dest_upd.assert_called_once_with(self.master_path, self.dest_path) | ||||
|         self.assertFalse(mock_link.called) | ||||
|         mock_download.assert_called_once_with( | ||||
|             self.cache, self.uuid, self.master_path, self.dest_path, | ||||
|             ctx=None, force_raw=True) | ||||
|         self.assertTrue(mock_clean_up.called) | ||||
|         mock_clean_up.assert_called_once_with(self.cache) | ||||
|  | ||||
|     @mock.patch.object(image_cache.ImageCache, 'clean_up', autospec=True) | ||||
|     @mock.patch.object(image_cache.ImageCache, '_download_image', | ||||
|                        autospec=True) | ||||
|     def test_fetch_image_master_exists(self, mock_download, mock_clean_up, | ||||
|                                        mock_fetch): | ||||
|         touch(self.master_path) | ||||
|     @mock.patch.object(os, 'link', autospec=True) | ||||
|     @mock.patch.object(image_cache, '_delete_dest_path_if_stale', | ||||
|                        return_value=True, autospec=True) | ||||
|     @mock.patch.object(image_cache, '_delete_master_path_if_stale', | ||||
|                        return_value=False, autospec=True) | ||||
|     def test_fetch_image_both_master_and_dest_out_of_date( | ||||
|             self, mock_cache_upd, mock_dest_upd, mock_link, mock_download, | ||||
|             mock_clean_up): | ||||
|         self.cache.fetch_image(self.uuid, self.dest_path) | ||||
|         self.assertFalse(mock_download.called) | ||||
|         self.assertFalse(mock_fetch.called) | ||||
|         self.assertTrue(os.path.isfile(self.dest_path)) | ||||
|         self.assertEqual(os.stat(self.dest_path).st_ino, | ||||
|                          os.stat(self.master_path).st_ino) | ||||
|         self.assertFalse(mock_clean_up.called) | ||||
|  | ||||
|     @mock.patch.object(image_cache.ImageCache, 'clean_up', autospec=True) | ||||
|     @mock.patch.object(image_cache.ImageCache, '_download_image', | ||||
|                        autospec=True) | ||||
|     def test_fetch_image(self, mock_download, mock_clean_up, | ||||
|                          mock_fetch): | ||||
|         self.cache.fetch_image(self.uuid, self.dest_path) | ||||
|         self.assertFalse(mock_fetch.called) | ||||
|         mock_cache_upd.assert_called_once_with(self.master_path, self.uuid, | ||||
|                                                None) | ||||
|         mock_dest_upd.assert_called_once_with(self.master_path, self.dest_path) | ||||
|         self.assertFalse(mock_link.called) | ||||
|         mock_download.assert_called_once_with( | ||||
|             self.cache, self.uuid, self.master_path, self.dest_path, | ||||
|             ctx=None, force_raw=True) | ||||
|         self.assertTrue(mock_clean_up.called) | ||||
|         mock_clean_up.assert_called_once_with(self.cache) | ||||
|  | ||||
|     @mock.patch.object(image_cache.ImageCache, 'clean_up', autospec=True) | ||||
|     @mock.patch.object(image_cache.ImageCache, '_download_image', | ||||
|                        autospec=True) | ||||
|     def test_fetch_image_not_uuid(self, mock_download, mock_clean_up, | ||||
|                                   mock_fetch): | ||||
|     def test_fetch_image_not_uuid(self, mock_download, mock_clean_up): | ||||
|         href = u'http://abc.com/ubuntu.qcow2' | ||||
|         href_encoded = href.encode('utf-8') if six.PY2 else href | ||||
|         href_converted = str(uuid.uuid5(uuid.NAMESPACE_URL, href_encoded)) | ||||
|         master_path = os.path.join(self.master_dir, href_converted) | ||||
|         self.cache.fetch_image(href, self.dest_path) | ||||
|         self.assertFalse(mock_fetch.called) | ||||
|         mock_download.assert_called_once_with( | ||||
|             self.cache, href, master_path, self.dest_path, | ||||
|             ctx=None, force_raw=True) | ||||
|         self.assertTrue(mock_clean_up.called) | ||||
|  | ||||
|     @mock.patch.object(image_cache, '_fetch', autospec=True) | ||||
|     def test__download_image(self, mock_fetch): | ||||
|         def _fake_fetch(ctx, uuid, tmp_path, *args): | ||||
|             self.assertEqual(self.uuid, uuid) | ||||
| @@ -167,6 +175,119 @@ class TestImageCacheFetch(base.TestCase): | ||||
|             self.assertEqual("TEST", fp.read()) | ||||
|  | ||||
|  | ||||
| @mock.patch.object(os, 'unlink', autospec=True) | ||||
| class TestUpdateImages(base.TestCase): | ||||
|  | ||||
|     def setUp(self): | ||||
|         super(TestUpdateImages, self).setUp() | ||||
|         self.master_dir = tempfile.mkdtemp() | ||||
|         self.dest_dir = tempfile.mkdtemp() | ||||
|         self.dest_path = os.path.join(self.dest_dir, 'dest') | ||||
|         self.uuid = uuidutils.generate_uuid() | ||||
|         self.master_path = os.path.join(self.master_dir, self.uuid) | ||||
|  | ||||
|     @mock.patch.object(os.path, 'exists', return_value=False, autospec=True) | ||||
|     @mock.patch.object(image_service, 'get_image_service', autospec=True) | ||||
|     def test__delete_master_path_if_stale_glance_img_not_cached( | ||||
|             self, mock_gis, mock_path_exists, mock_unlink): | ||||
|         res = image_cache._delete_master_path_if_stale(self.master_path, | ||||
|                                                        self.uuid, None) | ||||
|         self.assertFalse(mock_gis.called) | ||||
|         self.assertFalse(mock_unlink.called) | ||||
|         mock_path_exists.assert_called_once_with(self.master_path) | ||||
|         self.assertFalse(res) | ||||
|  | ||||
|     @mock.patch.object(os.path, 'exists', return_value=True, autospec=True) | ||||
|     @mock.patch.object(image_service, 'get_image_service', autospec=True) | ||||
|     def test__delete_master_path_if_stale_glance_img( | ||||
|             self, mock_gis, mock_path_exists, mock_unlink): | ||||
|         res = image_cache._delete_master_path_if_stale(self.master_path, | ||||
|                                                        self.uuid, None) | ||||
|         self.assertFalse(mock_gis.called) | ||||
|         self.assertFalse(mock_unlink.called) | ||||
|         mock_path_exists.assert_called_once_with(self.master_path) | ||||
|         self.assertTrue(res) | ||||
|  | ||||
|     @mock.patch.object(image_service, 'get_image_service', autospec=True) | ||||
|     def test__delete_master_path_if_stale_no_master(self, mock_gis, | ||||
|                                                     mock_unlink): | ||||
|         res = image_cache._delete_master_path_if_stale(self.master_path, | ||||
|                                                        'http://11', None) | ||||
|         self.assertFalse(mock_gis.called) | ||||
|         self.assertFalse(mock_unlink.called) | ||||
|         self.assertFalse(res) | ||||
|  | ||||
|     @mock.patch.object(image_service, 'get_image_service', autospec=True) | ||||
|     def test__delete_master_path_if_stale_no_updated_at(self, mock_gis, | ||||
|                                                         mock_unlink): | ||||
|         touch(self.master_path) | ||||
|         href = 'http://awesomefreeimages.al/img111' | ||||
|         mock_gis.return_value.show.return_value = {} | ||||
|         res = image_cache._delete_master_path_if_stale(self.master_path, href, | ||||
|                                                        None) | ||||
|         mock_gis.assert_called_once_with(href, context=None) | ||||
|         self.assertFalse(mock_unlink.called) | ||||
|         self.assertTrue(res) | ||||
|  | ||||
|     @mock.patch.object(image_service, 'get_image_service', autospec=True) | ||||
|     def test__delete_master_path_if_stale_master_up_to_date(self, mock_gis, | ||||
|                                                             mock_unlink): | ||||
|         touch(self.master_path) | ||||
|         href = 'http://awesomefreeimages.al/img999' | ||||
|         mock_gis.return_value.show.return_value = { | ||||
|             'updated_at': datetime.datetime(1999, 11, 15, 8, 12, 31) | ||||
|         } | ||||
|         res = image_cache._delete_master_path_if_stale(self.master_path, href, | ||||
|                                                        None) | ||||
|         mock_gis.assert_called_once_with(href, context=None) | ||||
|         self.assertFalse(mock_unlink.called) | ||||
|         self.assertTrue(res) | ||||
|  | ||||
|     @mock.patch.object(image_service, 'get_image_service', autospec=True) | ||||
|     def test__delete_master_path_if_stale_out_of_date(self, mock_gis, | ||||
|                                                       mock_unlink): | ||||
|         touch(self.master_path) | ||||
|         href = 'http://awesomefreeimages.al/img999' | ||||
|         mock_gis.return_value.show.return_value = { | ||||
|             'updated_at': datetime.datetime((datetime.datetime.utcnow().year | ||||
|                                              + 1), 11, 15, 8, 12, 31) | ||||
|         } | ||||
|         res = image_cache._delete_master_path_if_stale(self.master_path, href, | ||||
|                                                        None) | ||||
|         mock_gis.assert_called_once_with(href, context=None) | ||||
|         mock_unlink.assert_called_once_with(self.master_path) | ||||
|         self.assertFalse(res) | ||||
|  | ||||
|     def test__delete_dest_path_if_stale_no_dest(self, mock_unlink): | ||||
|         res = image_cache._delete_dest_path_if_stale(self.master_path, | ||||
|                                                      self.dest_path) | ||||
|         self.assertFalse(mock_unlink.called) | ||||
|         self.assertFalse(res) | ||||
|  | ||||
|     def test__delete_dest_path_if_stale_no_master(self, mock_unlink): | ||||
|         touch(self.dest_path) | ||||
|         res = image_cache._delete_dest_path_if_stale(self.master_path, | ||||
|                                                      self.dest_path) | ||||
|         mock_unlink.assert_called_once_with(self.dest_path) | ||||
|         self.assertFalse(res) | ||||
|  | ||||
|     def test__delete_dest_path_if_stale_out_of_date(self, mock_unlink): | ||||
|         touch(self.master_path) | ||||
|         touch(self.dest_path) | ||||
|         res = image_cache._delete_dest_path_if_stale(self.master_path, | ||||
|                                                      self.dest_path) | ||||
|         mock_unlink.assert_called_once_with(self.dest_path) | ||||
|         self.assertFalse(res) | ||||
|  | ||||
|     def test__delete_dest_path_if_stale_up_to_date(self, mock_unlink): | ||||
|         touch(self.master_path) | ||||
|         os.link(self.master_path, self.dest_path) | ||||
|         res = image_cache._delete_dest_path_if_stale(self.master_path, | ||||
|                                                      self.dest_path) | ||||
|         self.assertFalse(mock_unlink.called) | ||||
|         self.assertTrue(res) | ||||
|  | ||||
|  | ||||
| class TestImageCacheCleanUp(base.TestCase): | ||||
|  | ||||
|     def setUp(self): | ||||
|   | ||||
| @@ -94,12 +94,7 @@ class TestGlanceImageService(base.TestCase): | ||||
|     NOW_GLANCE_OLD_FORMAT = "2010-10-11T10:30:22" | ||||
|     NOW_GLANCE_FORMAT = "2010-10-11T10:30:22.000000" | ||||
|  | ||||
|     class tzinfo(datetime.tzinfo): | ||||
|         @staticmethod | ||||
|         def utcoffset(*args, **kwargs): | ||||
|             return datetime.timedelta() | ||||
|  | ||||
|     NOW_DATETIME = datetime.datetime(2010, 10, 11, 10, 30, 22, tzinfo=tzinfo()) | ||||
|     NOW_DATETIME = datetime.datetime(2010, 10, 11, 10, 30, 22) | ||||
|  | ||||
|     def setUp(self): | ||||
|         super(TestGlanceImageService, self).setUp() | ||||
|   | ||||
| @@ -10,6 +10,7 @@ | ||||
| #    License for the specific language governing permissions and limitations | ||||
| #    under the License. | ||||
|  | ||||
| import datetime | ||||
| import os | ||||
| import shutil | ||||
|  | ||||
| @@ -66,11 +67,28 @@ class HttpImageServiceTestCase(base.TestCase): | ||||
|         head_mock.assert_called_once_with(self.href) | ||||
|  | ||||
|     @mock.patch.object(requests, 'head', autospec=True) | ||||
|     def test_show(self, head_mock): | ||||
|     def _test_show(self, head_mock, mtime, mtime_date): | ||||
|         head_mock.return_value.status_code = 200 | ||||
|         head_mock.return_value.headers = { | ||||
|             'Content-Length': 100, | ||||
|             'Last-Modified': mtime | ||||
|         } | ||||
|         result = self.service.show(self.href) | ||||
|         head_mock.assert_called_with(self.href) | ||||
|         self.assertEqual({'size': 1, 'properties': {}}, result) | ||||
|         head_mock.assert_called_once_with(self.href) | ||||
|         self.assertEqual({'size': 100, 'updated_at': mtime_date, | ||||
|                           'properties': {}}, result) | ||||
|  | ||||
|     def test_show_rfc_822(self): | ||||
|         self._test_show(mtime='Tue, 15 Nov 2014 08:12:31 GMT', | ||||
|                         mtime_date=datetime.datetime(2014, 11, 15, 8, 12, 31)) | ||||
|  | ||||
|     def test_show_rfc_850(self): | ||||
|         self._test_show(mtime='Tuesday, 15-Nov-14 08:12:31 GMT', | ||||
|                         mtime_date=datetime.datetime(2014, 11, 15, 8, 12, 31)) | ||||
|  | ||||
|     def test_show_ansi_c(self): | ||||
|         self._test_show(mtime='Tue Nov 15 08:12:31 2014', | ||||
|                         mtime_date=datetime.datetime(2014, 11, 15, 8, 12, 31)) | ||||
|  | ||||
|     @mock.patch.object(requests, 'head', autospec=True) | ||||
|     def test_show_no_content_length(self, head_mock): | ||||
| @@ -132,15 +150,21 @@ class FileImageServiceTestCase(base.TestCase): | ||||
|                           self.service.validate_href, self.href) | ||||
|         path_exists_mock.assert_called_once_with(self.href_path) | ||||
|  | ||||
|     @mock.patch.object(os.path, 'getmtime', return_value=1431087909.1641912, | ||||
|                        autospec=True) | ||||
|     @mock.patch.object(os.path, 'getsize', return_value=42, autospec=True) | ||||
|     @mock.patch.object(image_service.FileImageService, 'validate_href', | ||||
|                        autospec=True) | ||||
|     def test_show(self, _validate_mock, getsize_mock): | ||||
|     def test_show(self, _validate_mock, getsize_mock, getmtime_mock): | ||||
|         _validate_mock.return_value = self.href_path | ||||
|         result = self.service.show(self.href) | ||||
|         getsize_mock.assert_called_once_with(self.href_path) | ||||
|         getmtime_mock.assert_called_once_with(self.href_path) | ||||
|         _validate_mock.assert_called_once_with(mock.ANY, self.href) | ||||
|         self.assertEqual({'size': 42, 'properties': {}}, result) | ||||
|         self.assertEqual({'size': 42, | ||||
|                           'updated_at': datetime.datetime(2015, 5, 8, | ||||
|                                                           12, 25, 9, 164191), | ||||
|                           'properties': {}}, result) | ||||
|  | ||||
|     @mock.patch.object(os, 'link', autospec=True) | ||||
|     @mock.patch.object(os, 'remove', autospec=True) | ||||
|   | ||||
| @@ -13,6 +13,7 @@ | ||||
| #    License for the specific language governing permissions and limitations | ||||
| #    under the License. | ||||
|  | ||||
| import datetime | ||||
| import errno | ||||
| import hashlib | ||||
| import os | ||||
| @@ -434,6 +435,14 @@ class GenericUtilsTestCase(base.TestCase): | ||||
|         # original value. | ||||
|         self.assertEqual(value, utils.safe_rstrip(value)) | ||||
|  | ||||
|     @mock.patch.object(os.path, 'getmtime', return_value=1439465889.4964755, | ||||
|                        autospec=True) | ||||
|     def test_unix_file_modification_datetime(self, mtime_mock): | ||||
|         expected = datetime.datetime(2015, 8, 13, 11, 38, 9, 496475) | ||||
|         self.assertEqual(expected, | ||||
|                          utils.unix_file_modification_datetime('foo')) | ||||
|         mtime_mock.assert_called_once_with('foo') | ||||
|  | ||||
|  | ||||
| class MkfsTestCase(base.TestCase): | ||||
|  | ||||
|   | ||||
| @@ -16,6 +16,7 @@ python-neutronclient<3,>=2.6.0 | ||||
| python-glanceclient>=0.18.0 | ||||
| python-keystoneclient>=1.6.0 | ||||
| python-swiftclient>=2.2.0 | ||||
| pytz>=2013.6 | ||||
| stevedore>=1.5.0 # Apache-2.0 | ||||
| pysendfile>=2.0.0 | ||||
| websockify>=0.6.0 | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Jenkins
					Jenkins