endpoints.v2: new fs layout for data interface
Fixes QUAY-656
This commit is contained in:
parent
570c72db17
commit
ecd5996386
6 changed files with 229 additions and 228 deletions
|
@ -28,6 +28,8 @@ def ping():
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
|
||||||
from endpoints.v1 import index
|
from endpoints.v1 import (
|
||||||
from endpoints.v1 import registry
|
index,
|
||||||
from endpoints.v1 import tag
|
registry,
|
||||||
|
tag,
|
||||||
|
)
|
||||||
|
|
|
@ -13,11 +13,11 @@ from auth.permissions import (ModifyRepositoryPermission, UserAdminPermission,
|
||||||
ReadRepositoryPermission, CreateRepositoryPermission,
|
ReadRepositoryPermission, CreateRepositoryPermission,
|
||||||
repository_read_grant, repository_write_grant)
|
repository_read_grant, repository_write_grant)
|
||||||
from auth.signedgrant import generate_signed_token
|
from auth.signedgrant import generate_signed_token
|
||||||
from data.interfaces.v1 import pre_oci_model as model
|
|
||||||
from endpoints.common import parse_repository_name
|
from endpoints.common import parse_repository_name
|
||||||
from endpoints.decorators import anon_protect, anon_allowed
|
from endpoints.decorators import anon_protect, anon_allowed
|
||||||
from endpoints.notificationhelper import spawn_notification
|
from endpoints.notificationhelper import spawn_notification
|
||||||
from endpoints.v1 import v1_bp
|
from endpoints.v1 import v1_bp
|
||||||
|
from endpoints.v1.models_pre_oci import pre_oci_model as model
|
||||||
from util.audit import track_and_log
|
from util.audit import track_and_log
|
||||||
from util.http import abort
|
from util.http import abort
|
||||||
from util.names import REPOSITORY_NAME_REGEX
|
from util.names import REPOSITORY_NAME_REGEX
|
||||||
|
|
219
endpoints/v1/models_interface.py
Normal file
219
endpoints/v1/models_interface.py
Normal file
|
@ -0,0 +1,219 @@
|
||||||
|
from abc import ABCMeta, abstractmethod
|
||||||
|
from collections import namedtuple
|
||||||
|
|
||||||
|
from six import add_metaclass
|
||||||
|
|
||||||
|
|
||||||
|
class Repository(namedtuple('Repository', ['id', 'name', 'namespace_name', 'description',
|
||||||
|
'is_public', 'kind'])):
|
||||||
|
"""
|
||||||
|
Repository represents a namespaced collection of tags.
|
||||||
|
:type id: int
|
||||||
|
:type name: string
|
||||||
|
:type namespace_name: string
|
||||||
|
:type description: string
|
||||||
|
:type is_public: bool
|
||||||
|
:type kind: string
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
@add_metaclass(ABCMeta)
|
||||||
|
class DockerRegistryV1DataInterface(object):
|
||||||
|
"""
|
||||||
|
Interface that represents all data store interactions required by a Docker Registry v1.
|
||||||
|
"""
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def placement_locations_and_path_docker_v1(self, namespace_name, repo_name, image_id):
|
||||||
|
"""
|
||||||
|
Returns all the placements for the image with the given V1 Docker ID, found under the given
|
||||||
|
repository or None if no image was found.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def docker_v1_metadata(self, namespace_name, repo_name, image_id):
|
||||||
|
"""
|
||||||
|
Returns various pieces of metadata associated with an image with the given V1 Docker ID,
|
||||||
|
including the checksum and its V1 JSON metadata.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def update_docker_v1_metadata(self, namespace_name, repo_name, image_id, created_date_str,
|
||||||
|
comment, command, compat_json, parent_image_id=None):
|
||||||
|
"""
|
||||||
|
Updates various pieces of V1 metadata associated with a particular image.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def storage_exists(self, namespace_name, repo_name, image_id):
|
||||||
|
"""
|
||||||
|
Returns whether storage already exists for the image with the V1 Docker ID under the given
|
||||||
|
repository.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def store_docker_v1_checksums(self, namespace_name, repo_name, image_id, checksum,
|
||||||
|
content_checksum):
|
||||||
|
"""
|
||||||
|
Stores the various V1 checksums for the image with the V1 Docker ID.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def is_image_uploading(self, namespace_name, repo_name, image_id):
|
||||||
|
"""
|
||||||
|
Returns whether the image with the V1 Docker ID is currently marked as uploading.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def update_image_uploading(self, namespace_name, repo_name, image_id, is_uploading):
|
||||||
|
"""
|
||||||
|
Marks the image with the V1 Docker ID with the given uploading status.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def update_image_sizes(self, namespace_name, repo_name, image_id, size, uncompressed_size):
|
||||||
|
"""
|
||||||
|
Updates the sizing information for the image with the given V1 Docker ID.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def get_image_size(self, namespace_name, repo_name, image_id):
|
||||||
|
"""
|
||||||
|
Returns the wire size of the image with the given Docker V1 ID.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def create_bittorrent_pieces(self, namespace_name, repo_name, image_id, pieces_bytes):
|
||||||
|
"""
|
||||||
|
Saves the BitTorrent piece hashes for the image with the given Docker V1 ID.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def image_ancestry(self, namespace_name, repo_name, image_id):
|
||||||
|
"""
|
||||||
|
Returns a list containing the full ancestry of Docker V1 IDs, in order, for the image with the
|
||||||
|
given Docker V1 ID.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def repository_exists(self, namespace_name, repo_name):
|
||||||
|
"""
|
||||||
|
Returns whether the repository with the given name and namespace exists.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def create_or_link_image(self, username, namespace_name, repo_name, image_id, storage_location):
|
||||||
|
"""
|
||||||
|
Adds the given image to the given repository, by either linking to an existing image visible to
|
||||||
|
the user with the given username, or creating a new one if no existing image matches.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def create_temp_hidden_tag(self, namespace_name, repo_name, image_id, expiration):
|
||||||
|
"""
|
||||||
|
Creates a hidden tag under the matching namespace pointing to the image with the given V1 Docker
|
||||||
|
ID.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def list_tags(self, namespace_name, repo_name):
|
||||||
|
"""
|
||||||
|
Returns all the tags defined in the repository with the given namespace and name.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def create_or_update_tag(self, namespace_name, repo_name, image_id, tag_name):
|
||||||
|
"""
|
||||||
|
Creates or updates a tag under the matching repository to point to the image with the given
|
||||||
|
Docker V1 ID.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def find_image_id_by_tag(self, namespace_name, repo_name, tag_name):
|
||||||
|
"""
|
||||||
|
Returns the Docker V1 image ID for the HEAD image for the tag with the given name under the
|
||||||
|
matching repository, or None if none.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def delete_tag(self, namespace_name, repo_name, tag_name):
|
||||||
|
"""
|
||||||
|
Deletes the given tag from the given repository.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def load_token(self, token):
|
||||||
|
"""
|
||||||
|
Loads the data associated with the given (deprecated) access token, and, if
|
||||||
|
found returns True.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def verify_robot(self, username, token):
|
||||||
|
"""
|
||||||
|
Returns True if the given robot username and token match an existing robot
|
||||||
|
account.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def change_user_password(self, user, new_password):
|
||||||
|
"""
|
||||||
|
Changes the password associated with the given user.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def get_repository(self, namespace_name, repo_name):
|
||||||
|
"""
|
||||||
|
Returns the repository with the given name under the given namespace or None
|
||||||
|
if none.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def create_repository(self, namespace_name, repo_name, user=None):
|
||||||
|
"""
|
||||||
|
Creates a new repository under the given namespace with the given name, for
|
||||||
|
the given user.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def repository_is_public(self, namespace_name, repo_name):
|
||||||
|
"""
|
||||||
|
Returns whether the repository with the given name under the given namespace
|
||||||
|
is public. If no matching repository was found, returns False.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def validate_oauth_token(self, token):
|
||||||
|
""" Returns whether the given OAuth token validates. """
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def get_sorted_matching_repositories(self, search_term, filter_username=None, offset=0, limit=25):
|
||||||
|
"""
|
||||||
|
Returns a sorted list of repositories matching the given search term.
|
||||||
|
"""
|
||||||
|
pass
|
|
@ -1,229 +1,9 @@
|
||||||
from abc import ABCMeta, abstractmethod
|
|
||||||
from collections import namedtuple
|
|
||||||
|
|
||||||
from six import add_metaclass
|
|
||||||
|
|
||||||
from app import app, storage as store
|
from app import app, storage as store
|
||||||
from data import model
|
from data import model
|
||||||
from data.model import db_transaction
|
from endpoints.v1.models_interface import DockerRegistryV1DataInterface, Repository
|
||||||
from util.morecollections import AttrDict
|
from util.morecollections import AttrDict
|
||||||
|
|
||||||
|
|
||||||
class Repository(namedtuple('Repository', ['id', 'name', 'namespace_name', 'description',
|
|
||||||
'is_public', 'kind'])):
|
|
||||||
"""
|
|
||||||
Repository represents a namespaced collection of tags.
|
|
||||||
:type id: int
|
|
||||||
:type name: string
|
|
||||||
:type namespace_name: string
|
|
||||||
:type description: string
|
|
||||||
:type is_public: bool
|
|
||||||
:type kind: string
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
@add_metaclass(ABCMeta)
|
|
||||||
class DockerRegistryV1DataInterface(object):
|
|
||||||
"""
|
|
||||||
Interface that represents all data store interactions required by a Docker Registry v1.
|
|
||||||
"""
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def placement_locations_and_path_docker_v1(self, namespace_name, repo_name, image_id):
|
|
||||||
"""
|
|
||||||
Returns all the placements for the image with the given V1 Docker ID, found under the given
|
|
||||||
repository or None if no image was found.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def docker_v1_metadata(self, namespace_name, repo_name, image_id):
|
|
||||||
"""
|
|
||||||
Returns various pieces of metadata associated with an image with the given V1 Docker ID,
|
|
||||||
including the checksum and its V1 JSON metadata.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def update_docker_v1_metadata(self, namespace_name, repo_name, image_id, created_date_str,
|
|
||||||
comment, command, compat_json, parent_image_id=None):
|
|
||||||
"""
|
|
||||||
Updates various pieces of V1 metadata associated with a particular image.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def storage_exists(self, namespace_name, repo_name, image_id):
|
|
||||||
"""
|
|
||||||
Returns whether storage already exists for the image with the V1 Docker ID under the given
|
|
||||||
repository.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def store_docker_v1_checksums(self, namespace_name, repo_name, image_id, checksum,
|
|
||||||
content_checksum):
|
|
||||||
"""
|
|
||||||
Stores the various V1 checksums for the image with the V1 Docker ID.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def is_image_uploading(self, namespace_name, repo_name, image_id):
|
|
||||||
"""
|
|
||||||
Returns whether the image with the V1 Docker ID is currently marked as uploading.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def update_image_uploading(self, namespace_name, repo_name, image_id, is_uploading):
|
|
||||||
"""
|
|
||||||
Marks the image with the V1 Docker ID with the given uploading status.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def update_image_sizes(self, namespace_name, repo_name, image_id, size, uncompressed_size):
|
|
||||||
"""
|
|
||||||
Updates the sizing information for the image with the given V1 Docker ID.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def get_image_size(self, namespace_name, repo_name, image_id):
|
|
||||||
"""
|
|
||||||
Returns the wire size of the image with the given Docker V1 ID.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def create_bittorrent_pieces(self, namespace_name, repo_name, image_id, pieces_bytes):
|
|
||||||
"""
|
|
||||||
Saves the BitTorrent piece hashes for the image with the given Docker V1 ID.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def image_ancestry(self, namespace_name, repo_name, image_id):
|
|
||||||
"""
|
|
||||||
Returns a list containing the full ancestry of Docker V1 IDs, in order, for the image with the
|
|
||||||
given Docker V1 ID.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def repository_exists(self, namespace_name, repo_name):
|
|
||||||
"""
|
|
||||||
Returns whether the repository with the given name and namespace exists.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def create_or_link_image(self, username, namespace_name, repo_name, image_id, storage_location):
|
|
||||||
"""
|
|
||||||
Adds the given image to the given repository, by either linking to an existing image visible to
|
|
||||||
the user with the given username, or creating a new one if no existing image matches.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def create_temp_hidden_tag(self, namespace_name, repo_name, image_id, expiration):
|
|
||||||
"""
|
|
||||||
Creates a hidden tag under the matching namespace pointing to the image with the given V1 Docker
|
|
||||||
ID.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def list_tags(self, namespace_name, repo_name):
|
|
||||||
"""
|
|
||||||
Returns all the tags defined in the repository with the given namespace and name.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def create_or_update_tag(self, namespace_name, repo_name, image_id, tag_name):
|
|
||||||
"""
|
|
||||||
Creates or updates a tag under the matching repository to point to the image with the given
|
|
||||||
Docker V1 ID.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def find_image_id_by_tag(self, namespace_name, repo_name, tag_name):
|
|
||||||
"""
|
|
||||||
Returns the Docker V1 image ID for the HEAD image for the tag with the given name under the
|
|
||||||
matching repository, or None if none.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def delete_tag(self, namespace_name, repo_name, tag_name):
|
|
||||||
"""
|
|
||||||
Deletes the given tag from the given repository.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def load_token(self, token):
|
|
||||||
"""
|
|
||||||
Loads the data associated with the given (deprecated) access token, and, if
|
|
||||||
found returns True.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def verify_robot(self, username, token):
|
|
||||||
"""
|
|
||||||
Returns True if the given robot username and token match an existing robot
|
|
||||||
account.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def change_user_password(self, user, new_password):
|
|
||||||
"""
|
|
||||||
Changes the password associated with the given user.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def get_repository(self, namespace_name, repo_name):
|
|
||||||
"""
|
|
||||||
Returns the repository with the given name under the given namespace or None
|
|
||||||
if none.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def create_repository(self, namespace_name, repo_name, user=None):
|
|
||||||
"""
|
|
||||||
Creates a new repository under the given namespace with the given name, for
|
|
||||||
the given user.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def repository_is_public(self, namespace_name, repo_name):
|
|
||||||
"""
|
|
||||||
Returns whether the repository with the given name under the given namespace
|
|
||||||
is public. If no matching repository was found, returns False.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def validate_oauth_token(self, token):
|
|
||||||
""" Returns whether the given OAuth token validates. """
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def get_sorted_matching_repositories(self, search_term, filter_username=None, offset=0, limit=25):
|
|
||||||
"""
|
|
||||||
Returns a sorted list of repositories matching the given search term.
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
class PreOCIModel(DockerRegistryV1DataInterface):
|
class PreOCIModel(DockerRegistryV1DataInterface):
|
||||||
"""
|
"""
|
||||||
PreOCIModel implements the data model for the v1 Docker Registry protocol using a database schema
|
PreOCIModel implements the data model for the v1 Docker Registry protocol using a database schema
|
||||||
|
@ -274,7 +54,7 @@ class PreOCIModel(DockerRegistryV1DataInterface):
|
||||||
if repo_image is None or repo_image.storage is None:
|
if repo_image is None or repo_image.storage is None:
|
||||||
return
|
return
|
||||||
|
|
||||||
with db_transaction():
|
with model.db_transaction():
|
||||||
repo_image.storage.content_checksum = content_checksum
|
repo_image.storage.content_checksum = content_checksum
|
||||||
repo_image.v1_checksum = checksum
|
repo_image.v1_checksum = checksum
|
||||||
repo_image.storage.save()
|
repo_image.storage.save()
|
|
@ -14,9 +14,9 @@ from auth.permissions import (ReadRepositoryPermission,
|
||||||
ModifyRepositoryPermission)
|
ModifyRepositoryPermission)
|
||||||
from auth.registry_jwt_auth import get_granted_username
|
from auth.registry_jwt_auth import get_granted_username
|
||||||
from data import model, database
|
from data import model, database
|
||||||
from data.interfaces.v1 import pre_oci_model as model
|
|
||||||
from digest import checksums
|
from digest import checksums
|
||||||
from endpoints.v1 import v1_bp
|
from endpoints.v1 import v1_bp
|
||||||
|
from endpoints.v1.models_pre_oci import pre_oci_model as model
|
||||||
from endpoints.decorators import anon_protect
|
from endpoints.decorators import anon_protect
|
||||||
from util.http import abort, exact_abort
|
from util.http import abort, exact_abort
|
||||||
from util.registry.filelike import SocketReader
|
from util.registry.filelike import SocketReader
|
||||||
|
|
|
@ -8,10 +8,10 @@ from auth.decorators import process_auth
|
||||||
from auth.permissions import (ReadRepositoryPermission,
|
from auth.permissions import (ReadRepositoryPermission,
|
||||||
ModifyRepositoryPermission)
|
ModifyRepositoryPermission)
|
||||||
from data import model
|
from data import model
|
||||||
from data.interfaces.v1 import pre_oci_model as model
|
|
||||||
from endpoints.common import parse_repository_name
|
from endpoints.common import parse_repository_name
|
||||||
from endpoints.decorators import anon_protect
|
from endpoints.decorators import anon_protect
|
||||||
from endpoints.v1 import v1_bp
|
from endpoints.v1 import v1_bp
|
||||||
|
from endpoints.v1.models_pre_oci import pre_oci_model as model
|
||||||
from util.audit import track_and_log
|
from util.audit import track_and_log
|
||||||
from util.names import TAG_ERROR, TAG_REGEX
|
from util.names import TAG_ERROR, TAG_REGEX
|
||||||
|
|
||||||
|
|
Reference in a new issue