mypy: cmd/manage.py

Change-Id: Iadc48face76e3ab930d76821d0e15ad76178fd71
This commit is contained in:
Eric Harney 2022-05-19 09:08:56 -04:00
parent b6fdb10f35
commit 64489722d5
2 changed files with 84 additions and 39 deletions

View File

@ -50,6 +50,8 @@
"""CLI interface for cinder management.""" """CLI interface for cinder management."""
from __future__ import annotations
import collections import collections
import collections.abc as collections_abc import collections.abc as collections_abc
import errno import errno
@ -60,6 +62,8 @@ import os
import re import re
import sys import sys
import time import time
import typing
from typing import Any, Callable, Optional, Tuple, Union # noqa: H301
from oslo_config import cfg from oslo_config import cfg
from oslo_db import exception as db_exc from oslo_db import exception as db_exc
@ -101,6 +105,7 @@ OVO_VERSION = ovo_base.OBJ_VERSIONS.get_current()
# Decorators for actions # Decorators for actions
@typing.no_type_check
def args(*args, **kwargs): def args(*args, **kwargs):
args = list(args) args = list(args)
if not args[0].startswith('-') and '-' in args[0]: if not args[0].startswith('-') and '-' in args[0]:
@ -118,7 +123,7 @@ class HostCommands(object):
@args('zone', nargs='?', default=None, @args('zone', nargs='?', default=None,
help='Availability Zone (default: %(default)s)') help='Availability Zone (default: %(default)s)')
def list(self, zone=None): def list(self, zone: Optional[str] = None) -> None:
"""Show a list of all physical hosts. """Show a list of all physical hosts.
Can be filtered by zone. Can be filtered by zone.
@ -129,7 +134,7 @@ class HostCommands(object):
services = objects.ServiceList.get_all(ctxt) services = objects.ServiceList.get_all(ctxt)
if zone: if zone:
services = [s for s in services if s.availability_zone == zone] services = [s for s in services if s.availability_zone == zone]
hosts = [] hosts: list[dict[str, Any]] = []
for srv in services: for srv in services:
if not [h for h in hosts if h['host'] == srv['host']]: if not [h for h in hosts if h['host'] == srv['host']]:
hosts.append(srv) hosts.append(srv)
@ -152,7 +157,8 @@ class DbCommands(object):
# preceed any element of the "online_migrations" tuple, like this: # preceed any element of the "online_migrations" tuple, like this:
# # Added in Queens remove in Rocky # # Added in Queens remove in Rocky
# db.service_uuids_online_data_migration, # db.service_uuids_online_data_migration,
online_migrations = ( online_migrations: Tuple[Callable[[context.RequestContext, int],
Tuple[int, int]], ...] = (
# TODO: (Z Release) Remove next line and this comment # TODO: (Z Release) Remove next line and this comment
# TODO: (Y Release) Uncomment next line and remove this comment # TODO: (Y Release) Uncomment next line and remove this comment
# db.remove_temporary_admin_metadata_data_migration, # db.remove_temporary_admin_metadata_data_migration,
@ -172,7 +178,9 @@ class DbCommands(object):
help='Update RPC and Objects versions when doing offline upgrades, ' help='Update RPC and Objects versions when doing offline upgrades, '
'with this we no longer need to restart the services twice ' 'with this we no longer need to restart the services twice '
'after the upgrade to prevent ServiceTooOld exceptions.') 'after the upgrade to prevent ServiceTooOld exceptions.')
def sync(self, version=None, bump_versions=False): def sync(self,
version: Optional[int] = None,
bump_versions: bool = False) -> None:
"""Sync the database up to the most recent version.""" """Sync the database up to the most recent version."""
if version is not None and version > db.MAX_INT: if version is not None and version > db.MAX_INT:
print(_('Version should be less than or equal to ' print(_('Version should be less than or equal to '
@ -199,13 +207,13 @@ class DbCommands(object):
print(_('Error during service version bump: %s') % ex) print(_('Error during service version bump: %s') % ex)
sys.exit(2) sys.exit(2)
def version(self): def version(self) -> None:
"""Print the current database version.""" """Print the current database version."""
print(db_migration.db_version()) print(db_migration.db_version())
@args('age_in_days', type=int, @args('age_in_days', type=int,
help='Purge deleted rows older than age in days') help='Purge deleted rows older than age in days')
def purge(self, age_in_days): def purge(self, age_in_days: int) -> None:
"""Purge deleted rows older than a given age from cinder tables.""" """Purge deleted rows older than a given age from cinder tables."""
age_in_days = int(age_in_days) age_in_days = int(age_in_days)
if age_in_days < 0: if age_in_days < 0:
@ -223,7 +231,9 @@ class DbCommands(object):
"logs for more details.")) "logs for more details."))
sys.exit(1) sys.exit(1)
def _run_migration(self, ctxt, max_count): def _run_migration(self,
ctxt: context.RequestContext,
max_count: int) -> Tuple[dict, bool]:
ran = 0 ran = 0
exceptions = False exceptions = False
migrations = {} migrations = {}
@ -254,7 +264,7 @@ class DbCommands(object):
@args('--max_count', metavar='<number>', dest='max_count', type=int, @args('--max_count', metavar='<number>', dest='max_count', type=int,
help='Maximum number of objects to consider.') help='Maximum number of objects to consider.')
def online_data_migrations(self, max_count=None): def online_data_migrations(self, max_count: Optional[int] = None) -> None:
"""Perform online data migrations for the release in batches.""" """Perform online data migrations for the release in batches."""
ctxt = context.get_admin_context() ctxt = context.get_admin_context()
if max_count is not None: if max_count is not None:
@ -269,7 +279,7 @@ class DbCommands(object):
ran = None ran = None
exceptions = False exceptions = False
migration_info = {} migration_info: dict[str, Any] = {}
while ran is None or ran != 0: while ran is None or ran != 0:
migrations, exceptions = self._run_migration(ctxt, max_count) migrations, exceptions = self._run_migration(ctxt, max_count)
ran = 0 ran = 0
@ -311,8 +321,10 @@ class DbCommands(object):
help='Change the active backend ID (default: %(default)s).') help='Change the active backend ID (default: %(default)s).')
@args('--backend-host', required=True, @args('--backend-host', required=True,
help='The backend host name.') help='The backend host name.')
def reset_active_backend(self, enable_replication, active_backend_id, def reset_active_backend(self,
backend_host): enable_replication: bool,
active_backend_id: Optional[str],
backend_host: str) -> None:
"""Reset the active backend for a host.""" """Reset the active backend for a host."""
ctxt = context.get_admin_context() ctxt = context.get_admin_context()
@ -336,7 +348,7 @@ class QuotaCommands(object):
@args('--project-id', default=None, @args('--project-id', default=None,
help=('The ID of the project where we want to sync the quotas ' help=('The ID of the project where we want to sync the quotas '
'(defaults to all projects).')) '(defaults to all projects).'))
def check(self, project_id): def check(self, project_id: Optional[str]) -> None:
"""Check if quotas and reservations are correct """Check if quotas and reservations are correct
This action checks quotas and reservations, for a specific project or This action checks quotas and reservations, for a specific project or
@ -356,7 +368,7 @@ class QuotaCommands(object):
@args('--project-id', default=None, @args('--project-id', default=None,
help=('The ID of the project where we want to sync the quotas ' help=('The ID of the project where we want to sync the quotas '
'(defaults to all projects).')) '(defaults to all projects).'))
def sync(self, project_id): def sync(self, project_id: Optional[str]) -> None:
"""Fix quotas and reservations """Fix quotas and reservations
This action refreshes existing quota usage and reservation count for a This action refreshes existing quota usage and reservation count for a
@ -373,7 +385,9 @@ class QuotaCommands(object):
self._check_sync(project_id, do_fix=True) self._check_sync(project_id, do_fix=True)
@db_api.main_context_manager.reader @db_api.main_context_manager.reader
def _get_quota_projects(self, context, project_id): def _get_quota_projects(self,
context: context.RequestContext,
project_id: Optional[str]) -> list[str]:
"""Get project ids that have quota_usage entries.""" """Get project ids that have quota_usage entries."""
if project_id: if project_id:
model = models.QuotaUsage model = models.QuotaUsage
@ -402,7 +416,10 @@ class QuotaCommands(object):
project_ids = [row.project_id for row in projects] project_ids = [row.project_id for row in projects]
return project_ids return project_ids
def _get_usages(self, context, resources, project_id): def _get_usages(self,
ctxt: context.RequestContext,
resources,
project_id: str) -> list:
"""Get data necessary to check out of sync quota usage. """Get data necessary to check out of sync quota usage.
Returns a list QuotaUsage instances for the specific project Returns a list QuotaUsage instances for the specific project
@ -414,7 +431,10 @@ class QuotaCommands(object):
).filter_by(project_id=project_id).with_for_update().all() ).filter_by(project_id=project_id).with_for_update().all()
return usages return usages
def _get_reservations(self, context, project_id, usage_id): def _get_reservations(self,
ctxt: context.RequestContext,
project_id: str,
usage_id: str) -> list:
"""Get reservations for a given project and usage id.""" """Get reservations for a given project and usage id."""
reservations = ( reservations = (
db_api.model_query( db_api.model_query(
@ -428,7 +448,10 @@ class QuotaCommands(object):
) )
return reservations return reservations
def _check_duplicates(self, context, usages, do_fix): def _check_duplicates(self,
context: context.RequestContext,
usages,
do_fix: bool) -> tuple[list, bool]:
"""Look for duplicated quota used entries (bug#1484343) """Look for duplicated quota used entries (bug#1484343)
If we have duplicates and we are fixing them, then we reassign the If we have duplicates and we are fixing them, then we reassign the
@ -470,7 +493,7 @@ class QuotaCommands(object):
result.append(keep_usage) result.append(keep_usage)
return result, duplicates_found return result, duplicates_found
def _check_sync(self, project_id, do_fix): def _check_sync(self, project_id: Optional[str], do_fix: bool) -> bool:
"""Check the quotas and reservations optionally fixing them.""" """Check the quotas and reservations optionally fixing them."""
ctxt = context.get_admin_context() ctxt = context.get_admin_context()
@ -496,7 +519,11 @@ class QuotaCommands(object):
return discrepancy return discrepancy
@db_api.main_context_manager.reader @db_api.main_context_manager.reader
def _check_project_sync(self, context, project, do_fix, resources): def _check_project_sync(self,
context: context.RequestContext,
project: str,
do_fix: bool,
resources) -> bool:
print('Processing quota usage for project %s' % project) print('Processing quota usage for project %s' % project)
discrepancy = False discrepancy = False
@ -579,7 +606,7 @@ class VolumeCommands(object):
@args('volume_id', @args('volume_id',
help='Volume ID to be deleted') help='Volume ID to be deleted')
def delete(self, volume_id): def delete(self, volume_id: str) -> None:
"""Delete a volume, bypassing the check that it must be available.""" """Delete a volume, bypassing the check that it must be available."""
ctxt = context.get_admin_context() ctxt = context.get_admin_context()
volume = objects.Volume.get_by_id(ctxt, volume_id) volume = objects.Volume.get_by_id(ctxt, volume_id)
@ -604,7 +631,7 @@ class VolumeCommands(object):
'the format host@backend#pool') 'the format host@backend#pool')
@args('--newhost', required=True, help='New volume host name in the ' @args('--newhost', required=True, help='New volume host name in the '
'format host@backend#pool') 'format host@backend#pool')
def update_host(self, currenthost, newhost): def update_host(self, currenthost: str, newhost: str) -> None:
"""Modify the host name associated with a volume. """Modify the host name associated with a volume.
Particularly to recover from cases where one has moved Particularly to recover from cases where one has moved
@ -627,7 +654,7 @@ class ConfigCommands(object):
@args('param', nargs='?', default=None, @args('param', nargs='?', default=None,
help='Configuration parameter to display (default: %(default)s)') help='Configuration parameter to display (default: %(default)s)')
def list(self, param=None): def list(self, param: Optional[str] = None) -> None:
"""List parameters configured for cinder. """List parameters configured for cinder.
Lists all parameters configured for cinder unless an optional argument Lists all parameters configured for cinder unless an optional argument
@ -646,7 +673,7 @@ class ConfigCommands(object):
class BackupCommands(object): class BackupCommands(object):
"""Methods for managing backups.""" """Methods for managing backups."""
def list(self): def list(self) -> None:
"""List all backups. """List all backups.
List all backups (including ones in progress) and the host List all backups (including ones in progress) and the host
@ -683,7 +710,7 @@ class BackupCommands(object):
@args('--currenthost', required=True, help='Existing backup host name') @args('--currenthost', required=True, help='Existing backup host name')
@args('--newhost', required=True, help='New backup host name') @args('--newhost', required=True, help='New backup host name')
def update_backup_host(self, currenthost, newhost): def update_backup_host(self, currenthost: str, newhost: str) -> None:
"""Modify the host name associated with a backup. """Modify the host name associated with a backup.
Particularly to recover from cases where one has moved Particularly to recover from cases where one has moved
@ -738,7 +765,7 @@ class ServiceCommands(BaseCommand):
help='Service to delete from the host.') help='Service to delete from the host.')
@args('host_name', type=str, @args('host_name', type=str,
help='Host from which to remove the service.') help='Host from which to remove the service.')
def remove(self, binary, host_name): def remove(self, binary: str, host_name: str) -> Optional[int]:
"""Completely removes a service.""" """Completely removes a service."""
ctxt = context.get_admin_context() ctxt = context.get_admin_context()
try: try:
@ -753,10 +780,12 @@ class ServiceCommands(BaseCommand):
print(_("Service %(service)s on host %(host)s removed.") % print(_("Service %(service)s on host %(host)s removed.") %
{'service': binary, 'host': host_name}) {'service': binary, 'host': host_name})
return None
class ClusterCommands(BaseCommand): class ClusterCommands(BaseCommand):
"""Methods for managing clusters.""" """Methods for managing clusters."""
def list(self): def list(self) -> None:
"""Show a list of all cinder services.""" """Show a list of all cinder services."""
ctxt = context.get_admin_context() ctxt = context.get_admin_context()
clusters = objects.ClusterList.get_all(ctxt, services_summary=True) clusters = objects.ClusterList.get_all(ctxt, services_summary=True)
@ -783,7 +812,10 @@ class ClusterCommands(BaseCommand):
@args('binary', type=str, @args('binary', type=str,
help='Service to delete from the cluster.') help='Service to delete from the cluster.')
@args('cluster-name', type=str, help='Cluster to delete.') @args('cluster-name', type=str, help='Cluster to delete.')
def remove(self, recursive, binary, cluster_name): def remove(self,
recursive: bool,
binary: str,
cluster_name: str) -> Optional[int]:
"""Completely removes a cluster.""" """Completely removes a cluster."""
ctxt = context.get_admin_context() ctxt = context.get_admin_context()
try: try:
@ -812,13 +844,18 @@ class ClusterCommands(BaseCommand):
'removed.') % {'msg': msg, 'num': len(cluster.services)}) 'removed.') % {'msg': msg, 'num': len(cluster.services)})
print(msg) print(msg)
return None
@args('--full-rename', dest='partial', @args('--full-rename', dest='partial',
action='store_false', default=True, action='store_false', default=True,
help='Do full cluster rename instead of just replacing provided ' help='Do full cluster rename instead of just replacing provided '
'current cluster name and preserving backend and/or pool info.') 'current cluster name and preserving backend and/or pool info.')
@args('current', help='Current cluster name.') @args('current', help='Current cluster name.')
@args('new', help='New cluster name.') @args('new', help='New cluster name.')
def rename(self, partial, current, new): def rename(self,
partial: bool,
current: Optional[str],
new: Optional[str]) -> Optional[int]:
"""Rename cluster name for Volumes and Consistency Groups. """Rename cluster name for Volumes and Consistency Groups.
Useful when you want to rename a cluster, particularly when the Useful when you want to rename a cluster, particularly when the
@ -851,13 +888,15 @@ class ClusterCommands(BaseCommand):
print(msg % {'current': current}) print(msg % {'current': current})
return 2 return 2
return None
class ConsistencyGroupCommands(object): class ConsistencyGroupCommands(object):
"""Methods for managing consistency groups.""" """Methods for managing consistency groups."""
@args('--currenthost', required=True, help='Existing CG host name') @args('--currenthost', required=True, help='Existing CG host name')
@args('--newhost', required=True, help='New CG host name') @args('--newhost', required=True, help='New CG host name')
def update_cg_host(self, currenthost, newhost): def update_cg_host(self, currenthost: str, newhost: str) -> None:
"""Modify the host name associated with a Consistency Group. """Modify the host name associated with a Consistency Group.
Particularly to recover from cases where one has moved Particularly to recover from cases where one has moved
@ -878,7 +917,9 @@ class UtilCommands(object):
"""Generic utils.""" """Generic utils."""
@staticmethod @staticmethod
def _get_resources_locks(): def _get_resources_locks() -> Tuple[collections.defaultdict,
collections.defaultdict,
collections.defaultdict]:
"""Get all vol/snap/backup file lock paths.""" """Get all vol/snap/backup file lock paths."""
backup_locks_prefix = 'cinder-cleanup_incomplete_backups_' backup_locks_prefix = 'cinder-cleanup_incomplete_backups_'
oslo_dir = os.path.abspath(cfg.CONF.oslo_concurrency.lock_path) oslo_dir = os.path.abspath(cfg.CONF.oslo_concurrency.lock_path)
@ -890,8 +931,8 @@ class UtilCommands(object):
if tooz_dir != oslo_dir: if tooz_dir != oslo_dir:
filenames += glob.glob(os.path.join(tooz_dir, 'cinder-*')) filenames += glob.glob(os.path.join(tooz_dir, 'cinder-*'))
volumes = collections.defaultdict(list) volumes: collections.defaultdict = collections.defaultdict(list)
snapshots = collections.defaultdict(list) snapshots: collections.defaultdict = collections.defaultdict(list)
backups = collections.defaultdict(list) backups = collections.defaultdict(list)
matcher = re.compile('.*?([0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-' matcher = re.compile('.*?([0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-'
'[0-9a-f]{4}-[0-9a-f]{12}).*?', re.IGNORECASE) '[0-9a-f]{4}-[0-9a-f]{12}).*?', re.IGNORECASE)
@ -908,7 +949,7 @@ class UtilCommands(object):
return volumes, snapshots, backups return volumes, snapshots, backups
def _exclude_running_backups(self, backups): def _exclude_running_backups(self, backups: dict) -> None:
"""Remove backup entries from the dict for running backup services.""" """Remove backup entries from the dict for running backup services."""
for backup_pgrp in list(backups.keys()): for backup_pgrp in list(backups.keys()):
# The PGRP is the same as the PID of the parent process, so we know # The PGRP is the same as the PID of the parent process, so we know
@ -928,7 +969,7 @@ class UtilCommands(object):
@args('--services-offline', dest='online', @args('--services-offline', dest='online',
action='store_false', default=True, action='store_false', default=True,
help='All locks can be deleted as Cinder services are not running.') help='All locks can be deleted as Cinder services are not running.')
def clean_locks(self, online): def clean_locks(self, online: bool) -> None:
"""Clean file locks for vols, snaps, and backups on the current host. """Clean file locks for vols, snaps, and backups on the current host.
Should be run on any host where we are running a Cinder service (API, Should be run on any host where we are running a Cinder service (API,
@ -955,6 +996,8 @@ class UtilCommands(object):
self.ctxt = context.get_admin_context() self.ctxt = context.get_admin_context()
# Find volume and snapshots ids, and backups PGRP based on the existing # Find volume and snapshots ids, and backups PGRP based on the existing
# file locks # file locks
volumes: Union[collections.defaultdict, dict]
snapshots: Union[collections.defaultdict, dict]
volumes, snapshots, backups = self._get_resources_locks() volumes, snapshots, backups = self._get_resources_locks()
# If services are online we cannot delete locks for existing resources # If services are online we cannot delete locks for existing resources
@ -999,7 +1042,7 @@ CATEGORIES = {
} }
def methods_of(obj): def methods_of(obj) -> list:
"""Return non-private methods from an object. """Return non-private methods from an object.
Get all callable methods of an object that don't start with underscore Get all callable methods of an object that don't start with underscore
@ -1007,13 +1050,14 @@ def methods_of(obj):
""" """
result = [] result = []
for i in dir(obj): for i in dir(obj):
if isinstance(getattr(obj, i), if (isinstance(getattr(obj, i),
collections_abc.Callable) and not i.startswith('_'): collections_abc.Callable) and # type: ignore
not i.startswith('_')):
result.append((i, getattr(obj, i))) result.append((i, getattr(obj, i)))
return result return result
def missing_action(help_func): def missing_action(help_func: Callable) -> Callable:
def wrapped(): def wrapped():
help_func() help_func()
exit(2) exit(2)
@ -1033,7 +1077,7 @@ def add_command_parsers(subparsers):
for (action, action_fn) in methods_of(command_object): for (action, action_fn) in methods_of(command_object):
parser = category_subparsers.add_parser(action) parser = category_subparsers.add_parser(action)
action_kwargs = [] action_kwargs: list = []
for args, kwargs in getattr(action_fn, 'args', []): for args, kwargs in getattr(action_fn, 'args', []):
parser.add_argument(*args, **kwargs) parser.add_argument(*args, **kwargs)

View File

@ -9,6 +9,7 @@ cinder/context.py
cinder/coordination.py cinder/coordination.py
cinder/cmd/api.py cinder/cmd/api.py
cinder/cmd/backup.py cinder/cmd/backup.py
cinder/cmd/manage.py
cinder/cmd/scheduler.py cinder/cmd/scheduler.py
cinder/cmd/status.py cinder/cmd/status.py
cinder/cmd/volume.py cinder/cmd/volume.py