Merge pull request #1780 from coreos-inc/createupdatetag

Raise a 409 if we try to insert a tag twice at the same time
This commit is contained in:
josephschorr 2016-08-29 16:06:57 -04:00 committed by GitHub
commit da0b0a18e7
5 changed files with 36 additions and 12 deletions

View file

@ -97,6 +97,10 @@ class ServiceNameInvalid(DataModelException):
pass pass
class TagAlreadyCreatedException(DataModelException):
pass
class TooManyLoginAttemptsException(Exception): class TooManyLoginAttemptsException(Exception):
def __init__(self, message, retry_after): def __init__(self, message, retry_after):
super(TooManyLoginAttemptsException, self).__init__(message) super(TooManyLoginAttemptsException, self).__init__(message)

View file

@ -2,8 +2,9 @@ import logging
from uuid import uuid4 from uuid import uuid4
from peewee import IntegrityError
from data.model import (image, db_transaction, DataModelException, _basequery, from data.model import (image, db_transaction, DataModelException, _basequery,
InvalidManifestException) InvalidManifestException, TagAlreadyCreatedException)
from data.database import (RepositoryTag, Repository, Image, ImageStorage, Namespace, TagManifest, from data.database import (RepositoryTag, Repository, Image, ImageStorage, Namespace, TagManifest,
RepositoryNotification, Label, TagManifestLabel, get_epoch_timestamp, RepositoryNotification, Label, TagManifestLabel, get_epoch_timestamp,
db_for_update) db_for_update)
@ -96,8 +97,12 @@ def create_or_update_tag(namespace_name, repository_name, tag_name, tag_docker_i
except Image.DoesNotExist: except Image.DoesNotExist:
raise DataModelException('Invalid image with id: %s' % tag_docker_image_id) raise DataModelException('Invalid image with id: %s' % tag_docker_image_id)
return RepositoryTag.create(repository=repo, image=image_obj, name=tag_name, try:
lifetime_start_ts=now_ts, reversion=reversion) return RepositoryTag.create(repository=repo, image=image_obj, name=tag_name,
lifetime_start_ts=now_ts, reversion=reversion)
except IntegrityError:
msg = 'Tag with name %s and lifetime start %s under repository %s/%s already exists'
raise TagAlreadyCreatedException(msg % (tag_name, now_ts, namespace_name, repository_name))
def create_temporary_hidden_tag(repo, image_obj, expiration_s): def create_temporary_hidden_tag(repo, image_obj, expiration_s):
@ -255,19 +260,16 @@ def revert_tag(repo_obj, tag_name, docker_image_id):
def store_tag_manifest(namespace, repo_name, tag_name, docker_image_id, manifest_digest, def store_tag_manifest(namespace, repo_name, tag_name, docker_image_id, manifest_digest,
manifest_data): manifest_data):
""" Stores a tag manifest for a specific tag name in the database. Returns the TagManifest """ Stores a tag manifest for a specific tag name in the database. Returns the TagManifest
object, as well as a boolean indicating whether the TagManifest was created or updated. object, as well as a boolean indicating whether the TagManifest was created.
""" """
with db_transaction(): with db_transaction():
tag = create_or_update_tag(namespace, repo_name, tag_name, docker_image_id) tag = create_or_update_tag(namespace, repo_name, tag_name, docker_image_id)
try: try:
manifest = TagManifest.get(digest=manifest_digest) manifest = TagManifest.get(digest=manifest_digest)
if manifest.tag == tag:
return manifest, False
manifest.tag = tag manifest.tag = tag
manifest.save() manifest.save()
return manifest, True return manifest, False
except TagManifest.DoesNotExist: except TagManifest.DoesNotExist:
return TagManifest.create(tag=tag, digest=manifest_digest, json_data=manifest_data), True return TagManifest.create(tag=tag, digest=manifest_digest, json_data=manifest_data), True

View file

@ -99,6 +99,13 @@ class SizeInvalid(V2RegistryException):
detail) detail)
class TagAlreadyExists(V2RegistryException):
def __init__(self, detail=None):
super(TagAlreadyExists, self).__init__('TAG_ALREADY_EXISTS',
'tag was already pushed',
detail,
409)
class TagInvalid(V2RegistryException): class TagInvalid(V2RegistryException):
def __init__(self, detail=None): def __init__(self, detail=None):
super(TagInvalid, self).__init__('TAG_INVALID', super(TagInvalid, self).__init__('TAG_INVALID',

View file

@ -20,13 +20,14 @@ from endpoints.common import parse_repository_name
from endpoints.decorators import anon_protect from endpoints.decorators import anon_protect
from endpoints.v2 import v2_bp, require_repo_read, require_repo_write from endpoints.v2 import v2_bp, require_repo_read, require_repo_write
from endpoints.v2.errors import (BlobUnknown, ManifestInvalid, ManifestUnknown, TagInvalid, from endpoints.v2.errors import (BlobUnknown, ManifestInvalid, ManifestUnknown, TagInvalid,
NameInvalid) NameInvalid, TagAlreadyExists)
from endpoints.trackhelper import track_and_log from endpoints.trackhelper import track_and_log
from endpoints.notificationhelper import spawn_notification from endpoints.notificationhelper import spawn_notification
from util.registry.replication import queue_storage_replication from util.registry.replication import queue_storage_replication
from util.names import VALID_TAG_PATTERN from util.names import VALID_TAG_PATTERN
from digest import digest_tools from digest import digest_tools
from data import model from data import model
from data.model import TagAlreadyCreatedException
from data.database import RepositoryTag from data.database import RepositoryTag
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
@ -451,9 +452,16 @@ def _write_manifest_itself(namespace_name, repo_name, manifest):
# Store the manifest pointing to the tag. # Store the manifest pointing to the tag.
manifest_digest = manifest.digest manifest_digest = manifest.digest
leaf_layer_id = images_map[layers[-1].v1_metadata.docker_id].docker_image_id leaf_layer_id = images_map[layers[-1].v1_metadata.docker_id].docker_image_id
tag_manifest, manifest_created = model.tag.store_tag_manifest(namespace_name, repo_name, tag_name,
leaf_layer_id, manifest_digest, try:
manifest.bytes) tag_manifest, manifest_created = model.tag.store_tag_manifest(namespace_name, repo_name,
tag_name, leaf_layer_id,
manifest_digest, manifest.bytes)
except TagAlreadyCreatedException:
logger.warning('Tag %s was already created under repository %s/%s pointing to image %s',
tag_name, namespace_name, repo_name, leaf_layer_id)
raise TagAlreadyExists()
if manifest_created: if manifest_created:
for key, value in layers[-1].v1_metadata.labels.iteritems(): for key, value in layers[-1].v1_metadata.labels.iteritems():
model.label.create_manifest_label(tag_manifest, key, value, 'manifest') model.label.create_manifest_label(tag_manifest, key, value, 'manifest')

View file

@ -1149,6 +1149,9 @@ class V2RegistryTests(V2RegistryPullMixin, V2RegistryPushMixin, RegistryTestsMix
(_, digest) = self.do_push('devtable', 'newrepo', 'devtable', 'password', images=images) (_, digest) = self.do_push('devtable', 'newrepo', 'devtable', 'password', images=images)
# Push again to verify no duplication.
(_, digest) = self.do_push('devtable', 'newrepo', 'devtable', 'password', images=images)
self.conduct_api_login('devtable', 'password') self.conduct_api_login('devtable', 'password')
labels = self.conduct('GET', '/api/v1/repository/devtable/newrepo/manifest/' + digest + '/labels').json() labels = self.conduct('GET', '/api/v1/repository/devtable/newrepo/manifest/' + digest + '/labels').json()
self.assertEquals(3, len(labels['labels'])) self.assertEquals(3, len(labels['labels']))