From 03440d9f6e801650f7283e5eb9f40c12c8913cf5 Mon Sep 17 00:00:00 2001 From: Dmitry Tantsur Date: Fri, 3 Sep 2021 11:49:00 +0200 Subject: [PATCH] Expand the driver contributor documentation * Add more information into the deploy and clean steps page. * Link from the driver contributor page to the deploy/clean steps page. * Move vendor passthru admin docs where they belong, adding CLI. Change-Id: I2e9f98addabfeed9adf2495c3b3546c2b53d7962 --- doc/source/admin/index.rst | 1 + doc/source/admin/vendor-passthru.rst | 113 +++++++++++++++++++ doc/source/contributor/deploy-steps.rst | 143 ++++++++++++++++++++++-- doc/source/contributor/drivers.rst | 88 ++------------- 4 files changed, 258 insertions(+), 87 deletions(-) create mode 100644 doc/source/admin/vendor-passthru.rst diff --git a/doc/source/admin/index.rst b/doc/source/admin/index.rst index 3e11d35818..a6ac6182dd 100644 --- a/doc/source/admin/index.rst +++ b/doc/source/admin/index.rst @@ -32,6 +32,7 @@ the services. Booting a Ramdisk or an ISO Deploying with anaconda deploy interface Hardware Burn-in + Vendor Passthru Drivers, Hardware Types and Hardware Interfaces ----------------------------------------------- diff --git a/doc/source/admin/vendor-passthru.rst b/doc/source/admin/vendor-passthru.rst new file mode 100644 index 0000000000..7aa85da116 --- /dev/null +++ b/doc/source/admin/vendor-passthru.rst @@ -0,0 +1,113 @@ +Vendor Passthru +=============== + +The bare metal service allows drivers to expose vendor-specific API known as +*vendor passthru*. + +Node Vendor Passthru +-------------------- + +Drivers may implement a passthrough API, which is accessible via +the ``/v1/nodes//vendor_passthru?method={METHOD}`` +endpoint. Beyond basic checking, Ironic does not introspect the message +body and simply "passes it through" to the relevant driver. + +A method: + +* can support one or more HTTP methods (for example, GET, POST) + +* is asynchronous or synchronous + + + For asynchronous methods, a 202 (Accepted) HTTP status code is returned + to indicate that the request was received, accepted and is being acted + upon. No body is returned in the response. + + + For synchronous methods, a 200 (OK) HTTP status code is returned to + indicate that the request was fulfilled. The response may include a body. + +* can require an exclusive lock on the node. This only occurs if the method + doesn't specify require_exclusive_lock=False in the decorator. If an + exclusive lock is held on the node, other requests for the node will be + delayed and may fail with an HTTP 409 (Conflict) error code. + +This endpoint exposes a node's driver directly, and as such, it is +expressly not part of Ironic's standard REST API. There is only a +single HTTP endpoint exposed, and the semantics of the message body +are determined solely by the driver. Ironic makes no guarantees about +backwards compatibility; this is solely up to the discretion of each +driver's author. + +To get information about all the methods available via the vendor_passthru +endpoint for a particular node, use CLI: + +.. code-block:: console + + $ baremetal node passthru list + +-----------------------+------------------------+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+------------------------+ + | Name | Supported HTTP methods | Async | Description | Response is attachment | + +-----------------------+------------------------+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+------------------------+ + | create_subscription | POST | False | Creates a subscription on a node. Required argument: a dictionary of {'destination': 'destination_url'} | False | + | delete_subscription | DELETE | False | Delete a subscription on a node. Required argument: a dictionary of {'id': 'subscription_bmc_id'} | False | + | eject_vmedia | POST | True | Eject a virtual media device. If no device is provided then all attached devices will be ejected. Optional arguments: 'boot_device' - the boot device to eject, either 'cd', 'dvd', 'usb', or 'floppy' | False | + | get_all_subscriptions | GET | False | Returns all subscriptions on the node. | False | + | get_subscription | GET | False | Get a subscription on the node. Required argument: a dictionary of {'id': 'subscription_bmc_id'} | False | + +-----------------------+------------------------+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+------------------------+ + +The response will contain information for each method, +such as the method's name, a description, the HTTP methods supported, +and whether it's asynchronous or synchronous. + +You can call a method with CLI, for example: + +.. code-block:: console + + $ baremetal node passthru call eject_vmedia + +Driver Vendor Passthru +---------------------- + +Drivers may implement an API for requests not related to any node, +at ``/v1/drivers//vendor_passthru?method={METHOD}``. + +A method: + +* can support one or more HTTP methods (for example, GET, POST) + +* is asynchronous or synchronous + + + For asynchronous methods, a 202 (Accepted) HTTP status code is + returned to indicate that the request was received, accepted and is + being acted upon. No body is returned in the response. + + + For synchronous methods, a 200 (OK) HTTP status code is returned + to indicate that the request was fulfilled. The response may include + a body. + +.. note:: + Unlike methods in `Node Vendor Passthru`_, a request does not lock any + resource, so it will not delay other requests and will not fail with an + HTTP 409 (Conflict) error code. + +Ironic makes no guarantees about the semantics of the message BODY sent +to this endpoint. That is left up to each driver's author. + +To get information about all the methods available via the driver +vendor_passthru endpoint, use CLI: + +.. code-block:: console + + $ baremetal driver passthru list redfish + +The response will contain information for each method, +such as the method's name, a description, the HTTP methods supported, +and whether it's asynchronous or synchronous. + +.. warning:: + Currently only the methods available in the default interfaces of the + hardware type are available. + +You can call a method with CLI, for example: + +.. code-block:: console + + $ baremetal driver passthru call diff --git a/doc/source/contributor/deploy-steps.rst b/doc/source/contributor/deploy-steps.rst index ae79e56f18..be5b960f07 100644 --- a/doc/source/contributor/deploy-steps.rst +++ b/doc/source/contributor/deploy-steps.rst @@ -1,5 +1,8 @@ -Developing a new Deploy Step -============================ +Developing deploy and clean steps +================================= + +Deploy steps basics +------------------- To support customized deployment step, implement a new method in an interface class and use the decorator ``deploy_step`` defined in @@ -78,13 +81,139 @@ The above command outputs the ``driver_internal_info`` as following:: } } -.. note:: - - Similarly, clean steps can be implemented using the ``clean_step`` - decorator. - In-band deploy steps (deploy steps that are run inside the ramdisk) have to be implemented in a custom :ironic-python-agent-doc:`IPA hardware manager `. All in-band deploy steps must have priorities between 41 and 99, see :ref:`node-deployment-core-steps` for details. + +Clean steps basics +------------------ + +Clean steps are written similarly to deploy steps, but are executed during +:doc:`cleaning `. Steps with priority > 0 are executed during +automated cleaning, all steps can be executed explicitly during manual +cleaning. Unlike deploy steps, clean steps are commonly found in these +interfaces: + +``bios`` + Steps that apply BIOS settings, see `Implementing BIOS settings`_. +``deploy`` + Steps that undo the effect of deployment (e.g. erase disks). +``management`` + Additional steps that use the node's BMC, such as out-of-band firmware + update or BMC reset. +``raid`` + Steps that build or tear down RAID, see `Implementing RAID`_. + +.. note:: + When designing a new step for your driver, try to make it consistent with + existing steps on other drivers. + +Just as deploy steps, in-band clean steps have to be +implemented in a custom :ironic-python-agent-doc:`IPA hardware manager +`. + +Implementing RAID +----------------- + +RAID is implemented via deploy and clean steps in the ``raid`` interfaces. +By convention they have the following signatures: + +.. code-block:: python + + from ironic.drivers import base + + class MyRAID(base.RAIDInterface): + + @base.clean_step(priority=0, abortable=False, argsinfo={ + 'create_root_volume': { + 'description': ( + 'This specifies whether to create the root volume. ' + 'Defaults to `True`.' + ), + 'required': False + }, + 'create_nonroot_volumes': { + 'description': ( + 'This specifies whether to create the non-root volumes. ' + 'Defaults to `True`.' + ), + 'required': False + }, + 'delete_existing': { + 'description': ( + 'Setting this to `True` indicates to delete existing RAID ' + 'configuration prior to creating the new configuration. ' + 'Default value is `False`.' + ), + 'required': False, + } + }) + def create_configuration(self, task, create_root_volume=True, + create_nonroot_volumes=True, + delete_existing=False): + pass + + @base.clean_step(priority=0) + @base.deploy_step(priority=0) + def delete_configuration(self, task): + pass + + @base.deploy_step(priority=0, + argsinfo=base.RAID_APPLY_CONFIGURATION_ARGSINFO) + def apply_configuration(self, task, raid_config, + create_root_volume=True, + create_nonroot_volumes=False, + delete_existing=False): + pass + +Notes: + +* ``create_configuration`` only works as a clean step, during deployment + ``apply_configuration`` is used instead. +* ``apply_configuration`` accepts the target RAID configuration explicitly, + while ``create_configuration`` uses the node's ``target_raid_config`` field. +* Priorities default to 0 since RAID should not be built by default. + +Implementing BIOS settings +-------------------------- + +BIOS is implemented via deploy and clean steps in the ``raid`` interfaces. +By convention they have the following signatures: + +.. code-block:: python + + from ironic.drivers import base + + _APPLY_CONFIGURATION_ARGSINFO = { + 'settings': { + 'description': ( + 'A list of BIOS settings to be applied' + ), + 'required': True + } + } + + class MyBIOS(base.BIOSInterface): + + @base.clean_step(priority=0) + @base.deploy_step(priority=0) + @base.cache_bios_settings + def factory_reset(self, task): + pass + + @base.clean_step(priority=0, argsinfo=_APPLY_CONFIGURATION_ARGSINFO) + @base.deploy_step(priority=0, argsinfo=_APPLY_CONFIGURATION_ARGSINFO) + @base.cache_bios_settings + def apply_configuration(self, task, settings): + pass + +Notes: + +* Both ``factory_reset`` and ``apply_configuration`` can be used as deploy + and clean steps. +* The ``cache_bios_settings`` decorator is used to ensure that the settings + cached in the ironic database is updated. +* Priorities default to 0 since BIOS settings should not be modified + by default. diff --git a/doc/source/contributor/drivers.rst b/doc/source/contributor/drivers.rst index bd1b4f7c35..95b0b21867 100644 --- a/doc/source/contributor/drivers.rst +++ b/doc/source/contributor/drivers.rst @@ -121,88 +121,16 @@ create entry points for them in the ``setup.cfg`` file:: ironic.hardware.interfaces.management = my-management = ironic.drivers.modules.my_hardware:MyManagement +Deploy and clean steps +---------------------- + +Significant parts of the bare metal functionality is implemented via +:doc:`deploy steps ` or :doc:`clean steps +`. See :doc:`deploy-steps` for information on how to write +them. + Supported Drivers ----------------- For a list of supported drivers (those that are continuously tested on every upstream commit) please consult the :doc:`drivers page `. - -Node Vendor Passthru --------------------- - -Drivers may implement a passthrough API, which is accessible via -the ``/v1/nodes//vendor_passthru?method={METHOD}`` -endpoint. Beyond basic checking, Ironic does not introspect the message -body and simply "passes it through" to the relevant driver. - -A method: - -* can support one or more HTTP methods (for example, GET, POST) - -* is asynchronous or synchronous - - + For asynchronous methods, a 202 (Accepted) HTTP status code is returned - to indicate that the request was received, accepted and is being acted - upon. No body is returned in the response. - - + For synchronous methods, a 200 (OK) HTTP status code is returned to - indicate that the request was fulfilled. The response may include a body. - -* can require an exclusive lock on the node. This only occurs if the method - doesn't specify require_exclusive_lock=False in the decorator. If an - exclusive lock is held on the node, other requests for the node will be - delayed and may fail with an HTTP 409 (Conflict) error code. - -This endpoint exposes a node's driver directly, and as such, it is -expressly not part of Ironic's standard REST API. There is only a -single HTTP endpoint exposed, and the semantics of the message body -are determined solely by the driver. Ironic makes no guarantees about -backwards compatibility; this is solely up to the discretion of each -driver's author. - -To get information about all the methods available via the vendor_passthru -endpoint for a particular node, you can issue an HTTP GET request:: - - GET /v1/nodes//vendor_passthru/methods - -The response's JSON body will contain information for each method, -such as the method's name, a description, the HTTP methods supported, -and whether it's asynchronous or synchronous. - - -Driver Vendor Passthru ----------------------- - -Drivers may implement an API for requests not related to any node, -at ``/v1/drivers//vendor_passthru?method={METHOD}``. - -A method: - -* can support one or more HTTP methods (for example, GET, POST) - -* is asynchronous or synchronous - - + For asynchronous methods, a 202 (Accepted) HTTP status code is - returned to indicate that the request was received, accepted and is - being acted upon. No body is returned in the response. - - + For synchronous methods, a 200 (OK) HTTP status code is returned - to indicate that the request was fulfilled. The response may include - a body. - -.. note:: - Unlike methods in `Node Vendor Passthru`_, a request does not lock any - resource, so it will not delay other requests and will not fail with an - HTTP 409 (Conflict) error code. - -Ironic makes no guarantees about the semantics of the message BODY sent -to this endpoint. That is left up to each driver's author. - -To get information about all the methods available via the driver -vendor_passthru endpoint, you can issue an HTTP GET request:: - - GET /v1/drivers//vendor_passthru/methods - -The response's JSON body will contain information for each method, -such as the method's name, a description, the HTTP methods supported, -and whether it's asynchronous or synchronous.