diff --git a/docs/changes.rst b/docs/changes.rst index e29724ae..77332def 100644 --- a/docs/changes.rst +++ b/docs/changes.rst @@ -54,6 +54,12 @@ Released: not yet commands have been grouped to be more easily identifiable. This required adding the "click-option-group" Python package to the dependencies. +* Added command group 'imageprofile' for operations on image activation + profiles in classic mode CPCs. + +* Added command group 'loadprofile' for operations on load activation + profiles in classic mode CPCs. + **Cleanup:** **Known issues:** diff --git a/zhmccli/__init__.py b/zhmccli/__init__.py index f4186b7a..aa68cf63 100644 --- a/zhmccli/__init__.py +++ b/zhmccli/__init__.py @@ -26,6 +26,8 @@ from ._cmd_unmanaged_cpc import * # noqa: F401 from ._cmd_ldap_server_definition import * # noqa: F401 from ._cmd_lpar import * # noqa: F401 +from ._cmd_imageprofile import * # noqa: F401 +from ._cmd_loadprofile import * # noqa: F401 from ._cmd_partition import * # noqa: F401 from ._cmd_adapter import * # noqa: F401 from ._cmd_port import * # noqa: F401 diff --git a/zhmccli/_cmd_imageprofile.py b/zhmccli/_cmd_imageprofile.py new file mode 100644 index 00000000..ea3f1477 --- /dev/null +++ b/zhmccli/_cmd_imageprofile.py @@ -0,0 +1,1729 @@ +# Copyright 2023 IBM Corp. All Rights Reserved. +# +# 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. + +""" +Commands for image activation profiles for CPCs in classic mode. +""" + +from __future__ import absolute_import + +import click +from click_option_group import optgroup + +import zhmcclient +from .zhmccli import cli +from ._helper import print_properties, print_resources, abort_if_false, \ + options_to_properties, original_options, COMMAND_OPTIONS_METAVAR, \ + click_exception, add_options, LIST_OPTIONS +from ._cmd_cpc import find_cpc +from ._cmd_certificates import find_certificate + + +# Defaults for image activation profile creation +DEFAULT_IFL_PROCESSORS = 1 +DEFAULT_INITIAL_MEMORY_MB = 1024 +DEFAULT_MAXIMUM_MEMORY_MB = 1024 +DEFAULT_PROCESSOR_MODE = 'shared' +PARTITION_TYPES = ['ssc', 'linux', 'zvm'] +DEFAULT_PARTITION_TYPE = 'linux' +DEFAULT_SSC_BOOT = 'installer' +DEFAULT_PROCESSING_WEIGHT = 100 +MIN_PROCESSING_WEIGHT = 1 +MAX_PROCESSING_WEIGHT = 999 +MIN_BOOT_TIMEOUT = 60 +MAX_BOOT_TIMEOUT = 600 + + +def find_imageprofile(cmd_ctx, client, cpc_or_name, imageprofile_name): + """ + Find an image activation profile by name and return its resource object. + """ + if isinstance(cpc_or_name, zhmcclient.Cpc): + cpc = cpc_or_name + else: + cpc = find_cpc(cmd_ctx, client, cpc_or_name) + # The CPC must not be in DPM mode. We don't check that because it would + # cause a GET to the CPC resource that we otherwise don't need. + try: + imageprofile = cpc.image_activation_profiles.find( + name=imageprofile_name) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + return imageprofile + + +@cli.group('imageprofile', options_metavar=COMMAND_OPTIONS_METAVAR) +def imageprofile_group(): + """ + Command group for managing image activation profiles (classic mode only). + + "Image activation profiles" are used to activate LPARs, and optionally + to load (= boot, IPL) an operating system (also known as "control program") + into the LPAR. Image activation profiles include definitions for the + resources to be assigned to the LPAR, and also definitions for loading + the operating system. + + The commands in this group work only on CPCs in classic mode. + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + + +@imageprofile_group.command('list', options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='[CPC]', required=False) +@add_options(LIST_OPTIONS) +@click.pass_obj +def imageprofile_list(cmd_ctx, cpc, **options): + """ + List the image activation profiles in a CPC or in all managed CPCs. + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + cmd_ctx.execute_cmd(lambda: cmd_imageprofile_list(cmd_ctx, cpc, options)) + + +@imageprofile_group.command('show', options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='CPC') +@click.argument('imageprofile', type=str, metavar='IMAGEPROFILE') +@click.pass_obj +def imageprofile_show(cmd_ctx, cpc, imageprofile, **options): + """ + Show details of an image activation profile. + + The following properties are shown in addition to those returned by the HMC: + + \b + - 'parent-name' - Name of the parent CPC. + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + cmd_ctx.execute_cmd( + lambda: cmd_imageprofile_show(cmd_ctx, cpc, imageprofile, options)) + + +@imageprofile_group.command('update', options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='CPC') +@click.argument('imageprofile', type=str, metavar='IMAGEPROFILE') +@optgroup.group('General options') +@optgroup.option('--description', type=str, required=False, + help='The new description for the image activation profile.') +@optgroup.option('--operating-mode', type=str, metavar='OPMODE', required=False, + help='The new operating mode for the LPAR:\n' + '\n' + '\b\n' + ' - "general" - TBD (on z14 or later)\n' + ' - "esa390" - TBD (on z13 or later)\n' + ' - "esa390-tpf" - TBD (on z13 or later)\n' + ' - "coupling-facility" - TBD\n' + ' - "linux-only" - TBD\n' + ' - "zvm" - TBD\n' + ' - "zaware" - TBD (on z13 or later)\n' + ' - "ssc" - TBD (on z13 or later)\n') +@optgroup.option('--load-at-activation', type=bool, required=False, + help='The new indicator for whether the LPAR will be loaded ' + 'at the end of the activation.') +@optgroup.option('--partition-id', type=str, required=False, + # Property: 'partition-identifier' + help='The new partition identifier (two-digit hexadecimal) to ' + 'be used for the LPAR.') +@optgroup.option('--local-cluster-name', type=str, required=False, + help='The name of the new CP management cluster for the LPAR.') +@optgroup.option('--group-profile', type=str, required=False, + # Property: 'group-profile-uri' (name -> URI) + help='The name of the new Group profile to be associated with ' + 'this profile. An empty string will cause no Group profile ' + 'to be associated with this profile. ' + 'An associated Group profile provides the group capacity ' + 'value for the LPAR.') +@optgroup.option('--liccc-validation-enabled', type=bool, required=False, + help='The new indicator for validating that the image profile ' + 'data conforms to the current maximum LICCC configuration.') +@optgroup.option('--lpar-isolation', type=bool, required=False, + # Property: 'logical-partition-isolation-control' + help='The new LPAR isolation indicator. When True, ' + 'reconfigurable channel paths assigned to the LPAR are ' + 'reserved for its exclusive use.') +@optgroup.group('Boot configuration') +@optgroup.option('--ipl-type', type=str, metavar='IPLTYPE', required=False, + help='The new IPL type for the operating system. The IPL type ' + 'defines from where the control program for the LPAR will be ' + 'loaded:\n' + '\n' + '\b\n' + ' - "ipltype-scsi" - SCSI list-directed OS load\n' + ' - "ipltype-scsidump" - SCSI list-directed OS dump\n' + ' - "ipltype-nvmeload" – NVMe list-directed OS load ' + '(z15 or later)\n' + ' - "ipltype-nvmedump" – NVMe list-directed OS dump ' + '(z15 or later)\n' + ' - "ipltype-tape-load" - Tape Channel Command Word (CCW) ' + 'OS load (z16 or later)\n' + ' - "ipltype-tape-dump" - Tape Channel Command Word (CCW) ' + 'OS dump (z16 or later)\n' + ' - "ipltype-eckd-ccw-load" - ECKD Channel Command Word ' + '(CCW) OS load (z16 or later)\n' + ' - "ipltype-eckd-ccw-dump" - ECKD Channel Command Word ' + '(CCW) OS dump (z16 or later)\n' + ' - "ipltype-eckd-ld-load" - ECKD list-directed OS load ' + '(z16 or later)\n' + ' - "ipltype-eckd-ld-dump" - ECKD list-directed OS dump ' + '(z16 or later)\n' + ' - "ipltype-standard" - Channel Command Word (CCW) ' + 'standard load. This is present for\n' + ' compatibility and will be interpreted as one of the ' + 'new values.\n') +@optgroup.option('--ipl-address', type=str, required=False, + help='The new IPL address. ' + 'For NVMe IPL types, this is the FID of the NVMe device ' + 'as 4 hex characters in any lexical case. ' + 'For any other IPL types, this is the device number of the ' + 'boot device as 4 or 5 hex characters in any lexical case. ' + 'If empty, the IPL address will be taken from the load ' + 'activation profile.') +@optgroup.option('--ipl-parameter', type=str, required=False, + help='The new IPL parameter. This is a short (0-8 characters) ' + 'string passed to the control program. Valid characters are ' + '"0"-"9", "A"-"Z", blank (" ") and period ("."). On z14 and ' + 'later, the characters "@", "$", and "#" are valid in ' + 'addition.') +@optgroup.option('--load-parameters', type=str, required=False, + # Property: 'os-specific-load-parameters' + help='The new operating system-specific load parameters. This ' + 'is a longer string string passed to the control program. ' + 'Only used for the list-directed IPL types "ipltype-scsi", ' + '"ipltype-scsidump", "ipltype-nvmeload", "ipltype-nvmedump", ' + '"ipltype-eckd-ld-load", or "ipltype-eckd-ld-dump".') +@optgroup.option('--load-timeout', type=int, required=False, + help='The new amount of time, in seconds, to wait for the ' + 'Load to complete (60-600).') +@optgroup.option('--store-status-indicator', type=bool, required=False, + help='The new store-status indicator. If True, the status is ' + 'stored before performing the Load.') +@optgroup.option('--secure-boot', type=bool, required=False, + help='The new secure-boot indicator. It indicates whether the ' + 'software signature of the control program will be verified ' + 'using the secure-boot certificate(s) assigned to the LPAR. ' + 'Only used for the list-directed IPL types "ipltype-scsi", ' + '"ipltype-scsidump", "ipltype-nvmeload", "ipltype-nvmedump", ' + '"ipltype-eckd-ld-load", or "ipltype-eckd-ld-dump".') +@optgroup.option('--disk-partition-id', type=str, required=False, + # Property: 'disk-partition-id' + 'disk-partition-id-automatic' + help='The new disk partition number (also called the boot ' + 'program selector), in decimal. The special value "auto" ' + '(only valid on z16 or later) causes the disk partition ' + 'number to be determined automatically. ' + 'Only used for the list-directed IPL types "ipltype-scsi", ' + '"ipltype-scsidump", "ipltype-nvmeload", "ipltype-nvmedump", ' + '"ipltype-eckd-ld-load", or "ipltype-eckd-ld-dump".') +@optgroup.option('--worldwide-port-name', type=str, required=False, + help='The new worldwide port name (WWPN) of the SCSI boot ' + 'device, in hexadecimal. Only used for the SCSI IPL types ' + '"ipltype-scsi" or "ipltype-scsidump".') +@optgroup.option('--logical-unit-number', type=str, required=False, + help='The new logical unit number of the SCSI boot device, ' + 'in hexadecimal. Only used for the SCSI IPL types ' + '"ipltype-scsi" or "ipltype-scsidump".') +@optgroup.option('--boot-record-lba', type=str, required=False, + help='The new boot record logical block address the SCSI/NVMe ' + 'boot device, in hexadecimal. Only used for SCSI and NVMe IPL ' + 'types "ipltype-scsi", "ipltype-scsidump", ' + '"ipltype-nvmeload", or "ipltype-nvmedump".') +@optgroup.option('--boot-record-location-cylinder', type=str, required=False, + help='The new boot record location cylinder value, in ' + 'hexadecimal. Only used for ECKD IPL types ' + '"ipltype-eckd-ld-load" or "ipltype-eckd-ld-dump".') +@optgroup.option('--boot-record-location-head', type=str, required=False, + help='The new boot record location head value, in ' + 'hexadecimal. Only used for ECKD IPL types ' + '"ipltype-eckd-ld-load" or "ipltype-eckd-ld-dump".') +@optgroup.option('--boot-record-location-record', type=str, required=False, + help='The new boot record location record value, in ' + 'hexadecimal. Only used for ECKD IPL types ' + '"ipltype-eckd-ld-load" or "ipltype-eckd-ld-dump".') +@optgroup.option('--boot-record-location-use-volume-label', type=bool, + required=False, + help='The new indicator whether the boot record location ' + 'cylinder, head, and record should be determined by the ' + 'volume label.') +@optgroup.group('CPU configuration') +@optgroup.option('--processor-usage', + type=click.Choice(['dedicated', 'shared']), required=False, + help='The new indicator for how processors are allocated to ' + 'the LPAR:\n' + '\n' + '\b\n' + ' - "dedicated" - The processors are exclusively available ' + 'to this LPAR.\n' + ' - "shared" - The processors are shareable across LPARs.\n') +@optgroup.option('--wlm-enabled', type=bool, required=False, + # Property: 'workload-manager-enabled' + help='The new indicator for enabling z/OS Workload Manager. ' + 'If True, z/OS Workload Manager is allowed to change ' + 'processing weight related properties of the LPAR after ' + 'activation.') +@optgroup.option('--defined-capacity', type=bool, required=False, + # Property: 'defined-capacity' + help='The new defined capacity of the LPAR ' + '(in MSU/h). This specifies how much capacity the LPAR is to ' + 'be managed to by z/OS Workload Manager for the purpose of ' + 'software pricing. 0 means that no defined capacity is ' + 'specified for this LPAR.') +@optgroup.option('--shared-cp-processors', type=int, required=False, + # Property: 'number-shared-general-purpose-processors' + help='The new number of shared CP (general purpose) ' + 'processors to be allocated to the LPAR at activation.') +@optgroup.option('--reserved-shared-cp-processors', type=int, required=False, + # Property: 'number-reserved-shared-general-purpose-processors' + help='The new number of shared CP (general purpose) ' + 'processors to be reserved for the LPAR, which can be ' + 'dynamically configured after activation.') +@optgroup.option('--initial-cp-processing-weight', type=int, required=False, + # Property: 'initial-processing-weight' + help='The new initial processing weight for CP (general ' + 'purpose) processors (1-999).') +@optgroup.option('--minimum-cp-processing-weight', type=int, required=False, + # Property: 'minimum-processing-weight' + help='The new minimum processing weight for CP (general ' + 'purpose) processors (0-999).') +@optgroup.option('--maximum-cp-processing-weight', type=int, required=False, + # Property: 'maximum-processing-weight' + help='The new maximum processing weight for CP (general ' + 'purpose) processors (0-999).') +@optgroup.option('--cp-processing-weight-capped', type=bool, required=False, + # Property: 'initial-processing-weight-capped' + help='The new indicator for whether the CP (general purpose) ' + 'processing weight is capped. If True, the processing weight ' + 'is an upper limit. If False, the processing weight is a ' + 'target that can be exceeded if excess CP processor resources ' + 'are available.') +@optgroup.option('--absolute-cp-capping', type=float, required=False, + # Property: 'absolute-general-purpose-capping' + # -> absolute-capping object + help='The new value for absolute CP (general purpose) ' + 'processor capping. A numeric value prevents the partition ' + 'from using any more than the specified number of physical ' + 'CP processors. An empty string disables absolute CP ' + 'processor capping.') +@optgroup.option('--dedicated-cp-processors', type=int, required=False, + # Property: 'number-dedicated-general-purpose-processors' + help='The new number of dedicated CP (general purpose) ' + 'processors to be allocated for the LPAR\'s exclusive use ' + 'at activation.') +@optgroup.option('--reserved-dedicated-cp-processors', type=int, required=False, + # Prop: 'number-reserved-dedicated-general-purpose-processors' + help='The new number of dedicated CP (general purpose) ' + 'processors to be reserved for the LPAR, which can be ' + 'dynamically configured after activation.') +# +@optgroup.option('--shared-ifl-processors', type=int, required=False, + # Property: 'number-shared-ifl-processors' + help='The new number of shared IFL (Integrated Facility for ' + 'Linux) processors to be allocated to the LPAR at activation.') +@optgroup.option('--reserved-shared-ifl-processors', type=int, required=False, + # Property: 'number-reserved-shared-ifl-processors' + help='The new number of shared IFL (Integrated Facility for ' + 'Linux) processors to be reserved for the LPAR, which can be ' + 'dynamically configured after activation.') +@optgroup.option('--initial-ifl-processing-weight', type=int, required=False, + # Property: 'initial-ifl-processing-weight' + help='The new initial processing weight for IFL (Integrated ' + 'Facility for Linux) processors (1-999).') +@optgroup.option('--minimum-ifl-processing-weight', type=int, required=False, + # Property: 'minimum-ifl-processing-weight' + help='The new minimum processing weight for IFL (Integrated ' + 'Facility for Linux) processors (0-999).') +@optgroup.option('--maximum-ifl-processing-weight', type=int, required=False, + # Property: 'maximum-ifl-processing-weight' + help='The new maximum processing weight for IFL (Integrated ' + 'Facility for Linux) processors (0-999).') +@optgroup.option('--ifl-processing-weight-capped', type=bool, required=False, + # Property: 'initial-ifl-processing-weight-capped' + help='The new indicator for whether the IFL (Integrated ' + 'Facility for Linux) processing weight is capped. If True, ' + 'the processing weight is an upper limit. If False, the ' + 'processing weight is a target that can be exceeded if ' + 'excess IFL processor resources are available.') +@optgroup.option('--absolute-ifl-capping', type=float, required=False, + # Property: 'absolute-ifl-capping' -> absolute-capping object + help='The new value for absolute IFL (Integrated Facility for ' + 'Linux) processor capping. A numeric value prevents the ' + 'partition from using any more than the specified number of ' + 'physical IFL processors. An empty string disables absolute ' + 'IFL processor capping.') +@optgroup.option('--dedicated-ifl-processors', type=int, required=False, + # Property: 'number-dedicated-ifl-processors' + help='The new number of dedicated IFL (Integrated Facility ' + 'for Linux) processors to be allocated for the LPAR\'s ' + 'exclusive use at activation.') +@optgroup.option('--reserved-dedicated-ifl-processors', type=int, + required=False, + # Property: 'number-reserved-dedicated-ifl-processors' + help='The new number of dedicated IFL (Integrated Facility ' + 'for Linux) processors to be reserved for the LPAR, which ' + 'can be dynamically configured after activation.') +# +@optgroup.option('--shared-zaap-processors', type=int, required=False, + # Property: 'number-shared-aap-processors' + help='The new number of shared zAAP (z Application Assist ' + 'Processor) processors to be allocated to the LPAR at ' + 'activation.') +@optgroup.option('--reserved-shared-zaap-processors', type=int, required=False, + # Property: 'number-reserved-shared-aap-processors' + help='The new number of shared zAAP (z Application Assist ' + 'Processor) processors to be reserved for the LPAR, which can ' + 'be dynamically configured after activation.') +@optgroup.option('--initial-zaap-processing-weight', type=int, required=False, + # Property: 'initial-aap-processing-weight' + help='The new initial processing weight for zAAP (z ' + 'Application Assist Processor) processors (1-999).') +@optgroup.option('--minimum-zaap-processing-weight', type=int, required=False, + # Property: 'minimum-aap-processing-weight' + help='The new minimum processing weight for zAAP (z ' + 'Application Assist Processor) processors (0-999).') +@optgroup.option('--maximum-zaap-processing-weight', type=int, required=False, + # Property: 'maximum-aap-processing-weight' + help='The new maximum processing weight for zAAP (z ' + 'Application Assist Processor) processors (0-999).') +@optgroup.option('--zaap-processing-weight-capped', type=bool, required=False, + # Property: 'initial-aap-processing-weight-capped' + help='The new indicator for whether the zAAP (z ' + 'Application Assist Processor) processing weight is capped. ' + 'If True, the processing weight is an upper limit. If False, ' + 'the processing weight is a target that can be exceeded if ' + 'excess zAAP processor resources are available.') +@optgroup.option('--absolute-zaap-capping', type=float, required=False, + # Property: 'absolute-aap-capping' -> absolute-capping object + help='The new value for absolute zAAP (z Application Assist ' + 'Processor) processor capping. A numeric value prevents the ' + 'partition from using any more than the specified number of ' + 'physical zAAP processors. An empty string disables absolute ' + 'zAAP processor capping.') +@optgroup.option('--dedicated-zaap-processors', type=int, required=False, + # Property: 'number-dedicated-aap-processors' + help='The new number of dedicated zAAP (z Application Assist ' + 'Processor) processors to be allocated for the LPAR\'s ' + 'exclusive use at activation.') +@optgroup.option('--reserved-dedicated-zaap-processors', type=int, + # Property: 'number-reserved-dedicated-aap-processors' + required=False, + help='The new number of dedicated zAAP (z Application Assist ' + 'Processor) processors to be reserved for the LPAR, which ' + 'can be dynamically configured after activation.') +# +@optgroup.option('--shared-ziip-processors', type=int, required=False, + # Property: 'number-shared-ziip-processors' + help='The new number of shared zIIP (z Integrated ' + 'Information Processor) processors to be allocated to the ' + 'LPAR at activation.') +@optgroup.option('--reserved-shared-ziip-processors', type=int, required=False, + # Property: 'number-reserved-shared-ziip-processors' + help='The new number of shared zIIP (z Integrated ' + 'Information Processor) processors to be reserved for the ' + 'LPAR, which can be dynamically configured after activation.') +@optgroup.option('--initial-ziip-processing-weight', type=int, required=False, + # Property: 'initial-ziip-processing-weight' + help='The new initial processing weight for zIIP (z ' + 'Integrated Information Processor) processors (1-999).') +@optgroup.option('--minimum-ziip-processing-weight', type=int, required=False, + # Property: 'minimum-ziip-processing-weight' + help='The new minimum processing weight for zIIP (z ' + 'Integrated Information Processor) processors (0-999).') +@optgroup.option('--maximum-ziip-processing-weight', type=int, required=False, + # Property: 'maximum-ziip-processing-weight' + help='The new maximum processing weight for zIIP (z ' + 'Integrated Information Processor) processors (0-999).') +@optgroup.option('--ziip-processing-weight-capped', type=bool, required=False, + # Property: 'initial-ziip-processing-weight-capped' + help='The new indicator for whether the zIIP (z Integrated ' + 'Information Processor) processing weight is capped. ' + 'If True, the processing weight is an upper limit. If False, ' + 'the processing weight is a target that can be exceeded if ' + 'excess zIIP processor resources are available.') +@optgroup.option('--absolute-ziip-capping', type=float, required=False, + # Property: 'absolute-ziip-capping' -> absolute-capping object + help='The new value for absolute zIIP (z Integrated ' + 'Information Processor) processor capping. A numeric value ' + 'prevents the partition from using any more than the ' + 'specified number of physical zIIP processors. An empty ' + 'string disables absolute zIIP processor capping.') +@optgroup.option('--dedicated-ziip-processors', type=int, required=False, + # Property: 'number-dedicated-ziip-processors' + help='The new number of dedicated zIIP (z Integrated ' + 'Information Processor) processors to be allocated for the ' + 'LPAR\'s exclusive use at activation.') +@optgroup.option('--reserved-dedicated-ziip-processors', type=int, + # Property: 'number-reserved-dedicated-ziip-processors' + required=False, + help='The new number of dedicated zIIP (z Integrated ' + 'Information Processor) processors to be reserved for the ' + 'LPAR, which can be dynamically configured after activation.') +# +@optgroup.option('--shared-icf-processors', type=int, required=False, + # Property: 'number-shared-icf-processors' + help='The new number of shared ICF (Integrated Coupling ' + 'Facility) processors to be allocated to the LPAR at ' + 'activation.') +@optgroup.option('--reserved-shared-icf-processors', type=int, required=False, + # Property: 'number-reserved-shared-icf-processors' + help='The new number of shared ICF (Integrated Coupling ' + 'Facility) processors to be reserved for the ' + 'LPAR, which can be dynamically configured after activation.') +@optgroup.option('--initial-icf-processing-weight', type=int, required=False, + # Property: 'initial-internal-cf-processing-weight' + help='The new initial processing weight for ICF (Integrated ' + 'Coupling Facility) processors (1-999).') +@optgroup.option('--minimum-icf-processing-weight', type=int, required=False, + # Property: 'minimum-internal-cf-processing-weight' + help='The new minimum processing weight for ICF (Integrated ' + 'Coupling Facility) processors (0-999).') +@optgroup.option('--maximum-icf-processing-weight', type=int, required=False, + # Property: 'maximum-internal-cf-processing-weight' + help='The new maximum processing weight for ICF (Integrated ' + 'Coupling Facility) processors (0-999).') +@optgroup.option('--icf-processing-weight-capped', type=bool, required=False, + # Property: 'initial-internal-cf-processing-weight-capped' + help='The new indicator for whether the ICF (Integrated ' + 'Coupling Facility) processing weight is capped. ' + 'If True, the processing weight is an upper limit. If False, ' + 'the processing weight is a target that can be exceeded if ' + 'excess ICF processor resources are available.') +@optgroup.option('--absolute-icf-capping', type=float, required=False, + # Property: 'absolute-internal-cf-capping' + # -> absolute-capping object + help='The new value for absolute ICF (Integrated Coupling ' + 'Facility) processor capping. A numeric value ' + 'prevents the partition from using any more than the ' + 'specified number of physical ICF processors. An empty ' + 'string disables absolute ICF processor capping.') +@optgroup.option('--dedicated-icf-processors', type=int, required=False, + # Property: 'number-dedicated-icf-processors' + help='The new number of dedicated ICF (Integrated Coupling ' + 'Facility) processors to be allocated for the LPAR\'s ' + 'exclusive use at activation.') +@optgroup.option('--reserved-dedicated-icf-processors', type=int, + # Property: 'number-reserved-dedicated-icf-processors' + required=False, + help='The new number of dedicated ICF (Integrated Coupling ' + 'Facility) processors to be reserved for the LPAR, which ' + 'can be dynamically configured after activation.') +# +@optgroup.group('Memory configuration') +@optgroup.option('--central-storage', type=int, required=False, + help='The new amount of central storage (in MiB) to be ' + 'allocated for the LPAR\'s exclusive use at activation.') +@optgroup.option('--reserved-central-storage', type=int, required=False, + help='The new amount of central storage (in MiB) that is ' + 'dynamically reconfigurable to the LPAR after activation.') +@optgroup.option('--expanded-storage', type=int, required=False, + help='The new amount of expanded storage (in MiB) to be ' + 'allocated for the LPAR\'s exclusive use at activation.') +@optgroup.option('--reserved-expanded-storage', type=int, required=False, + help='The new amount of expanded storage (in MiB) that is ' + 'dynamically reconfigurable to the LPAR after activation.') +@optgroup.option('--initial-vfm-storage', type=int, required=False, + help='The new amount of Virtual Flash Memory (VFM) storage ' + '(in GiB) to be allocated for the LPAR\'s exclusive use at ' + 'activation.') +@optgroup.option('--maximum-vfm-storage', type=int, required=False, + help='The new maximum amount of VFM storage (in GiB) that can ' + 'be allocated to the LPAR while it is running.') +@optgroup.option('--central-storage-origin', type=str, required=False, + # Property: 'user-specified-central-storage-origin' (bool), + # 'central-storage-origin' (long) + help='The new user-specified central storage origin address, ' + 'as hexadecimal. An empty string disables the user ' + 'specification of the central storage origin.') +# +@optgroup.group('I/O priority configuration') +@optgroup.option('--minimum-io-priority-queuing', type=int, required=False, + help='The new minimum I/O priority queuing (0-255).') +@optgroup.option('--maximum-io-priority-queuing', type=int, required=False, + help='The new maximum I/O priority queuing (0-255).') +# +@optgroup.group('Special permission configuration') +@optgroup.option('--access-basic-cpu-counter', type=bool, required=False, + # Property: 'basic-cpu-counter-authorization-control' + help='The new indicator for enabling access to the ' + 'basic CPU counter facility for the LPAR.') +@optgroup.option('--access-problem-state-cpu-counter', type=bool, + required=False, + # Property: 'problem-state-cpu-counter-authorization-control' + help='The new indicator for enabling access to the ' + 'problem state CPU counter facility for the LPAR.') +@optgroup.option('--access-crypto-activity-cpu-counter', type=bool, + required=False, + # Property: 'crypto-activity-cpu-counter-authorization-control' + help='The new indicator for enabling access to the ' + 'crypto activity CPU counter facility for the LPAR.') +@optgroup.option('--access-extended-cpu-counter', type=bool, required=False, + # Property: 'extended-cpu-counter-authorization-control' + help='The new indicator for enabling access to the ' + 'extended CPU counter facility for the LPAR.') +@optgroup.option('--access-coprocessor-cpu-counter', type=bool, required=False, + # Property: 'coprocessor-cpu-counter-authorization-control' + help='The new indicator for enabling access to the ' + 'coprocessor group CPU counter facility for the LPAR.') +@optgroup.option('--access-diagnostic-sampling', type=bool, required=False, + # Property: 'diagnostic-sampling-authorization-control' + help='The new indicator for enabling access to the ' + 'diagnostic sampling facility for the LPAR.') +@optgroup.option('--access-basic-cpu-sampling', type=bool, required=False, + # Property: 'basic-cpu-sampling-authorization-control' + help='The new indicator for enabling access to the ' + 'basic CPU sampling facility for the LPAR.') +@optgroup.option('--permit-aes-key-import', type=bool, required=False, + # Property: 'permit-aes-key-import-functions' + help='The new indicator for permitting the ' + 'importing of AES keys for the LPAR.') +@optgroup.option('--permit-des-key-import', type=bool, required=False, + # Property: 'permit-des-key-import-functions' + help='The new indicator for permitting the ' + 'importing of DES keys for the LPAR.') +@optgroup.option('--permit-ecc-key-import', type=bool, required=False, + # Property: 'permit-ecc-key-import-functions' + help='The new indicator for permitting the ' + 'importing of ECC keys for the LPAR.') +@optgroup.option('--access-global-performance-data', type=bool, required=False, + # Property: 'global-performance-data-authorization-control' + help='The new indicator for enabling access to the processor ' + 'activity data for all other LPARs on the same CPC.') +@optgroup.option('--permit-global-iocds', type=bool, required=False, + # Property: 'io-configuration-authorization-control' + help='The new indicator for permitting to read and write any ' + 'IOCDS in the configuration.') +@optgroup.option('--permit-cross-partition-authority', type=bool, + required=False, + # Property: 'cross-partition-authority-authorization-control' + help='The new indicator for permitting to issue control ' + 'program instructions that reset or deactivate other LPARs.') +@optgroup.option('--permit-bcpii-send-commands', type=bool, required=False, + # Property: 'security-bcpii-send-commands' + help='The new indicator for whether the LPAR is permitted to ' + 'send commands through BCPii.') +@optgroup.option('--permit-bcpii-receive-commands', required=False, + # Property: 'security-bcpii-receive-commands', (bool) + # 'security-bcpii-receive-selection' (enum) + type=click.Choice(['none', 'all', 'bcpii-list']), + help='The new indicator that determines whether and which ' + 'commands the LPAR is permitted to receive through BCPii:\n' + '\n' + '\b\n' + ' - "none" – (default) No BCPii commands are allowed.\n' + ' - "all" – all BCPii commands are allowed.\n' + ' - "bcpii-list" – only the BCPii commands in the list are ' + 'allowed.\n') +@optgroup.option('--bcpii-receive-partitions', type=str, required=False, + # Property: 'security-bcpii-receive-partition-list' + # -> add to list + help='The new list of LPARs from which the LPAR activated by ' + 'this profile is allowed to receive commands through BCPii ' + '(if generally permitted). The LPARs in the string are ' + 'comma-separated, and each LPAR is referenced in the format ' + '"NETID.SYSTEM.LPAR".') +@optgroup.group('Clock configuration') +@optgroup.option('--clock-type', required=False, + type=click.Choice(['standard', 'lpar']), + help='The new clock type for the LPAR: ' + '"standard" - Use the CPC\'s time source. ' + '"lpar" - Use an offset from the External Time Source.') +@optgroup.option('--time-offset-direction', required=False, + # Property: 'time-offset-increase-decrease' + type=click.Choice(['increase', 'decrease']), + help='The new clock type for the LPAR:\n' + '\n' + '\b\n' + ' - "increase" - LPAR clock is ahead of the External ' + 'Time Source.\n' + ' - "decrease" - LPAR clock is behind the External ' + 'Time Source.\n') +@optgroup.option('--time-offset-days', type=int, required=False, + help='The new number of days the LPAR\'s clock ' + 'is to be offset from the External Time Source.') +@optgroup.option('--time-offset-hours', type=int, required=False, + help='The new number of hours the LPAR\'s clock ' + 'is to be offset from the External Time Source.') +@optgroup.option('--time-offset-minutes', type=int, required=False, + help='The new number of minutes the LPAR\'s ' + 'clock is to be offset from the External Time Source.') +@optgroup.group('zAware configuration') +@optgroup.option('--zaware-host-name', type=str, required=False, + help='The new hostname for IBM zAware. ' + 'Empty string sets no hostname. ' + '(only for LPARs in zaware activation mode).') +@optgroup.option('--zaware-master-userid', type=str, required=False, + help='The new master userid for IBM zAware. ' + 'Empty string sets no master userid. ' + '(only for LPARs in zaware activation mode).') +@optgroup.option('--zaware-master-password', type=str, required=False, + help='The new master password for IBM zAware. ' + 'Empty string sets no master password. ' + '(only for LPARs in zaware activation mode).') +# TODO: Support for 'zaware-network-info' property +# TODO: Support for 'zaware-gateway-info' property +# TODO: Support for 'zaware-dns-info' property +@optgroup.group('SSC configuration (only applicable to SSC partitions)') +@optgroup.option('--ssc-host-name', type=str, required=False, + help='The new hostname for the SSC appliance. ' + 'Empty string sets no hostname.') +@optgroup.option('--ssc-master-userid', type=str, required=False, + help='The new master userid for the SSC appliance. ' + 'Empty string sets no master userid.') +@optgroup.option('--ssc-master-password', type=str, required=False, + help='The new master password for the SSC appliance. ' + 'Empty string sets no master password.') +@optgroup.option('--ssc-boot-selection', required=False, + type=click.Choice(['installer', 'appliance']), + help='The new boot selection for the SSC LPAR.') +# TODO: Support for 'ssc-network-info' property +# TODO: Support for 'ssc-gateway-info' property +# TODO: Support for 'ssc-gateway-ipv6-info' property +# TODO: Support for 'ssc-dns-info' property +# TODO: SEPARATE assigned-crypto-domains Array of assigned-crypto-domain +# objects - Specifies the crypto domains and their access modes to be +# assigned to the LPAR once activated. +# TODO: SEPARATE assigned-cryptos — Array of assigned-crypto objects - Specifies +# the crypto adapters to be assigned to the LPAR once activated. +# TODO: SEPARATE assigned-certificate-uris (c)(pc) Array of String/ URI - Array +# of URIs referring to the certificates that are assigned to this image +# activation profile, or an empty array if there are no assigned +# certificates. +@click.pass_obj +def imageprofile_update(cmd_ctx, cpc, imageprofile, **options): + """ + Update the properties of an image activation profile. + + Only the properties will be changed for which a corresponding option is + specified, so the default for all options is not to change properties. + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + + \b + Limitations: + * The processor capping/sharing/weight related properties cannot be + updated. + * The network-related properties for zaware and ssc cannot be updated. + * The --zaware-master-password and --ssc-master-password options do not + ask for the password. + """ + cmd_ctx.execute_cmd( + lambda: cmd_imageprofile_update(cmd_ctx, cpc, imageprofile, options)) + + +@imageprofile_group.command('create', options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='CPC') +@optgroup.group('General options') +@optgroup.option('--name', type=str, required=False, + help='The name for the new image activation profile.') +@optgroup.option('--copy-name', type=str, required=False, + help='The name of an existing image activation profile whose ' + 'properties are used to initialize the new profile before ' + 'any other options update them. If this option is not used, ' + 'the DEFAULT image profile will be used to initialize the ' + 'new profile.') +@optgroup.option('--description', type=str, required=False, + help='The description for the new image activation profile.') +@optgroup.option('--operating-mode', type=str, metavar='OPMODE', required=False, + help='The operating mode for the LPAR:\n' + '\n' + '\b\n' + ' - "general" - TBD (on z14 or later)\n' + ' - "coupling-facility" - TBD\n' + ' - "linux-only" - TBD\n' + ' - "zvm" - TBD\n' + ' - "ssc" - TBD (on z13 or later)\n' + 'Note that only this subset of operating modes can be used ' + 'when creating image activation profiles.') +@optgroup.option('--load-at-activation', type=bool, required=False, + help='The indicator for whether the LPAR will be loaded ' + 'at the end of the activation.') +@optgroup.option('--partition-id', type=str, required=False, + # Property: 'partition-identifier' + help='The partition identifier (two-digit hexadecimal) to ' + 'be used for the LPAR.') +@optgroup.option('--local-cluster-name', type=str, required=False, + help='The name of the CP management cluster for the LPAR.') +@optgroup.option('--group-profile', type=str, required=False, + # Property: 'group-profile-uri' (name -> URI) + help='The name of the Group profile to be associated with ' + 'this profile. An empty string will cause no Group profile ' + 'to be associated with this profile. ' + 'An associated Group profile provides the group capacity ' + 'value for the LPAR.') +@optgroup.option('--liccc-validation-enabled', type=bool, required=False, + help='The indicator for validating that the image profile ' + 'data conforms to the current maximum LICCC configuration.') +@optgroup.option('--lpar-isolation', type=bool, required=False, + # Property: 'logical-partition-isolation-control' + help='The LPAR isolation indicator. When True, ' + 'reconfigurable channel paths assigned to the LPAR are ' + 'reserved for its exclusive use.') +@optgroup.group('Boot configuration') +@optgroup.option('--ipl-type', type=str, metavar='IPLTYPE', required=False, + help='The IPL type for the operating system. The IPL type ' + 'defines from where the control program for the LPAR will be ' + 'loaded:\n' + '\n' + '\b\n' + ' - "ipltype-scsi" - SCSI list-directed OS load\n' + ' - "ipltype-scsidump" - SCSI list-directed OS dump\n' + ' - "ipltype-nvmeload" – NVMe list-directed OS load ' + '(z15 or later)\n' + ' - "ipltype-nvmedump" – NVMe list-directed OS dump ' + '(z15 or later)\n' + ' - "ipltype-tape-load" - Tape Channel Command Word (CCW) ' + 'OS load (z16 or later)\n' + ' - "ipltype-tape-dump" - Tape Channel Command Word (CCW) ' + 'OS dump (z16 or later)\n' + ' - "ipltype-eckd-ccw-load" - ECKD Channel Command Word ' + '(CCW) OS load (z16 or later)\n' + ' - "ipltype-eckd-ccw-dump" - ECKD Channel Command Word ' + '(CCW) OS dump (z16 or later)\n' + ' - "ipltype-eckd-ld-load" - ECKD list-directed OS load ' + '(z16 or later)\n' + ' - "ipltype-eckd-ld-dump" - ECKD list-directed OS dump ' + '(z16 or later)\n' + ' - "ipltype-standard" - Channel Command Word (CCW) ' + 'standard load. This is present for\n' + ' compatibility and will be interpreted as one of the ' + 'new values.\n') +@optgroup.option('--ipl-address', type=str, required=False, + help='The IPL address. ' + 'For NVMe IPL types, this is the FID of the NVMe device ' + 'as 4 hex characters in any lexical case. ' + 'For any other IPL types, this is the device number of the ' + 'boot device as 4 or 5 hex characters in any lexical case. ' + 'If empty, the IPL address will be taken from the load ' + 'activation profile.') +@optgroup.option('--load-parameter', type=str, required=False, + # Property: 'ipl-parameter' + help='The load parameter. This is a short (0-8 characters) ' + 'string passed to the control program. Valid characters are ' + '"0"-"9", "A"-"Z", blank (" ") and period ("."). On z14 and ' + 'later, the characters "@", "$", and "#" are valid in ' + 'addition.') +@optgroup.option('--load-parameters', type=str, required=False, + # Property: 'os-specific-load-parameters' + help='The operating system-specific load parameters. This ' + 'is a longer string string passed to the control program. ' + 'Only used for the list-directed IPL types "ipltype-scsi", ' + '"ipltype-scsidump", "ipltype-nvmeload", "ipltype-nvmedump", ' + '"ipltype-eckd-ld-load", or "ipltype-eckd-ld-dump".') +@optgroup.option('--load-timeout', type=int, required=False, + help='The amount of time, in seconds, to wait for the ' + 'Load to complete (60-600).') +@optgroup.option('--store-status-indicator', type=bool, required=False, + help='The store-status indicator. If True, the status is ' + 'stored before performing the Load.') +@optgroup.option('--secure-boot', type=bool, required=False, + help='The secure-boot indicator. It indicates whether the ' + 'software signature of the control program will be verified ' + 'using the secure-boot certificate(s) assigned to the LPAR. ' + 'Only used for the list-directed IPL types "ipltype-scsi", ' + '"ipltype-scsidump", "ipltype-nvmeload", "ipltype-nvmedump", ' + '"ipltype-eckd-ld-load", or "ipltype-eckd-ld-dump".') +@optgroup.option('--disk-partition-id', type=str, required=False, + # Property: 'disk-partition-id' + 'disk-partition-id-automatic' + help='The disk partition number (also called the boot ' + 'program selector), in decimal. The special value "auto" ' + '(only valid on z16 or later) causes the disk partition ' + 'number to be determined automatically. ' + 'Only used for the list-directed IPL types "ipltype-scsi", ' + '"ipltype-scsidump", "ipltype-nvmeload", "ipltype-nvmedump", ' + '"ipltype-eckd-ld-load", or "ipltype-eckd-ld-dump".') +@optgroup.option('--worldwide-port-name', type=str, required=False, + help='The worldwide port name (WWPN) of the SCSI boot ' + 'device, in hexadecimal. Only used for the SCSI IPL types ' + '"ipltype-scsi" or "ipltype-scsidump".') +@optgroup.option('--logical-unit-number', type=str, required=False, + help='The logical unit number of the SCSI boot device, ' + 'in hexadecimal. Only used for the SCSI IPL types ' + '"ipltype-scsi" or "ipltype-scsidump".') +@optgroup.option('--boot-record-lba', type=str, required=False, + help='The boot record logical block address the SCSI/NVMe ' + 'boot device, in hexadecimal. Only used for SCSI and NVMe IPL ' + 'types "ipltype-scsi", "ipltype-scsidump", ' + '"ipltype-nvmeload", or "ipltype-nvmedump".') +@optgroup.option('--boot-record-location-cylinder', type=str, required=False, + help='The boot record location cylinder value, in ' + 'hexadecimal. Only used for ECKD IPL types ' + '"ipltype-eckd-ld-load" or "ipltype-eckd-ld-dump".') +@optgroup.option('--boot-record-location-head', type=str, required=False, + help='The boot record location head value, in ' + 'hexadecimal. Only used for ECKD IPL types ' + '"ipltype-eckd-ld-load" or "ipltype-eckd-ld-dump".') +@optgroup.option('--boot-record-location-record', type=str, required=False, + help='The boot record location record value, in ' + 'hexadecimal. Only used for ECKD IPL types ' + '"ipltype-eckd-ld-load" or "ipltype-eckd-ld-dump".') +@optgroup.option('--boot-record-location-use-volume-label', type=bool, + required=False, + help='The indicator whether the boot record location ' + 'cylinder, head, and record should be determined by the ' + 'volume label.') +@optgroup.group('CPU configuration') +@optgroup.option('--processor-usage', + type=click.Choice(['dedicated', 'shared']), required=False, + help='The indicator for how processors are allocated to ' + 'the LPAR:\n' + '\n' + '\b\n' + ' - "dedicated" - The processors are exclusively available ' + 'to this LPAR.\n' + ' - "shared" - The processors are shareable across LPARs.\n') +@optgroup.option('--wlm-enabled', type=bool, required=False, + # Property: 'workload-manager-enabled' + help='The indicator for enabling z/OS Workload Manager. ' + 'If True, z/OS Workload Manager is allowed to change ' + 'processing weight related properties of the LPAR after ' + 'activation.') +@optgroup.option('--defined-capacity', type=bool, required=False, + # Property: 'defined-capacity' + help='The defined capacity of the LPAR ' + '(in MSU/h). This specifies how much capacity the LPAR is to ' + 'be managed to by z/OS Workload Manager for the purpose of ' + 'software pricing. 0 means that no defined capacity is ' + 'specified for this LPAR.') +@optgroup.option('--shared-cp-processors', type=int, required=False, + # Property: 'number-shared-general-purpose-processors' + help='The number of shared CP (general purpose) ' + 'processors to be allocated to the LPAR at activation.') +@optgroup.option('--reserved-shared-cp-processors', type=int, required=False, + # Property: 'number-reserved-shared-general-purpose-processors' + help='The number of shared CP (general purpose) ' + 'processors to be reserved for the LPAR, which can be ' + 'dynamically configured after activation.') +@optgroup.option('--initial-cp-processing-weight', type=int, required=False, + # Property: 'initial-processing-weight' + help='The initial processing weight for CP (general ' + 'purpose) processors (1-999).') +@optgroup.option('--minimum-cp-processing-weight', type=int, required=False, + # Property: 'minimum-processing-weight' + help='The minimum processing weight for CP (general ' + 'purpose) processors (0-999).') +@optgroup.option('--maximum-cp-processing-weight', type=int, required=False, + # Property: 'maximum-processing-weight' + help='The maximum processing weight for CP (general ' + 'purpose) processors (0-999).') +@optgroup.option('--cp-processing-weight-capped', type=bool, required=False, + # Property: 'initial-processing-weight-capped' + help='The indicator for whether the CP (general purpose) ' + 'processing weight is capped. If True, the processing weight ' + 'is an upper limit. If False, the processing weight is a ' + 'target that can be exceeded if excess CP processor resources ' + 'are available.') +@optgroup.option('--absolute-cp-capping', type=float, required=False, + # Property: 'absolute-general-purpose-capping' + # -> absolute-capping object + help='The value for absolute CP (general purpose) ' + 'processor capping. A numeric value prevents the partition ' + 'from using any more than the specified number of physical ' + 'CP processors. An empty string disables absolute CP ' + 'processor capping.') +@optgroup.option('--dedicated-cp-processors', type=int, required=False, + # Property: 'number-dedicated-general-purpose-processors' + help='The number of dedicated CP (general purpose) ' + 'processors to be allocated for the LPAR\'s exclusive use ' + 'at activation.') +@optgroup.option('--reserved-dedicated-cp-processors', type=int, required=False, + # Prop: 'number-reserved-dedicated-general-purpose-processors' + help='The number of dedicated CP (general purpose) ' + 'processors to be reserved for the LPAR, which can be ' + 'dynamically configured after activation.') +# +@optgroup.option('--shared-ifl-processors', type=int, required=False, + # Property: 'number-shared-ifl-processors' + help='The number of shared IFL (Integrated Facility for ' + 'Linux) processors to be allocated to the LPAR at activation.') +@optgroup.option('--reserved-shared-ifl-processors', type=int, required=False, + # Property: 'number-reserved-shared-ifl-processors' + help='The number of shared IFL (Integrated Facility for ' + 'Linux) processors to be reserved for the LPAR, which can be ' + 'dynamically configured after activation.') +@optgroup.option('--initial-ifl-processing-weight', type=int, required=False, + # Property: 'initial-ifl-processing-weight' + help='The initial processing weight for IFL (Integrated ' + 'Facility for Linux) processors (1-999).') +@optgroup.option('--minimum-ifl-processing-weight', type=int, required=False, + # Property: 'minimum-ifl-processing-weight' + help='The minimum processing weight for IFL (Integrated ' + 'Facility for Linux) processors (0-999).') +@optgroup.option('--maximum-ifl-processing-weight', type=int, required=False, + # Property: 'maximum-ifl-processing-weight' + help='The maximum processing weight for IFL (Integrated ' + 'Facility for Linux) processors (0-999).') +@optgroup.option('--ifl-processing-weight-capped', type=bool, required=False, + # Property: 'initial-ifl-processing-weight-capped' + help='The indicator for whether the IFL (Integrated ' + 'Facility for Linux) processing weight is capped. If True, ' + 'the processing weight is an upper limit. If False, the ' + 'processing weight is a target that can be exceeded if ' + 'excess IFL processor resources are available.') +@optgroup.option('--absolute-ifl-capping', type=float, required=False, + # Property: 'absolute-ifl-capping' -> absolute-capping object + help='The value for absolute IFL (Integrated Facility for ' + 'Linux) processor capping. A numeric value prevents the ' + 'partition from using any more than the specified number of ' + 'physical IFL processors. An empty string disables absolute ' + 'IFL processor capping.') +@optgroup.option('--dedicated-ifl-processors', type=int, required=False, + # Property: 'number-dedicated-ifl-processors' + help='The number of dedicated IFL (Integrated Facility ' + 'for Linux) processors to be allocated for the LPAR\'s ' + 'exclusive use at activation.') +@optgroup.option('--reserved-dedicated-ifl-processors', type=int, + required=False, + # Property: 'number-reserved-dedicated-ifl-processors' + help='The number of dedicated IFL (Integrated Facility ' + 'for Linux) processors to be reserved for the LPAR, which ' + 'can be dynamically configured after activation.') +# +@optgroup.option('--shared-zaap-processors', type=int, required=False, + # Property: 'number-shared-aap-processors' + help='The number of shared zAAP (z Application Assist ' + 'Processor) processors to be allocated to the LPAR at ' + 'activation.') +@optgroup.option('--reserved-shared-zaap-processors', type=int, required=False, + # Property: 'number-reserved-shared-aap-processors' + help='The number of shared zAAP (z Application Assist ' + 'Processor) processors to be reserved for the LPAR, which can ' + 'be dynamically configured after activation.') +@optgroup.option('--initial-zaap-processing-weight', type=int, required=False, + # Property: 'initial-aap-processing-weight' + help='The initial processing weight for zAAP (z ' + 'Application Assist Processor) processors (1-999).') +@optgroup.option('--minimum-zaap-processing-weight', type=int, required=False, + # Property: 'minimum-aap-processing-weight' + help='The minimum processing weight for zAAP (z ' + 'Application Assist Processor) processors (0-999).') +@optgroup.option('--maximum-zaap-processing-weight', type=int, required=False, + # Property: 'maximum-aap-processing-weight' + help='The maximum processing weight for zAAP (z ' + 'Application Assist Processor) processors (0-999).') +@optgroup.option('--zaap-processing-weight-capped', type=bool, required=False, + # Property: 'initial-aap-processing-weight-capped' + help='The indicator for whether the zAAP (z ' + 'Application Assist Processor) processing weight is capped. ' + 'If True, the processing weight is an upper limit. If False, ' + 'the processing weight is a target that can be exceeded if ' + 'excess zAAP processor resources are available.') +@optgroup.option('--absolute-zaap-capping', type=float, required=False, + # Property: 'absolute-aap-capping' -> absolute-capping object + help='The value for absolute zAAP (z Application Assist ' + 'Processor) processor capping. A numeric value prevents the ' + 'partition from using any more than the specified number of ' + 'physical zAAP processors. An empty string disables absolute ' + 'zAAP processor capping.') +@optgroup.option('--dedicated-zaap-processors', type=int, required=False, + # Property: 'number-dedicated-aap-processors' + help='The number of dedicated zAAP (z Application Assist ' + 'Processor) processors to be allocated for the LPAR\'s ' + 'exclusive use at activation.') +@optgroup.option('--reserved-dedicated-zaap-processors', type=int, + # Property: 'number-reserved-dedicated-aap-processors' + required=False, + help='The number of dedicated zAAP (z Application Assist ' + 'Processor) processors to be reserved for the LPAR, which ' + 'can be dynamically configured after activation.') +# +@optgroup.option('--shared-ziip-processors', type=int, required=False, + # Property: 'number-shared-ziip-processors' + help='The number of shared zIIP (z Integrated ' + 'Information Processor) processors to be allocated to the ' + 'LPAR at activation.') +@optgroup.option('--reserved-shared-ziip-processors', type=int, required=False, + # Property: 'number-reserved-shared-ziip-processors' + help='The number of shared zIIP (z Integrated ' + 'Information Processor) processors to be reserved for the ' + 'LPAR, which can be dynamically configured after activation.') +@optgroup.option('--initial-ziip-processing-weight', type=int, required=False, + # Property: 'initial-ziip-processing-weight' + help='The initial processing weight for zIIP (z ' + 'Integrated Information Processor) processors (1-999).') +@optgroup.option('--minimum-ziip-processing-weight', type=int, required=False, + # Property: 'minimum-ziip-processing-weight' + help='The minimum processing weight for zIIP (z ' + 'Integrated Information Processor) processors (0-999).') +@optgroup.option('--maximum-ziip-processing-weight', type=int, required=False, + # Property: 'maximum-ziip-processing-weight' + help='The maximum processing weight for zIIP (z ' + 'Integrated Information Processor) processors (0-999).') +@optgroup.option('--ziip-processing-weight-capped', type=bool, required=False, + # Property: 'initial-ziip-processing-weight-capped' + help='The indicator for whether the zIIP (z Integrated ' + 'Information Processor) processing weight is capped. ' + 'If True, the processing weight is an upper limit. If False, ' + 'the processing weight is a target that can be exceeded if ' + 'excess zIIP processor resources are available.') +@optgroup.option('--absolute-ziip-capping', type=float, required=False, + # Property: 'absolute-ziip-capping' -> absolute-capping object + help='The value for absolute zIIP (z Integrated ' + 'Information Processor) processor capping. A numeric value ' + 'prevents the partition from using any more than the ' + 'specified number of physical zIIP processors. An empty ' + 'string disables absolute zIIP processor capping.') +@optgroup.option('--dedicated-ziip-processors', type=int, required=False, + # Property: 'number-dedicated-ziip-processors' + help='The number of dedicated zIIP (z Integrated ' + 'Information Processor) processors to be allocated for the ' + 'LPAR\'s exclusive use at activation.') +@optgroup.option('--reserved-dedicated-ziip-processors', type=int, + # Property: 'number-reserved-dedicated-ziip-processors' + required=False, + help='The number of dedicated zIIP (z Integrated ' + 'Information Processor) processors to be reserved for the ' + 'LPAR, which can be dynamically configured after activation.') +# +@optgroup.option('--shared-icf-processors', type=int, required=False, + # Property: 'number-shared-icf-processors' + help='The number of shared ICF (Integrated Coupling ' + 'Facility) processors to be allocated to the LPAR at ' + 'activation.') +@optgroup.option('--reserved-shared-icf-processors', type=int, required=False, + # Property: 'number-reserved-shared-icf-processors' + help='The number of shared ICF (Integrated Coupling ' + 'Facility) processors to be reserved for the ' + 'LPAR, which can be dynamically configured after activation.') +@optgroup.option('--initial-icf-processing-weight', type=int, required=False, + # Property: 'initial-internal-cf-processing-weight' + help='The initial processing weight for ICF (Integrated ' + 'Coupling Facility) processors (1-999).') +@optgroup.option('--minimum-icf-processing-weight', type=int, required=False, + # Property: 'minimum-internal-cf-processing-weight' + help='The minimum processing weight for ICF (Integrated ' + 'Coupling Facility) processors (0-999).') +@optgroup.option('--maximum-icf-processing-weight', type=int, required=False, + # Property: 'maximum-internal-cf-processing-weight' + help='The maximum processing weight for ICF (Integrated ' + 'Coupling Facility) processors (0-999).') +@optgroup.option('--icf-processing-weight-capped', type=bool, required=False, + # Property: 'initial-internal-cf-processing-weight-capped' + help='The indicator for whether the ICF (Integrated ' + 'Coupling Facility) processing weight is capped. ' + 'If True, the processing weight is an upper limit. If False, ' + 'the processing weight is a target that can be exceeded if ' + 'excess ICF processor resources are available.') +@optgroup.option('--absolute-icf-capping', type=float, required=False, + # Property: 'absolute-internal-cf-capping' + # -> absolute-capping object + help='The value for absolute ICF (Integrated Coupling ' + 'Facility) processor capping. A numeric value ' + 'prevents the partition from using any more than the ' + 'specified number of physical ICF processors. An empty ' + 'string disables absolute ICF processor capping.') +@optgroup.option('--dedicated-icf-processors', type=int, required=False, + # Property: 'number-dedicated-icf-processors' + help='The number of dedicated ICF (Integrated Coupling ' + 'Facility) processors to be allocated for the LPAR\'s ' + 'exclusive use at activation.') +@optgroup.option('--reserved-dedicated-icf-processors', type=int, + # Property: 'number-reserved-dedicated-icf-processors' + required=False, + help='The number of dedicated ICF (Integrated Coupling ' + 'Facility) processors to be reserved for the LPAR, which ' + 'can be dynamically configured after activation.') +# +@optgroup.group('Memory configuration') +@optgroup.option('--central-storage', type=int, required=False, + help='New amount of central storage (in MiB) to be allocated ' + 'for the LPAR\'s exclusive use at activation.') +@optgroup.option('--reserved-central-storage', type=int, required=False, + help='New amount of central storage (in MiB) that is ' + 'dynamically reconfigurable to the LPAR after activation.') +@optgroup.option('--expanded-storage', type=int, required=False, + help='New amount of expanded storage (in MiB) to be allocated ' + 'for the LPAR\'s exclusive use at activation.') +@optgroup.option('--reserved-expanded-storage', type=int, required=False, + help='New amount of expanded storage (in MiB) that is ' + 'dynamically reconfigurable to the LPAR after activation.') +@optgroup.option('--initial-vfm-storage', type=int, required=False, + help='New amount of Virtual Flash Memory (VFM) storage ' + '(in GiB) to be allocated for the LPAR\'s exclusive use at ' + 'activation.') +@optgroup.option('--maximum-vfm-storage', type=int, required=False, + help='New maximum amount of VFM storage (in GiB) that can be ' + 'allocated to the LPAR while it is running.') +@optgroup.option('--central-storage-origin', type=str, required=False, + # Property: 'user-specified-central-storage-origin' (bool), + # 'central-storage-origin' (long) + help='New user-specified central storage origin address, as ' + 'hexadecimal. An empty string disables the user specification ' + 'of the central storage origin.') +# +@optgroup.group('I/O priority configuration') +@optgroup.option('--minimum-io-priority-queuing', type=int, required=False, + help='The minimum I/O priority queuing (0-255).') +@optgroup.option('--maximum-io-priority-queuing', type=int, required=False, + help='The maximum I/O priority queuing (0-255).') +# +@optgroup.group('Special permission configuration') +@optgroup.option('--access-basic-cpu-counter', type=bool, required=False, + # Property: 'basic-cpu-counter-authorization-control' + help='The indicator for enabling access to the ' + 'basic CPU counter facility for the LPAR.') +@optgroup.option('--access-problem-state-cpu-counter', type=bool, + required=False, + # Property: 'problem-state-cpu-counter-authorization-control' + help='The indicator for enabling access to the ' + 'problem state CPU counter facility for the LPAR.') +@optgroup.option('--access-crypto-activity-cpu-counter', type=bool, + required=False, + # Property: 'crypto-activity-cpu-counter-authorization-control' + help='The indicator for enabling access to the ' + 'crypto activity CPU counter facility for the LPAR.') +@optgroup.option('--access-extended-cpu-counter', type=bool, required=False, + # Property: 'extended-cpu-counter-authorization-control' + help='The indicator for enabling access to the ' + 'extended CPU counter facility for the LPAR.') +@optgroup.option('--access-coprocessor-cpu-counter', type=bool, required=False, + # Property: 'coprocessor-cpu-counter-authorization-control' + help='The indicator for enabling access to the ' + 'coprocessor group CPU counter facility for the LPAR.') +@optgroup.option('--access-diagnostic-sampling', type=bool, required=False, + # Property: 'diagnostic-sampling-authorization-control' + help='The indicator for enabling access to the ' + 'diagnostic sampling facility for the LPAR.') +@optgroup.option('--access-basic-cpu-sampling', type=bool, required=False, + # Property: 'basic-cpu-sampling-authorization-control' + help='The indicator for enabling access to the ' + 'basic CPU sampling facility for the LPAR.') +@optgroup.option('--permit-aes-key-import', type=bool, required=False, + # Property: 'permit-aes-key-import-functions' + help='The indicator for permitting the ' + 'importing of AES keys for the LPAR.') +@optgroup.option('--permit-des-key-import', type=bool, required=False, + # Property: 'permit-des-key-import-functions' + help='The indicator for permitting the ' + 'importing of DES keys for the LPAR.') +@optgroup.option('--permit-ecc-key-import', type=bool, required=False, + # Property: 'permit-ecc-key-import-functions' + help='The indicator for permitting the ' + 'importing of ECC keys for the LPAR.') +@optgroup.option('--access-global-performance-data', type=bool, required=False, + # Property: 'global-performance-data-authorization-control' + help='The indicator for enabling access to the processor ' + 'activity data for all other LPARs on the same CPC.') +@optgroup.option('--permit-global-iocds', type=bool, required=False, + # Property: 'io-configuration-authorization-control' + help='The indicator for permitting to read and write any ' + 'IOCDS in the configuration.') +@optgroup.option('--permit-cross-partition-authority', type=bool, + required=False, + # Property: 'cross-partition-authority-authorization-control' + help='The indicator for permitting to issue control ' + 'program instructions that reset or deactivate other LPARs.') +@optgroup.option('--permit-bcpii-send-commands', type=bool, required=False, + # Property: 'security-bcpii-send-commands' + help='Determines whether the LPAR is permitted to send ' + 'commands through BCPii.') +@optgroup.option('--permit-bcpii-receive-commands', required=False, + # Property: 'security-bcpii-receive-commands', (bool) + # 'security-bcpii-receive-selection' (enum) + type=click.Choice(['none', 'all', 'bcpii-list']), + help='Determines whether and which commands the LPAR is ' + 'permitted to receive through BCPii:\n' + '\n' + '\b\n' + ' - "none" – (default) No BCPii commands are allowed.\n' + ' - "all" – all BCPii commands are allowed.\n' + ' - "bcpii-list" – only the BCPii commands in the list are ' + 'allowed.\n') +@optgroup.option('--bcpii-receive-partitions', type=str, required=False, + # Property: 'security-bcpii-receive-partition-list' + # -> add to list + help='Comma-separated list of LPARs from which the LPAR ' + 'activated by this profile is allowed to receive commands ' + 'through BCPii (if generally permitted). ' + 'Each LPAR is referenced in the format "NETID.SYSTEM.LPAR".') +@optgroup.group('Clock configuration') +@optgroup.option('--clock-type', required=False, + type=click.Choice(['standard', 'lpar']), + help='The clock type for the LPAR: ' + '"standard" - Use the CPC\'s time source. ' + '"lpar" - Use an offset from the External Time Source.') +@optgroup.option('--time-offset-direction', required=False, + # Property: 'time-offset-increase-decrease' + type=click.Choice(['increase', 'decrease']), + help='The clock type for the LPAR:\n' + '\n' + '\b\n' + ' - "increase" - LPAR clock is ahead of the External ' + 'Time Source.\n' + ' - "decrease" - LPAR clock is behind the External ' + 'Time Source.\n') +@optgroup.option('--time-offset-days', type=int, required=False, + help='The number of days the LPAR\'s clock ' + 'is to be offset from the External Time Source.') +@optgroup.option('--time-offset-hours', type=int, required=False, + help='The number of hours the LPAR\'s clock ' + 'is to be offset from the External Time Source.') +@optgroup.option('--time-offset-minutes', type=int, required=False, + help='The number of minutes the LPAR\'s ' + 'clock is to be offset from the External Time Source.') +@optgroup.group('zAware configuration') +@optgroup.option('--zaware-host-name', type=str, required=False, + help='The hostname for IBM zAware. ' + 'Empty string sets no hostname. ' + '(only for LPARs in zaware activation mode).') +@optgroup.option('--zaware-master-userid', type=str, required=False, + help='The master userid for IBM zAware. ' + 'Empty string sets no master userid. ' + '(only for LPARs in zaware activation mode).') +@optgroup.option('--zaware-master-password', type=str, required=False, + help='The master password for IBM zAware. ' + 'Empty string sets no master password. ' + '(only for LPARs in zaware activation mode).') +# TODO: Support for 'zaware-network-info' property +# TODO: Support for 'zaware-gateway-info' property +# TODO: Support for 'zaware-dns-info' property +@optgroup.group('SSC configuration (only applicable to SSC partitions)') +@optgroup.option('--ssc-host-name', type=str, required=False, + help='The hostname for the SSC appliance. ' + 'Empty string sets no hostname.') +@optgroup.option('--ssc-master-userid', type=str, required=False, + help='The master userid for the SSC appliance. ' + 'Empty string sets no master userid.') +@optgroup.option('--ssc-master-password', type=str, required=False, + help='The master password for the SSC appliance. ' + 'Empty string sets no master password.') +@optgroup.option('--ssc-boot-selection', required=False, + type=click.Choice(['installer', 'appliance']), + help='The boot selection for the SSC LPAR.') +# TODO: Support for 'ssc-network-info' property +# TODO: Support for 'ssc-gateway-info' property +# TODO: Support for 'ssc-gateway-ipv6-info' property +# TODO: Support for 'ssc-dns-info' property +# TODO: SEPARATE assigned-crypto-domains Array of assigned-crypto-domain +# objects - Specifies the crypto domains and their access modes to be +# assigned to the LPAR once activated. +# TODO: SEPARATE assigned-cryptos — Array of assigned-crypto objects - Specifies +# the crypto adapters to be assigned to the LPAR once activated. +# TODO: SEPARATE assigned-certificate-uris (c)(pc) Array of String/ URI - Array +# of URIs referring to the certificates that are assigned to this image +# activation profile, or an empty array if there are no assigned +# certificates. +@click.pass_obj +def imageprofile_create(cmd_ctx, cpc, **options): + """ + Create an image activation profile in a CPC (Only HMC 2.16 and later). + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + cmd_ctx.execute_cmd(lambda: cmd_imageprofile_create(cmd_ctx, cpc, options)) + + +@imageprofile_group.command('delete', options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='CPC') +@click.argument('imageprofile', type=str, metavar='IMAGEPROFILE') +@click.option('-y', '--yes', is_flag=True, callback=abort_if_false, + expose_value=False, + help='Skip prompt to confirm deletion of the image activation ' + 'profile.', + prompt='Are you sure you want to delete this image activation ' + 'profile ?') +@click.pass_obj +def imageprofile_delete(cmd_ctx, cpc, imageprofile): + """ + Delete an image activation profile (Only HMC 2.16 and later). + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + cmd_ctx.execute_cmd( + lambda: cmd_imageprofile_delete(cmd_ctx, cpc, imageprofile)) + + +@imageprofile_group.command('assign-certificate', + options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='CPC') +@click.argument('imageprofile', type=str, metavar='IMAGEPROFILE') +@click.option('--certificate', type=str, required=True, + help='The name of the certificate.') +@click.pass_obj +def imageprofile_assign_certificate(cmd_ctx, cpc, imageprofile, **options): + """ + Assign a certificate to an image activation profile (Only HMC 2.16 and + later). + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + cmd_ctx.execute_cmd( + lambda: cmd_imageprofile_assign_certificate( + cmd_ctx, cpc, imageprofile, options)) + + +@imageprofile_group.command('unassign-certificate', + options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='CPC') +@click.argument('imageprofile', type=str, metavar='IMAGEPROFILE') +@click.option('--certificate', type=str, required=True, + help='The name of the certificate.') +@click.pass_obj +def imageprofile_unassign_certificate(cmd_ctx, cpc, imageprofile, **options): + """ + Unassign a certificate from an image activation profile (Only HMC 2.16 and + later). + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + cmd_ctx.execute_cmd( + lambda: cmd_imageprofile_unassign_certificate( + cmd_ctx, cpc, imageprofile, options)) + + +def cmd_imageprofile_list(cmd_ctx, cpc_name, options): + # pylint: disable=missing-function-docstring + + client = zhmcclient.Client(cmd_ctx.session) + + if cpc_name: + cpc = find_cpc(cmd_ctx, client, cpc_name) + cpcs = [cpc] + else: + try: + cpcs = client.cpcs.list() + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + # Prepare the additions dict of dicts. It contains additional + # (=non-resource) property values by property name and by resource URI. + # Depending on options, some of them will not be populated. + additions = {'cpc': {}} + + show_list = [ + 'name', + 'cpc', + ] + if not options['names_only']: + show_list.extend([ + 'operating-mode', + 'ipl-type', + 'workload-manager-enabled', + 'load-at-activation', + 'description', + ]) + if options['uri']: + show_list.extend([ + 'element-uri', + ]) + + imageprofiles = [] + for cpc in cpcs: + try: + imageprofiles.extend(cpc.image_activation_profiles.list()) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + for imageprofile in imageprofiles: + cpc = imageprofile.manager.parent + additions['cpc'][imageprofile.uri] = cpc.name + + try: + print_resources( + cmd_ctx, imageprofiles, cmd_ctx.output_format, show_list, + additions, all=options['all']) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + +def cmd_imageprofile_show(cmd_ctx, cpc_name, imageprofile_name, options): + # pylint: disable=missing-function-docstring,unused-argument + + client = zhmcclient.Client(cmd_ctx.session) + imageprofile = find_imageprofile( + cmd_ctx, client, cpc_name, imageprofile_name) + + try: + imageprofile.pull_full_properties() + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + properties = dict(imageprofile.properties) + + # Add artificial property 'parent-name' + properties['parent-name'] = cpc_name + + print_properties(cmd_ctx, properties, cmd_ctx.output_format) + + +def cmd_imageprofile_update(cmd_ctx, cpc_name, imageprofile_name, options): + # pylint: disable=missing-function-docstring + # pylint: disable=unreachable + + raise NotImplementedError # TODO: Implement + + client = zhmcclient.Client(cmd_ctx.session) + imageprofile = find_imageprofile( + cmd_ctx, client, cpc_name, imageprofile_name) + + name_map = { + 'next-activation-profile': 'next-activation-profile-name', + 'acceptable-status': None, + } + org_options = original_options(options) + properties = options_to_properties(org_options, name_map) + + if org_options['zaware-host-name'] == '': + properties['zaware-host-name'] = None + if org_options['zaware-master-userid'] == '': + properties['zaware-master-userid'] = None + if org_options['zaware-master-password'] == '': + properties['zaware-master-password'] = None + + if org_options['ssc-host-name'] == '': + properties['ssc-host-name'] = None + if org_options['ssc-master-userid'] == '': + properties['ssc-master-userid'] = None + if org_options['ssc-master-password'] == '': + properties['ssc-master-password'] = None + + if org_options['acceptable-status'] is not None: + status_list = org_options['acceptable-status'].split(',') + status_list = [item for item in status_list if item] + properties['acceptable-status'] = status_list + + if not properties: + cmd_ctx.spinner.stop() + click.echo("No properties specified for updating image activation " + "profile '{ip}' on CPC '{c}'.". + format(ip=imageprofile_name, c=cpc_name)) + return + + try: + imageprofile.update_properties(properties) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + cmd_ctx.spinner.stop() + click.echo("Image activation profile '{ip}' on CPC '{c}' has been updated.". + format(ip=imageprofile_name, c=cpc_name)) + + +def cmd_imageprofile_create(cmd_ctx, cpc_name, options): + # pylint: disable=missing-function-docstring + # pylint: disable=unreachable + + raise NotImplementedError # TODO: Implement + + client = zhmcclient.Client(cmd_ctx.session) + cpc = find_cpc(cmd_ctx, client, cpc_name) + + # The following options are handled specifically in this function (as + # opposed to be handled generically in options_to_properties()): + + # The options for booting from an FTP server. + # They need to be specified together, all of them are required. + boot_ftp_option_names = ( + 'boot-ftp-host', + 'boot-ftp-username', + 'boot-ftp-password', + 'boot-ftp-insfile', + ) + + # The option for booting from an HMC media file. + boot_media_option_names = ( + 'boot-media-file', + 'boot-media-type', + ) + + # Specially handled options + special_option_names = ( + 'imageprofile-id', + 'acceptable-status', + 'ssc-dns-servers', + 'ssc-ipv4-gateway', + 'ssc-ipv6-gateway', + 'cp-absolute-capping', + 'ifl-absolute-capping', + ) + + # Options handled in this function + special_opt_names = boot_ftp_option_names + boot_media_option_names + \ + special_option_names + name_map = dict((opt, None) for opt in special_opt_names) + + org_options = original_options(options) + properties = options_to_properties(org_options, name_map) + + # Used and missing options handled in this function + used_boot_ftp_opts = [ + '--' + name for name in boot_ftp_option_names + if org_options[name] is not None] + missing_boot_ftp_opts = [ + '--' + name for name in boot_ftp_option_names + if org_options[name] is None] + + used_boot_media_opts = [ + '--' + name for name in boot_media_option_names + if org_options[name] is not None] + + used_boot_opts = used_boot_ftp_opts + used_boot_media_opts + num_boot_devices = bool(used_boot_ftp_opts) + bool(used_boot_media_opts) + if num_boot_devices > 1: + raise click_exception( + "Boot from multiple devices specified: {opts}". + format(opts=', '.join(used_boot_opts)), + cmd_ctx.error_format) + + if used_boot_ftp_opts: + if missing_boot_ftp_opts: + raise click_exception( + "Boot from FTP server specified, but misses the following " + "options: {opts}". + format(opts=', '.join(missing_boot_ftp_opts)), + cmd_ctx.error_format) + properties['boot-device'] = 'ftp' + properties['boot-ftp-host'] = org_options['boot-ftp-host'] + properties['boot-ftp-username'] = org_options['boot-ftp-username'] + properties['boot-ftp-password'] = org_options['boot-ftp-password'] + properties['boot-ftp-insfile'] = org_options['boot-ftp-insfile'] + + elif used_boot_media_opts: + properties['boot-device'] = 'removable-media' + properties['boot-removable-media'] = org_options['boot-media-file'] + properties['boot-removable-media-type'] = org_options['boot-media-type'] + + else: + # boot-device="none" is the default + pass + + # Default for the number of processors + if 'ifl-processors' not in properties and \ + 'cp-processors' not in properties: + properties['ifl-processors'] = DEFAULT_IFL_PROCESSORS + + # Specially handled options + + if org_options['imageprofile-id'] == "auto": + properties['autogenerate-imageprofile-id'] = True + elif org_options['imageprofile-id'] is not None: + imageprofile_id = "{:02X}".format(org_options['imageprofile-id']) + properties['imageprofile-id'] = imageprofile_id + properties['autogenerate-imageprofile-id'] = False + + if org_options['acceptable-status'] is not None: + status_list = org_options['acceptable-status'].split(',') + status_list = [item for item in status_list if item] + properties['acceptable-status'] = status_list + + if org_options['ssc-dns-servers'] == '': + properties['ssc-dns-servers'] = [] + elif org_options['ssc-dns-servers'] is not None: + properties['ssc-dns-servers'] = \ + org_options['ssc-dns-servers'].split(',') + + if org_options['ssc-ipv4-gateway'] == '': + properties['ssc-ipv4-gateway'] = None + + if org_options['ssc-ipv6-gateway'] == '': + properties['ssc-ipv6-gateway'] = None + + if org_options['cp-absolute-capping'] == '': + properties['cp-absolute-processor-capping'] = False + elif org_options['cp-absolute-capping'] is not None: + properties['cp-absolute-processor-capping'] = True + properties['cp-absolute-processor-capping-value'] = \ + org_options['cp-absolute-capping'] + + if org_options['ifl-absolute-capping'] == '': + properties['ifl-absolute-processor-capping'] = False + elif org_options['ifl-absolute-capping'] is not None: + properties['ifl-absolute-processor-capping'] = True + properties['ifl-absolute-processor-capping-value'] = \ + org_options['ifl-absolute-capping'] + + try: + new_imageprofile = cpc.imageprofiles.create(properties) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + cmd_ctx.spinner.stop() + click.echo("New imageprofile '{ip}' on CPC '{c}' has been created.". + format(ip=new_imageprofile.properties['name'], c=cpc_name)) + + +def cmd_imageprofile_delete(cmd_ctx, cpc_name, imageprofile_name): + # pylint: disable=missing-function-docstring + + client = zhmcclient.Client(cmd_ctx.session) + imageprofile = find_imageprofile( + cmd_ctx, client, cpc_name, imageprofile_name) + + try: + imageprofile.delete() + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + cmd_ctx.spinner.stop() + click.echo("Image activation profile '{ip}' on CPC '{c}' has been deleted.". + format(ip=imageprofile_name, c=cpc_name)) + + +def cmd_imageprofile_assign_certificate( + cmd_ctx, cpc_name, imageprofile_name, options): + # pylint: disable=missing-function-docstring + + client = zhmcclient.Client(cmd_ctx.session) + imageprofile = find_imageprofile( + cmd_ctx, client, cpc_name, imageprofile_name) + + cert_name = options['certificate'] + cert = find_certificate(cmd_ctx, client, cert_name) + + try: + imageprofile.assign_certificate(cert) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + cmd_ctx.spinner.stop() + click.echo("Certificate '{cert}' was assigned to image activation " + "profile '{ip}' on CPC '{c}.". + format(cert=cert_name, ip=imageprofile_name, c=cpc_name)) + + +def cmd_imageprofile_unassign_certificate( + cmd_ctx, cpc_name, imageprofile_name, options): + # pylint: disable=missing-function-docstring + + client = zhmcclient.Client(cmd_ctx.session) + imageprofile = find_imageprofile( + cmd_ctx, client, cpc_name, imageprofile_name) + + cert_name = options['certificate'] + cert = find_certificate(cmd_ctx, client, cert_name) + + try: + imageprofile.unassign_certificate(cert) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + cmd_ctx.spinner.stop() + click.echo("Certificate '{cert}' was unassigned from image activation " + "profile '{ip}' on CPC '{c}.". + format(cert=cert_name, ip=imageprofile_name, c=cpc_name)) diff --git a/zhmccli/_cmd_loadprofile.py b/zhmccli/_cmd_loadprofile.py new file mode 100644 index 00000000..b7bd8c59 --- /dev/null +++ b/zhmccli/_cmd_loadprofile.py @@ -0,0 +1,473 @@ +# Copyright 2023 IBM Corp. All Rights Reserved. +# +# 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. + +""" +Commands for load activation profiles for CPCs in classic mode. +""" + +from __future__ import absolute_import + +import click +from click_option_group import optgroup + +import zhmcclient +from .zhmccli import cli +from ._helper import print_properties, print_resources, abort_if_false, \ + options_to_properties, original_options, COMMAND_OPTIONS_METAVAR, \ + click_exception, add_options, LIST_OPTIONS +from ._cmd_cpc import find_cpc + + +# Defaults for load activation profile creation +DEFAULT_IFL_PROCESSORS = 1 +DEFAULT_INITIAL_MEMORY_MB = 1024 +DEFAULT_MAXIMUM_MEMORY_MB = 1024 +DEFAULT_PROCESSOR_MODE = 'shared' +PARTITION_TYPES = ['ssc', 'linux', 'zvm'] +DEFAULT_PARTITION_TYPE = 'linux' +DEFAULT_SSC_BOOT = 'installer' +DEFAULT_PROCESSING_WEIGHT = 100 +MIN_PROCESSING_WEIGHT = 1 +MAX_PROCESSING_WEIGHT = 999 +MIN_BOOT_TIMEOUT = 60 +MAX_BOOT_TIMEOUT = 600 + + +def find_loadprofile(cmd_ctx, client, cpc_or_name, loadprofile_name): + """ + Find a load activation profile by name and return its resource object. + """ + if isinstance(cpc_or_name, zhmcclient.Cpc): + cpc = cpc_or_name + else: + cpc = find_cpc(cmd_ctx, client, cpc_or_name) + # The CPC must not be in DPM mode. We don't check that because it would + # cause a GET to the CPC resource that we otherwise don't need. + try: + loadprofile = cpc.load_activation_profiles.find( + name=loadprofile_name) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + return loadprofile + + +@cli.group('loadprofile', options_metavar=COMMAND_OPTIONS_METAVAR) +def loadprofile_group(): + """ + Command group for managing load activation profiles (classic mode only). + + "Load activation profiles" are used to load (= boot, IPL) an operating + system (also known as "control program") into an LPAR. Load activation + profiles include definitions for loading the operating system. + + The commands in this group work only on CPCs in classic mode. + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + + +@loadprofile_group.command('list', options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='[CPC]', required=False) +@add_options(LIST_OPTIONS) +@click.pass_obj +def loadprofile_list(cmd_ctx, cpc, **options): + """ + List the load activation profiles in a CPC or in all managed CPCs. + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + cmd_ctx.execute_cmd(lambda: cmd_loadprofile_list(cmd_ctx, cpc, options)) + + +@loadprofile_group.command('show', options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='CPC') +@click.argument('loadprofile', type=str, metavar='IMAGEPROFILE') +@click.pass_obj +def loadprofile_show(cmd_ctx, cpc, loadprofile, **options): + """ + Show details of a load activation profile. + + The following properties are shown in addition to those returned by the HMC: + + \b + - 'parent-name' - Name of the parent CPC. + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + cmd_ctx.execute_cmd( + lambda: cmd_loadprofile_show(cmd_ctx, cpc, loadprofile, options)) + + +@loadprofile_group.command('update', options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='CPC') +@click.argument('loadprofile', type=str, metavar='IMAGEPROFILE') +@optgroup.group('General options') +# TODO: Add options +@click.pass_obj +def loadprofile_update(cmd_ctx, cpc, loadprofile, **options): + """ + Update the properties of a load activation profile. + + Only the properties will be changed for which a corresponding option is + specified, so the default for all options is not to change properties. + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + + \b + Limitations: + * The processor capping/sharing/weight related properties cannot be + updated. + * The network-related properties for zaware and ssc cannot be updated. + * The --zaware-master-password and --ssc-master-password options do not + ask for the password. + """ + cmd_ctx.execute_cmd( + lambda: cmd_loadprofile_update(cmd_ctx, cpc, loadprofile, options)) + + +@loadprofile_group.command('create', options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='CPC') +@optgroup.group('General options') +@click.option('--name', type=str, required=True, + help='The name of the new load activation profile.') +# TODO: Add options +@click.pass_obj +def loadprofile_create(cmd_ctx, cpc, **options): + """ + Create a load activation profile in a CPC (Only HMC 2.16 and later). + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + cmd_ctx.execute_cmd(lambda: cmd_loadprofile_create(cmd_ctx, cpc, options)) + + +@loadprofile_group.command('delete', options_metavar=COMMAND_OPTIONS_METAVAR) +@click.argument('cpc', type=str, metavar='CPC') +@click.argument('loadprofile', type=str, metavar='IMAGEPROFILE') +@click.option('-y', '--yes', is_flag=True, callback=abort_if_false, + expose_value=False, + help='Skip prompt to confirm deletion of the load activation ' + 'profile.', + prompt='Are you sure you want to delete this load activation ' + 'profile ?') +@click.pass_obj +def loadprofile_delete(cmd_ctx, cpc, loadprofile): + """ + Delete a load activation profile (Only HMC 2.16 and later). + + In addition to the command-specific options shown in this help text, the + general options (see 'zhmc --help') can also be specified right after the + 'zhmc' command name. + """ + cmd_ctx.execute_cmd( + lambda: cmd_loadprofile_delete(cmd_ctx, cpc, loadprofile)) + + +def cmd_loadprofile_list(cmd_ctx, cpc_name, options): + # pylint: disable=missing-function-docstring + + client = zhmcclient.Client(cmd_ctx.session) + + if cpc_name: + cpc = find_cpc(cmd_ctx, client, cpc_name) + cpcs = [cpc] + else: + try: + cpcs = client.cpcs.list() + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + # Prepare the additions dict of dicts. It contains additional + # (=non-resource) property values by property name and by resource URI. + # Depending on options, some of them will not be populated. + additions = {'cpc': {}} + + show_list = [ + 'name', + 'cpc', + ] + if not options['names_only']: + show_list.extend([ + 'operating-mode', + 'ipl-type', + 'workload-manager-enabled', + 'load-at-activation', + 'description', + ]) + if options['uri']: + show_list.extend([ + 'element-uri', + ]) + + loadprofiles = [] + for cpc in cpcs: + try: + loadprofiles.extend(cpc.load_activation_profiles.list()) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + for loadprofile in loadprofiles: + cpc = loadprofile.manager.parent + additions['cpc'][loadprofile.uri] = cpc.name + + try: + print_resources( + cmd_ctx, loadprofiles, cmd_ctx.output_format, show_list, + additions, all=options['all']) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + +def cmd_loadprofile_show(cmd_ctx, cpc_name, loadprofile_name, options): + # pylint: disable=missing-function-docstring,unused-argument + + client = zhmcclient.Client(cmd_ctx.session) + loadprofile = find_loadprofile( + cmd_ctx, client, cpc_name, loadprofile_name) + + try: + loadprofile.pull_full_properties() + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + properties = dict(loadprofile.properties) + + # Add artificial property 'parent-name' + properties['parent-name'] = cpc_name + + print_properties(cmd_ctx, properties, cmd_ctx.output_format) + + +def cmd_loadprofile_update(cmd_ctx, cpc_name, loadprofile_name, options): + # pylint: disable=missing-function-docstring + # pylint: disable=unreachable + + raise NotImplementedError # TODO: Implement + + client = zhmcclient.Client(cmd_ctx.session) + loadprofile = find_loadprofile( + cmd_ctx, client, cpc_name, loadprofile_name) + + name_map = { + 'next-activation-profile': 'next-activation-profile-name', + 'acceptable-status': None, + } + org_options = original_options(options) + properties = options_to_properties(org_options, name_map) + + if org_options['zaware-host-name'] == '': + properties['zaware-host-name'] = None + if org_options['zaware-master-userid'] == '': + properties['zaware-master-userid'] = None + if org_options['zaware-master-password'] == '': + properties['zaware-master-password'] = None + + if org_options['ssc-host-name'] == '': + properties['ssc-host-name'] = None + if org_options['ssc-master-userid'] == '': + properties['ssc-master-userid'] = None + if org_options['ssc-master-password'] == '': + properties['ssc-master-password'] = None + + if org_options['acceptable-status'] is not None: + status_list = org_options['acceptable-status'].split(',') + status_list = [item for item in status_list if item] + properties['acceptable-status'] = status_list + + if not properties: + cmd_ctx.spinner.stop() + click.echo("No properties specified for updating load activation " + "profile '{lp}' on CPC '{c}'.". + format(lp=loadprofile_name, c=cpc_name)) + return + + try: + loadprofile.update_properties(properties) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + cmd_ctx.spinner.stop() + click.echo("Load activation profile '{lp}' on CPC '{c}' has been updated.". + format(lp=loadprofile_name, c=cpc_name)) + + +def cmd_loadprofile_create(cmd_ctx, cpc_name, options): + # pylint: disable=missing-function-docstring + # pylint: disable=unreachable + + raise NotImplementedError # TODO: Implement + + client = zhmcclient.Client(cmd_ctx.session) + cpc = find_cpc(cmd_ctx, client, cpc_name) + + # The following options are handled specifically in this function (as + # opposed to be handled generically in options_to_properties()): + + # The options for booting from an FTP server. + # They need to be specified together, all of them are required. + boot_ftp_option_names = ( + 'boot-ftp-host', + 'boot-ftp-username', + 'boot-ftp-password', + 'boot-ftp-insfile', + ) + + # The option for booting from an HMC media file. + boot_media_option_names = ( + 'boot-media-file', + 'boot-media-type', + ) + + # Specially handled options + special_option_names = ( + 'loadprofile-id', + 'acceptable-status', + 'ssc-dns-servers', + 'ssc-ipv4-gateway', + 'ssc-ipv6-gateway', + 'cp-absolute-capping', + 'ifl-absolute-capping', + ) + + # Options handled in this function + special_opt_names = boot_ftp_option_names + boot_media_option_names + \ + special_option_names + name_map = dict((opt, None) for opt in special_opt_names) + + org_options = original_options(options) + properties = options_to_properties(org_options, name_map) + + # Used and missing options handled in this function + used_boot_ftp_opts = [ + '--' + name for name in boot_ftp_option_names + if org_options[name] is not None] + missing_boot_ftp_opts = [ + '--' + name for name in boot_ftp_option_names + if org_options[name] is None] + + used_boot_media_opts = [ + '--' + name for name in boot_media_option_names + if org_options[name] is not None] + + used_boot_opts = used_boot_ftp_opts + used_boot_media_opts + num_boot_devices = bool(used_boot_ftp_opts) + bool(used_boot_media_opts) + if num_boot_devices > 1: + raise click_exception( + "Boot from multiple devices specified: {opts}". + format(opts=', '.join(used_boot_opts)), + cmd_ctx.error_format) + + if used_boot_ftp_opts: + if missing_boot_ftp_opts: + raise click_exception( + "Boot from FTP server specified, but misses the following " + "options: {opts}". + format(opts=', '.join(missing_boot_ftp_opts)), + cmd_ctx.error_format) + properties['boot-device'] = 'ftp' + properties['boot-ftp-host'] = org_options['boot-ftp-host'] + properties['boot-ftp-username'] = org_options['boot-ftp-username'] + properties['boot-ftp-password'] = org_options['boot-ftp-password'] + properties['boot-ftp-insfile'] = org_options['boot-ftp-insfile'] + + elif used_boot_media_opts: + properties['boot-device'] = 'removable-media' + properties['boot-removable-media'] = org_options['boot-media-file'] + properties['boot-removable-media-type'] = org_options['boot-media-type'] + + else: + # boot-device="none" is the default + pass + + # Default for the number of processors + if 'ifl-processors' not in properties and \ + 'cp-processors' not in properties: + properties['ifl-processors'] = DEFAULT_IFL_PROCESSORS + + # Specially handled options + + if org_options['loadprofile-id'] == "auto": + properties['autogenerate-loadprofile-id'] = True + elif org_options['loadprofile-id'] is not None: + loadprofile_id = "{:02X}".format(org_options['loadprofile-id']) + properties['loadprofile-id'] = loadprofile_id + properties['autogenerate-loadprofile-id'] = False + + if org_options['acceptable-status'] is not None: + status_list = org_options['acceptable-status'].split(',') + status_list = [item for item in status_list if item] + properties['acceptable-status'] = status_list + + if org_options['ssc-dns-servers'] == '': + properties['ssc-dns-servers'] = [] + elif org_options['ssc-dns-servers'] is not None: + properties['ssc-dns-servers'] = \ + org_options['ssc-dns-servers'].split(',') + + if org_options['ssc-ipv4-gateway'] == '': + properties['ssc-ipv4-gateway'] = None + + if org_options['ssc-ipv6-gateway'] == '': + properties['ssc-ipv6-gateway'] = None + + if org_options['cp-absolute-capping'] == '': + properties['cp-absolute-processor-capping'] = False + elif org_options['cp-absolute-capping'] is not None: + properties['cp-absolute-processor-capping'] = True + properties['cp-absolute-processor-capping-value'] = \ + org_options['cp-absolute-capping'] + + if org_options['ifl-absolute-capping'] == '': + properties['ifl-absolute-processor-capping'] = False + elif org_options['ifl-absolute-capping'] is not None: + properties['ifl-absolute-processor-capping'] = True + properties['ifl-absolute-processor-capping-value'] = \ + org_options['ifl-absolute-capping'] + + try: + new_loadprofile = cpc.loadprofiles.create(properties) + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + cmd_ctx.spinner.stop() + click.echo("New loadprofile '{lp}' on CPC '{c}' has been created.". + format(lp=new_loadprofile.properties['name'], c=cpc_name)) + + +def cmd_loadprofile_delete(cmd_ctx, cpc_name, loadprofile_name): + # pylint: disable=missing-function-docstring + + client = zhmcclient.Client(cmd_ctx.session) + loadprofile = find_loadprofile( + cmd_ctx, client, cpc_name, loadprofile_name) + + try: + loadprofile.delete() + except zhmcclient.Error as exc: + raise click_exception(exc, cmd_ctx.error_format) + + cmd_ctx.spinner.stop() + click.echo("Load activation profile '{lp}' on CPC '{c}' has been deleted.". + format(lp=loadprofile_name, c=cpc_name))