1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212
|
.. image:: images/hpe_logo2.png
:width: 150pt
==============
Advanced Usage
==============
The following sections describe advanced usage of the python-ilorest-library.
A deeper dive into the Client Classes
=====================================
The **RedfishClient** and **LegacyRestClient** are both based on the **RestClient** class.
The only difference between the two is that one is configured for *Legacy Rest* and the other one is configured for *Redfish*.
So it is possible to directly use the **RestClient** class instead if you want.
For other REST APIs, it is even possible to use the **RestClientBase** class as a client.
Modifying Remote Connection Settings
------------------------------------
A remote connection eventually turns into a *urllib3* **PoolManager** class. Because of this setting,
any extra arguments passed to the Client class that **PoolManager** takes (and any urllib3 adds in the future)are added to the *urllib3* connection.
Useful arguments include:
* **timeout** (in seconds)
* **retries**
* **ca_certs**
For a full list of arguments possible, see urllib3's `documentation <https://urllib3.readthedocs.io/en/latest/reference/index.html?highlight=PoolManager#urllib3.PoolManager>`_.
The following example sets a **RedfishClient** with a timeout of 30 seconds and five retries.
>>> REST_OBJ = redfish.RedfishClient(base_url=iLO_host, username=login_account, password=login_password, timeout=30, retries=5)
Proxy Support
-------------
The python-ilorest-library **version 2.5.0 and greater** supports both HTTP and SOCKS proxies. If you are using a socks proxy, you must `install the [socks] add-on <Installation-Guide.html#with-socks-proxy-support>`_.
>>> REST_OBJ = redfish.RedfishClient(base_url="https://10.0.0.100", username="username", password="password", proxy='http://proxy.proxy.com:8080')
SOCKS proxies must start with `socks`
>>> REST_OBJ = redfish.RedfishClient(base_url="https://10.0.0.100", username="username", password="password", proxy='socks5://proxy.proxy.com:8080')
HTTPS verification
------------------
By default the python-ilorest-library creates an unverified context for HTTPS connections.
To verify the HTTPS certificate, pass an optional argument to urllib3 through a Client object, **ca_certs**.
>>> REST_OBJ = redfish.RedfishClient(base_url="https://hostname", username="username", password="password", ca_certs='C:\<pathto>\CA.crt')
For a full description of possible arguments in ca_certs, see `urllib3's documentation on the subject <https://urllib3.readthedocs.io/en/latest/user-guide.html#certificate-verification>`_.
A deeper dive into the Response Class
=====================================
The **Response** class was already mentioned briefly in the `Quick Start <Quick-Start.html#response-object>`_ section, but this section provides more details about the useful properties of the Response class.
You can use the **getheader()** function to search for a specific header in a response. *getheader is case insensitive*
>>> response.getheader('link')
'</redfish/v1/SchemaStore/en/ComputerSystem.json>; rel=describedby'
The **ori** property returns the response body in its original form, and it is useful for binary responses.
>>> response.ori
The **obj** property returns the response body in dot notation.
>>> resp_obj = response.obj
>>> resp_obj.AssetTag
u'assettag'
For keys with special characters, the dictionary notation still works.
>>> resp_obj.Bios['@odata.id']
u'/redfish/v1/systems/1/bios'
Response Object
---------------
A class has been implemented within the library to handle HTTP response codes from iLO for patch, post and put type requests. HTTP response data is parsed and then combined with the relevant error response resource as associated with the applied properties (categorized by type and MessageId).
Error response data is guided by HTTP status codes. See `List of HTTP status codes <https://en.wikipedia.org/wiki/List_of_HTTP_status_codes>`_ for more information:
* **200** response codes indicate success. Generally these types of responses include simple messages, such as "successful" or "one or more properties have been changed, an ilo reset is required for the properties to take effect" (on some rare occassions no response message or an empty response string will occur).
* **400** response codes indicate client failures such as an invalid post or patch, or the resource could not be modified. Additional detailed information will likely accompany the response message.
.. note:: Response codes outside of 200 and 400 have not been extensively tested or witnessed. Responses for HTTP response codes under 300 redirects or 500 server errors might be observed; please feel free to note and share the results of such encounters by opening a GitHub issue.
Generally 400 error response codes result in a full error response. A response might include the following data; however, the entries are not guaranteed for every response:
* An error response message ID ("MessageId"), iLO response error message code identification.
* An error response message description ("Description"), essentially a quick synopsis of the issue.
* An error response message ("Message") describing the reason for the error in greater detail. The offending properties are noted as per the relevant error response resource.
* An error response resolution ("Resolution") describing steps to correct the error.
The following is an example response message for an error in the iLODateTime type regarding the property "NTPServers":
.. code-block:: TEXT
HTTP Response Code: [400]
MessageId: iLO.2.8.PropertyNotWritableOrUnknown
Description: The request included a value for a read-only or unknown property.
Message: The property "NTPServers" is a read-only property and cannot be assigned a value, or not valid for this resource.
Resolution: If the operation did not complete, remove the property from the request body and resubmit the request.
.. note:: The response data is also available when utilizing the iLO Channel Interface Driver (Chif).
Request Object
--------------
The **response** property of the Response object returns a **Request** object, which describes the request that generated the response.
>>> response.request
<redfish.rest.containers.RestRequest object at 0x00000000070C7E80>
The **url** property shows the actual system the call was made against.
>>> response.request.url
'https://10.0.0.100'
The **method** property shows the method applied.
>>> response.request.method
'PATCH'
The **path** property shows what path the method was made against.
>>> response.request.path
'/redfish/v1/systems/1'
The **body** property shows what body (if any) was passed along with the request.
>>> response.request.body
'{"AssetTag": "new tag"}'
Certificate Based Authentication (HPE iLO Supported Authentication Only)
========================================================================
The python-ilorest-library (**version 3.2.0 and greater**) supports the use of certificate based authentication when accessing an HPE iLO based BMC. Certificate based authentication utilizes Urllib3 TLS certificates passed to the PoolManager when an http client is initialized.
>>> REST_OBJ = redfish.RedfishClient(base_url="https://10.0.0.100", username="username", ca_cert_data=<dictionary of certificate data>)
Upon successful login, a session key is provided in return, by iLO, which can be used for all subsequent transactions.
>>> REST_OBJ.login(auth="session")
.. note:: This feature requires iLO 5 version 2.40 or greater. In addition to the version requirement the iLO internal RTC clock must have the correct date and time (NTP must be configured), have a valid "iLO Advanced Premium Security Edition" license key, have CAC/Smartcard Authentication enabled, incorporate a valid SSL certificate, as well as certificate thumbprint mappings for the relevant iLO management users which can be authenticated by the client certificate(s). Multiple users can share the same certificate, if so configured.
The dictionary of certificate data should include:
#. iLO User TLS Certificate (.pem suggested)
#. iLO User Root CA Certificate (if self signed) (.pem suggested)
#. Private Root CA Password (if available)
>>> {'ca_certs': <User Root CA>, 'cert_file': <User TLS Certificate>, 'key_file': <User Rooot CA key file>, 'key_password': <User Root CA key file password>}
General steps to create the necessary steps include:
#. Generate a root CA private key (.PEM) using RSA. This key can be password protected, which must be supplied in order to be used. Bit encryption level should be consistent.
#. Generate a user certificate private key. This key can be password protected, which must be supplied in order to be used.
#. Generate a root CA Certificate (.PEM), if using a self signed option. Alternatively this can be generated by a valid, trusted, certificate authority (CA) (IdenTrust, DigiCert, GoDaddy, GlobalSign, and Let's Encrypt just to name a few).
#. Configure and set X509 SSL certificate by generating a certificate signing request from iLO (CSR) and signing with the previously generated private root CA key.
#. Add X509 SSL certificate to iLO.
#. Add root CA Certificate to CAC/Smartcard and enable CAC Smartcard Authentication.
#. Generate user certificate.
#. Add user certificate mappings to iLO
Multi-part Form Data
====================
.. note:: Multi-part Form data is only supported on iLO with a **Remote Object**.
Multi-part form data is sent as a list of tuples.
Standard fields are a tuple of `(field, value)`
>>> ('field1': 'value')
Json data is just another value
>>> ('jsondata', {'key1': 'value1', 'key2': 'value2'})
Files & binary data are similar to the standard fields, except instead of a value we have another tuple.
This value tuple has 2 or 3 items `(filename, filedata)` or optionally include the MIME type explicitly `(filename, filedata , MIME type)`
>>> ('filedata', ('filename.data', '<binary data>', 'application/octet-stream'))
So putting it all together in a single form data request:
>>> [('field1': 'value'), ('jsondata', {'key1': 'value1', 'key2': 'value2'}), ('filedata', ('filename.data', '<binary data>', 'application/octet-stream'))]
For an example of Multi-part form data in action, see the `example for uploading firmware to the iLO repository <https://github.com/HewlettPackard/python-ilorest-library/blob/master/examples/Redfish/upload_firmware_ilo_repository.py>`_.
Logging
=======
Logging of all request and responses is available through the redfish_logger function. First, import the logging and redfish_logger functions.
>>> import logging
>>> from redfish import redfish_logger
Then, specify the logger file name, format of the logger, and the logging level. See the `logging` python documentation for more information on these arguments.
>>> LOGGERFILE = "RedfishApiExamples.log"
>>> LOGGERFORMAT = "%(asctime)s - %(name)s - %(levelname)s - %(message)s"
>>> LOGGER = redfish_logger(LOGGERFILE, LOGGERFORMAT, logging.DEBUG)
After running some commands, your log file should be populated in the file specified at creation.
|