Files
python-ironicclient/ironicclient/osc/v1/baremetal_deploy_template.py
Abhishek Bongale fe9a35908c Fix field names for selection and json
Whenever we list baremetal nodes  with JSON format, there's an
inconsisency between the field names specified in the command and
the field names returned in the JSON output.
Users request field names (e.g. created_at) it should return as
created_at not (e.g. Created At) pretty Json format response.

This change modifies the ListBaremetalNode.take_action() to return the
field name (columns) instead of pretty labels. So that input and output
remains consistent.

Updated all the repeative objects to match change made in the
ListBaremetalNode.take_action() and removed unused labels using Claude.

Closes-bug: #2138466
Change-Id: Ia66f08a020ecf932f21775939c9b436ec147d738
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
Signed-off-by: Abhishek Bongale <abhishekbongale@outlook.com>
2026-02-03 12:27:26 +00:00

343 lines
12 KiB
Python

# 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.
#
import itertools
import json
import logging
from osc_lib.command import command
from osc_lib import utils as oscutils
from ironicclient.common.i18n import _
from ironicclient.common import utils
from ironicclient import exc
from ironicclient.v1 import resource_fields as res_fields
_DEPLOY_STEPS_HELP = _(
"The deploy steps. May be the path to a YAML file containing the deploy "
"steps; OR '-', with the deploy steps being read from standard "
"input; OR a JSON string. The value should be a list of deploy-step "
"dictionaries; each dictionary should have keys 'interface', 'step', "
"'args' and 'priority'.")
class CreateBaremetalDeployTemplate(command.ShowOne):
"""Create a new deploy template"""
log = logging.getLogger(__name__ + ".CreateBaremetalDeployTemplate")
def get_parser(self, prog_name):
parser = super(CreateBaremetalDeployTemplate, self).get_parser(
prog_name)
parser.add_argument(
'name',
metavar='<name>',
help=_('Unique name for this deploy template. Must be a valid '
'trait name')
)
parser.add_argument(
'--uuid',
dest='uuid',
metavar='<uuid>',
help=_('UUID of the deploy template.'))
parser.add_argument(
'--extra',
metavar="<key=value>",
action='append',
help=_("Record arbitrary key/value metadata. "
"Can be specified multiple times."))
parser.add_argument(
'--steps',
metavar="<steps>",
required=True,
help=_DEPLOY_STEPS_HELP
)
return parser
def take_action(self, parsed_args):
self.log.debug("take_action(%s)", parsed_args)
baremetal_client = self.app.client_manager.baremetal
steps = utils.handle_json_arg(parsed_args.steps, 'deploy steps')
field_list = ['name', 'uuid', 'extra']
fields = dict((k, v) for (k, v) in vars(parsed_args).items()
if k in field_list and v is not None)
fields = utils.args_array_to_dict(fields, 'extra')
template = baremetal_client.deploy_template.create(steps=steps,
**fields)
data = dict([(f, getattr(template, f, '')) for f in
res_fields.DEPLOY_TEMPLATE_DETAILED_RESOURCE.fields])
return self.dict2columns(data)
class ShowBaremetalDeployTemplate(command.ShowOne):
"""Show baremetal deploy template details."""
log = logging.getLogger(__name__ + ".ShowBaremetalDeployTemplate")
def get_parser(self, prog_name):
parser = super(ShowBaremetalDeployTemplate, self).get_parser(prog_name)
parser.add_argument(
"template",
metavar="<template>",
help=_("Name or UUID of the deploy template.")
)
parser.add_argument(
'--fields',
nargs='+',
dest='fields',
metavar='<field>',
action='append',
choices=res_fields.DEPLOY_TEMPLATE_DETAILED_RESOURCE.fields,
default=[],
help=_("One or more deploy template fields. Only these fields "
"will be fetched from the server.")
)
return parser
def take_action(self, parsed_args):
self.log.debug("take_action(%s)", parsed_args)
baremetal_client = self.app.client_manager.baremetal
fields = list(itertools.chain.from_iterable(parsed_args.fields))
fields = fields if fields else None
template = baremetal_client.deploy_template.get(
parsed_args.template, fields=fields)._info
template.pop("links", None)
return zip(*sorted(template.items()))
class SetBaremetalDeployTemplate(command.Command):
"""Set baremetal deploy template properties."""
log = logging.getLogger(__name__ + ".SetBaremetalDeployTemplate")
def get_parser(self, prog_name):
parser = super(SetBaremetalDeployTemplate, self).get_parser(prog_name)
parser.add_argument(
'template',
metavar='<template>',
help=_("Name or UUID of the deploy template")
)
parser.add_argument(
'--name',
metavar='<name>',
help=_('Set unique name of the deploy template. Must be a valid '
'trait name.')
)
parser.add_argument(
'--steps',
metavar="<steps>",
help=_DEPLOY_STEPS_HELP
)
parser.add_argument(
"--extra",
metavar="<key=value>",
action='append',
help=_('Extra to set on this baremetal deploy template '
'(repeat option to set multiple extras).'),
)
return parser
def take_action(self, parsed_args):
self.log.debug("take_action(%s)", parsed_args)
baremetal_client = self.app.client_manager.baremetal
properties = []
if parsed_args.name:
name = ["name=%s" % parsed_args.name]
properties.extend(utils.args_array_to_patch('add', name))
if parsed_args.steps:
steps = utils.handle_json_arg(parsed_args.steps, 'deploy steps')
steps = ["steps=%s" % json.dumps(steps)]
properties.extend(utils.args_array_to_patch('add', steps))
if parsed_args.extra:
properties.extend(utils.args_array_to_patch(
'add', ['extra/' + x for x in parsed_args.extra]))
if properties:
baremetal_client.deploy_template.update(parsed_args.template,
properties)
else:
self.log.warning("Please specify what to set.")
class UnsetBaremetalDeployTemplate(command.Command):
"""Unset baremetal deploy template properties."""
log = logging.getLogger(__name__ + ".UnsetBaremetalDeployTemplate")
def get_parser(self, prog_name):
parser = super(UnsetBaremetalDeployTemplate, self).get_parser(
prog_name)
parser.add_argument(
'template',
metavar='<template>',
help=_("Name or UUID of the deploy template")
)
parser.add_argument(
"--extra",
metavar="<key>",
action='append',
help=_('Extra to unset on this baremetal deploy template '
'(repeat option to unset multiple extras).'),
)
return parser
def take_action(self, parsed_args):
self.log.debug("take_action(%s)", parsed_args)
baremetal_client = self.app.client_manager.baremetal
properties = []
if parsed_args.extra:
properties.extend(utils.args_array_to_patch('remove',
['extra/' + x for x in parsed_args.extra]))
if properties:
baremetal_client.deploy_template.update(parsed_args.template,
properties)
else:
self.log.warning("Please specify what to unset.")
class DeleteBaremetalDeployTemplate(command.Command):
"""Delete deploy template(s)."""
log = logging.getLogger(__name__ + ".DeleteBaremetalDeployTemplate")
def get_parser(self, prog_name):
parser = super(DeleteBaremetalDeployTemplate, self).get_parser(
prog_name)
parser.add_argument(
"templates",
metavar="<template>",
nargs="+",
help=_("Name(s) or UUID(s) of the deploy template(s) to delete.")
)
return parser
def take_action(self, parsed_args):
self.log.debug("take_action(%s)", parsed_args)
baremetal_client = self.app.client_manager.baremetal
failures = []
for template in parsed_args.templates:
try:
baremetal_client.deploy_template.delete(template)
print(_('Deleted deploy template %s') % template)
except exc.ClientException as e:
failures.append(_("Failed to delete deploy template "
"%(template)s: %(error)s")
% {'template': template, 'error': e})
if failures:
raise exc.ClientException("\n".join(failures))
class ListBaremetalDeployTemplate(command.Lister):
"""List baremetal deploy templates."""
log = logging.getLogger(__name__ + ".ListBaremetalDeployTemplate")
def get_parser(self, prog_name):
parser = super(ListBaremetalDeployTemplate, self).get_parser(prog_name)
parser.add_argument(
'--limit',
metavar='<limit>',
type=int,
help=_('Maximum number of deploy templates to return per request, '
'0 for no limit. Default is the maximum number used '
'by the Baremetal API Service.')
)
parser.add_argument(
'--marker',
metavar='<template>',
help=_('DeployTemplate UUID (for example, of the last deploy '
'template in the list from a previous request). Returns '
'the list of deploy templates after this UUID.')
)
parser.add_argument(
'--sort',
metavar="<key>[:<direction>]",
help=_('Sort output by specified deploy template fields and '
'directions (asc or desc) (default: asc). Multiple fields '
'and directions can be specified, separated by comma.')
)
display_group = parser.add_mutually_exclusive_group()
display_group.add_argument(
'--long',
dest='detail',
action='store_true',
default=False,
help=_("Show detailed information about deploy templates.")
)
display_group.add_argument(
'--fields',
nargs='+',
dest='fields',
metavar='<field>',
action='append',
default=[],
choices=res_fields.DEPLOY_TEMPLATE_DETAILED_RESOURCE.fields,
help=_("One or more deploy template fields. Only these fields "
"will be fetched from the server. Can not be used when "
"'--long' is specified.")
)
return parser
def take_action(self, parsed_args):
self.log.debug("take_action(%s)", parsed_args)
client = self.app.client_manager.baremetal
columns = res_fields.DEPLOY_TEMPLATE_RESOURCE.fields
params = {}
if parsed_args.limit is not None and parsed_args.limit < 0:
raise exc.CommandError(
_('Expected non-negative --limit, got %s') %
parsed_args.limit)
params['limit'] = parsed_args.limit
params['marker'] = parsed_args.marker
if parsed_args.detail:
params['detail'] = parsed_args.detail
columns = res_fields.DEPLOY_TEMPLATE_DETAILED_RESOURCE.fields
elif parsed_args.fields:
params['detail'] = False
fields = itertools.chain.from_iterable(parsed_args.fields)
resource = res_fields.Resource(list(fields))
columns = resource.fields
params['fields'] = columns
self.log.debug("params(%s)", params)
data = client.deploy_template.list(**params)
data = oscutils.sort_items(data, parsed_args.sort)
return (columns,
(oscutils.get_item_properties(s, columns) for s in data))