2014-10-17 18:35:17 +00:00
|
|
|
import unittest
|
2015-02-12 21:05:45 +00:00
|
|
|
import time
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-08-01 22:42:55 +00:00
|
|
|
from playhouse.test_utils import assert_query_count
|
|
|
|
|
2014-10-17 18:35:17 +00:00
|
|
|
from app import app, storage
|
|
|
|
from initdb import setup_database_for_testing, finished_database_for_testing
|
|
|
|
from data import model, database
|
2016-07-25 21:02:00 +00:00
|
|
|
from data.database import Image, ImageStorage, DerivedStorageForImage
|
2015-11-19 21:01:36 +00:00
|
|
|
from endpoints.v2.manifest import _generate_and_store_manifest
|
|
|
|
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
ADMIN_ACCESS_USER = 'devtable'
|
|
|
|
PUBLIC_USER = 'public'
|
|
|
|
|
|
|
|
REPO = 'somerepo'
|
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
|
|
|
|
class assert_no_new_dangling_storages(object):
|
|
|
|
""" Specialized assertion for ensuring that GC cleans up all dangling storages.
|
|
|
|
"""
|
|
|
|
def __init__(self):
|
|
|
|
self.existing_count = 0
|
|
|
|
|
|
|
|
def _get_dangling_count(self):
|
|
|
|
storage_ids = set([current.id for current in ImageStorage.select()])
|
|
|
|
referneced_by_image = set([image.storage_id for image in Image.select()])
|
|
|
|
referenced_by_derived = set([derived.derivative_id for derived in DerivedStorageForImage.select()])
|
|
|
|
|
|
|
|
return len(storage_ids - referneced_by_image - referenced_by_derived)
|
|
|
|
|
|
|
|
def __enter__(self):
|
|
|
|
self.existing_count = self._get_dangling_count()
|
|
|
|
return self
|
|
|
|
|
|
|
|
def __exit__(self, exc_type, exc_val, exc_tb):
|
|
|
|
updated_count = self._get_dangling_count()
|
|
|
|
assert updated_count == self.existing_count
|
|
|
|
|
|
|
|
|
2015-06-19 18:55:44 +00:00
|
|
|
class TestGarbageCollection(unittest.TestCase):
|
2015-02-12 19:44:01 +00:00
|
|
|
@staticmethod
|
|
|
|
def _set_tag_expiration_policy(namespace, expiration_s):
|
2015-07-15 21:25:41 +00:00
|
|
|
namespace_user = model.user.get_user(namespace)
|
|
|
|
model.user.change_user_tag_expiration(namespace_user, expiration_s)
|
2015-02-11 20:02:50 +00:00
|
|
|
|
2015-02-12 19:44:01 +00:00
|
|
|
def setUp(self):
|
2014-10-17 18:35:17 +00:00
|
|
|
setup_database_for_testing(self)
|
2015-02-12 19:44:01 +00:00
|
|
|
|
|
|
|
self._set_tag_expiration_policy(ADMIN_ACCESS_USER, 0)
|
|
|
|
self._set_tag_expiration_policy(PUBLIC_USER, 0)
|
|
|
|
|
2014-10-17 18:35:17 +00:00
|
|
|
self.app = app.test_client()
|
|
|
|
self.ctx = app.test_request_context()
|
|
|
|
self.ctx.__enter__()
|
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
finished_database_for_testing(self)
|
|
|
|
self.ctx.__exit__(True, None, None)
|
|
|
|
|
|
|
|
def createImage(self, docker_image_id, repository_obj, username):
|
|
|
|
preferred = storage.preferred_locations[0]
|
2015-07-15 21:25:41 +00:00
|
|
|
image = model.image.find_create_or_link_image(docker_image_id, repository_obj, username, {},
|
|
|
|
preferred)
|
2014-10-17 18:35:17 +00:00
|
|
|
image.storage.uploading = False
|
|
|
|
image.storage.save()
|
|
|
|
|
|
|
|
# Create derived images as well.
|
2015-11-24 17:44:07 +00:00
|
|
|
model.image.find_or_create_derived_storage(image, 'squash', preferred)
|
|
|
|
model.image.find_or_create_derived_storage(image, 'aci', preferred)
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-01-12 17:15:07 +00:00
|
|
|
# Add some torrent info.
|
|
|
|
try:
|
|
|
|
model.storage.save_torrent_info(image.storage, 1, 'helloworld')
|
|
|
|
model.storage.save_torrent_info(image.storage, 2, 'helloworlds!')
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
2014-11-13 18:20:21 +00:00
|
|
|
# Add some additional placements to the image.
|
|
|
|
for location_name in ['local_eu']:
|
|
|
|
location = database.ImageStorageLocation.get(name=location_name)
|
|
|
|
|
|
|
|
try:
|
|
|
|
database.ImageStoragePlacement.get(location=location, storage=image.storage)
|
|
|
|
except:
|
|
|
|
continue
|
2014-11-24 21:07:38 +00:00
|
|
|
|
2014-11-13 18:20:21 +00:00
|
|
|
database.ImageStoragePlacement.create(location=location, storage=image.storage)
|
|
|
|
|
2014-10-17 18:35:17 +00:00
|
|
|
return image.storage
|
|
|
|
|
|
|
|
def createRepository(self, namespace=ADMIN_ACCESS_USER, name=REPO, **kwargs):
|
2015-07-15 21:25:41 +00:00
|
|
|
user = model.user.get_user(namespace)
|
|
|
|
repo = model.repository.create_repository(namespace, name, user)
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
# Populate the repository with the tags.
|
|
|
|
image_map = {}
|
|
|
|
for tag_name in kwargs:
|
|
|
|
image_ids = kwargs[tag_name]
|
|
|
|
parent = None
|
|
|
|
|
|
|
|
for image_id in image_ids:
|
|
|
|
if not image_id in image_map:
|
|
|
|
image_map[image_id] = self.createImage(image_id, repo, namespace)
|
|
|
|
|
2015-09-15 15:53:31 +00:00
|
|
|
v1_metadata = {
|
|
|
|
'id': image_id,
|
|
|
|
}
|
|
|
|
if parent is not None:
|
|
|
|
v1_metadata['parent'] = parent.docker_image_id
|
|
|
|
|
2014-10-17 18:35:17 +00:00
|
|
|
# Set the ancestors for the image.
|
2015-09-15 15:53:31 +00:00
|
|
|
parent = model.image.set_image_metadata(image_id, namespace, name, '', '', '', v1_metadata,
|
2015-07-15 21:25:41 +00:00
|
|
|
parent=parent)
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
# Set the tag for the image.
|
2015-07-15 21:25:41 +00:00
|
|
|
model.tag.create_or_update_tag(namespace, name, tag_name, image_ids[-1])
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
return repo
|
|
|
|
|
2014-10-17 21:48:31 +00:00
|
|
|
def gcNow(self, repository):
|
2015-07-15 21:25:41 +00:00
|
|
|
model.repository.garbage_collect_repository(repository.namespace_user.username, repository.name)
|
2014-10-17 21:48:31 +00:00
|
|
|
|
2015-06-19 18:55:44 +00:00
|
|
|
def deleteTag(self, repository, tag, perform_gc=True):
|
2015-07-15 21:25:41 +00:00
|
|
|
model.tag.delete_tag(repository.namespace_user.username, repository.name, tag)
|
2015-06-19 18:55:44 +00:00
|
|
|
if perform_gc:
|
|
|
|
model.repository.garbage_collect_repository(repository.namespace_user.username,
|
|
|
|
repository.name)
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
def moveTag(self, repository, tag, docker_image_id):
|
2015-07-15 21:25:41 +00:00
|
|
|
model.tag.create_or_update_tag(repository.namespace_user.username, repository.name, tag,
|
2015-02-11 20:02:50 +00:00
|
|
|
docker_image_id)
|
2015-07-15 21:25:41 +00:00
|
|
|
model.repository.garbage_collect_repository(repository.namespace_user.username, repository.name)
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
def assertNotDeleted(self, repository, *args):
|
|
|
|
for docker_image_id in args:
|
2015-07-15 21:25:41 +00:00
|
|
|
self.assertTrue(bool(model.image.get_image_by_id(repository.namespace_user.username,
|
|
|
|
repository.name, docker_image_id)))
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
def assertDeleted(self, repository, *args):
|
|
|
|
for docker_image_id in args:
|
|
|
|
try:
|
|
|
|
# Verify the image is missing when accessed by the repository.
|
2015-07-15 21:25:41 +00:00
|
|
|
model.image.get_image_by_id(repository.namespace_user.username, repository.name,
|
|
|
|
docker_image_id)
|
2015-02-11 20:02:50 +00:00
|
|
|
except model.DataModelException:
|
2014-10-17 18:35:17 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
self.fail('Expected image %s to be deleted' % docker_image_id)
|
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
|
2015-06-19 18:55:44 +00:00
|
|
|
def test_has_garbage(self):
|
|
|
|
""" Remove all existing repositories, then add one without garbage, check, then add one with
|
|
|
|
garbage, and check again.
|
|
|
|
"""
|
|
|
|
# Delete all existing repos.
|
|
|
|
for repo in database.Repository.select():
|
2015-12-18 19:18:52 +00:00
|
|
|
model.repository.purge_repository(repo.namespace_user.username, repo.name)
|
2015-06-19 18:55:44 +00:00
|
|
|
|
|
|
|
# Change the time machine expiration on the namespace.
|
|
|
|
(database.User.update(removed_tag_expiration_s=1000000000)
|
|
|
|
.where(database.User.username == ADMIN_ACCESS_USER)
|
|
|
|
.execute())
|
|
|
|
|
|
|
|
# Create a repository without any garbage.
|
|
|
|
repository = self.createRepository(latest=['i1', 'i2', 'i3'])
|
|
|
|
|
|
|
|
# Ensure that no repositories are returned by the has garbage check.
|
2016-08-01 22:22:38 +00:00
|
|
|
self.assertIsNone(model.repository.find_repository_with_garbage(1000000000))
|
2015-06-19 18:55:44 +00:00
|
|
|
|
|
|
|
# Delete a tag.
|
|
|
|
self.deleteTag(repository, 'latest', perform_gc=False)
|
|
|
|
|
|
|
|
# There should still not be any repositories with garbage, due to time machine.
|
2016-08-01 22:22:38 +00:00
|
|
|
self.assertIsNone(model.repository.find_repository_with_garbage(1000000000))
|
2015-06-19 18:55:44 +00:00
|
|
|
|
|
|
|
# Change the time machine expiration on the namespace.
|
|
|
|
(database.User.update(removed_tag_expiration_s=0)
|
|
|
|
.where(database.User.username == ADMIN_ACCESS_USER)
|
|
|
|
.execute())
|
|
|
|
|
|
|
|
# Now we should find the repository for GC.
|
2016-08-01 22:22:38 +00:00
|
|
|
repository = model.repository.find_repository_with_garbage(0)
|
2015-06-19 18:55:44 +00:00
|
|
|
self.assertIsNotNone(repository)
|
|
|
|
self.assertEquals(REPO, repository.name)
|
|
|
|
|
|
|
|
# GC the repository.
|
|
|
|
model.repository.garbage_collect_repository(repository.namespace_user.username, repository.name)
|
|
|
|
|
|
|
|
# There should now be no repositories with garbage.
|
2016-08-01 22:22:38 +00:00
|
|
|
self.assertIsNone(model.repository.find_repository_with_garbage(0))
|
2015-06-19 18:55:44 +00:00
|
|
|
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-08-01 22:42:55 +00:00
|
|
|
def test_find_garbage_policy_functions(self):
|
|
|
|
with assert_query_count(1):
|
|
|
|
one_policy = model.repository.get_random_gc_policy()
|
|
|
|
all_policies = model.repository._get_gc_expiration_policies()
|
|
|
|
self.assertIn(one_policy, all_policies)
|
|
|
|
|
|
|
|
|
2014-10-17 18:35:17 +00:00
|
|
|
def test_one_tag(self):
|
|
|
|
""" Create a repository with a single tag, then remove that tag and verify that the repository
|
|
|
|
is now empty. """
|
2016-07-25 21:02:00 +00:00
|
|
|
with assert_no_new_dangling_storages():
|
|
|
|
repository = self.createRepository(latest=['i1', 'i2', 'i3'])
|
|
|
|
self.deleteTag(repository, 'latest')
|
|
|
|
self.assertDeleted(repository, 'i1', 'i2', 'i3')
|
|
|
|
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
def test_two_tags_unshared_images(self):
|
|
|
|
""" Repository has two tags with no shared images between them. """
|
2016-07-25 21:02:00 +00:00
|
|
|
with assert_no_new_dangling_storages():
|
|
|
|
repository = self.createRepository(latest=['i1', 'i2', 'i3'], other=['f1', 'f2'])
|
|
|
|
self.deleteTag(repository, 'latest')
|
|
|
|
self.assertDeleted(repository, 'i1', 'i2', 'i3')
|
|
|
|
self.assertNotDeleted(repository, 'f1', 'f2')
|
|
|
|
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
def test_two_tags_shared_images(self):
|
|
|
|
""" Repository has two tags with shared images. Deleting the tag should only remove the
|
|
|
|
unshared images.
|
|
|
|
"""
|
2016-07-25 21:02:00 +00:00
|
|
|
with assert_no_new_dangling_storages():
|
|
|
|
repository = self.createRepository(latest=['i1', 'i2', 'i3'], other=['i1', 'f1'])
|
|
|
|
self.deleteTag(repository, 'latest')
|
|
|
|
self.assertDeleted(repository, 'i2', 'i3')
|
|
|
|
self.assertNotDeleted(repository, 'i1', 'f1')
|
|
|
|
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
def test_unrelated_repositories(self):
|
|
|
|
""" Two repositories with different images. Removing the tag from one leaves the other's
|
|
|
|
images intact.
|
|
|
|
"""
|
2016-07-25 21:02:00 +00:00
|
|
|
with assert_no_new_dangling_storages():
|
|
|
|
repository1 = self.createRepository(latest=['i1', 'i2', 'i3'], name='repo1')
|
|
|
|
repository2 = self.createRepository(latest=['j1', 'j2', 'j3'], name='repo2')
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
self.deleteTag(repository1, 'latest')
|
|
|
|
|
|
|
|
self.assertDeleted(repository1, 'i1', 'i2', 'i3')
|
|
|
|
self.assertNotDeleted(repository2, 'j1', 'j2', 'j3')
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_related_repositories(self):
|
|
|
|
""" Two repositories with shared images. Removing the tag from one leaves the other's
|
|
|
|
images intact.
|
|
|
|
"""
|
2016-07-25 21:02:00 +00:00
|
|
|
with assert_no_new_dangling_storages():
|
|
|
|
repository1 = self.createRepository(latest=['i1', 'i2', 'i3'], name='repo1')
|
|
|
|
repository2 = self.createRepository(latest=['i1', 'i2', 'j1'], name='repo2')
|
|
|
|
|
|
|
|
self.deleteTag(repository1, 'latest')
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
self.assertDeleted(repository1, 'i3')
|
|
|
|
self.assertNotDeleted(repository2, 'i1', 'i2', 'j1')
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_inaccessible_repositories(self):
|
|
|
|
""" Two repositories under different namespaces should result in the images being deleted
|
|
|
|
but not completely removed from the database.
|
|
|
|
"""
|
2016-07-25 21:02:00 +00:00
|
|
|
with assert_no_new_dangling_storages():
|
|
|
|
repository1 = self.createRepository(namespace=ADMIN_ACCESS_USER, latest=['i1', 'i2', 'i3'])
|
|
|
|
repository2 = self.createRepository(namespace=PUBLIC_USER, latest=['i1', 'i2', 'i3'])
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
self.deleteTag(repository1, 'latest')
|
|
|
|
self.assertDeleted(repository1, 'i1', 'i2', 'i3')
|
|
|
|
self.assertNotDeleted(repository2, 'i1', 'i2', 'i3')
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_multiple_shared_images(self):
|
|
|
|
""" Repository has multiple tags with shared images. Selectively deleting the tags, and
|
|
|
|
verifying at each step.
|
|
|
|
"""
|
2016-07-25 21:02:00 +00:00
|
|
|
with assert_no_new_dangling_storages():
|
|
|
|
repository = self.createRepository(latest=['i1', 'i2', 'i3'], other=['i1', 'f1', 'f2'],
|
|
|
|
third=['t1', 't2', 't3'], fourth=['i1', 'f1'])
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
# Delete tag other. Should delete f2, since it is not shared.
|
|
|
|
self.deleteTag(repository, 'other')
|
|
|
|
self.assertDeleted(repository, 'f2')
|
|
|
|
self.assertNotDeleted(repository, 'i1', 'i2', 'i3', 't1', 't2', 't3', 'f1')
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
# Move tag fourth to i3. This should remove f1 since it is no longer referenced.
|
|
|
|
self.moveTag(repository, 'fourth', 'i3')
|
|
|
|
self.assertDeleted(repository, 'f1')
|
|
|
|
self.assertNotDeleted(repository, 'i1', 'i2', 'i3', 't1', 't2', 't3')
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
# Delete tag 'latest'. This should do nothing since fourth is on the same branch.
|
|
|
|
self.deleteTag(repository, 'latest')
|
|
|
|
self.assertNotDeleted(repository, 'i1', 'i2', 'i3', 't1', 't2', 't3')
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
# Delete tag 'third'. This should remove t1->t3.
|
|
|
|
self.deleteTag(repository, 'third')
|
|
|
|
self.assertDeleted(repository, 't1', 't2', 't3')
|
|
|
|
self.assertNotDeleted(repository, 'i1', 'i2', 'i3')
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
# Add tag to i1.
|
|
|
|
self.moveTag(repository, 'newtag', 'i1')
|
|
|
|
self.assertNotDeleted(repository, 'i1', 'i2', 'i3')
|
2014-10-17 18:35:17 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
# Delete tag 'fourth'. This should remove i2 and i3.
|
|
|
|
self.deleteTag(repository, 'fourth')
|
|
|
|
self.assertDeleted(repository, 'i2', 'i3')
|
|
|
|
self.assertNotDeleted(repository, 'i1')
|
|
|
|
|
|
|
|
# Delete tag 'newtag'. This should remove the remaining image.
|
|
|
|
self.deleteTag(repository, 'newtag')
|
|
|
|
self.assertDeleted(repository, 'i1')
|
2014-10-17 18:35:17 +00:00
|
|
|
|
|
|
|
|
2014-10-17 21:48:31 +00:00
|
|
|
def test_empty_gc(self):
|
2016-07-25 21:02:00 +00:00
|
|
|
with assert_no_new_dangling_storages():
|
|
|
|
repository = self.createRepository(latest=['i1', 'i2', 'i3'], other=['i1', 'f1', 'f2'],
|
|
|
|
third=['t1', 't2', 't3'], fourth=['i1', 'f1'])
|
|
|
|
|
|
|
|
self.gcNow(repository)
|
|
|
|
self.assertNotDeleted(repository, 'i1', 'i2', 'i3', 't1', 't2', 't3', 'f1', 'f2')
|
2014-10-17 21:48:31 +00:00
|
|
|
|
2014-10-17 21:49:58 +00:00
|
|
|
|
2015-02-12 21:05:45 +00:00
|
|
|
def test_time_machine_no_gc(self):
|
|
|
|
""" Repository has two tags with shared images. Deleting the tag should not remove any images
|
|
|
|
"""
|
2016-07-25 21:02:00 +00:00
|
|
|
with assert_no_new_dangling_storages():
|
|
|
|
repository = self.createRepository(latest=['i1', 'i2', 'i3'], other=['i1', 'f1'])
|
|
|
|
self._set_tag_expiration_policy(repository.namespace_user.username, 60*60*24)
|
|
|
|
|
|
|
|
self.deleteTag(repository, 'latest')
|
|
|
|
self.assertNotDeleted(repository, 'i2', 'i3')
|
|
|
|
self.assertNotDeleted(repository, 'i1', 'f1')
|
2015-02-12 21:05:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_time_machine_gc(self):
|
|
|
|
""" Repository has two tags with shared images. Deleting the second tag should cause the images
|
|
|
|
for the first deleted tag to gc.
|
|
|
|
"""
|
2016-07-25 21:02:00 +00:00
|
|
|
with assert_no_new_dangling_storages():
|
|
|
|
repository = self.createRepository(latest=['i1', 'i2', 'i3'], other=['i1', 'f1'])
|
|
|
|
|
|
|
|
self._set_tag_expiration_policy(repository.namespace_user.username, 1)
|
2015-02-12 21:05:45 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
self.deleteTag(repository, 'latest')
|
|
|
|
self.assertNotDeleted(repository, 'i2', 'i3')
|
|
|
|
self.assertNotDeleted(repository, 'i1', 'f1')
|
2015-02-12 21:05:45 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
time.sleep(2)
|
2015-02-12 21:05:45 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
self.deleteTag(repository, 'other') # This will cause the images associated with latest to gc
|
|
|
|
self.assertDeleted(repository, 'i2', 'i3')
|
|
|
|
self.assertNotDeleted(repository, 'i1', 'f1')
|
2015-02-12 21:05:45 +00:00
|
|
|
|
|
|
|
|
2015-11-19 21:01:36 +00:00
|
|
|
def test_manifest_gc(self):
|
2016-07-25 21:02:00 +00:00
|
|
|
with assert_no_new_dangling_storages():
|
|
|
|
repository = self.createRepository(latest=['i1', 'i2', 'i3'], other=['i1', 'f1'])
|
|
|
|
_generate_and_store_manifest(ADMIN_ACCESS_USER, REPO, 'latest')
|
2015-11-19 21:01:36 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
self._set_tag_expiration_policy(repository.namespace_user.username, 0)
|
2015-11-19 21:01:36 +00:00
|
|
|
|
2016-07-25 21:02:00 +00:00
|
|
|
self.deleteTag(repository, 'latest')
|
|
|
|
self.assertDeleted(repository, 'i2', 'i3')
|
2015-11-19 21:01:36 +00:00
|
|
|
|
2015-02-12 21:05:45 +00:00
|
|
|
|
2014-10-17 18:35:17 +00:00
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|