Merge pull request #1511 from coreos-inc/location-cache
Use a cache for ImageStorageLocation
This commit is contained in:
commit
bda5d7ae29
2 changed files with 40 additions and 22 deletions
|
@ -2,8 +2,8 @@ from uuid import uuid4
|
||||||
|
|
||||||
from data.model import (tag, _basequery, BlobDoesNotExist, InvalidBlobUpload, db_transaction,
|
from data.model import (tag, _basequery, BlobDoesNotExist, InvalidBlobUpload, db_transaction,
|
||||||
storage as storage_model, InvalidImageException)
|
storage as storage_model, InvalidImageException)
|
||||||
from data.database import (Repository, Namespace, ImageStorage, Image, ImageStorageLocation,
|
from data.database import (Repository, Namespace, ImageStorage, Image, ImageStoragePlacement,
|
||||||
ImageStoragePlacement, BlobUpload)
|
BlobUpload)
|
||||||
|
|
||||||
|
|
||||||
def get_repo_blob_by_digest(namespace, repo_name, blob_digest):
|
def get_repo_blob_by_digest(namespace, repo_name, blob_digest):
|
||||||
|
@ -75,6 +75,6 @@ def get_blob_upload(namespace, repo_name, upload_uuid):
|
||||||
|
|
||||||
def initiate_upload(namespace, repo_name, uuid, location_name, storage_metadata):
|
def initiate_upload(namespace, repo_name, uuid, location_name, storage_metadata):
|
||||||
repo = _basequery.get_existing_repository(namespace, repo_name)
|
repo = _basequery.get_existing_repository(namespace, repo_name)
|
||||||
location = ImageStorageLocation.get(name=location_name)
|
location = storage_model.get_image_location_for_name(location_name)
|
||||||
return BlobUpload.create(repository=repo, location=location, uuid=uuid,
|
return BlobUpload.create(repository=repo, location=location.id, uuid=uuid,
|
||||||
storage_metadata=storage_metadata)
|
storage_metadata=storage_metadata)
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
from peewee import JOIN_LEFT_OUTER, fn, SQL, IntegrityError
|
from peewee import JOIN_LEFT_OUTER, fn, SQL, IntegrityError
|
||||||
|
from cachetools import lru_cache
|
||||||
|
from collections import namedtuple
|
||||||
|
|
||||||
from data.model import (config, db_transaction, InvalidImageException, TorrentInfoDoesNotExist,
|
from data.model import (config, db_transaction, InvalidImageException, TorrentInfoDoesNotExist,
|
||||||
DataModelException, _basequery)
|
DataModelException, _basequery)
|
||||||
|
@ -11,11 +13,33 @@ from data.database import (ImageStorage, Image, ImageStoragePlacement, ImageStor
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
_Location = namedtuple('location', ['id', 'name'])
|
||||||
|
|
||||||
|
@lru_cache(maxsize=1)
|
||||||
|
def get_image_locations():
|
||||||
|
location_map = {}
|
||||||
|
for location in ImageStorageLocation.select():
|
||||||
|
location_tuple = _Location(location.id, location.name)
|
||||||
|
location_map[location.id] = location_tuple
|
||||||
|
location_map[location.name] = location_tuple
|
||||||
|
|
||||||
|
return location_map
|
||||||
|
|
||||||
|
|
||||||
|
def get_image_location_for_name(location_name):
|
||||||
|
locations = get_image_locations()
|
||||||
|
return locations[location_name]
|
||||||
|
|
||||||
|
|
||||||
|
def get_image_location_for_id(location_id):
|
||||||
|
locations = get_image_locations()
|
||||||
|
return locations[location_id]
|
||||||
|
|
||||||
|
|
||||||
def add_storage_placement(storage, location_name):
|
def add_storage_placement(storage, location_name):
|
||||||
""" Adds a storage placement for the given storage at the given location. """
|
""" Adds a storage placement for the given storage at the given location. """
|
||||||
location = ImageStorageLocation.get(name=location_name)
|
location = get_image_location_for_name(location_name)
|
||||||
ImageStoragePlacement.create(location=location, storage=storage)
|
ImageStoragePlacement.create(location=location.id, storage=storage)
|
||||||
|
|
||||||
|
|
||||||
def garbage_collect_storage(storage_id_whitelist):
|
def garbage_collect_storage(storage_id_whitelist):
|
||||||
|
@ -23,7 +47,8 @@ def garbage_collect_storage(storage_id_whitelist):
|
||||||
return
|
return
|
||||||
|
|
||||||
def placements_query_to_paths_set(placements_query):
|
def placements_query_to_paths_set(placements_query):
|
||||||
return {(placement.location.name, get_layer_path(placement.storage))
|
return {(get_image_location_for_id(placement.location_id).name,
|
||||||
|
get_layer_path(placement.storage))
|
||||||
for placement in placements_query}
|
for placement in placements_query}
|
||||||
|
|
||||||
def orphaned_storage_query(select_base_query, candidates, group_by):
|
def orphaned_storage_query(select_base_query, candidates, group_by):
|
||||||
|
@ -43,14 +68,10 @@ def garbage_collect_storage(storage_id_whitelist):
|
||||||
# Track all of the data that should be removed from blob storage
|
# Track all of the data that should be removed from blob storage
|
||||||
placements_to_remove = list(orphaned_storage_query(ImageStoragePlacement
|
placements_to_remove = list(orphaned_storage_query(ImageStoragePlacement
|
||||||
.select(ImageStoragePlacement,
|
.select(ImageStoragePlacement,
|
||||||
ImageStorage,
|
ImageStorage)
|
||||||
ImageStorageLocation)
|
|
||||||
.join(ImageStorageLocation)
|
|
||||||
.switch(ImageStoragePlacement)
|
|
||||||
.join(ImageStorage),
|
.join(ImageStorage),
|
||||||
storage_id_whitelist,
|
storage_id_whitelist,
|
||||||
(ImageStorage.id, ImageStoragePlacement.id,
|
(ImageStorage.id, ImageStoragePlacement.id)))
|
||||||
ImageStorageLocation.id)))
|
|
||||||
|
|
||||||
paths_to_remove = placements_query_to_paths_set(placements_to_remove)
|
paths_to_remove = placements_query_to_paths_set(placements_to_remove)
|
||||||
|
|
||||||
|
@ -97,8 +118,8 @@ def garbage_collect_storage(storage_id_whitelist):
|
||||||
|
|
||||||
def create_v1_storage(location_name):
|
def create_v1_storage(location_name):
|
||||||
storage = ImageStorage.create(cas_path=False)
|
storage = ImageStorage.create(cas_path=False)
|
||||||
location = ImageStorageLocation.get(name=location_name)
|
location = get_image_location_for_name(location_name)
|
||||||
ImageStoragePlacement.create(location=location, storage=storage)
|
ImageStoragePlacement.create(location=location.id, storage=storage)
|
||||||
storage.locations = {location_name}
|
storage.locations = {location_name}
|
||||||
return storage
|
return storage
|
||||||
|
|
||||||
|
@ -125,8 +146,7 @@ def lookup_storage_signature(storage, signature_kind_name):
|
||||||
|
|
||||||
def _get_storage(query_modifier):
|
def _get_storage(query_modifier):
|
||||||
query = (ImageStoragePlacement
|
query = (ImageStoragePlacement
|
||||||
.select(ImageStoragePlacement, ImageStorage, ImageStorageLocation)
|
.select(ImageStoragePlacement, ImageStorage)
|
||||||
.join(ImageStorageLocation)
|
|
||||||
.switch(ImageStoragePlacement)
|
.switch(ImageStoragePlacement)
|
||||||
.join(ImageStorage))
|
.join(ImageStorage))
|
||||||
|
|
||||||
|
@ -136,8 +156,8 @@ def _get_storage(query_modifier):
|
||||||
raise InvalidImageException()
|
raise InvalidImageException()
|
||||||
|
|
||||||
found = placements[0].storage
|
found = placements[0].storage
|
||||||
found.locations = {placement.location.name for placement in placements}
|
found.locations = {get_image_location_for_id(placement.location_id).name
|
||||||
|
for placement in placements}
|
||||||
return found
|
return found
|
||||||
|
|
||||||
|
|
||||||
|
@ -253,12 +273,10 @@ def set_image_storage_metadata(docker_image_id, namespace_name, repository_name,
|
||||||
def get_storage_locations(uuid):
|
def get_storage_locations(uuid):
|
||||||
query = (ImageStoragePlacement
|
query = (ImageStoragePlacement
|
||||||
.select()
|
.select()
|
||||||
.join(ImageStorageLocation)
|
|
||||||
.switch(ImageStoragePlacement)
|
|
||||||
.join(ImageStorage)
|
.join(ImageStorage)
|
||||||
.where(ImageStorage.uuid == uuid))
|
.where(ImageStorage.uuid == uuid))
|
||||||
|
|
||||||
return [location.location.name for location in query]
|
return [get_image_location_for_id(placement.location_id).name for placement in query]
|
||||||
|
|
||||||
|
|
||||||
def save_torrent_info(storage_object, piece_length, pieces):
|
def save_torrent_info(storage_object, piece_length, pieces):
|
||||||
|
|
Reference in a new issue