Merge pull request #1511 from coreos-inc/location-cache

Use a cache for ImageStorageLocation
This commit is contained in:
josephschorr 2016-06-09 14:03:07 -04:00
commit bda5d7ae29
2 changed files with 40 additions and 22 deletions

View file

@ -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)

View file

@ -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):