Skip to content

Commit

Permalink
Implement object hook to get event object information from respective…
Browse files Browse the repository at this point in the history
… sources
  • Loading branch information
raviks789 committed May 17, 2024
1 parent e044065 commit 59dd1e6
Show file tree
Hide file tree
Showing 7 changed files with 206 additions and 64 deletions.
119 changes: 119 additions & 0 deletions library/Notifications/Hook/EventsObjectsInfoHook.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,119 @@
<?php

/* Icinga Notifications Web | (c) 2024 Icinga GmbH | GPLv2 */

namespace Icinga\Module\Notifications\Hook;

use Exception;
use Icinga\Application\Hook;
use Icinga\Application\Logger;
use Icinga\Module\Notifications\Model\Objects;
use Icinga\Module\Notifications\Model\Source;
use ipl\Html\HtmlString;
use ipl\Html\ValidHtml;
use ipl\Orm\Query;

abstract class EventsObjectsInfoHook
{
/** @var array<string, ValidHtml> */
protected static $objectNameWidgets = [];

/**
* Get the object name widgets for the objects using the object ID tags
*
* @param array<string, array<string, string>> $objectIdTags
*
* @return array<string, ValidHtml>
*/
abstract public function getObjectNameWidgets(array $objectIdTags): array;

/**
* Get the source of the objects
*
* @return string
*/
abstract public function getSource(): string;

/**
* Get the object list item widget for the given object ID tag
*
* @param array<string, string> $objectIdTag
*
* @return ValidHtml
*/
abstract public function getHtmlForObject(array $objectIdTag): ValidHtml;

/**
* Register object name widgets for the objects using the object ID tags
*
* @param array<string, array<string, array<string, string>>> $objectIdTags
*
* @return void
*/
final public static function registerObjectNameWidgets(array $objectIdTags): void
{
$objectDisplayNames = [];
foreach (Hook::all('Notifications\\EventsObjectsInfo') as $hook) {
/** @var self $hook */
try {
$objectDisplayNames = array_merge(
$objectDisplayNames,
$hook->getObjectNameWidgets($objectIdTags[$hook->getSource()])
);
} catch (Exception $e) {
Logger::error('Failed to load hook %s:', get_class($hook), $e);
}
}

self::$objectNameWidgets = $objectDisplayNames;
}

/**
* Get the display name for the given object
*
* @param Objects $obj
*
* @return ValidHtml
*/
final public static function getObjectNameWidget(Objects $obj): ValidHtml
{
/** @var Query|Source $source */
$source = $obj->source;
if ($source instanceof Query) {
$source = $obj->source->first();
}

$objectTags = [];

foreach ($obj->id_tags as $tag => $value) {
$objectTags[] = sprintf('%s=%s', $tag, $value);
}

return self::$objectNameWidgets[$source->type][$obj->id] ?? new HtmlString(implode(', ', $objectTags));
}

/**
* Get the object list item widget for the object using the object ID tag
*
* @param string $source
* @param array<string, string> $objectIdTag
*
* @return ?ValidHtml
*/
final public static function getObjectListItemWidget(string $source, array $objectIdTag): ?ValidHtml
{
$objectListItemWidget = null;
foreach (Hook::all('Notifications\\EventsObjectsInfo') as $hook) {
/** @var self $hook */
try {
if ($source === $hook->getSource()) {
$objectListItemWidget = $hook->getHtmlForObject($objectIdTag);
}
} catch (Exception $e) {
Logger::error('Failed to load hook %s:', get_class($hook), $e);
}
}

return $objectListItemWidget;
}
}
11 changes: 2 additions & 9 deletions library/Notifications/Model/Objects.php
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,8 @@

namespace Icinga\Module\Notifications\Model;

use Icinga\Module\Notifications\Hook\EventsObjectsInfoHook;
use Icinga\Module\Notifications\Model\Behavior\IdTagAggregator;
use ipl\Html\HtmlString;
use ipl\Html\ValidHtml;
use ipl\Orm\Behavior\Binary;
use ipl\Orm\Behaviors;
Expand Down Expand Up @@ -91,13 +91,6 @@ public function createRelations(Relations $relations)

public function getName(): ValidHtml
{
//TODO: Once hooks are available, they should render the tags accordingly
$objectTags = [];

foreach ($this->id_tags as $tag => $value) {
$objectTags[] = sprintf('%s=%s', $tag, $value);
}

return new HtmlString(implode(', ', $objectTags));
return EventsObjectsInfoHook::getObjectNameWidget($this);
}
}
38 changes: 13 additions & 25 deletions library/Notifications/Widget/Detail/EventDetail.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,15 +5,18 @@
namespace Icinga\Module\Notifications\Widget\Detail;

use Icinga\Date\DateFormatter;
use Icinga\Module\Notifications\Hook\EventsObjectsInfoHook;
use Icinga\Module\Notifications\Model\Event;
use Icinga\Module\Notifications\Model\Incident;
use Icinga\Module\Notifications\Model\Objects;
use Icinga\Module\Notifications\Model\Source;
use Icinga\Module\Notifications\Widget\EventSourceBadge;
use Icinga\Module\Notifications\Widget\ItemList\IncidentList;
use InvalidArgumentException;
use ipl\Html\BaseHtmlElement;
use ipl\Html\Html;
use ipl\Html\ValidHtml;
use ipl\Web\Url;
use ipl\Web\Widget\HorizontalKeyValue;
use ipl\Web\Widget\Link;
use ipl\Web\Widget\StateBall;
Expand Down Expand Up @@ -80,37 +83,22 @@ protected function createMessage(): array
/** @return ValidHtml[] */
protected function createRelatedObject(): array
{
//TODO(sd): This is just placeholder. Add hook implementation instead
$relatedObj = Html::tag('ul', ['class' => ['item-list', 'action-list'], 'data-base-target' => '_next']);

/** @var Objects $obj */
$obj = $this->event->object;

/** @var string $objUrl */
$objUrl = $obj->url;
$relatedObj->add(
Html::tag(
'li',
['class' => 'list-item', 'data-action-item' => true],
[ //TODO(sd): fix stateball
Html::tag('div', ['class' => 'visual'], new StateBall('down', StateBall::SIZE_LARGE)),
Html::tag(
'div',
['class' => 'main'],
Html::tag('header')
->add(Html::tag(
'div',
['class' => 'title'],
new Link($obj->getName(), $objUrl, ['class' => 'subject'])
))
)
]
)
);
/** @var Source $source */
$source = $obj->source;

$objWidget = EventsObjectsInfoHook::getObjectListItemWidget($source->name, $obj->id_tags);
$objUrl = Url::fromPath($obj->url);

return [
Html::tag('h2', t('Related Object')),
$relatedObj
$objWidget ?? (new Link(
$obj->getName(),
$objUrl->isExternal() ? $objUrl->getAbsoluteUrl() : $objUrl->getRelativeUrl(),
['class' => 'subject']
))->setBaseTarget('_next')
];
}

Expand Down
42 changes: 14 additions & 28 deletions library/Notifications/Widget/Detail/IncidentDetail.php
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,10 @@

namespace Icinga\Module\Notifications\Widget\Detail;

use Icinga\Module\Notifications\Hook\EventsObjectsInfoHook;
use Icinga\Module\Notifications\Model\Incident;
use Icinga\Module\Notifications\Model\Objects;
use Icinga\Module\Notifications\Model\Source;
use Icinga\Module\Notifications\Widget\EventSourceBadge;
use Icinga\Module\Notifications\Widget\ItemList\IncidentContactList;
use Icinga\Module\Notifications\Widget\ItemList\IncidentHistoryList;
Expand All @@ -14,8 +16,8 @@
use ipl\Html\Html;
use ipl\Html\HtmlElement;
use ipl\Html\Table;
use ipl\Web\Url;
use ipl\Web\Widget\Link;
use ipl\Web\Widget\StateBall;

class IncidentDetail extends BaseHtmlElement
{
Expand Down Expand Up @@ -52,39 +54,23 @@ protected function createContacts()

protected function createRelatedObject()
{
//TODO(sd): Add hook implementation
$list = Html::tag('ul', ['class' => ['item-list', 'minimal', 'action-list'], 'data-base-target' => '_next']);

/** @var Objects $obj */
$obj = $this->incident->object;

/** @var string $objUrl */
$objUrl = $obj->url;
$list->add(Html::tag(
'li',
['class' => 'list-item', 'data-action-item' => true],
[ //TODO(sd): fix stateball
Html::tag(
'div',
['class' => 'visual'],
new StateBall('down', StateBall::SIZE_LARGE)
),
Html::tag(
'div',
['class' => 'main'],
Html::tag('header')
->add(Html::tag(
'div',
['class' => 'title'],
new Link($obj->getName(), $objUrl, ['class' => 'subject'])
))
)
]
));
/** @var Source $source */
$source = $obj->source;

$objWidget = EventsObjectsInfoHook::getObjectListItemWidget($source->name, $obj->id_tags);

/** @var Url $objUrl */
$objUrl = Url::fromPath($obj->url);
return [
Html::tag('h2', t('Object')),
$list
$objWidget ?? (new Link(
$obj->getName(),
$objUrl->isExternal() ? $objUrl->getAbsoluteUrl() : $objUrl->getRelativeUrl(),
['class' => 'subject']
))->setBaseTarget('_next')
];
}

Expand Down
37 changes: 37 additions & 0 deletions library/Notifications/Widget/ItemList/EventList.php
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,11 @@

use Icinga\Module\Notifications\Common\LoadMore;
use Icinga\Module\Notifications\Common\NoSubjectLink;
use Icinga\Module\Notifications\Hook\EventsObjectsInfoHook;
use Icinga\Module\Notifications\Model\Event;
use Icinga\Module\Notifications\Model\Objects;
use Icinga\Module\Notifications\Model\Source;
use ipl\Html\ValidHtml;
use ipl\Orm\ResultSet;
use ipl\Web\Common\BaseItemList;

Expand All @@ -19,6 +24,12 @@ class EventList extends BaseItemList
/** @var ResultSet */
protected $data;

/** @var array<string, array<string, array<string, string>>> Object ID tags for each source */
public $objectIdTagsCache = [];

/** @var array<string, ValidHtml> Object display names obtained from the corresponding web modules of the sources */
public $renderedObjectDisplayNames = [];

public function __construct(ResultSet $data)
{
parent::__construct($data);
Expand All @@ -27,10 +38,36 @@ public function __construct(ResultSet $data)
protected function init(): void
{
$this->data = $this->getIterator($this->data);

$this->on(self::ON_ITEM_ADD, function (EventListItem $item, Event $data) {
/** @var Objects $obj */
$obj = $data->object;
/** @var Source $src */
$src = $obj->source;
$this->objectIdTagsCache[$src->type][$obj->id] = $obj->id_tags;
});

$this->on(self::ON_ASSEMBLED, function () {
EventsObjectsInfoHook::registerObjectNameWidgets(
$this->objectIdTagsCache
);
});
}

protected function getItemClass(): string
{
return EventListItem::class;
}

/**
* Get the rendered object display name for the given object ID
*
* @param string $objectID
*
* @return ?ValidHtml
*/
public function getRenderedObjectDisplayName(string $objectID): ?ValidHtml
{
return $this->renderedObjectDisplayNames[$objectID] ?? null;
}
}
7 changes: 5 additions & 2 deletions library/Notifications/Widget/ItemList/EventListItem.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
namespace Icinga\Module\Notifications\Widget\ItemList;

use Icinga\Module\Notifications\Common\Links;
use Icinga\Module\Notifications\Hook\EventsObjectsInfoHook;
use Icinga\Module\Notifications\Model\Event;
use Icinga\Module\Notifications\Model\Incident;
use Icinga\Module\Notifications\Model\Objects;
Expand All @@ -13,6 +14,7 @@
use Icinga\Module\Notifications\Widget\SourceIcon;
use InvalidArgumentException;
use ipl\Html\BaseHtmlElement;
use ipl\Html\DeferredText;
use ipl\Html\Html;
use ipl\Stdlib\Str;
use ipl\Web\Common\BaseListItem;
Expand Down Expand Up @@ -105,14 +107,15 @@ protected function assembleTitle(BaseHtmlElement $title): void

/** @var Objects $obj */
$obj = $this->item->object;
$name = $obj->getName();
$name = (new DeferredText(function () use ($obj) {
return $obj->getName();
}))->setEscaped();
if (! $this->list->getNoSubjectLink()) {
$content = new Link($name, Links::event($this->item->id), ['class' => 'subject']);
} else {
$content = Html::tag('span', ['class' => 'subject'], $name);
}

$msg = null;
if ($this->item->severity === null) {
$description = strtolower(trim($this->item->message ?? ''));
if (Str::startsWith($description, 'incident reached age')) {
Expand Down
Loading

0 comments on commit 59dd1e6

Please sign in to comment.