How we use the powerful new Icinga PHP Library to build the Icinga DB Web UI

by | May 13, 2020

When building Icinga DB Web we completely rewrote the basic Icinga Web UI. The “monitoring module”, which it was called before.
While this existed for some time and the and the underlying concepts were already used in the Icinga Director, we took the chance and brought our Web UI to a new level.

These are the basic concepts

In Icinga IPL every UI element is represented by a PHP class. Besides the logic for handling data, all IPL components feature an assemble() method, where the markup for the UI is put together and then later rendered as HTML. Components can use other IPL components as well.
This offers developers the following advantages.

  • more object oriented approach to building UI
  • consolidated look and feel of UI elements
  • more control over single elements
  • default UI areas, such as the controls area and the status bar can be filled with elements.
  • every IPL component can be used throughout Icinga Web
  • you can use IPL to build components for your own modules more quickly and efficiently
  • updated functionality for IPL components

How the new Host List is built

HostsController

Throughout Icinga Web there are basic UI areas, that can be filled with components:

  • Controls
  • Content
  • Footer


indexAction()
The HostsController features the indexAction() for displaying the hosts list. In this method, the data is set up and passed to the respective IPL components. The created IPL components are then added to the specific areas of the view.

Lets have a look at the code

First the title of the view is set. With the variable $db a new database connection is created. After that the host items are retrieved from the database.

$this->setTitle(t('Hosts'));
$compact = $this->view->compact;
$db = $this->getDb();
$hosts = Host::on($db)->with('state');
…

Afterwards the single control components are set up.

$limitControl = $this->createLimitControl();
$paginationControl = $this->createPaginationControl($hosts);
$sortControl = $this->createSortControl(
    $hosts,
    [ … ]
);
$viewModeSwitcher = $this->createViewModeSwitcher();
$filterControl = $this->createFilterControl($hosts);

And then added to the control area of the view.

$this->addControl($paginationControl);
$this->addControl($sortControl);
$this->addControl($limitControl);
$this->addControl($viewModeSwitcher);
$this->addControl($filterControl);
$this->addControl(new ContinueWith($this->getFilter(), Links::hostsDetails()));

The HostList object is created and gets passed the results of the hosts items from the database.

$results = $hosts->execute();
$hostList = (new HostList($results))
    ->setViewMode($viewModeSwitcher->getViewMode());
$this->addContent($hostList);

HostList

Now let’s see how the HostList is created. It’s a child class of StateList, which is also a child class of BaseItemList. The latter features the `assemble` method, which is used in the HostList.
assemble()
The assemble() method is where the markup is rendered. If one component is using other components, they can be created and added here as well.

protected function getItemClass()
{
    switch ($this->getViewMode()) {
        case 'minimal':
            return HostListItemMinimal::class;
        case 'detailed':
            return HostListItemDetailed::class;
        default:
            return HostListItem::class;
    }
}

Depending on the view mode, we have different item classes for the Host List to use. This is determined in the getItemClass() method.
Depending on the view mode the actual host list items are created and then added to the view.

foreach ($this->data as $data) {
/** @var BaseListItem|BaseTableRowItem $item */
    $item = new $itemClass($data, $this);
    $this->add($item);
}
if ($this->isEmpty()) {
    $this->setTag('div');
    $this->add(new EmptyState(t('No items found.')));
}

Have a look at the code

You can clone the Icinga DB Web repository from GitHub. Go here

You May Also Like…

Subscribe to our Newsletter

A monthly digest of the latest Icinga news, releases, articles and community topics.