Files
distcloud-client/distributedcloud-client/dcmanagerclient/commands/v1/subcloud_manager.py
Tao Liu 19f027179c Support subcloud deploy upload the common files
Add new CLI commands to upload and show the subcloud
deploy common files:
dcmanager subcloud-deploy upload \
--deploy-playbook <play book> \
--deploy-chart <helm chart> \
--deploy-overrides <overrides>

dcmanager subcloud-deploy show

Changes to the subcloud add commands
dcmanager subcloud add \
--bootstrap-address oam_ip_address_of_subclouds_controller-0 \
--bootstrap-values <file> \
--deploy-config <file> \
--sysadmin-password sysadmin_password \
--install-values <file> \
--bmc-password bmc_password

The password is base64 encoded in the REST API request.
The files are sent using multipart/form-data in the REST request.
The file contents are processed by the API server.

Depends-On: https://review.opendev.org/#/c/720589/
Closes-Bug: 1864508

Change-Id: Id92ee8b631789b4949b9682586060ce424983e88
Signed-off-by: Tao Liu <tao.liu@windriver.com>
2020-04-23 09:38:51 -04:00

434 lines
14 KiB
Python

# Copyright (c) 2017 Ericsson AB.
#
# 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 base64
import getpass
import os
from osc_lib.command import command
from dcmanagerclient.commands.v1 import base
from dcmanagerclient import exceptions
def format(subcloud=None):
columns = (
'id',
'name',
'management',
'availability',
'deploy status',
'sync'
)
if subcloud:
data = (
subcloud.subcloud_id,
subcloud.name,
subcloud.management_state,
subcloud.availability_status,
subcloud.deploy_status,
subcloud.sync_status
)
else:
data = (tuple('<none>' for _ in range(len(columns))),)
return columns, data
def detail_format(subcloud=None):
columns = (
'id',
'name',
'description',
'location',
'software_version',
'management',
'availability',
'deploy_status',
'management_subnet',
'management_start_ip',
'management_end_ip',
'management_gateway_ip',
'systemcontroller_gateway_ip',
'group_id',
'created_at',
'updated_at',
)
if subcloud:
data = (
subcloud.subcloud_id,
subcloud.name,
subcloud.description,
subcloud.location,
subcloud.software_version,
subcloud.management_state,
subcloud.availability_status,
subcloud.deploy_status,
subcloud.management_subnet,
subcloud.management_start_ip,
subcloud.management_end_ip,
subcloud.management_gateway_ip,
subcloud.systemcontroller_gateway_ip,
subcloud.group_id,
subcloud.created_at,
subcloud.updated_at,
)
for listitem, sync_status in enumerate(subcloud.endpoint_sync_status
):
added_field = (sync_status['endpoint_type'] +
"_sync_status",)
added_value = (sync_status['sync_status'],)
columns += tuple(added_field)
data += tuple(added_value)
if subcloud.oam_floating_ip != "unavailable":
columns += ('oam_floating_ip',)
data += (subcloud.oam_floating_ip,)
else:
data = (tuple('<none>' for _ in range(len(columns))),)
return columns, data
class AddSubcloud(base.DCManagerShowOne):
"""Add a new subcloud."""
def _get_format_function(self):
return detail_format
def get_parser(self, prog_name):
parser = super(AddSubcloud, self).get_parser(prog_name)
parser.add_argument(
'--bootstrap-address',
required=True,
help='IP address for initial subcloud controller.'
)
parser.add_argument(
'--bootstrap-values',
required=True,
help='YAML file containing subcloud configuration settings. '
'Can be either a local file path or a URL.'
)
parser.add_argument(
'--deploy-config',
required=False,
help='YAML file containing subcloud variables to be passed to the '
'deploy playbook.'
)
parser.add_argument(
'--install-values',
required=False,
help='YAML file containing subcloud variables required for remote '
'install playbook.'
)
parser.add_argument(
'--sysadmin-password',
required=False,
help='sysadmin password of the subcloud to be configured, '
'if not provided you will be prompted.'
)
parser.add_argument(
'--bmc-password',
required=False,
help='bmc password of the subcloud to be configured, '
'if not provided you will be prompted.'
)
parser.add_argument(
'--group',
required=False,
help='Name or ID of subcloud group.'
)
return parser
def _get_resources(self, parsed_args):
dcmanager_client = self.app.client_manager.subcloud_manager
files = dict()
data = dict()
data['bootstrap-address'] = parsed_args.bootstrap_address
# Get the install values yaml file
if parsed_args.install_values is not None:
if not os.path.isfile(parsed_args.install_values):
error_msg = "install-values does not exist: %s" % \
parsed_args.install_values
raise exceptions.DCManagerClientException(error_msg)
files['install_values'] = parsed_args.install_values
# Get the bootstrap values yaml file
if not os.path.isfile(parsed_args.bootstrap_values):
error_msg = "bootstrap-values does not exist: %s" % \
parsed_args.bootstrap_values
raise exceptions.DCManagerClientException(error_msg)
files['bootstrap_values'] = parsed_args.bootstrap_values
# Get the deploy config yaml file
if parsed_args.deploy_config is not None:
if not os.path.isfile(parsed_args.deploy_config):
error_msg = "deploy-config does not exist: %s" % \
parsed_args.deploy_config
raise exceptions.DCManagerClientException(error_msg)
files['deploy_config'] = parsed_args.deploy_config
# Prompt the user for the subcloud's password if it isn't provided
if parsed_args.sysadmin_password is not None:
data['sysadmin_password'] = base64.b64encode(
parsed_args.sysadmin_password.encode("utf-8"))
else:
while True:
password = getpass.getpass(
"Enter the sysadmin password for the subcloud: ")
if len(password) < 1:
print("Password cannot be empty")
continue
confirm = getpass.getpass(
"Re-enter sysadmin password to confirm: ")
if password != confirm:
print("Passwords did not match")
continue
data["sysadmin_password"] = base64.b64encode(
password.encode("utf-8"))
break
if parsed_args.install_values is not None:
if parsed_args.bmc_password is not None:
data['bmc_password'] = base64.b64encode(
parsed_args.bmc_password.encode("utf-8"))
else:
while True:
password = getpass.getpass(
"Enter the bmc password for the subcloud: ")
if len(password) < 1:
print("Password cannot be empty")
continue
confirm = getpass.getpass(
"Re-enter bmc password to confirm: ")
if password != confirm:
print("Passwords did not match")
continue
data["bmc_password"] = base64.b64encode(
password.encode("utf-8"))
break
if parsed_args.group is not None:
data['group_id'] = parsed_args.group
return dcmanager_client.subcloud_manager.add_subcloud(files=files,
data=data)
class ListSubcloud(base.DCManagerLister):
"""List subclouds."""
def _get_format_function(self):
return format
def get_parser(self, prog_name):
parser = super(ListSubcloud, self).get_parser(prog_name)
return parser
def _get_resources(self, parsed_args):
dcmanager_client = self.app.client_manager.subcloud_manager
return dcmanager_client.subcloud_manager.list_subclouds()
class ShowSubcloud(base.DCManagerShowOne):
"""Show the details of a subcloud."""
def _get_format_function(self):
return detail_format
def get_parser(self, prog_name):
parser = super(ShowSubcloud, self).get_parser(prog_name)
parser.add_argument(
'subcloud',
help='Name or ID of subcloud to view the details.'
)
parser.add_argument(
'-d', '--detail',
action='store_true',
help="Show additional details for a subcloud"
)
return parser
def _get_resources(self, parsed_args):
subcloud_ref = parsed_args.subcloud
dcmanager_client = self.app.client_manager.subcloud_manager
if parsed_args.detail:
return dcmanager_client.subcloud_manager.\
subcloud_additional_details(subcloud_ref)
else:
return dcmanager_client.subcloud_manager.\
subcloud_detail(subcloud_ref)
class DeleteSubcloud(command.Command):
"""Delete subcloud details from the database."""
def get_parser(self, prog_name):
parser = super(DeleteSubcloud, self).get_parser(prog_name)
parser.add_argument(
'subcloud',
help='Name or ID of the subcloud to delete.'
)
return parser
def take_action(self, parsed_args):
subcloud_ref = parsed_args.subcloud
dcmanager_client = self.app.client_manager.subcloud_manager
try:
dcmanager_client.subcloud_manager.delete_subcloud(subcloud_ref)
except Exception as e:
print(e)
error_msg = "Unable to delete subcloud %s" % (subcloud_ref)
raise exceptions.DCManagerClientException(error_msg)
class UnmanageSubcloud(base.DCManagerShowOne):
"""Unmanage a subcloud."""
def _get_format_function(self):
return detail_format
def get_parser(self, prog_name):
parser = super(UnmanageSubcloud, self).get_parser(prog_name)
parser.add_argument(
'subcloud',
help='Name or ID of the subcloud to unmanage.'
)
return parser
def _get_resources(self, parsed_args):
subcloud_ref = parsed_args.subcloud
dcmanager_client = self.app.client_manager.subcloud_manager
kwargs = dict()
kwargs['management-state'] = 'unmanaged'
try:
return dcmanager_client.subcloud_manager.update_subcloud(
subcloud_ref, **kwargs)
except Exception as e:
print(e)
error_msg = "Unable to unmanage subcloud %s" % (subcloud_ref)
raise exceptions.DCManagerClientException(error_msg)
class ManageSubcloud(base.DCManagerShowOne):
"""Manage a subcloud."""
def _get_format_function(self):
return detail_format
def get_parser(self, prog_name):
parser = super(ManageSubcloud, self).get_parser(prog_name)
parser.add_argument(
'subcloud',
help='Name or ID of the subcloud to manage.'
)
return parser
def _get_resources(self, parsed_args):
subcloud_ref = parsed_args.subcloud
dcmanager_client = self.app.client_manager.subcloud_manager
kwargs = dict()
kwargs['management-state'] = 'managed'
try:
return dcmanager_client.subcloud_manager.update_subcloud(
subcloud_ref, **kwargs)
except Exception as e:
print(e)
error_msg = "Unable to manage subcloud %s" % (subcloud_ref)
raise exceptions.DCManagerClientException(error_msg)
class UpdateSubcloud(base.DCManagerShowOne):
"""Update attributes of a subcloud."""
def _get_format_function(self):
return detail_format
def get_parser(self, prog_name):
parser = super(UpdateSubcloud, self).get_parser(prog_name)
parser.add_argument(
'subcloud',
help='Name or ID of the subcloud to update.'
)
parser.add_argument(
'--description',
required=False,
help='Description of subcloud.'
)
parser.add_argument(
'--location',
required=False,
help='Location of subcloud.'
)
parser.add_argument(
'--group',
required=False,
help='Name or ID of subcloud group.'
)
return parser
def _get_resources(self, parsed_args):
subcloud_ref = parsed_args.subcloud
dcmanager_client = self.app.client_manager.subcloud_manager
kwargs = dict()
if parsed_args.description:
kwargs['description'] = parsed_args.description
if parsed_args.location:
kwargs['location'] = parsed_args.location
if parsed_args.group:
kwargs['group_id'] = parsed_args.group
if len(kwargs) == 0:
error_msg = "Nothing to update"
raise exceptions.DCManagerClientException(error_msg)
try:
return dcmanager_client.subcloud_manager.update_subcloud(
subcloud_ref, **kwargs)
except Exception as e:
print(e)
error_msg = "Unable to update subcloud %s" % (subcloud_ref)
raise exceptions.DCManagerClientException(error_msg)