9. Security

9.1. Handling of HMC credentials in zhmcclient

The following figure shows a node that runs a user-written Python application “my-hmc-app” using the zhmcclient package that drives operations against the HMC and subscribes for notifications from the HMC.

_images/security1.svg

The figure shows which security relevant init arguments are passed to the two zhmcclient objects, and which of those are subsequently available as properties of those objects. It is the responsibility of the application using the zhmcclient package to make sure that the HMC userid and password, and the resulting HMC session ID are handled in a secure manner.

The zhmcclient.Session object uses the Python requests package for the HTTPS communication with the HMC. There are further Python libraries, OS-level libraries and network drivers in the OS involved that are not shown in the figure. The Session object at some point creates a session with the HMC and then stores the session ID as one of its properties. The password is not available as a property on that object, but it is stored as an internal attribute in order to perform automatic re-logon if the HMC session expires.

The zhmcclient.NotificationReceiver object uses the Python stomp.py package for the STOMP communication with the HMC. There are further Python libraries, OS-level libraries and network drivers in the OS involved that are not shown in the figure. The userid and password are stored as internal attributes in the NotificationReceiver object, and are passed on to the stomp.Connection object when connecting to the HMC for the purpose of establishing the STOMP session.

9.2. HMC Web Services API operations

This section covers the HTTPS communication between the ‘zhmcclient’ package and the HMC Web Services API on port 6794. The ‘zhmcclient’ package uses the Python ‘requests’ package for this purpose.

9.2.1. SSL/TLS protocol version

The HMC supports HTTPS at its Web Services API port, i.e. it requires the use of SSL/TLS-based sockets. The HMC can be configured to require particular TLS versions. It is recommended to use the highest TLS version, but at least TLS 1.2. This can be configured in the HMC task “Customize Console Services”. See also the HMC Security book and Chapter 3 “Invoking API operations” in the HMC API book.

The ‘zhmcclient’ package uses the default settings of the ‘requests’ package regarding the SSL/TLS version, which causes the highest version supported by both client and HMC to be used. The highest supported SSL/TLS version used by the client is determined by the OpenSSL version used by Python on the client side. OpenSSL 1.0.1 or higher is required to support TLS 1.2. You can display the OpenSSL version used by Python using this command:

$ python -c "import ssl; print(ssl.OPENSSL_VERSION)"
OpenSSL 1.1.1i  8 Dec 2020

9.2.2. HMC certificate

By default, the HMC is configured with a self-signed certificate. That is the X.509 certificate presented by the HMC as the server certificate during SSL/TLS handshake at its Web Services API.

Starting with version 0.31, the zhmcclient will reject self-signed certificates by default.

The HMC should be configured to use a CA-verifiable certificate. This can be done in the HMC task “Certificate Management”. See also the HMC Security book and Chapter 3 “Invoking API operations” in the HMC API book.

Starting with version 0.31, the zhmcclient provides a control knob for the verification of the HMC certificate via the verify_cert init parameter of the zhmcclient.Session class. That init parameter can be set to:

  • False: Do not verify the HMC certificate. Not verifying the HMC certificate means the zhmcclient will not detect hostname mismatches, expired certificates, revoked certificates, or otherwise invalid certificates. Since this mode makes the connection vulnerable to man-in-the-middle attacks, it is insecure and should not be used in production environments.

  • True (default): Verify the HMC certificate using the CA certificates from the first of these locations:

  • string: Path name of a certificate file or directory. Verify the HMC certificate using the CA certificates in that file or directory.

If a certificate file is specified (using any of the ways listed above), that file must be in PEM format and must contain all CA certificates that are supposed to be used. Usually they are in the order from leaf to root, but that is not a hard requirement. The single certificates are concatenated in the file.

If a certificate directory is specified (using any of the ways listed above), it must contain PEM files with all CA certificates that are supposed to be used, and copies of the PEM files or symbolic links to them in the hashed format created by the OpenSSL command c_rehash.

An X.509 certificate in PEM format is base64-encoded, begins with the line -----BEGIN CERTIFICATE-----, and ends with the line -----END CERTIFICATE-----. More information about the PEM format is for example on this www.ssl.com page or in this serverfault.com answer.

Since the zhmcclient package uses the ‘requests’ package for the communication with the Web Services API of the HMC, the behavior described above actually comes from the ‘requests’ package. Unfortunately, its documentation about certificate verification is somewhat brief, see SSL Cert Verification.

Note that setting the REQUESTS_CA_BUNDLE or CURL_CA_BUNDLE environment variables influences other programs that use these variables, too.

9.2.3. Cipher suites

During SSL/TLS handshake, the cipher suite to be used for various aspects in the communication is negotiated between the HMC and the ‘zhmcclient’ client.

The set of cipher suites enabled for the HMC Web Services API can be configured in the HMC task “Certificate Management”. See also the HMC Security book for details.

The ‘zhmcclient’ package uses the default cipher suites of the ‘requests’ package, which are the default cipher suites used by the standard Python ‘ssl’ module. By default, the CPython implementation uses OpenSSL. Python PEP 644 targeted for Python 3.10 contains information about which versions of Python support which versions of OpenSSL.

As of Python 3.9, there is no function yet that lists the supported ciphers.

You can display the OpenSSL version used by the Python on your system with this command:

$ python -c "import ssl; print(ssl.OPENSSL_VERSION)"
OpenSSL 1.1.1i  8 Dec 2020

The TLS 1.2 compatible ciphers that are supported by OpenSSL on your system can be listed with this command:

$ openssl ciphers -tls1_2 -s -v | sort
AES128-GCM-SHA256  TLSv1.2  Kx=RSA  Au=RSA  Enc=AESGCM(128)  Mac=AEAD
AES128-SHA         SSLv3    Kx=RSA  Au=RSA  Enc=AES(128)     Mac=SHA1
AES128-SHA256      TLSv1.2  Kx=RSA  Au=RSA  Enc=AES(128)     Mac=SHA256
. . .

The SSL/TLS version shown in the output is the minimum SSL/TLS protocol version needed to use the cipher, not the actual version that is used.

Brief expansion of the output field names used by this command:

  • Kx = Key Exchange

  • Au = Authentication

  • Enc = Encryption

  • Mac = Message Authentication Code

9.3. HMC Web Services API notifications

The HMC Web Services API supports notifications that are sent from the HMC to a client. The HMC supports a choice of protocols for this purpose:

  • Protocols following the JMS (Java Message Service) architecture:

    • STOMP (Streaming Text Oriented Messaging Protocol) over SSL connections, on port 61612.

    • OpenWire over SSL connections, on port 61617.

  • SSE (Server-Sent Events), using a long-lived HTTPS connection on port 6794. Support for this protocol has been added in HMC version 2.16.

These protocols can be enabled on the HMC task “Customize API Settings”. See also the HMC Security book for details.

The ‘zhmcclient’ package supports the STOMP protocol for HMC notifications and uses the Python ‘stomp.py’ package for this purpose.

The STOMP protocol uses SSL/TLS sockets, so there is a TLS handshake that happens. The HMC uses the same TLS related settings for STOMP that it uses for the HTTPS operations. The zhmcclient package configures the stomp.py package to negotiate the highest possible TLS version for the STOMP protocol with the HMC.

The zhmcclient package version 1.22.0 has added support for enabling CA certificate validation for the STOMP protocol by adding a verify_cert init parameter to zhmcclient.NotificationReceiver. For backwards compatibility reasons, the validation is disabled by default. As a result, no detection of invalid HMC certificates, hostname mismatches, etc. is performed by default. Therefore, it is recommended to enable CA certificate validation by specifying the verify_cert init parameter with a value other than False.