 7f3827f24d
			
		
	
	7f3827f24d
	
	
	
		
			
			Add the following misc. changes to dcorch and dcmanager components:
- Cache the master resource in dcorch audit
- Consolidate the openstack drivers to common module, combine the
  dcmanager and dcorch sysinv client. (Note: the sdk driver that
  used by nova, neutron and cinder will be cleaned as part of
  story 2006588).
- Update the common sdk driver:
  . in order to avoid creating new keystone client multiple times
  . to add a option for caching region clients, in addition to the
    keystone client
  . finally, to randomize the token early renewal duration
- Change subcloud audit manager, patch audit manager,
  and sw update manager to:
  utilize the sdk driver which caches the keystone client and token
Test cases:
1. Manage/unmanage subclouds
2. Platform resources sync and audit
3. Verify the keystone token is cached until the token is
   expired
4. Add/delete subclouds
5. Managed subcloud goes offline/online (power off/on)
6. Managed subcloud goes offline/online (delete/add a static route)
7. Apply a patch to all subclouds via patch Orchestration
Story: 2007267
Task: 38865
Change-Id: I75e0cf66a797a65faf75e7c64dafb07f54c2df06
Signed-off-by: Tao Liu <tao.liu@windriver.com>
		
	
		
			
				
	
	
		
			149 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			149 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Copyright 2015 Huawei Technologies Co., Ltd.
 | |
| #
 | |
| # 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.
 | |
| #
 | |
| # Copyright (c) 2017-2020 Wind River Systems, Inc.
 | |
| #
 | |
| # The right to copy, distribute, modify, or otherwise make use
 | |
| # of this software may be licensed only pursuant to the terms
 | |
| # of an applicable Wind River license agreement.
 | |
| #
 | |
| 
 | |
| import grp
 | |
| import itertools
 | |
| import os
 | |
| import pwd
 | |
| import six.moves
 | |
| import tsconfig.tsconfig as tsc
 | |
| 
 | |
| from oslo_concurrency import lockutils
 | |
| from oslo_config import cfg
 | |
| from oslo_log import log as logging
 | |
| 
 | |
| from dccommon import consts as dccommon_consts
 | |
| from dccommon.drivers.openstack import vim
 | |
| from dcmanager.common import exceptions
 | |
| from dcmanager.db import api as db_api
 | |
| from dcorch.common import consts as dcorch_consts
 | |
| 
 | |
| LOG = logging.getLogger(__name__)
 | |
| 
 | |
| DC_MANAGER_USERNAME = "root"
 | |
| DC_MANAGER_GRPNAME = "root"
 | |
| 
 | |
| 
 | |
| def get_import_path(cls):
 | |
|     return cls.__module__ + "." + cls.__name__
 | |
| 
 | |
| 
 | |
| # Returns a iterator of tuples containing batch_size number of objects in each
 | |
| def get_batch_projects(batch_size, project_list, fillvalue=None):
 | |
|     args = [iter(project_list)] * batch_size
 | |
|     return six.moves.zip_longest(fillvalue=fillvalue, *args)
 | |
| 
 | |
| 
 | |
| # to do validate the quota limits
 | |
| def validate_quota_limits(payload):
 | |
|     for resource in payload:
 | |
|         # Check valid resource name
 | |
|         if resource not in itertools.chain(dcorch_consts.CINDER_QUOTA_FIELDS,
 | |
|                                            dcorch_consts.NOVA_QUOTA_FIELDS,
 | |
|                                            dcorch_consts.NEUTRON_QUOTA_FIELDS):
 | |
|             raise exceptions.InvalidInputError
 | |
|         # Check valid quota limit value in case for put/post
 | |
|         if isinstance(payload, dict) and (not isinstance(
 | |
|                 payload[resource], int) or payload[resource] <= 0):
 | |
|             raise exceptions.InvalidInputError
 | |
| 
 | |
| 
 | |
| def get_sw_update_opts(context,
 | |
|                        for_sw_update=False, subcloud_id=None):
 | |
|         """Get sw update options for a subcloud
 | |
| 
 | |
|         :param context: request context object.
 | |
|         :param for_sw_update: return the default options if subcloud options
 | |
|                               are empty. Useful for retrieving sw update
 | |
|                               options on application of patch strategy.
 | |
|         :param subcloud_id: id of subcloud.
 | |
| 
 | |
|         """
 | |
| 
 | |
|         if subcloud_id is None:
 | |
|             # Requesting defaults. Return constants if no entry in db.
 | |
|             sw_update_opts_ref = db_api.sw_update_opts_default_get(context)
 | |
|             if not sw_update_opts_ref:
 | |
|                 sw_update_opts_dict = vim.SW_UPDATE_OPTS_CONST_DEFAULT
 | |
|                 return sw_update_opts_dict
 | |
|         else:
 | |
|             # requesting subcloud options
 | |
|             sw_update_opts_ref = db_api.sw_update_opts_get(context,
 | |
|                                                            subcloud_id)
 | |
|             if sw_update_opts_ref:
 | |
|                 subcloud_name = db_api.subcloud_get(context, subcloud_id).name
 | |
|                 return db_api.sw_update_opts_w_name_db_model_to_dict(
 | |
|                     sw_update_opts_ref, subcloud_name)
 | |
|             elif for_sw_update:
 | |
|                 sw_update_opts_ref = db_api.sw_update_opts_default_get(context)
 | |
|                 if not sw_update_opts_ref:
 | |
|                     sw_update_opts_dict = vim.SW_UPDATE_OPTS_CONST_DEFAULT
 | |
|                     return sw_update_opts_dict
 | |
|             else:
 | |
|                 raise exceptions.SubcloudPatchOptsNotFound(
 | |
|                     subcloud_id=subcloud_id)
 | |
| 
 | |
|         return db_api.sw_update_opts_w_name_db_model_to_dict(
 | |
|             sw_update_opts_ref, dccommon_consts.SW_UPDATE_DEFAULT_TITLE)
 | |
| 
 | |
| 
 | |
| def ensure_lock_path():
 | |
|     # Determine the oslo_concurrency lock path:
 | |
|     # 1) First, from the oslo_concurrency section of the config
 | |
|     #    a) If not set via an option default or config file, oslo_concurrency
 | |
|     #       sets it to the OSLO_LOCK_PATH env variable
 | |
|     # 2) Then if not set, set it to a specific directory under
 | |
|     #    tsc.VOLATILE_PATH
 | |
| 
 | |
|     if cfg.CONF.oslo_concurrency.lock_path:
 | |
|         lock_path = cfg.CONF.oslo_concurrency.lock_path
 | |
|     else:
 | |
|         lock_path = os.path.join(tsc.VOLATILE_PATH, "dcmanager")
 | |
| 
 | |
|     if not os.path.isdir(lock_path):
 | |
|         try:
 | |
|             uid = pwd.getpwnam(DC_MANAGER_USERNAME).pw_uid
 | |
|             gid = grp.getgrnam(DC_MANAGER_GRPNAME).gr_gid
 | |
|             os.makedirs(lock_path)
 | |
|             os.chown(lock_path, uid, gid)
 | |
|             LOG.info("Created directory=%s" % lock_path)
 | |
| 
 | |
|         except OSError as e:
 | |
|             LOG.exception("makedir %s OSError=%s encountered" %
 | |
|                           (lock_path, e))
 | |
|             return None
 | |
| 
 | |
|     return lock_path
 | |
| 
 | |
| 
 | |
| def synchronized(name, external=True, fair=False):
 | |
|     if external:
 | |
|         prefix = 'DCManager-'
 | |
|         lock_path = ensure_lock_path()
 | |
|     else:
 | |
|         prefix = None
 | |
|         lock_path = None
 | |
| 
 | |
|     return lockutils.synchronized(name, lock_file_prefix=prefix,
 | |
|                                   external=external, lock_path=lock_path,
 | |
|                                   semaphores=None, delay=0.01, fair=fair)
 |