mirror of https://github.com/Icinga/icinga2.git
1033 lines
36 KiB
C++
1033 lines
36 KiB
C++
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
|
|
|
#include "icinga/downtime.hpp"
|
|
#include "icinga/host.hpp"
|
|
#include "icinga/service.hpp"
|
|
#include <BoostTestTargetConfig.h>
|
|
#include <iostream>
|
|
#include <sstream>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
using namespace icinga;
|
|
|
|
BOOST_AUTO_TEST_SUITE(icinga_checkresult)
|
|
|
|
static CheckResult::Ptr MakeCheckResult(ServiceState state)
|
|
{
|
|
CheckResult::Ptr cr = new CheckResult();
|
|
|
|
cr->SetState(state);
|
|
|
|
double now = Utility::GetTime();
|
|
cr->SetScheduleStart(now);
|
|
cr->SetScheduleEnd(now);
|
|
cr->SetExecutionStart(now);
|
|
cr->SetExecutionEnd(now);
|
|
|
|
return cr;
|
|
}
|
|
|
|
static void NotificationHandler(const Checkable::Ptr& checkable, NotificationType type)
|
|
{
|
|
std::cout << "Notification triggered: " << Notification::NotificationTypeToString(type) << std::endl;
|
|
|
|
checkable->SetExtension("requested_notifications", true);
|
|
checkable->SetExtension("notification_type", type);
|
|
}
|
|
|
|
static void CheckNotification(const Checkable::Ptr& checkable, bool expected, NotificationType type = NotificationRecovery)
|
|
{
|
|
BOOST_CHECK((expected && checkable->GetExtension("requested_notifications").ToBool()) || (!expected && !checkable->GetExtension("requested_notifications").ToBool()));
|
|
|
|
if (expected && checkable->GetExtension("requested_notifications").ToBool())
|
|
BOOST_CHECK(checkable->GetExtension("notification_type") == type);
|
|
|
|
checkable->SetExtension("requested_notifications", false);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(host_1attempt)
|
|
{
|
|
boost::signals2::connection c = Checkable::OnNotificationsRequested.connect([](const Checkable::Ptr& checkable, NotificationType type,
|
|
const CheckResult::Ptr&, const String&, const String&, const MessageOrigin::Ptr&) {
|
|
NotificationHandler(checkable, type);
|
|
});
|
|
|
|
Host::Ptr host = new Host();
|
|
host->SetActive(true);
|
|
host->SetMaxCheckAttempts(1);
|
|
host->Activate();
|
|
host->SetAuthority(true);
|
|
host->SetStateRaw(ServiceOK);
|
|
host->SetStateType(StateTypeHard);
|
|
|
|
std::cout << "Before first check result (ok, hard)" << std::endl;
|
|
BOOST_CHECK(host->GetState() == HostUp);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, false);
|
|
|
|
std::cout << "First check result (unknown)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceUnknown));
|
|
BOOST_CHECK(host->GetState() == HostDown);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, true, NotificationProblem);
|
|
|
|
std::cout << "Second check result (ok)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(host->GetState() == HostUp);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, true, NotificationRecovery);
|
|
|
|
std::cout << "Third check result (critical)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(host->GetState() == HostDown);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, true, NotificationProblem);
|
|
|
|
std::cout << "Fourth check result (ok)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(host->GetState() == HostUp);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, true, NotificationRecovery);
|
|
|
|
c.disconnect();
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(host_2attempts)
|
|
{
|
|
boost::signals2::connection c = Checkable::OnNotificationsRequested.connect([](const Checkable::Ptr& checkable, NotificationType type,
|
|
const CheckResult::Ptr&, const String&, const String&, const MessageOrigin::Ptr&) {
|
|
NotificationHandler(checkable, type);
|
|
});
|
|
|
|
Host::Ptr host = new Host();
|
|
host->SetActive(true);
|
|
host->SetMaxCheckAttempts(2);
|
|
host->Activate();
|
|
host->SetAuthority(true);
|
|
host->SetStateRaw(ServiceOK);
|
|
host->SetStateType(StateTypeHard);
|
|
|
|
std::cout << "Before first check result (ok, hard)" << std::endl;
|
|
BOOST_CHECK(host->GetState() == HostUp);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, false);
|
|
|
|
std::cout << "First check result (unknown)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceUnknown));
|
|
BOOST_CHECK(host->GetState() == HostDown);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeSoft);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, false);
|
|
|
|
std::cout << "Second check result (critical)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(host->GetState() == HostDown);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, true, NotificationProblem);
|
|
|
|
std::cout << "Third check result (ok)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(host->GetState() == HostUp);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, true, NotificationRecovery);
|
|
|
|
std::cout << "Fourth check result (critical)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(host->GetState() == HostDown);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeSoft);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, false);
|
|
|
|
std::cout << "Fifth check result (ok)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(host->GetState() == HostUp);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, false);
|
|
|
|
c.disconnect();
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(host_3attempts)
|
|
{
|
|
boost::signals2::connection c = Checkable::OnNotificationsRequested.connect([](const Checkable::Ptr& checkable, NotificationType type,
|
|
const CheckResult::Ptr&, const String&, const String&, const MessageOrigin::Ptr&) {
|
|
NotificationHandler(checkable, type);
|
|
});
|
|
|
|
Host::Ptr host = new Host();
|
|
host->SetActive(true);
|
|
host->SetMaxCheckAttempts(3);
|
|
host->Activate();
|
|
host->SetAuthority(true);
|
|
host->SetStateRaw(ServiceOK);
|
|
host->SetStateType(StateTypeHard);
|
|
|
|
std::cout << "Before first check result (ok, hard)" << std::endl;
|
|
BOOST_CHECK(host->GetState() == HostUp);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, false);
|
|
|
|
std::cout << "First check result (unknown)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceUnknown));
|
|
BOOST_CHECK(host->GetState() == HostDown);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeSoft);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, false);
|
|
|
|
std::cout << "Second check result (critical)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(host->GetState() == HostDown);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeSoft);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 2);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, false);
|
|
|
|
std::cout << "Third check result (critical)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(host->GetState() == HostDown);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, true, NotificationProblem);
|
|
|
|
std::cout << "Fourth check result (ok)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(host->GetState() == HostUp);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, true, NotificationRecovery);
|
|
|
|
std::cout << "Fifth check result (critical)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(host->GetState() == HostDown);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeSoft);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, false);
|
|
|
|
std::cout << "Sixth check result (ok)" << std::endl;
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(host->GetState() == HostUp);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(host->IsReachable() == true);
|
|
CheckNotification(host, false);
|
|
|
|
c.disconnect();
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(service_1attempt)
|
|
{
|
|
boost::signals2::connection c = Checkable::OnNotificationsRequested.connect([](const Checkable::Ptr& checkable, NotificationType type,
|
|
const CheckResult::Ptr&, const String&, const String&, const MessageOrigin::Ptr&) {
|
|
NotificationHandler(checkable, type);
|
|
});
|
|
|
|
Service::Ptr service = new Service();
|
|
service->SetActive(true);
|
|
service->SetMaxCheckAttempts(1);
|
|
service->Activate();
|
|
service->SetAuthority(true);
|
|
service->SetStateRaw(ServiceOK);
|
|
service->SetStateType(StateTypeHard);
|
|
|
|
std::cout << "Before first check result (ok, hard)" << std::endl;
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, false);
|
|
|
|
std::cout << "First check result (unknown)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceUnknown));
|
|
BOOST_CHECK(service->GetState() == ServiceUnknown);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, true, NotificationProblem);
|
|
|
|
std::cout << "Second check result (ok)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, true, NotificationRecovery);
|
|
|
|
std::cout << "Third check result (critical)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(service->GetState() == ServiceCritical);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, true, NotificationProblem);
|
|
|
|
std::cout << "Fourth check result (ok)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, true, NotificationRecovery);
|
|
|
|
c.disconnect();
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(service_2attempts)
|
|
{
|
|
boost::signals2::connection c = Checkable::OnNotificationsRequested.connect([](const Checkable::Ptr& checkable, NotificationType type,
|
|
const CheckResult::Ptr&, const String&, const String&, const MessageOrigin::Ptr&) {
|
|
NotificationHandler(checkable, type);
|
|
});
|
|
|
|
Service::Ptr service = new Service();
|
|
service->SetActive(true);
|
|
service->SetMaxCheckAttempts(2);
|
|
service->Activate();
|
|
service->SetAuthority(true);
|
|
service->SetStateRaw(ServiceOK);
|
|
service->SetStateType(StateTypeHard);
|
|
|
|
std::cout << "Before first check result (ok, hard)" << std::endl;
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, false);
|
|
|
|
std::cout << "First check result (unknown)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceUnknown));
|
|
BOOST_CHECK(service->GetState() == ServiceUnknown);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeSoft);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, false);
|
|
|
|
std::cout << "Second check result (critical)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(service->GetState() == ServiceCritical);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, true, NotificationProblem);
|
|
|
|
std::cout << "Third check result (ok)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, true, NotificationRecovery);
|
|
|
|
std::cout << "Fourth check result (critical)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(service->GetState() == ServiceCritical);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeSoft);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, false);
|
|
|
|
std::cout << "Fifth check result (ok)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, false);
|
|
|
|
c.disconnect();
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(service_3attempts)
|
|
{
|
|
boost::signals2::connection c = Checkable::OnNotificationsRequested.connect([](const Checkable::Ptr& checkable, NotificationType type,
|
|
const CheckResult::Ptr&, const String&, const String&, const MessageOrigin::Ptr&) {
|
|
NotificationHandler(checkable, type);
|
|
});
|
|
|
|
Service::Ptr service = new Service();
|
|
service->SetActive(true);
|
|
service->SetMaxCheckAttempts(3);
|
|
service->Activate();
|
|
service->SetAuthority(true);
|
|
service->SetStateRaw(ServiceOK);
|
|
service->SetStateType(StateTypeHard);
|
|
|
|
std::cout << "Before first check result (ok, hard)" << std::endl;
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, false);
|
|
|
|
std::cout << "First check result (unknown)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceUnknown));
|
|
BOOST_CHECK(service->GetState() == ServiceUnknown);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeSoft);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, false);
|
|
|
|
std::cout << "Second check result (critical)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(service->GetState() == ServiceCritical);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeSoft);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 2);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, false);
|
|
|
|
std::cout << "Third check result (critical)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(service->GetState() == ServiceCritical);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, true, NotificationProblem);
|
|
|
|
std::cout << "Fourth check result (ok)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, true, NotificationRecovery);
|
|
|
|
std::cout << "Fifth check result (critical)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
BOOST_CHECK(service->GetState() == ServiceCritical);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeSoft);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, false);
|
|
|
|
std::cout << "Sixth check result (ok)" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
BOOST_CHECK(service->IsReachable() == true);
|
|
CheckNotification(service, false);
|
|
|
|
c.disconnect();
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(host_flapping_notification)
|
|
{
|
|
#ifndef I2_DEBUG
|
|
BOOST_WARN_MESSAGE(false, "This test can only be run in a debug build!");
|
|
#else /* I2_DEBUG */
|
|
boost::signals2::connection c = Checkable::OnNotificationsRequested.connect([](const Checkable::Ptr& checkable, NotificationType type,
|
|
const CheckResult::Ptr&, const String&, const String&, const MessageOrigin::Ptr&) {
|
|
NotificationHandler(checkable, type);
|
|
});
|
|
|
|
int timeStepInterval = 60;
|
|
|
|
Host::Ptr host = new Host();
|
|
host->SetActive(true);
|
|
host->Activate();
|
|
host->SetAuthority(true);
|
|
host->SetStateRaw(ServiceOK);
|
|
host->SetStateType(StateTypeHard);
|
|
host->SetEnableFlapping(true);
|
|
|
|
/* Initialize start time */
|
|
Utility::SetTime(0);
|
|
|
|
std::cout << "Before first check result (ok, hard)" << std::endl;
|
|
BOOST_CHECK(host->GetState() == HostUp);
|
|
BOOST_CHECK(host->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(host->GetCheckAttempt() == 1);
|
|
|
|
Utility::IncrementTime(timeStepInterval);
|
|
|
|
std::cout << "Inserting flapping check results" << std::endl;
|
|
|
|
for (int i = 0; i < 10; i++) {
|
|
ServiceState state = (i % 2 == 0 ? ServiceOK : ServiceCritical);
|
|
host->ProcessCheckResult(MakeCheckResult(state));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
}
|
|
|
|
BOOST_CHECK(host->IsFlapping() == true);
|
|
|
|
CheckNotification(host, true, NotificationFlappingStart);
|
|
|
|
std::cout << "Now calm down..." << std::endl;
|
|
|
|
for (int i = 0; i < 20; i++) {
|
|
host->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
}
|
|
|
|
CheckNotification(host, true, NotificationFlappingEnd);
|
|
|
|
|
|
c.disconnect();
|
|
|
|
#endif /* I2_DEBUG */
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(service_flapping_notification)
|
|
{
|
|
#ifndef I2_DEBUG
|
|
BOOST_WARN_MESSAGE(false, "This test can only be run in a debug build!");
|
|
#else /* I2_DEBUG */
|
|
boost::signals2::connection c = Checkable::OnNotificationsRequested.connect([](const Checkable::Ptr& checkable, NotificationType type,
|
|
const CheckResult::Ptr&, const String&, const String&, const MessageOrigin::Ptr&) {
|
|
NotificationHandler(checkable, type);
|
|
});
|
|
|
|
int timeStepInterval = 60;
|
|
|
|
Service::Ptr service = new Service();
|
|
service->SetActive(true);
|
|
service->Activate();
|
|
service->SetAuthority(true);
|
|
service->SetStateRaw(ServiceOK);
|
|
service->SetStateType(StateTypeHard);
|
|
service->SetEnableFlapping(true);
|
|
|
|
/* Initialize start time */
|
|
Utility::SetTime(0);
|
|
|
|
std::cout << "Before first check result (ok, hard)" << std::endl;
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
|
|
Utility::IncrementTime(timeStepInterval);
|
|
|
|
std::cout << "Inserting flapping check results" << std::endl;
|
|
|
|
for (int i = 0; i < 10; i++) {
|
|
ServiceState state = (i % 2 == 0 ? ServiceOK : ServiceCritical);
|
|
service->ProcessCheckResult(MakeCheckResult(state));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
}
|
|
|
|
BOOST_CHECK(service->IsFlapping() == true);
|
|
|
|
CheckNotification(service, true, NotificationFlappingStart);
|
|
|
|
|
|
|
|
std::cout << "Now calm down..." << std::endl;
|
|
|
|
for (int i = 0; i < 20; i++) {
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
}
|
|
|
|
CheckNotification(service, true, NotificationFlappingEnd);
|
|
|
|
c.disconnect();
|
|
|
|
#endif /* I2_DEBUG */
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(service_flapping_problem_notifications)
|
|
{
|
|
#ifndef I2_DEBUG
|
|
BOOST_WARN_MESSAGE(false, "This test can only be run in a debug build!");
|
|
#else /* I2_DEBUG */
|
|
boost::signals2::connection c = Checkable::OnNotificationsRequested.connect([](const Checkable::Ptr& checkable, NotificationType type,
|
|
const CheckResult::Ptr&, const String&, const String&, const MessageOrigin::Ptr&) {
|
|
NotificationHandler(checkable, type);
|
|
});
|
|
|
|
int timeStepInterval = 60;
|
|
|
|
Service::Ptr service = new Service();
|
|
service->Activate();
|
|
service->SetAuthority(true);
|
|
service->SetStateRaw(ServiceOK);
|
|
service->SetStateType(StateTypeHard);
|
|
service->SetEnableFlapping(true);
|
|
service->SetMaxCheckAttempts(3);
|
|
|
|
/* Initialize start time */
|
|
Utility::SetTime(0);
|
|
|
|
std::cout << "Before first check result (ok, hard)" << std::endl;
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
|
|
Utility::IncrementTime(timeStepInterval);
|
|
|
|
std::cout << "Inserting flapping check results" << std::endl;
|
|
|
|
for (int i = 0; i < 10; i++) {
|
|
ServiceState state = (i % 2 == 0 ? ServiceOK : ServiceCritical);
|
|
service->ProcessCheckResult(MakeCheckResult(state));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
}
|
|
|
|
BOOST_CHECK(service->IsFlapping() == true);
|
|
|
|
CheckNotification(service, true, NotificationFlappingStart);
|
|
|
|
//Insert enough check results to get into hard problem state but staying flapping
|
|
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
|
|
|
|
BOOST_CHECK(service->IsFlapping() == true);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetState() == ServiceCritical);
|
|
|
|
CheckNotification(service, false, NotificationProblem);
|
|
|
|
// Calm down
|
|
while (service->IsFlapping()) {
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
}
|
|
|
|
CheckNotification(service, true, NotificationFlappingEnd);
|
|
|
|
/* Intended behaviour is a Problem notification being sent as well, but there are is a Problem:
|
|
* We don't know whether the Object was Critical before we started flapping and sent out a Notification.
|
|
* A notification will not be sent, no matter how many criticals follow.
|
|
*
|
|
* service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
* CheckNotification(service, true, NotificationProblem);
|
|
* ^ This fails, no notification will be sent
|
|
*
|
|
* There is also a different issue, when we receive a OK check result, a Recovery Notification will be sent
|
|
* since the service went from hard critical into soft ok. Yet there is no fitting critical notification.
|
|
* This should not happen:
|
|
*
|
|
* service->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
* CheckNotification(service, false, NotificationRecovery);
|
|
* ^ This fails, recovery is sent
|
|
*/
|
|
|
|
BOOST_CHECK(service->IsFlapping() == false);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetState() == ServiceCritical);
|
|
|
|
// Known failure, see #5713
|
|
// CheckNotification(service, true, NotificationProblem);
|
|
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
|
|
// Known failure, see #5713
|
|
// CheckNotification(service, true, NotificationRecovery);
|
|
|
|
c.disconnect();
|
|
|
|
#endif /* I2_DEBUG */
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(service_flapping_ok_into_bad)
|
|
{
|
|
#ifndef I2_DEBUG
|
|
BOOST_WARN_MESSAGE(false, "This test can only be run in a debug build!");
|
|
#else /* I2_DEBUG */
|
|
boost::signals2::connection c = Checkable::OnNotificationsRequested.connect([](const Checkable::Ptr& checkable, NotificationType type,
|
|
const CheckResult::Ptr&, const String&, const String&, const MessageOrigin::Ptr&) {
|
|
NotificationHandler(checkable, type);
|
|
});
|
|
|
|
int timeStepInterval = 60;
|
|
|
|
Service::Ptr service = new Service();
|
|
service->Activate();
|
|
service->SetAuthority(true);
|
|
service->SetStateRaw(ServiceOK);
|
|
service->SetStateType(StateTypeHard);
|
|
service->SetEnableFlapping(true);
|
|
service->SetMaxCheckAttempts(3);
|
|
|
|
/* Initialize start time */
|
|
Utility::SetTime(0);
|
|
|
|
std::cout << "Before first check result (ok, hard)" << std::endl;
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
|
|
Utility::IncrementTime(timeStepInterval);
|
|
|
|
std::cout << "Inserting flapping check results" << std::endl;
|
|
|
|
for (int i = 0; i < 10; i++) {
|
|
ServiceState state = (i % 2 == 0 ? ServiceOK : ServiceCritical);
|
|
service->ProcessCheckResult(MakeCheckResult(state));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
}
|
|
|
|
BOOST_CHECK(service->IsFlapping() == true);
|
|
|
|
CheckNotification(service, true, NotificationFlappingStart);
|
|
|
|
//Insert enough check results to get into hard problem state but staying flapping
|
|
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
|
|
|
|
BOOST_CHECK(service->IsFlapping() == true);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetState() == ServiceCritical);
|
|
|
|
CheckNotification(service, false, NotificationProblem);
|
|
|
|
// Calm down
|
|
while (service->IsFlapping()) {
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
}
|
|
|
|
CheckNotification(service, true, NotificationFlappingEnd);
|
|
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
|
|
BOOST_CHECK(service->IsFlapping() == false);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetState() == ServiceCritical);
|
|
|
|
// We expect a problem notification here
|
|
// Known failure, see #5713
|
|
// CheckNotification(service, true, NotificationProblem);
|
|
|
|
c.disconnect();
|
|
|
|
#endif /* I2_DEBUG */
|
|
}
|
|
BOOST_AUTO_TEST_CASE(service_flapping_ok_over_bad_into_ok)
|
|
{
|
|
#ifndef I2_DEBUG
|
|
BOOST_WARN_MESSAGE(false, "This test can only be run in a debug build!");
|
|
#else /* I2_DEBUG */
|
|
boost::signals2::connection c = Checkable::OnNotificationsRequested.connect([](const Checkable::Ptr& checkable, NotificationType type,
|
|
const CheckResult::Ptr&, const String&, const String&, const MessageOrigin::Ptr&) {
|
|
NotificationHandler(checkable, type);
|
|
});
|
|
|
|
int timeStepInterval = 60;
|
|
|
|
Service::Ptr service = new Service();
|
|
service->Activate();
|
|
service->SetAuthority(true);
|
|
service->SetStateRaw(ServiceOK);
|
|
service->SetStateType(StateTypeHard);
|
|
service->SetEnableFlapping(true);
|
|
service->SetMaxCheckAttempts(3);
|
|
|
|
/* Initialize start time */
|
|
Utility::SetTime(0);
|
|
|
|
std::cout << "Before first check result (ok, hard)" << std::endl;
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetCheckAttempt() == 1);
|
|
|
|
Utility::IncrementTime(timeStepInterval);
|
|
|
|
std::cout << "Inserting flapping check results" << std::endl;
|
|
|
|
for (int i = 0; i < 10; i++) {
|
|
ServiceState state = (i % 2 == 0 ? ServiceOK : ServiceCritical);
|
|
service->ProcessCheckResult(MakeCheckResult(state));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
}
|
|
|
|
BOOST_CHECK(service->IsFlapping() == true);
|
|
|
|
CheckNotification(service, true, NotificationFlappingStart);
|
|
|
|
//Insert enough check results to get into hard problem state but staying flapping
|
|
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
|
|
|
|
BOOST_CHECK(service->IsFlapping() == true);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetState() == ServiceCritical);
|
|
|
|
CheckNotification(service, false, NotificationProblem);
|
|
|
|
// Calm down
|
|
while (service->IsFlapping()) {
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceCritical));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
}
|
|
|
|
CheckNotification(service, true, NotificationFlappingEnd);
|
|
|
|
service->ProcessCheckResult(MakeCheckResult(ServiceOK));
|
|
Utility::IncrementTime(timeStepInterval);
|
|
|
|
BOOST_CHECK(service->IsFlapping() == false);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
BOOST_CHECK(service->GetState() == ServiceOK);
|
|
|
|
// There should be no recovery
|
|
// Known failure, see #5713
|
|
// CheckNotification(service, false, NotificationRecovery);
|
|
|
|
c.disconnect();
|
|
|
|
#endif /* I2_DEBUG */
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(suppressed_notification)
|
|
{
|
|
/* Tests that suppressed notifications on a Checkable are sent after the suppression ends if and only if the first
|
|
* hard state after the suppression is different from the last hard state before the suppression. The test works
|
|
* by bringing a service in a defined hard state, creating a downtime, performing some state changes, removing the
|
|
* downtime, bringing the service into another defined hard state (if not already) and checking the requested
|
|
* notifications.
|
|
*/
|
|
|
|
struct NotificationLog {
|
|
std::vector<std::pair<NotificationType, ServiceState>> GetAndClear() {
|
|
std::lock_guard<std::mutex> lock (mutex);
|
|
|
|
std::vector<std::pair<NotificationType, ServiceState>> ret;
|
|
std::swap(ret, log);
|
|
return ret;
|
|
}
|
|
|
|
void Add(std::pair<NotificationType, ServiceState> notification) {
|
|
std::lock_guard<std::mutex> lock (mutex);
|
|
|
|
log.emplace_back(notification);
|
|
}
|
|
|
|
private:
|
|
std::mutex mutex;
|
|
std::vector<std::pair<NotificationType, ServiceState>> log;
|
|
};
|
|
|
|
const std::vector<ServiceState> states {ServiceOK, ServiceWarning, ServiceCritical, ServiceUnknown};
|
|
|
|
for (bool isVolatile : {false, true}) {
|
|
for (int checkAttempts : {1, 2}) {
|
|
for (ServiceState initialState : states) {
|
|
for (ServiceState s1 : states)
|
|
for (ServiceState s2 : states)
|
|
for (ServiceState s3 : states) {
|
|
const std::vector<ServiceState> sequence {s1, s2, s3};
|
|
|
|
std::string testcase;
|
|
|
|
{
|
|
std::ostringstream buf;
|
|
buf << "volatile=" << isVolatile
|
|
<< " checkAttempts=" << checkAttempts
|
|
<< " sequence={" << Service::StateToString(initialState);
|
|
|
|
for (ServiceState s : sequence) {
|
|
buf << " " << Service::StateToString(s);
|
|
}
|
|
|
|
buf << "}";
|
|
testcase = buf.str();
|
|
}
|
|
|
|
std::cout << "Test case: " << testcase << std::endl;
|
|
|
|
// Create host and service for the test.
|
|
Host::Ptr host = new Host();
|
|
host->SetName("suppressed_notifications");
|
|
host->Register();
|
|
|
|
Service::Ptr service = new Service();
|
|
service->SetHostName(host->GetName());
|
|
service->SetName("service");
|
|
service->SetActive(true);
|
|
service->SetVolatile(isVolatile);
|
|
service->SetMaxCheckAttempts(checkAttempts);
|
|
service->Activate();
|
|
service->SetAuthority(true);
|
|
service->Register();
|
|
|
|
host->OnAllConfigLoaded();
|
|
service->OnAllConfigLoaded();
|
|
|
|
// Bring service into the initial hard state.
|
|
for (int i = 0; i < checkAttempts; i++) {
|
|
std::cout << " ProcessCheckResult("
|
|
<< Service::StateToString(initialState) << ")" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(initialState));
|
|
}
|
|
|
|
BOOST_CHECK(service->GetState() == initialState);
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
|
|
/* Keep track of all notifications requested from now on.
|
|
*
|
|
* Boost.Signal2 handler may still be executing from another thread after they were disconnected.
|
|
* Make the structures accessed by the handlers shared pointers so that they remain valid as long
|
|
* as they may be accessed from one of these handlers.
|
|
*/
|
|
auto notificationLog = std::make_shared<NotificationLog>();
|
|
|
|
boost::signals2::scoped_connection c (Checkable::OnNotificationsRequested.connect(
|
|
[notificationLog,service](
|
|
const Checkable::Ptr& checkable, NotificationType type, const CheckResult::Ptr& cr,
|
|
const String&, const String&, const MessageOrigin::Ptr&
|
|
) {
|
|
BOOST_CHECK_EQUAL(checkable, service);
|
|
std::cout << " -> OnNotificationsRequested(" << Notification::NotificationTypeToString(type)
|
|
<< ", " << Service::StateToString(cr->GetState()) << ")" << std::endl;
|
|
|
|
notificationLog->Add({type, cr->GetState()});
|
|
}
|
|
));
|
|
|
|
// Helper to assert which notifications were requested. Implicitly clears the stored notifications.
|
|
auto assertNotifications = [notificationLog](
|
|
const std::vector<std::pair<NotificationType, ServiceState>>& expected,
|
|
const std::string& extraMessage
|
|
) {
|
|
// Pretty-printer for the vectors of requested and expected notifications.
|
|
auto pretty = [](const std::vector<std::pair<NotificationType, ServiceState>>& vec) {
|
|
std::ostringstream s;
|
|
|
|
s << "{";
|
|
bool first = true;
|
|
for (const auto &v : vec) {
|
|
if (first) {
|
|
first = false;
|
|
} else {
|
|
s << ", ";
|
|
}
|
|
s << Notification::NotificationTypeToString(v.first)
|
|
<< "/" << Service::StateToString(v.second);
|
|
}
|
|
s << "}";
|
|
|
|
return s.str();
|
|
};
|
|
|
|
auto got (notificationLog->GetAndClear());
|
|
|
|
BOOST_CHECK_MESSAGE(got == expected, "expected=" << pretty(expected)
|
|
<< " got=" << pretty(got)
|
|
<< (extraMessage.empty() ? "" : " ") << extraMessage);
|
|
};
|
|
|
|
// Start a downtime for the service.
|
|
std::cout << " Downtime Start" << std::endl;
|
|
Downtime::Ptr downtime = new Downtime();
|
|
downtime->SetHostName(host->GetName());
|
|
downtime->SetServiceName(service->GetName());
|
|
downtime->SetName("downtime");
|
|
downtime->SetFixed(true);
|
|
downtime->SetStartTime(Utility::GetTime() - 3600);
|
|
downtime->SetEndTime(Utility::GetTime() + 3600);
|
|
service->RegisterDowntime(downtime);
|
|
downtime->Register();
|
|
downtime->OnAllConfigLoaded();
|
|
downtime->TriggerDowntime(Utility::GetTime());
|
|
|
|
BOOST_CHECK(service->IsInDowntime());
|
|
|
|
// Process check results for the state sequence.
|
|
for (ServiceState s : sequence) {
|
|
std::cout << " ProcessCheckResult(" << Service::StateToString(s) << ")" << std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(s));
|
|
BOOST_CHECK(service->GetState() == s);
|
|
if (checkAttempts == 1) {
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
}
|
|
}
|
|
|
|
assertNotifications({}, "(no notifications in downtime)");
|
|
|
|
if (service->GetSuppressedNotifications()) {
|
|
BOOST_CHECK_EQUAL(service->GetStateBeforeSuppression(), initialState);
|
|
}
|
|
|
|
// Remove the downtime.
|
|
std::cout << " Downtime End" << std::endl;
|
|
service->UnregisterDowntime(downtime);
|
|
downtime->Unregister();
|
|
BOOST_CHECK(!service->IsInDowntime());
|
|
|
|
if (service->GetStateType() == icinga::StateTypeSoft) {
|
|
// When the current state is a soft state, no notification should be sent just yet.
|
|
std::cout << " FireSuppressedNotifications()" << std::endl;
|
|
service->FireSuppressedNotifications();
|
|
|
|
assertNotifications({}, testcase + " (should not fire in soft state)");
|
|
|
|
// Repeat the last check result until reaching a hard state.
|
|
for (int i = 0; i < checkAttempts && service->GetStateType() == StateTypeSoft; i++) {
|
|
std::cout << " ProcessCheckResult(" << Service::StateToString(sequence.back()) << ")"
|
|
<< std::endl;
|
|
service->ProcessCheckResult(MakeCheckResult(sequence.back()));
|
|
BOOST_CHECK(service->GetState() == sequence.back());
|
|
}
|
|
}
|
|
|
|
// The service should be in a hard state now and notifications should now be sent if applicable.
|
|
BOOST_CHECK(service->GetStateType() == StateTypeHard);
|
|
|
|
std::cout << " FireSuppressedNotifications()" << std::endl;
|
|
service->FireSuppressedNotifications();
|
|
|
|
if (initialState != sequence.back()) {
|
|
NotificationType t = sequence.back() == ServiceOK ? NotificationRecovery : NotificationProblem;
|
|
assertNotifications({{t, sequence.back()}}, testcase);
|
|
} else {
|
|
assertNotifications({}, testcase);
|
|
}
|
|
|
|
// Remove host and service.
|
|
service->Unregister();
|
|
host->Unregister();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_SUITE_END()
|