Skip to content

Commit

Permalink
Add donut graphs to deployments, replica sets, daemon sets and stateful
Browse files Browse the repository at this point in the history
sets
  • Loading branch information
jhoxhaa committed Aug 4, 2023
1 parent 98abc4a commit 14bf366
Show file tree
Hide file tree
Showing 6 changed files with 228 additions and 0 deletions.
90 changes: 90 additions & 0 deletions library/Kubernetes/Donut.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,90 @@
<?php

/* Icinga Kubernetes Web | (c) 2023 Icinga GmbH | GPLv2 */

namespace Icinga\Module\Kubernetes;

use ipl\Html\BaseHtmlElement;
use ipl\Html\Html;
use ipl\Html\HtmlString;
use ipl\Html\Table;

class Donut extends BaseHtmlElement
{
protected $tag = 'section';

protected $defaultAttributes = ['class' => 'donut'];

/**
* The donut data
*
* @var iterable
*/
protected $data = [];

/**
* @var string
*/
protected $heading;

/**
* @var int
*/
protected $headingLevel = 2;

/**
* @var callable
*/
protected $labelCallback;

/**
* Set the data to display
*
* @param iterable $data
*
* @return $this
*/
public function setData(iterable $data)
{
$this->data = $data;

return $this;
}

public function setHeading(string $heading, int $level)
{
$this->heading = $heading;
$this->headingLevel = $level;

return $this;
}

public function setLabelCallback(callable $callback)
{
$this->labelCallback = $callback;

return $this;
}

public function assemble()
{
$donut = new \Icinga\Chart\Donut();
$legend = new Table();

foreach ($this->data as $index => $value) {
$donut->addSlice((int) $value, ['class' => 'segment-' . $index]);
$legend->add(
[
Html::tag('span', ['class' => 'badge badge-' . $index]),
call_user_func($this->labelCallback, $index),
$value
]
);
}

$this->addHtml(new HtmlString($donut->render()), $legend);
if ($this->heading !== null) {
$this->addHtml(Html::tag("h{$this->headingLevel}", $this->heading));
}
}
}
20 changes: 20 additions & 0 deletions library/Kubernetes/Web/DaemonSetDetail.php
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
namespace Icinga\Module\Kubernetes\Web;

use Icinga\Module\Kubernetes\Common\Database;
use Icinga\Module\Kubernetes\Donut;
use Icinga\Module\Kubernetes\Model\DaemonSet;
use Icinga\Module\Kubernetes\Model\Event;
use Icinga\Module\Kubernetes\Model\ReplicaSet;
Expand Down Expand Up @@ -49,6 +50,25 @@ protected function assemble()
t('Created') => new TimeAgo($this->daemonSet->created->getTimestamp())
]));

$data = [
$this->daemonSet->number_available,
$this->daemonSet->number_ready - $this->daemonSet->number_available,
$this->daemonSet->desired_number_scheduled - $this->daemonSet->number_ready,
$this->daemonSet->number_unavailable
];
$labels = [
t('Available'),
t('Ready but not yet available'),
t('Not yet ready'),
t('Not yet scheduled or failing')
];
$donut = (new Donut())
->setData($data)
->setLabelCallback(function ($index) use ($labels){
return HtmlElement::create('span', null, $labels[$index]);
});
$this->addHtml($donut);

$this->addHtml(new ConditionTable($this->daemonSet, (new ReplicaSetCondition())->getColumnDefinitions()));

$this->addHtml(new HtmlElement(
Expand Down
23 changes: 23 additions & 0 deletions library/Kubernetes/Web/DeploymentDetail.php
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@

namespace Icinga\Module\Kubernetes\Web;

use Icinga\Module\Kubernetes\Donut;
use Icinga\Module\Kubernetes\Model\Deployment;
use Icinga\Module\Kubernetes\Model\DeploymentCondition;
use ipl\Html\Attributes;
Expand Down Expand Up @@ -46,6 +47,28 @@ protected function assemble()
t('Created') => new TimeAgo($this->deployment->created->getTimestamp())
]));

$data = [
$this->deployment->available_replicas,
$this->deployment->desired_replicas - $this->deployment->actual_replicas,
$this->deployment->actual_replicas - $this->deployment->ready_replicas,
$this->deployment->unavailable_replicas,
];

$donut = (new Donut())
->setData($data)
->setLabelCallback(function ($index) {
$labels = [
'Available',
'Ready but not yet available',
'Not yet ready',
'Not yet scheduled or failing'
];
return HtmlElement::create('span',
null,
$labels[$index]);
});
$this->addHtml($donut);

$this->addHtml(new ConditionTable($this->deployment, (new DeploymentCondition())->getColumnDefinitions()));

$this->addHtml(new HtmlElement(
Expand Down
23 changes: 23 additions & 0 deletions library/Kubernetes/Web/ReplicaSetDetail.php
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
use Icinga\Module\Kubernetes\Model\Event;
use Icinga\Module\Kubernetes\Model\ReplicaSet;
use Icinga\Module\Kubernetes\Model\replicaSetCondition;
use Icinga\Module\Kubernetes\Donut;
use ipl\Html\Attributes;
use ipl\Html\BaseHtmlElement;
use ipl\Html\HtmlElement;
Expand Down Expand Up @@ -46,6 +47,28 @@ protected function assemble()
t('Created') => new TimeAgo($this->replicaSet->created->getTimestamp())
]));

$data = [
$this->replicaSet->available_replicas,
$this->replicaSet->ready_replicas - $this->replicaSet->available_replicas,
$this->replicaSet->actual_replicas - $this->replicaSet->ready_replicas,
$this->replicaSet->desired_replicas - $this->replicaSet->actual_replicas,
];

$donut = (new Donut())
->setData($data)
->setLabelCallback(function ($index) {
$labels = [
'Available',
'Ready but not yet available',
'Not yet ready',
'Not yet scheduled or failing'
];
return HtmlElement::create('span',
null,
$labels[$index]);
});
$this->addHtml($donut);

$this->addHtml(new ConditionTable($this->replicaSet, (new ReplicaSetCondition())->getColumnDefinitions()));

$this->addHtml(new HtmlElement(
Expand Down
23 changes: 23 additions & 0 deletions library/Kubernetes/Web/StatefulSetDetail.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

namespace Icinga\Module\Kubernetes\Web;

use Icinga\Module\Kubernetes\Donut;
use Icinga\Module\Kubernetes\Model\StatefulSet;
use Icinga\Module\Kubernetes\Model\StatefulSetCondition;
use ipl\Html\Attributes;
Expand Down Expand Up @@ -42,6 +43,28 @@ protected function assemble()
t('Created') => new TimeAgo($this->statefulSet->created->getTimestamp())
]));

$data = [
$this->statefulSet->available_replicas,
$this->statefulSet->ready_replicas - $this->statefulSet->available_replicas,
$this->statefulSet->actual_replicas - $this->statefulSet->ready_replicas,
$this->statefulSet->desired_replicas - $this->statefulSet->actual_replicas
];

$donut = (new Donut())
->setData($data)
->setLabelCallback(function ($index) {
$labels = [
'Available',
'Ready but not yet available',
'Not yet ready',
'Not yet scheduled or failing'
];
return HtmlElement::create('span',
null,
$labels[$index]);
});
$this->addHtml($donut);

$this->addHtml(new ConditionTable($this->statefulSet, (new StatefulSetCondition())->getColumnDefinitions()));

$this->addHtml(new HtmlElement(
Expand Down
49 changes: 49 additions & 0 deletions public/css/module.less
Original file line number Diff line number Diff line change
Expand Up @@ -433,3 +433,52 @@ body {
#grid li:nth-child(6n+4) {
margin-left: 0.5%;
}

@donut-segment-color-0: #44bb77;
@donut-segment-color-1: #77aaff;
@donut-segment-color-2: #aa44ff;
@donut-segment-color-3: #ff5566;

.donut {
align-self: flex-start;
padding: 1em;


.donut-graph {
.segment-0 {
stroke: @donut-segment-color-0;
}

.segment-1 {
stroke: @donut-segment-color-1;
}

.segment-2 {
stroke: @donut-segment-color-2;
}

.segment-3 {
stroke: @donut-segment-color-3;
}
}

.badge {
height: 1.75em;

&.badge-0 {
background: @donut-segment-color-0;
}

&.badge-1 {
background: @donut-segment-color-1;
}

&.badge-2 {
background: @donut-segment-color-2;
}

&.badge-3 {
background: @donut-segment-color-3;
}
}
}

0 comments on commit 14bf366

Please sign in to comment.