zun/zun/db/sqlalchemy/api.py

1434 lines
56 KiB
Python

# Copyright 2013 Hewlett-Packard Development Company, L.P.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
"""SQLAlchemy storage backend."""
from oslo_db import exception as db_exc
from oslo_db.sqlalchemy import session as db_session
from oslo_db.sqlalchemy import utils as db_utils
from oslo_utils import importutils
from oslo_utils import strutils
from oslo_utils import timeutils
from oslo_utils import uuidutils
import sqlalchemy as sa
from sqlalchemy.orm import contains_eager
from sqlalchemy.orm.exc import MultipleResultsFound
from sqlalchemy.orm.exc import NoResultFound
from sqlalchemy.sql.expression import desc
from sqlalchemy.sql import func
from zun.common import consts
from zun.common import exception
from zun.common.i18n import _
import zun.conf
from zun.db.sqlalchemy import models
profiler_sqlalchemy = importutils.try_import('osprofiler.sqlalchemy')
CONF = zun.conf.CONF
_FACADE = None
def _create_facade_lazily():
global _FACADE
if _FACADE is None:
_FACADE = db_session.enginefacade.get_legacy_facade()
if profiler_sqlalchemy:
if CONF.profiler.enabled and CONF.profiler.trace_sqlalchemy:
profiler_sqlalchemy.add_tracing(sa, _FACADE.get_engine(), "db")
return _FACADE
def get_engine():
facade = _create_facade_lazily()
return facade.get_engine()
def get_session(**kwargs):
facade = _create_facade_lazily()
return facade.get_session(**kwargs)
def get_backend():
"""The backend is this module itself."""
return Connection()
def model_query(model, *args, **kwargs):
"""Query helper for simpler session usage.
:param session: if present, the session to use
"""
session = kwargs.get('session') or get_session()
query = session.query(model, *args)
return query
def add_identity_filter(query, value):
"""Adds an identity filter to a query.
Filters results by ID, if supplied value is a valid integer.
Otherwise attempts to filter results by UUID.
:param query: Initial query to add filter to.
:param value: Value for filtering results by.
:return: Modified query.
"""
if strutils.is_int_like(value):
return query.filter_by(id=value)
elif uuidutils.is_uuid_like(value):
return query.filter_by(uuid=value)
else:
raise exception.InvalidIdentity(identity=value)
def _paginate_query(model, limit=None, marker=None, sort_key=None,
sort_dir=None, query=None, default_sort_key='id'):
if not query:
query = model_query(model)
sort_keys = [default_sort_key]
if sort_key and sort_key not in sort_keys:
sort_keys.insert(0, sort_key)
try:
query = db_utils.paginate_query(query, model, limit, sort_keys,
marker=marker, sort_dir=sort_dir)
except db_exc.InvalidSortKey:
raise exception.InvalidParameterValue(
_('The sort_key value "%(key)s" is an invalid field for sorting')
% {'key': sort_key})
return query.all()
class Connection(object):
"""SqlAlchemy connection."""
def __init__(self):
pass
def _add_project_filters(self, context, query):
if context.is_admin and context.all_projects:
return query
if context.project_id:
query = query.filter_by(project_id=context.project_id)
else:
query = query.filter_by(user_id=context.user_id)
return query
def _add_filters(self, query, model, filters=None, filter_names=None):
"""Generic way to add filters to a Zun model"""
if not filters:
return query
if not filter_names:
filter_names = []
for name in filter_names:
if name in filters:
value = filters[name]
if isinstance(value, list):
column = getattr(model, name)
query = query.filter(column.in_(value))
else:
column = getattr(model, name)
query = query.filter(column == value)
return query
def _add_containers_filters(self, query, filters):
filter_names = ['name', 'image', 'project_id', 'user_id',
'memory', 'host', 'task_state', 'status',
'auto_remove', 'uuid', 'capsule_id']
return self._add_filters(query, models.Container, filters=filters,
filter_names=filter_names)
def list_containers(self, context, filters=None, limit=None,
marker=None, sort_key=None, sort_dir=None):
query = model_query(models.Container)
query = self._add_project_filters(context, query)
query = self._add_containers_filters(query, filters)
return _paginate_query(models.Container, limit, marker,
sort_key, sort_dir, query)
def _validate_unique_container_name(self, context, name):
if not CONF.compute.unique_container_name_scope:
return
lowername = name.lower()
base_query = model_query(models.Container).\
filter(func.lower(models.Container.name) == lowername)
if CONF.compute.unique_container_name_scope == 'project':
container_with_same_name = base_query.\
filter_by(project_id=context.project_id).count()
elif CONF.compute.unique_container_name_scope == 'global':
container_with_same_name = base_query.count()
else:
return
if container_with_same_name > 0:
raise exception.ContainerAlreadyExists(field='name',
value=lowername)
def create_container(self, context, values):
# ensure defaults are present for new containers
if not values.get('uuid'):
values['uuid'] = uuidutils.generate_uuid()
if values.get('name'):
self._validate_unique_container_name(context, values['name'])
container = models.Container()
container.update(values)
try:
container.save()
except db_exc.DBDuplicateEntry:
raise exception.ContainerAlreadyExists(field='UUID',
value=values['uuid'])
return container
def get_container_by_uuid(self, context, container_uuid):
query = model_query(models.Container)
query = self._add_project_filters(context, query)
query = query.filter_by(uuid=container_uuid)
try:
return query.one()
except NoResultFound:
raise exception.ContainerNotFound(container=container_uuid)
def get_container_by_name(self, context, container_name):
query = model_query(models.Container)
query = self._add_project_filters(context, query)
query = query.filter_by(name=container_name)
try:
return query.one()
except NoResultFound:
raise exception.ContainerNotFound(container=container_name)
except MultipleResultsFound:
raise exception.Conflict('Multiple containers exist with same '
'name. Please use the container uuid '
'instead.')
def destroy_container(self, context, container_id):
session = get_session()
with session.begin():
query = model_query(models.Container, session=session)
query = add_identity_filter(query, container_id)
count = query.delete()
if count != 1:
raise exception.ContainerNotFound(container_id)
def update_container(self, context, container_id, values):
# NOTE(dtantsur): this can lead to very strange errors
if 'uuid' in values:
msg = _("Cannot overwrite UUID for an existing Container.")
raise exception.InvalidParameterValue(err=msg)
if 'name' in values:
self._validate_unique_container_name(context, values['name'])
return self._do_update_container(container_id, values)
def _do_update_container(self, container_id, values):
session = get_session()
with session.begin():
query = model_query(models.Container, session=session)
query = add_identity_filter(query, container_id)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.ContainerNotFound(container=container_id)
ref.update(values)
return ref
def _add_volume_mappings_filters(self, query, filters):
filter_names = ['project_id', 'user_id', 'volume_id',
'container_path', 'container_uuid']
return self._add_filters(query, models.VolumeMapping, filters=filters,
filter_names=filter_names)
def list_volume_mappings(self, context, filters=None, limit=None,
marker=None, sort_key=None, sort_dir=None):
query = model_query(models.VolumeMapping)
query = query.join(models.Volume)
query = self._add_project_filters(context, query)
query = self._add_volume_filters(query, filters)
query = self._add_volume_mappings_filters(query, filters)
return _paginate_query(models.VolumeMapping, limit, marker,
sort_key, sort_dir, query)
def count_volume_mappings(self, context, **filters):
query = model_query(models.VolumeMapping)
query = self._add_project_filters(context, query)
query = self._add_volume_mappings_filters(query, filters)
return query.count()
def create_volume_mapping(self, context, values):
# ensure defaults are present for new volume_mappings
if not values.get('uuid'):
values['uuid'] = uuidutils.generate_uuid()
volume_mapping = models.VolumeMapping()
volume_mapping.update(values)
try:
volume_mapping.save()
except db_exc.DBDuplicateEntry:
raise exception.VolumeMappingAlreadyExists(field='UUID',
value=values['uuid'])
return volume_mapping
def get_volume_mapping_by_uuid(self, context, volume_mapping_uuid):
query = model_query(models.VolumeMapping)
query = self._add_project_filters(context, query)
query = query.filter_by(uuid=volume_mapping_uuid)
try:
return query.one()
except NoResultFound:
raise exception.VolumeMappingNotFound(volume_mapping_uuid)
def destroy_volume_mapping(self, context, volume_mapping_uuid):
session = get_session()
with session.begin():
query = model_query(models.VolumeMapping, session=session)
query = add_identity_filter(query, volume_mapping_uuid)
count = query.delete()
if count != 1:
raise exception.VolumeMappingNotFound(
volume_mapping_uuid)
def update_volume_mapping(self, context, volume_mapping_uuid, values):
# NOTE(dtantsur): this can lead to very strange errors
if 'uuid' in values:
msg = _("Cannot overwrite UUID for an existing VolumeMapping.")
raise exception.InvalidParameterValue(err=msg)
return self._do_update_volume_mapping(volume_mapping_uuid, values)
def _do_update_volume_mapping(self, volume_mapping_uuid, values):
session = get_session()
with session.begin():
query = model_query(models.VolumeMapping, session=session)
query = add_identity_filter(query, volume_mapping_uuid)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.VolumeMappingNotFound(volume_mapping_uuid)
ref.update(values)
return ref
def _add_volume_filters(self, query, filters):
filter_names = ['project_id', 'user_id', 'cinder_volume_id',
'volume_provider']
return self._add_filters(query, models.Volume, filters=filters,
filter_names=filter_names)
def create_volume(self, context, values):
# ensure defaults are present for new volume_mappings
if not values.get('uuid'):
values['uuid'] = uuidutils.generate_uuid()
volume = models.Volume()
volume.update(values)
try:
volume.save()
except db_exc.DBDuplicateEntry:
raise exception.VolumeAlreadyExists(field='UUID',
value=values['uuid'])
return volume
def get_volume_by_id(self, context, volume_id):
query = model_query(models.Volume)
query = self._add_project_filters(context, query)
query = query.filter_by(id=volume_id)
try:
return query.one()
except NoResultFound:
raise exception.VolumeNotFound(volume_id)
def destroy_volume(self, context, volume_uuid):
session = get_session()
with session.begin():
query = model_query(models.Volume, session=session)
query = add_identity_filter(query, volume_uuid)
count = query.delete()
if count != 1:
raise exception.VolumeNotFound(volume_uuid)
def update_volume(self, context, volume_uuid, values):
# NOTE(dtantsur): this can lead to very strange errors
if 'uuid' in values:
msg = _("Cannot overwrite UUID for an existing Volume.")
raise exception.InvalidParameterValue(err=msg)
return self._do_update_volume(volume_uuid, values)
def _do_update_volume(self, volume_uuid, values):
session = get_session()
with session.begin():
query = model_query(models.Volume, session=session)
query = add_identity_filter(query, volume_uuid)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.VolumeNotFound(volume_uuid)
ref.update(values)
return ref
def destroy_zun_service(self, host, binary):
session = get_session()
with session.begin():
query = model_query(models.ZunService, session=session)
query = query.filter_by(host=host, binary=binary)
count = query.delete()
if count != 1:
raise exception.ZunServiceNotFound(host=host, binary=binary)
def update_zun_service(self, host, binary, values):
session = get_session()
with session.begin():
query = model_query(models.ZunService, session=session)
query = query.filter_by(host=host, binary=binary)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.ZunServiceNotFound(host=host, binary=binary)
if 'report_count' in values:
if values['report_count'] > ref.report_count:
ref.last_seen_up = timeutils.utcnow()
ref.update(values)
return ref
def get_zun_service(self, host, binary):
query = model_query(models.ZunService)
query = query.filter_by(host=host, binary=binary)
try:
return query.one()
except NoResultFound:
return None
def create_zun_service(self, values):
zun_service = models.ZunService()
zun_service.update(values)
try:
zun_service.save()
except db_exc.DBDuplicateEntry:
raise exception.ZunServiceAlreadyExists(
host=zun_service.host, binary=zun_service.binary)
return zun_service
def _add_zun_service_filters(self, query, filters):
filter_names = ['disabled', 'host', 'binary', 'project_id', 'user_id']
return self._add_filters(query, models.ZunService, filters=filters,
filter_names=filter_names)
def list_zun_services(self, filters=None, limit=None, marker=None,
sort_key=None, sort_dir=None):
query = model_query(models.ZunService)
if filters:
query = self._add_zun_service_filters(query, filters)
return _paginate_query(models.ZunService, limit, marker,
sort_key, sort_dir, query)
def list_zun_services_by_binary(self, binary):
query = model_query(models.ZunService)
query = query.filter_by(binary=binary)
return _paginate_query(models.ZunService, query=query)
def destroy_image(self, context, uuid):
session = get_session()
with session.begin():
query = model_query(models.Image, session=session)
query = add_identity_filter(query, uuid)
count = query.delete()
if count != 1:
raise exception.ImageNotFound(uuid)
def pull_image(self, context, values):
# ensure defaults are present for new images
if not values.get('uuid'):
values['uuid'] = uuidutils.generate_uuid()
image = models.Image()
image.update(values)
try:
image.save()
except db_exc.DBDuplicateEntry:
raise exception.ImageAlreadyExists(tag=values['tag'],
repo=values['repo'])
return image
def update_image(self, image_id, values):
# NOTE(dtantsur): this can lead to very strange errors
if 'uuid' in values:
msg = _("Cannot overwrite UUID for an existing Image.")
raise exception.InvalidParameterValue(err=msg)
return self._do_update_image(image_id, values)
def _do_update_image(self, image_id, values):
session = get_session()
with session.begin():
query = model_query(models.Image, session=session)
query = add_identity_filter(query, image_id)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.ImageNotFound(image=image_id)
ref.update(values)
return ref
def _add_image_filters(self, query, filters):
filter_names = ['repo', 'project_id', 'user_id', 'size']
return self._add_filters(query, models.Image, filters=filters,
filter_names=filter_names)
def list_images(self, context, filters=None, limit=None, marker=None,
sort_key=None, sort_dir=None):
query = model_query(models.Image)
query = self._add_project_filters(context, query)
query = self._add_image_filters(query, filters)
return _paginate_query(models.Image, limit, marker, sort_key,
sort_dir, query)
def get_image_by_id(self, context, image_id):
query = model_query(models.Image)
query = self._add_project_filters(context, query)
query = query.filter_by(id=image_id)
try:
return query.one()
except NoResultFound:
raise exception.ImageNotFound(image=image_id)
def get_image_by_uuid(self, context, image_uuid):
query = model_query(models.Image)
query = self._add_project_filters(context, query)
query = query.filter_by(uuid=image_uuid)
try:
return query.one()
except NoResultFound:
raise exception.ImageNotFound(image=image_uuid)
def _add_resource_providers_filters(self, query, filters):
filter_names = ['name', 'root_provider', 'parent_provider', 'can_host']
return self._add_filters(query, models.ResourceProvider,
filters=filters,
filter_names=filter_names)
def list_resource_providers(self, context, filters=None, limit=None,
marker=None, sort_key=None, sort_dir=None):
query = model_query(models.ResourceProvider)
query = self._add_resource_providers_filters(query, filters)
return _paginate_query(models.ResourceProvider, limit, marker,
sort_key, sort_dir, query)
def create_resource_provider(self, context, values):
# ensure defaults are present for new resource providers
if not values.get('uuid'):
values['uuid'] = uuidutils.generate_uuid()
resource_provider = models.ResourceProvider()
resource_provider.update(values)
try:
resource_provider.save()
except db_exc.DBDuplicateEntry:
raise exception.ResourceProviderAlreadyExists(
field='UUID', value=values['uuid'])
return resource_provider
def get_resource_provider(self, context, provider_ident):
if uuidutils.is_uuid_like(provider_ident):
return self._get_resource_provider_by_uuid(context, provider_ident)
else:
return self._get_resource_provider_by_name(context, provider_ident)
def _get_resource_provider_by_uuid(self, context, provider_uuid):
query = model_query(models.ResourceProvider)
query = query.filter_by(uuid=provider_uuid)
try:
return query.one()
except NoResultFound:
raise exception.ResourceProviderNotFound(
resource_provider=provider_uuid)
def _get_resource_provider_by_name(self, context, provider_name):
query = model_query(models.ResourceProvider)
query = query.filter_by(name=provider_name)
try:
return query.one()
except NoResultFound:
raise exception.ResourceProviderNotFound(
resource_provider=provider_name)
except MultipleResultsFound:
raise exception.Conflict('Multiple resource providers exist with '
'same name. Please use the uuid instead.')
def destroy_resource_provider(self, context, provider_id):
session = get_session()
with session.begin():
query = model_query(models.ResourceProvider, session=session)
query = add_identity_filter(query, provider_id)
count = query.delete()
if count != 1:
raise exception.ResourceProviderNotFound(
resource_provider=provider_id)
def update_resource_provider(self, context, provider_id, values):
if 'uuid' in values:
msg = _("Cannot overwrite UUID for an existing ResourceProvider.")
raise exception.InvalidParameterValue(err=msg)
return self._do_update_resource_provider(provider_id, values)
def _do_update_resource_provider(self, provider_id, values):
session = get_session()
with session.begin():
query = model_query(models.ResourceProvider, session=session)
query = add_identity_filter(query, provider_id)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.ResourceProviderNotFound(
resource_provider=provider_id)
ref.update(values)
return ref
def list_resource_classes(self, context, limit=None, marker=None,
sort_key=None, sort_dir=None):
query = model_query(models.ResourceClass)
return _paginate_query(models.ResourceClass, limit, marker,
sort_key, sort_dir, query)
def create_resource_class(self, context, values):
resource = models.ResourceClass()
resource.update(values)
try:
resource.save()
except db_exc.DBDuplicateEntry:
raise exception.ResourceClassAlreadyExists(
field='uuid', value=values['uuid'])
return resource
def get_resource_class(self, context, resource_ident):
if uuidutils.is_uuid_like(resource_ident):
return self._get_resource_class_by_uuid(context, resource_ident)
else:
return self._get_resource_class_by_name(context, resource_ident)
def _get_resource_class_by_uuid(self, context, resource_uuid):
query = model_query(models.ResourceClass)
query = query.filter_by(uuid=resource_uuid)
try:
return query.one()
except NoResultFound:
raise exception.ResourceClassNotFound(
resource_class=resource_uuid)
def _get_resource_class_by_name(self, context, resource_name):
query = model_query(models.ResourceClass)
query = query.filter_by(name=resource_name)
try:
return query.one()
except NoResultFound:
raise exception.ResourceClassNotFound(resource_class=resource_name)
def destroy_resource_class(self, context, resource_id):
session = get_session()
with session.begin():
query = model_query(models.ResourceClass, session=session)
count = query.delete()
if count != 1:
raise exception.ResourceClassNotFound(
resource_class=str(resource_id))
def update_resource_class(self, context, resource_id, values):
session = get_session()
with session.begin():
query = model_query(models.ResourceClass, session=session)
query = query.filter_by(id=resource_id)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.ResourceClassNotFound(
resource_class=resource_id)
ref.update(values)
return ref
def _add_inventories_filters(self, query, filters):
filter_names = ['resource_provider_id', 'resource_class_id', 'total',
'reserved', 'min_unit', 'max_unit', 'step_size',
'allocation_ratio', 'is_nested']
return self._add_filters(query, models.Inventory, filters=filters,
filter_names=filter_names)
def list_inventories(self, context, filters=None, limit=None,
marker=None, sort_key=None, sort_dir=None):
session = get_session()
query = model_query(models.Inventory, session=session)
query = self._add_inventories_filters(query, filters)
query = query.join(models.Inventory.resource_provider)
query = query.options(contains_eager('resource_provider'))
return _paginate_query(models.Inventory, limit, marker,
sort_key, sort_dir, query)
def create_inventory(self, context, provider_id, values):
values['resource_provider_id'] = provider_id
inventory = models.Inventory()
inventory.update(values)
try:
inventory.save()
except db_exc.DBDuplicateEntry as e:
fields = {c: values[c] for c in e.columns}
raise exception.UniqueConstraintViolated(fields=fields)
return inventory
def get_inventory(self, context, inventory_id):
session = get_session()
query = model_query(models.Inventory, session=session)
query = query.join(models.Inventory.resource_provider)
query = query.options(contains_eager('resource_provider'))
query = query.filter_by(id=inventory_id)
try:
return query.one()
except NoResultFound:
raise exception.InventoryNotFound(inventory=inventory_id)
def destroy_inventory(self, context, inventory_id):
session = get_session()
with session.begin():
query = model_query(models.Inventory, session=session)
query = query.filter_by(id=inventory_id)
count = query.delete()
if count != 1:
raise exception.InventoryNotFound(inventory=inventory_id)
def update_inventory(self, context, inventory_id, values):
session = get_session()
with session.begin():
query = model_query(models.Inventory, session=session)
query = query.filter_by(id=inventory_id)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.InventoryNotFound(inventory=inventory_id)
ref.update(values)
return ref
def _add_allocations_filters(self, query, filters):
filter_names = ['resource_provider_id', 'resource_class_id',
'consumer_id', 'used', 'is_nested']
return self._add_filters(query, models.Allocation, filters=filters,
filter_names=filter_names)
def list_allocations(self, context, filters=None, limit=None,
marker=None, sort_key=None, sort_dir=None):
session = get_session()
query = model_query(models.Allocation, session=session)
query = self._add_allocations_filters(query, filters)
query = query.join(models.Allocation.resource_provider)
query = query.options(contains_eager('resource_provider'))
return _paginate_query(models.Allocation, limit, marker,
sort_key, sort_dir, query)
def create_allocation(self, context, values):
allocation = models.Allocation()
allocation.update(values)
try:
allocation.save()
except db_exc.DBDuplicateEntry as e:
fields = {c: values[c] for c in e.columns}
raise exception.UniqueConstraintViolated(fields=fields)
return allocation
def get_allocation(self, context, allocation_id):
session = get_session()
query = model_query(models.Allocation, session=session)
query = query.join(models.Allocation.resource_provider)
query = query.options(contains_eager('resource_provider'))
query = query.filter_by(id=allocation_id)
try:
return query.one()
except NoResultFound:
raise exception.AllocationNotFound(allocation=allocation_id)
def destroy_allocation(self, context, allocation_id):
session = get_session()
with session.begin():
query = model_query(models.Allocation, session=session)
query = query.filter_by(id=allocation_id)
count = query.delete()
if count != 1:
raise exception.AllocationNotFound(allocation=allocation_id)
def update_allocation(self, context, allocation_id, values):
session = get_session()
with session.begin():
query = model_query(models.Allocation, session=session)
query = query.filter_by(id=allocation_id)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.AllocationNotFound(allocation=allocation_id)
ref.update(values)
return ref
def _add_compute_nodes_filters(self, query, filters):
filter_names = ['hostname']
return self._add_filters(query, models.ComputeNode, filters=filters,
filter_names=filter_names)
def list_compute_nodes(self, context, filters=None, limit=None,
marker=None, sort_key=None, sort_dir=None):
query = model_query(models.ComputeNode)
query = self._add_compute_nodes_filters(query, filters)
return _paginate_query(models.ComputeNode, limit, marker,
sort_key, sort_dir, query,
default_sort_key='uuid')
def create_compute_node(self, context, values):
# ensure defaults are present for new compute nodes
if not values.get('uuid'):
values['uuid'] = uuidutils.generate_uuid()
compute_node = models.ComputeNode()
compute_node.update(values)
try:
compute_node.save()
except db_exc.DBDuplicateEntry:
raise exception.ComputeNodeAlreadyExists(
field='UUID', value=values['uuid'])
return compute_node
def get_compute_node(self, context, node_uuid):
query = model_query(models.ComputeNode)
query = query.filter_by(uuid=node_uuid)
try:
return query.one()
except NoResultFound:
raise exception.ComputeNodeNotFound(
compute_node=node_uuid)
def get_compute_node_by_hostname(self, context, hostname):
query = model_query(models.ComputeNode)
query = query.filter_by(hostname=hostname)
try:
return query.one()
except NoResultFound:
raise exception.ComputeNodeNotFound(
compute_node=hostname)
except MultipleResultsFound:
raise exception.Conflict('Multiple compute nodes exist with same '
'hostname. Please use the uuid instead.')
def destroy_compute_node(self, context, node_uuid):
session = get_session()
with session.begin():
query = model_query(models.ComputeNode, session=session)
query = query.filter_by(uuid=node_uuid)
count = query.delete()
if count != 1:
raise exception.ComputeNodeNotFound(
compute_node=node_uuid)
def update_compute_node(self, context, node_uuid, values):
if 'uuid' in values:
msg = _("Cannot overwrite UUID for an existing ComputeNode.")
raise exception.InvalidParameterValue(err=msg)
return self._do_update_compute_node(node_uuid, values)
def _do_update_compute_node(self, node_uuid, values):
session = get_session()
with session.begin():
query = model_query(models.ComputeNode, session=session)
query = query.filter_by(uuid=node_uuid)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.ComputeNodeNotFound(
compute_node=node_uuid)
ref.update(values)
return ref
def list_capsules(self, context, filters=None, limit=None,
marker=None, sort_key=None, sort_dir=None):
query = model_query(models.Capsule)
query = self._add_project_filters(context, query)
query = self._add_capsules_filters(query, filters)
return _paginate_query(models.Capsule, limit, marker,
sort_key, sort_dir, query)
def create_capsule(self, context, values):
# ensure defaults are present for new capsules
# here use the infra container uuid as the capsule uuid
if not values.get('uuid'):
values['uuid'] = uuidutils.generate_uuid()
capsule = models.Capsule()
capsule.update(values)
try:
capsule.save()
except db_exc.DBDuplicateEntry:
raise exception.CapsuleAlreadyExists(field='UUID',
value=values['uuid'])
return capsule
def get_capsule_by_uuid(self, context, capsule_uuid):
query = model_query(models.Capsule)
query = self._add_project_filters(context, query)
query = query.filter_by(uuid=capsule_uuid)
try:
return query.one()
except NoResultFound:
raise exception.CapsuleNotFound(capsule=capsule_uuid)
def get_capsule_by_meta_name(self, context, capsule_name):
query = model_query(models.Capsule)
query = self._add_project_filters(context, query)
query = query.filter_by(meta_name=capsule_name)
try:
return query.one()
except NoResultFound:
raise exception.CapsuleNotFound(capsule=capsule_name)
except MultipleResultsFound:
raise exception.Conflict('Multiple capsules exist with same '
'name. Please use the capsule uuid '
'instead.')
def destroy_capsule(self, context, capsule_id):
session = get_session()
with session.begin():
query = model_query(models.Capsule, session=session)
query = add_identity_filter(query, capsule_id)
count = query.delete()
if count != 1:
raise exception.CapsuleNotFound(capsule_id)
def update_capsule(self, context, capsule_id, values):
if 'uuid' in values:
msg = _("Cannot overwrite UUID for an existing Capsule.")
raise exception.InvalidParameterValue(err=msg)
return self._do_update_capsule_id(capsule_id, values)
def _do_update_capsule_id(self, capsule_id, values):
session = get_session()
with session.begin():
query = model_query(models.Capsule, session=session)
query = add_identity_filter(query, capsule_id)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.CapsuleNotFound(capsule=capsule_id)
ref.update(values)
return ref
def _add_capsules_filters(self, query, filters):
# filter_names = ['uuid', 'project_id', 'user_id', 'containers']
filter_names = ['uuid', 'project_id', 'user_id']
return self._add_filters(query, models.Capsule, filters=filters,
filter_names=filter_names)
def get_pci_device_by_addr(self, node_id, dev_addr):
pci_dev_ref = model_query(models.PciDevice).\
filter_by(compute_node_uuid=node_id).\
filter_by(address=dev_addr).\
first()
if not pci_dev_ref:
raise exception.PciDeviceNotFound(node_id=node_id,
address=dev_addr)
return pci_dev_ref
def get_pci_device_by_id(self, id):
pci_dev_ref = model_query(models.PciDevice).\
filter_by(id=id).\
first()
if not pci_dev_ref:
raise exception.PciDeviceNotFoundById(id=id)
return pci_dev_ref
def get_all_pci_device_by_node(self, node_id):
return model_query(models.PciDevice).\
filter_by(compute_node_uuid=node_id).\
all()
def get_all_pci_device_by_parent_addr(self, node_id, parent_addr):
return model_query(models.PciDevice).\
filter_by(compute_node_uuid=node_id).\
filter_by(parent_addr=parent_addr).\
all()
def get_all_pci_device_by_container_uuid(self, container_uuid):
return model_query(models.PciDevice).\
filter_by(status=consts.ALLOCATED).\
filter_by(container_uuid=container_uuid).\
all()
def destroy_pci_device(self, node_id, address):
session = get_session()
with session.begin():
query = model_query(models.PciDevice).\
filter_by(compute_node_uuid=node_id).\
filter_by(address=address)
count = query.delete()
if count != 1:
raise exception.PciDeviceNotFound(node_id=node_id,
address=address)
def update_pci_device(self, node_id, address, values):
query = model_query(models.PciDevice).\
filter_by(compute_node_uuid=node_id).\
filter_by(address=address)
if query.update(values) == 0:
device = models.PciDevice()
device.update(values)
device.save()
return query.one()
def action_start(self, context, values):
action = models.ContainerAction()
action.update(values)
action.save()
return action
def actions_get(self, context, container_uuid):
"""Get all container actions for the provided uuid."""
query = model_query(models.ContainerAction).\
filter_by(container_uuid=container_uuid)
actions = _paginate_query(models.ContainerAction, sort_dir='desc',
sort_key='created_at', query=query)
return actions
def action_get_by_request_id(self, context, container_uuid, request_id):
"""Get the action by request_id and given container."""
action = self._action_get_by_request_id(context, container_uuid,
request_id)
return action
def _action_get_by_request_id(self, context, container_uuid, request_id):
result = model_query(models.ContainerAction).\
filter_by(container_uuid=container_uuid).\
filter_by(request_id=request_id).\
first()
return result
def _action_get_last_created_by_container_uuid(self, context,
container_uuid):
result = model_query(models.ContainerAction).\
filter_by(container_uuid=container_uuid).\
order_by(desc("created_at"), desc("id")).\
first()
return result
def action_event_start(self, context, values):
"""Start an event on a container action."""
action = self._action_get_by_request_id(context,
values['container_uuid'],
values['request_id'])
# When zun-compute restarts, the request_id was different with
# request_id recorded in ContainerAction, so we can't get the original
# recode according to request_id. Try to get the last created action
# so that init_container can continue to finish the recovery action.
if not action and not context.project_id:
action = self._action_get_last_created_by_container_uuid(
context, values['container_uuid'])
if not action:
raise exception.ContainerActionNotFound(
request_id=values['request_id'],
container_uuid=values['container_uuid'])
values['action_id'] = action['id']
event = models.ContainerActionEvent()
event.update(values)
event.save()
return event
def action_event_finish(self, context, values):
"""Finish an event on a container action."""
action = self._action_get_by_request_id(context,
values['container_uuid'],
values['request_id'])
# When zun-compute restarts, the request_id was different with
# request_id recorded in ContainerAction, so we can't get the original
# recode according to request_id. Try to get the last created action
# so that init_container can continue to finish the recovery action.
if not action and not context.project_id:
action = self._action_get_last_created_by_container_uuid(
context, values['container_uuid'])
if not action:
raise exception.ContainerActionNotFound(
request_id=values['request_id'],
container_uuid=values['container_uuid'])
event = model_query(models.ContainerActionEvent).\
filter_by(action_id=action['id']).\
filter_by(event=values['event']).\
first()
if not event:
raise exception.ContainerActionEventNotFound(
action_id=action['id'], event=values['event'])
event.update(values)
event.save()
if values['result'].lower() == 'error':
action.update({'message': 'Error'})
action.save()
return event
def action_events_get(self, context, action_id):
query = model_query(models.ContainerActionEvent).\
filter_by(action_id=action_id)
events = _paginate_query(models.ContainerActionEvent, sort_dir='desc',
sort_key='created_at', query=query)
return events
def quota_create(self, context, project_id, resource, limit):
quota_ref = models.Quota()
quota_ref.project_id = project_id
quota_ref.resource = resource
quota_ref.hard_limit = limit
session = get_session()
try:
quota_ref.save(session=session)
except db_exc.DBDuplicateEntry:
raise exception.QuotaAlreadyExists(project_id=project_id,
resource=resource)
return quota_ref
def quota_get(self, context, project_id, resource):
session = get_session()
with session.begin():
query = model_query(models.Quota, session=session).\
filter_by(project_id=project_id).\
filter_by(resource=resource)
result = query.first()
if not result:
raise exception.ProjectQuotaNotFound(project_id=project_id)
return result
def quota_get_all_by_project(self, context, project_id):
session = get_session()
with session.begin():
rows = model_query(models.Quota, session=session).\
filter_by(project_id=project_id).\
all()
result = {'project_id': project_id}
for row in rows:
result[row.resource] = row.hard_limit
return result
def quota_update(self, context, project_id, resource, limit):
session = get_session()
with session.begin():
query = model_query(models.Quota, session=session).\
filter_by(project_id=project_id).\
filter_by(resource=resource)
result = query.update({'hard_limit': limit})
if not result:
raise exception.ProjectQuotaNotFound(project_id=project_id)
def quota_destroy(self, context, project_id, resource):
session = get_session()
with session.begin():
query = model_query(models.Quota, session=session).\
filter_by(project_id=project_id).\
filter_by(resource=resource)
query.delete()
def quota_destroy_all_by_project(self, context, project_id):
session = get_session()
with session.begin():
model_query(models.Quota, session=session).\
filter_by(project_id=project_id).\
delete()
model_query(models.QuotaUsage, session=session).\
filter_by(project_id=project_id).\
delete()
def quota_class_create(self, context, class_name, resource, limit):
quota_class_ref = models.QuotaClass()
quota_class_ref.class_name = class_name
quota_class_ref.resource = resource
quota_class_ref.hard_limit = limit
session = get_session()
with session.begin():
quota_class_ref.save(session=session)
return quota_class_ref
def quota_class_get(self, context, class_name, resource):
session = get_session()
with session.begin():
result = model_query(models.QuotaClass, session=session).\
filter_by(class_name=class_name).\
filter_by(resource=resource).\
first()
if not result:
raise exception.QuotaClassNotFound(class_name=class_name)
return result
def quota_class_get_default(self, context):
session = get_session()
with session.begin():
rows = model_query(models.QuotaClass, session=session).\
filter_by(class_name=consts.DEFAULT_QUOTA_CLASS_NAME).\
all()
result = {'class_name': consts.DEFAULT_QUOTA_CLASS_NAME}
for row in rows:
result[row.resource] = row.hard_limit
return result
def quota_class_get_all_by_name(self, context, class_name):
session = get_session()
with session.begin():
rows = model_query(models.QuotaClass, session=session).\
filter_by(class_name=class_name).\
all()
result = {'class_name': class_name}
for row in rows:
result[row.resource] = row.hard_limit
return result
def quota_class_update(self, context, class_name, resource, limit):
session = get_session()
with session.begin():
result = model_query(models.QuotaClass, session=session).\
filter_by(class_name=class_name).\
filter_by(resource=resource).\
update({'hard_limit': limit})
if not result:
raise exception.QuotaClassNotFound(class_name=class_name)
def quota_usage_get_all_by_project(self, context, project_id):
rows = model_query(models.QuotaUsage).\
filter_by(project_id=project_id).\
all()
result = {'project_id': project_id}
for row in rows:
result[row.resource] = dict(in_use=row.in_use,
reserved=row.reserved)
return result
def _add_networks_filters(self, query, filters):
filter_names = ['name', 'neutron_net_id', 'project_id', 'user_id']
return self._add_filters(query, models.Network, filters=filters,
filter_names=filter_names)
def list_networks(self, context, filters=None, limit=None,
marker=None, sort_key=None, sort_dir=None):
query = model_query(models.Network)
query = self._add_project_filters(context, query)
query = self._add_networks_filters(query, filters)
return _paginate_query(models.Network, limit, marker,
sort_key, sort_dir, query)
def create_network(self, context, values):
# ensure defaults are present for new networks
if not values.get('uuid'):
values['uuid'] = uuidutils.generate_uuid()
network = models.Network()
network.update(values)
try:
network.save()
except db_exc.DBDuplicateEntry as e:
if 'neutron_net_id' in e.columns:
raise exception.NetworkAlreadyExists(
field='neutron_net_id', value=values['neutron_net_id'])
else:
raise exception.NetworkAlreadyExists(
field='UUID', value=values['uuid'])
return network
def update_network(self, context, network_uuid, values):
# NOTE(dtantsur): this can lead to very strange errors
if 'uuid' in values:
msg = _("Cannot overwrite UUID for an existing docker network.")
raise exception.InvalidParameterValue(err=msg)
return self._do_update_network(network_uuid, values)
def _do_update_network(self, network_uuid, values):
session = get_session()
with session.begin():
query = model_query(models.Network, session=session)
query = add_identity_filter(query, network_uuid)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.NetworkNotFound(network=network_uuid)
ref.update(values)
return ref
def get_network_by_uuid(self, context, network_uuid):
query = model_query(models.Network)
query = self._add_project_filters(context, query)
query = query.filter_by(uuid=network_uuid)
try:
return query.one()
except NoResultFound:
raise exception.NetworkNotFound(network=network_uuid)
def destroy_network(self, context, network_uuid):
session = get_session()
with session.begin():
query = model_query(models.Network, session=session)
query = add_identity_filter(query, network_uuid)
count = query.delete()
if count != 1:
raise exception.NetworkNotFound(network=network_uuid)
def list_exec_instances(self, context, filters=None, limit=None,
marker=None, sort_key=None, sort_dir=None):
query = model_query(models.ExecInstance)
query = self._add_exec_instances_filters(query, filters)
return _paginate_query(models.ExecInstance, limit, marker,
sort_key, sort_dir, query)
def _add_exec_instances_filters(self, query, filters):
filter_names = ['container_id', 'exec_id', 'token']
return self._add_filters(query, models.ExecInstance, filters=filters,
filter_names=filter_names)
def create_exec_instance(self, context, values):
exec_inst = models.ExecInstance()
exec_inst.update(values)
try:
exec_inst.save()
except db_exc.DBDuplicateEntry:
raise exception.ExecInstanceAlreadyExists(
exec_id=values['exec_id'])
return exec_inst
def count_usage(self, context, project_id, flag):
session = get_session()
if flag == 'containers':
project_query = session.query(
func.count(models.Container.id)). \
filter_by(project_id=project_id)
elif flag in ['disk', 'cpu', 'memory']:
project_query = session.query(
func.sum(getattr(models.Container, flag))). \
filter_by(project_id=project_id)
return project_query.first()
def _add_registries_filters(self, query, filters):
filter_names = ['name', 'domain', 'username', 'project_id', 'user_id']
return self._add_filters(query, models.Registry, filters=filters,
filter_names=filter_names)
def list_registries(self, context, filters=None, limit=None,
marker=None, sort_key=None, sort_dir=None):
query = model_query(models.Registry)
query = self._add_project_filters(context, query)
query = self._add_registries_filters(query, filters)
return _paginate_query(models.Registry, limit, marker,
sort_key, sort_dir, query)
def create_registry(self, context, values):
# ensure defaults are present for new registries
if not values.get('uuid'):
values['uuid'] = uuidutils.generate_uuid()
registry = models.Registry()
registry.update(values)
try:
registry.save()
except db_exc.DBDuplicateEntry:
raise exception.RegistryAlreadyExists(
field='UUID', value=values['uuid'])
return registry
def update_registry(self, context, registry_uuid, values):
# NOTE(dtantsur): this can lead to very strange errors
if 'uuid' in values:
msg = _("Cannot overwrite UUID for an existing registry.")
raise exception.InvalidParameterValue(err=msg)
return self._do_update_registry(registry_uuid, values)
def _do_update_registry(self, registry_uuid, values):
session = get_session()
with session.begin():
query = model_query(models.Registry, session=session)
query = add_identity_filter(query, registry_uuid)
try:
ref = query.with_lockmode('update').one()
except NoResultFound:
raise exception.RegistryNotFound(registry=registry_uuid)
ref.update(values)
return ref
def get_registry_by_uuid(self, context, registry_uuid):
query = model_query(models.Registry)
query = self._add_project_filters(context, query)
query = query.filter_by(uuid=registry_uuid)
try:
return query.one()
except NoResultFound:
raise exception.RegistryNotFound(registry=registry_uuid)
def get_registry_by_name(self, context, registry_name):
query = model_query(models.Registry)
query = self._add_project_filters(context, query)
query = query.filter_by(name=registry_name)
try:
return query.one()
except NoResultFound:
raise exception.RegistryNotFound(registry=registry_name)
except MultipleResultsFound:
raise exception.Conflict('Multiple registries exist with same '
'name. Please use the registry uuid '
'instead.')
def destroy_registry(self, context, registry_uuid):
session = get_session()
with session.begin():
query = model_query(models.Registry, session=session)
query = add_identity_filter(query, registry_uuid)
count = query.delete()
if count != 1:
raise exception.RegistryNotFound(registry=registry_uuid)