Implemented notification request messages, notifications dictionary for hosts/services and state-based notifications.

This commit is contained in:
Gunnar Beutner 2013-02-09 15:20:10 +01:00
parent 3e7376576e
commit c44ad7b06b
20 changed files with 410 additions and 71 deletions

View File

@ -28,6 +28,11 @@ EXPORT_COMPONENT(notification, NotificationComponent);
*/
void NotificationComponent::Start(void)
{
m_Endpoint = Endpoint::MakeEndpoint("notification", false);
m_Endpoint->RegisterTopicHandler("icinga::SendNotifications",
boost::bind(&NotificationComponent::SendNotificationsRequestHandler, this, _2,
_3));
m_NotificationTimer = boost::make_shared<Timer>();
m_NotificationTimer->SetInterval(5);
m_NotificationTimer->OnTimerExpired.connect(boost::bind(&NotificationComponent::NotificationTimerHandler, this));
@ -50,3 +55,26 @@ void NotificationComponent::NotificationTimerHandler(void)
{
// TODO: implement
}
/**
* Processes icinga::SendNotifications messages.
*/
void NotificationComponent::SendNotificationsRequestHandler(const Endpoint::Ptr& sender,
const RequestMessage& request)
{
MessagePart params;
if (!request.GetParams(&params))
return;
String svc;
if (!params.Get("service", &svc))
return;
int type;
if (!params.Get("type", &type))
return;
Service::Ptr service = Service::GetByName(svc);
service->SendNotifications(static_cast<NotificationType>(type));
}

View File

@ -34,8 +34,10 @@ public:
private:
Timer::Ptr m_NotificationTimer;
Endpoint::Ptr m_Endpoint;
void NotificationTimerHandler(void);
void SendNotificationsRequestHandler(const Endpoint::Ptr& sender, const RequestMessage& request);
};
}

View File

@ -103,6 +103,18 @@ type Host {
}
},
%attribute dictionary "notifications" {
%attribute string "*",
%attribute dictionary "*" {
%require "notification",
%attribute string "notification",
%attribute dictionary "macros" {
%attribute string "*"
}
}
},
/* service attributes */
%attribute number "max_check_attempts",
%attribute number "check_interval",
@ -172,6 +184,18 @@ type Service {
%attribute dictionary "methods" {
%require "check",
%attribute string "check"
},
%attribute dictionary "notifications" {
%attribute string "*",
%attribute dictionary "*" {
%require "notification",
%attribute string "notification",
%attribute dictionary "macros" {
%attribute string "*"
}
}
}
}

View File

@ -105,6 +105,16 @@ ConfigItem::Ptr ConfigItemBuilder::Compile(void)
BOOST_THROW_EXCEPTION(invalid_argument(msgbuf.str()));
}
BOOST_FOREACH(const String& parent, m_Parents) {
ConfigItem::Ptr item = ConfigItem::GetObject(m_Type, parent);
if (!item) {
stringstream msgbuf;
msgbuf << "The parent config item '" + parent + "' does not exist: " << m_DebugInfo;
BOOST_THROW_EXCEPTION(invalid_argument(msgbuf.str()));
}
}
ExpressionList::Ptr exprl = boost::make_shared<ExpressionList>();
Expression execExpr("", OperatorExecute, m_ExpressionList, m_DebugInfo);

View File

@ -22,6 +22,8 @@ libicinga_la_SOURCES = \
macroprocessor.h \
notification.cpp \
notification.h \
notificationrequestmessage.cpp \
notificationrequestmessage.h \
nullchecktask.cpp \
nullchecktask.h \
pluginchecktask.cpp \

View File

@ -99,6 +99,8 @@ void ExternalCommandProcessor::Execute(double time, const String& command, const
RegisterCommand("DEL_SVC_COMMENT", &ExternalCommandProcessor::DelSvcComment);
RegisterCommand("DEL_ALL_HOST_COMMENTS", &ExternalCommandProcessor::DelAllHostComments);
RegisterCommand("DEL_ALL_SVC_COMMENTS", &ExternalCommandProcessor::DelAllSvcComments);
RegisterCommand("SEND_CUSTOM_HOST_NOTIFICATION", &ExternalCommandProcessor::SendCustomHostNotification);
RegisterCommand("SEND_CUSTOM_SVC_NOTIFICATION", &ExternalCommandProcessor::SendCustomSvcNotification);
m_Initialized = true;
}
@ -799,3 +801,28 @@ void ExternalCommandProcessor::DelAllSvcComments(double, const vector<String>& a
Logger::Write(LogInformation, "icinga", "Removing all comments for service " + service->GetName());
service->RemoveAllComments();
}
void ExternalCommandProcessor::SendCustomHostNotification(double time, const vector<String>& arguments)
{
if (arguments.size() < 4)
BOOST_THROW_EXCEPTION(invalid_argument("Expected 4 arguments."));
Host::Ptr host = Host::GetByName(arguments[0]);
Logger::Write(LogInformation, "icinga", "Sending custom notification for host " + host->GetName());
Service::Ptr service = host->GetHostCheckService();
if (service) {
service->RequestNotifications(NotificationCustom);
}
}
void ExternalCommandProcessor::SendCustomSvcNotification(double time, const vector<String>& arguments)
{
if (arguments.size() < 5)
BOOST_THROW_EXCEPTION(invalid_argument("Expected 5 arguments."));
Service::Ptr service = Service::GetByNamePair(arguments[0], arguments[1]);
Logger::Write(LogInformation, "icinga", "Sending custom notification for service " + service->GetName());
service->RequestNotifications(NotificationCustom);
}

View File

@ -22,7 +22,7 @@
namespace icinga
{
class I2_ICINGA_API ExternalCommandProcessor {
public:
static void Execute(const String& line);
@ -70,6 +70,8 @@ public:
static void DelSvcComment(double time, const vector<String>& arguments);
static void DelAllHostComments(double time, const vector<String>& arguments);
static void DelAllSvcComments(double time, const vector<String>& arguments);
static void SendCustomHostNotification(double time, const vector<String>& arguments);
static void SendCustomSvcNotification(double time, const vector<String>& arguments);
private:
typedef function<void (double time, const vector<String>& arguments)> Callback;

View File

@ -152,7 +152,7 @@ template<bool copyServiceAttrs, typename TDict>
static void CopyServiceAttributes(TDict serviceDesc, const ConfigItemBuilder::Ptr& builder)
{
/* TODO: we only need to copy macros if this is an inline definition,
* i.e. host->GetProperties() != service, however for now we just
* i.e. "typeid(serviceDesc)" != Service, however for now we just
* copy them anyway. */
Value macros = serviceDesc->Get("macros");
if (!macros.IsEmpty())
@ -194,7 +194,7 @@ void Host::UpdateSlaveServices(void)
if (!item || IsAbstract())
return;
Dictionary::Ptr oldServices = Get("convenience_services");
Dictionary::Ptr oldServices = Get("slave_services");
Dictionary::Ptr newServices;
newServices = boost::make_shared<Dictionary>();
@ -212,7 +212,7 @@ void Host::UpdateSlaveServices(void)
ConfigItemBuilder::Ptr builder = boost::make_shared<ConfigItemBuilder>(item->GetDebugInfo());
builder->SetType("Service");
builder->SetName(name);
builder->AddExpression("host_name", OperatorSet, item->GetName());
builder->AddExpression("host_name", OperatorSet, GetName());
builder->AddExpression("alias", OperatorSet, svcname);
builder->AddExpression("short_name", OperatorSet, svcname);
@ -252,7 +252,7 @@ void Host::UpdateSlaveServices(void)
}
}
Set("convenience_services", newServices);
Set("slave_services", newServices);
}
void Host::OnAttributeChanged(const String& name, const Value&)
@ -261,6 +261,11 @@ void Host::OnAttributeChanged(const String& name, const Value&)
HostGroup::InvalidateMembersCache();
else if (name == "services")
UpdateSlaveServices();
else if (name == "notifications") {
BOOST_FOREACH(const Service::Ptr& service, GetServices()) {
service->UpdateSlaveNotifications();
}
}
}
set<Service::Ptr> Host::GetServices(void) const

View File

@ -47,6 +47,7 @@ using boost::algorithm::is_any_of;
#include "icingaapplication.h"
#include "notification.h"
#include "notificationrequestmessage.h"
#include "host.h"
#include "hostgroup.h"

View File

@ -33,10 +33,11 @@
<ClCompile Include="icingaapplication.cpp" />
<ClCompile Include="macroprocessor.cpp" />
<ClCompile Include="notification.cpp" />
<ClCompile Include="notificationrequestmessage.cpp" />
<ClCompile Include="pluginchecktask.cpp" />
<ClCompile Include="nullchecktask.cpp" />
<ClCompile Include="pluginnotificationtask.cpp" />
<ClCompile Include="service-notifications.cpp" />
<ClCompile Include="service-notification.cpp" />
<ClCompile Include="service.cpp" />
<ClCompile Include="service-comment.cpp" />
<ClCompile Include="service-downtime.cpp" />
@ -52,6 +53,7 @@
<ClInclude Include="icingaapplication.h" />
<ClInclude Include="macroprocessor.h" />
<ClInclude Include="notification.h" />
<ClInclude Include="notificationrequestmessage.h" />
<ClInclude Include="pluginchecktask.h" />
<ClInclude Include="nullchecktask.h" />
<ClInclude Include="pluginnotificationtask.h" />
@ -208,4 +210,4 @@
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
<ImportGroup Label="ExtensionTargets">
</ImportGroup>
</Project>
</Project>

View File

@ -25,10 +25,14 @@ REGISTER_TYPE(Notification, NULL);
Notification::Notification(const Dictionary::Ptr& properties)
: DynamicObject(properties)
{ }
{
Service::InvalidateNotificationsCache();
}
Notification::~Notification(void)
{ }
{
Service::InvalidateNotificationsCache();
}
bool Notification::Exists(const String& name)
{
@ -66,13 +70,20 @@ Dictionary::Ptr Notification::GetMacros(void) const
return Get("macros");
}
void Notification::SendNotification(void)
void Notification::SendNotification(NotificationType type)
{
vector<Value> arguments;
arguments.push_back(static_cast<Notification::Ptr>(GetSelf()));
arguments.push_back(type);
ScriptTask::Ptr task;
task = InvokeMethod("notify", arguments, boost::bind(&Notification::NotificationCompletedHandler, this, _1));
if (!task) {
Logger::Write(LogWarning, "icinga", "Notification object '" + GetName() + "' doesn't have a 'notify' method.");
return;
}
if (!task->IsFinished()) {
/* We need to keep the task object alive until the completion handler is called. */

View File

@ -30,8 +30,11 @@ namespace icinga
*/
enum NotificationType
{
NotificationHost,
NotificationService
NotificationDowntimeStart,
NotificationDowntimeEnd,
NotificationDowntimeRemoved,
NotificationCustom,
NotificationStateChange
};
class Service;
@ -57,7 +60,7 @@ public:
String GetNotificationCommand(void) const;
Dictionary::Ptr GetMacros(void) const;
void SendNotification(void);
void SendNotification(NotificationType type);
private:
set<ScriptTask::Ptr> m_Tasks;

View File

@ -21,53 +21,26 @@
using namespace icinga;
map<String, set<Notification::WeakPtr> > Service::m_NotificationsCache;
bool Service::m_NotificationsCacheValid;
void Service::SendNotifications(void) const
String NotificationRequestMessage::GetService(void) const
{
BOOST_FOREACH(const Notification::Ptr& notification, GetNotifications()) {
notification->SendNotification();
}
String service;
Get("service", &service);
return service;
}
void Service::InvalidateNotificationsCache(void)
void NotificationRequestMessage::SetService(const String& service)
{
m_NotificationsCacheValid = false;
m_NotificationsCache.clear();
Set("service", service);
}
void Service::ValidateNotificationsCache(void)
NotificationType NotificationRequestMessage::GetType(void) const
{
if (m_NotificationsCacheValid)
return;
m_NotificationsCache.clear();
DynamicObject::Ptr object;
BOOST_FOREACH(tie(tuples::ignore, object), DynamicType::GetByName("Notification")->GetObjects()) {
const Notification::Ptr& notification = static_pointer_cast<Notification>(object);
m_NotificationsCache[notification->GetService()->GetName()].insert(notification);
}
m_NotificationsCacheValid = true;
int type;
Get("type", &type);
return static_cast<NotificationType>(type);
}
set<Notification::Ptr> Service::GetNotifications(void) const
void NotificationRequestMessage::SetType(NotificationType type)
{
set<Notification::Ptr> notifications;
ValidateNotificationsCache();
BOOST_FOREACH(const Notification::WeakPtr& wservice, m_NotificationsCache[GetName()]) {
Notification::Ptr notification = wservice.lock();
if (!notification)
continue;
notifications.insert(notification);
}
return notifications;
Set("type", type);
}

View File

@ -0,0 +1,46 @@
/******************************************************************************
* Icinga 2 *
* Copyright (C) 2012 Icinga Development Team (http://www.icinga.org/) *
* *
* This program is free software; you can redistribute it and/or *
* modify it under the terms of the GNU General Public License *
* as published by the Free Software Foundation; either version 2 *
* of the License, or (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program; if not, write to the Free Software Foundation *
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
******************************************************************************/
#ifndef NOTIFICATIONREQUESTMESSAGE_H
#define NOTIFICATIONREQUESTMESSAGE_H
namespace icinga
{
/**
* An API request for sending notifications.
*
* @ingroup icinga
*/
class I2_ICINGA_API NotificationRequestMessage : public MessagePart
{
public:
NotificationRequestMessage(void) : MessagePart() { }
NotificationRequestMessage(const MessagePart& message) : MessagePart(message) { }
String GetService(void) const;
void SetService(const String& service);
NotificationType GetType(void) const;
void SetType(NotificationType type);
};
}
#endif /* NOTIFICATIONREQUESTMESSAGE_H */

View File

@ -23,8 +23,9 @@ using namespace icinga;
REGISTER_SCRIPTFUNCTION("native::PluginNotification", &PluginNotificationTask::ScriptFunc);
PluginNotificationTask::PluginNotificationTask(const ScriptTask::Ptr& task, const Process::Ptr& process)
: m_Task(task), m_Process(process)
PluginNotificationTask::PluginNotificationTask(const ScriptTask::Ptr& task, const Process::Ptr& process,
const String& service, const String& command)
: m_Task(task), m_Process(process), m_ServiceName(service), m_Command(command)
{ }
void PluginNotificationTask::ScriptFunc(const ScriptTask::Ptr& task, const vector<Value>& arguments)
@ -32,10 +33,14 @@ void PluginNotificationTask::ScriptFunc(const ScriptTask::Ptr& task, const vecto
if (arguments.size() < 1)
BOOST_THROW_EXCEPTION(invalid_argument("Missing argument: Notification target must be specified."));
if (arguments.size() < 2)
BOOST_THROW_EXCEPTION(invalid_argument("Missing argument: Notification type must be specified."));
if (!arguments[0].IsObjectType<Notification>())
BOOST_THROW_EXCEPTION(invalid_argument("Argument must be a service."));
Notification::Ptr notification = arguments[0];
NotificationType type = static_cast<NotificationType>(static_cast<int>(arguments[1]));
String notificationCommand = notification->GetNotificationCommand();
@ -48,7 +53,7 @@ void PluginNotificationTask::ScriptFunc(const ScriptTask::Ptr& task, const vecto
Process::Ptr process = boost::make_shared<Process>(command);
PluginNotificationTask ct(task, process);
PluginNotificationTask ct(task, process, notification->GetService()->GetName(), command);
process->Start(boost::bind(&PluginNotificationTask::ProcessFinishedHandler, ct));
}
@ -60,8 +65,13 @@ void PluginNotificationTask::ProcessFinishedHandler(PluginNotificationTask ct)
try {
pr = ct.m_Process->GetResult();
if (pr.ExitStatus != 0)
Logger::Write(LogWarning, "icinga", "Notification command failed; output: " + pr.Output);
if (pr.ExitStatus != 0) {
stringstream msgbuf;
msgbuf << "Notification command '" << ct.m_Command << "' for service '"
<< ct.m_ServiceName << "' failed; exit status: "
<< pr.ExitStatus << ", output: " << pr.Output;
Logger::Write(LogWarning, "icinga", msgbuf.str());
}
ct.m_Task->FinishResult(Empty);
} catch (...) {

View File

@ -37,10 +37,14 @@ private:
static void ProcessFinishedHandler(PluginNotificationTask ct);
PluginNotificationTask(const ScriptTask::Ptr& task, const Process::Ptr& process);
PluginNotificationTask(const ScriptTask::Ptr& task, const Process::Ptr& process,
const String& service, const String& command);
ScriptTask::Ptr m_Task;
Process::Ptr m_Process;
String m_ServiceName;
String m_Command;
};
}

View File

@ -0,0 +1,184 @@
/******************************************************************************
* Icinga 2 *
* Copyright (C) 2012 Icinga Development Team (http://www.icinga.org/) *
* *
* This program is free software; you can redistribute it and/or *
* modify it under the terms of the GNU General Public License *
* as published by the Free Software Foundation; either version 2 *
* of the License, or (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program; if not, write to the Free Software Foundation *
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
******************************************************************************/
#include "i2-icinga.h"
using namespace icinga;
map<String, set<Notification::WeakPtr> > Service::m_NotificationsCache;
bool Service::m_NotificationsCacheValid;
void Service::RequestNotifications(NotificationType type) const
{
RequestMessage msg;
msg.SetMethod("icinga::SendNotifications");
NotificationRequestMessage params;
msg.SetParams(params);
params.SetService(GetName());
params.SetType(type);
Logger::Write(LogInformation, "icinga", "Sending notification anycast request for service '" + GetName() + "'");
EndpointManager::GetInstance()->SendAnycastMessage(Endpoint::Ptr(), msg);
}
void Service::SendNotifications(NotificationType type) const
{
Logger::Write(LogInformation, "icinga", "Sending notifications for service '" + GetName() + "'");
set<Notification::Ptr> notifications = GetNotifications();
if (notifications.size() == 0)
Logger::Write(LogInformation, "icinga", "Service '" + GetName() + "' does not have any notifications.");
BOOST_FOREACH(const Notification::Ptr& notification, notifications) {
notification->SendNotification(type);
}
}
void Service::InvalidateNotificationsCache(void)
{
m_NotificationsCacheValid = false;
m_NotificationsCache.clear();
}
void Service::ValidateNotificationsCache(void)
{
if (m_NotificationsCacheValid)
return;
m_NotificationsCache.clear();
DynamicObject::Ptr object;
BOOST_FOREACH(tie(tuples::ignore, object), DynamicType::GetByName("Notification")->GetObjects()) {
const Notification::Ptr& notification = static_pointer_cast<Notification>(object);
m_NotificationsCache[notification->GetService()->GetName()].insert(notification);
}
m_NotificationsCacheValid = true;
}
set<Notification::Ptr> Service::GetNotifications(void) const
{
set<Notification::Ptr> notifications;
ValidateNotificationsCache();
BOOST_FOREACH(const Notification::WeakPtr& wservice, m_NotificationsCache[GetName()]) {
Notification::Ptr notification = wservice.lock();
if (!notification)
continue;
notifications.insert(notification);
}
return notifications;
}
template<typename TDict>
static void CopyNotificationAttributes(TDict notificationDesc, const ConfigItemBuilder::Ptr& builder)
{
/* TODO: we only need to copy macros if this is an inline definition,
* i.e. "typeid(notificationDesc)" != Notification, however for now we just
* copy them anyway. */
Value macros = notificationDesc->Get("macros");
if (!macros.IsEmpty())
builder->AddExpression("macros", OperatorPlus, macros);
/*Value notificationInterval = notificationDesc->Get("notification_interval");
if (!notificationInterval.IsEmpty())
builder->AddExpression("notification_interval", OperatorSet, notificationInterval);*/
}
void Service::UpdateSlaveNotifications(void)
{
ConfigItem::Ptr item = ConfigItem::GetObject("Service", GetName());
/* Don't create slave notifications unless we own this object
* and it's not a template. */
if (!item || IsAbstract())
return;
Dictionary::Ptr oldNotifications = Get("slave_notifications");
Dictionary::Ptr newNotifications;
newNotifications = boost::make_shared<Dictionary>();
vector<Dictionary::Ptr> notificationDescsList;
notificationDescsList.push_back(GetHost()->Get("notifications"));
notificationDescsList.push_back(Get("notifications"));
BOOST_FOREACH(const Dictionary::Ptr& notificationDescs, notificationDescsList) {
if (!notificationDescs)
continue;
String nfcname;
Value nfcdesc;
BOOST_FOREACH(tie(nfcname, nfcdesc), notificationDescs) {
stringstream namebuf;
namebuf << GetName() << "-" << nfcname;
String name = namebuf.str();
ConfigItemBuilder::Ptr builder = boost::make_shared<ConfigItemBuilder>(item->GetDebugInfo());
builder->SetType("Notification");
builder->SetName(name);
builder->AddExpression("host_name", OperatorSet, GetHost()->GetName());
builder->AddExpression("service", OperatorSet, GetName());
CopyNotificationAttributes(this, builder);
if (nfcdesc.IsScalar()) {
builder->AddParent(nfcdesc);
} else if (nfcdesc.IsObjectType<Dictionary>()) {
Dictionary::Ptr notification = nfcdesc;
String parent = notification->Get("notification");
if (parent.IsEmpty())
parent = nfcname;
builder->AddParent(parent);
CopyNotificationAttributes(notification, builder);
} else {
BOOST_THROW_EXCEPTION(invalid_argument("Notification description must be either a string or a dictionary."));
}
ConfigItem::Ptr notificationItem = builder->Compile();
notificationItem->Commit();
newNotifications->Set(name, notificationItem);
}
}
if (oldNotifications) {
ConfigItem::Ptr notification;
BOOST_FOREACH(tie(tuples::ignore, notification), oldNotifications) {
if (!notification)
continue;
if (!newNotifications->Contains(notification->GetName()))
notification->Unregister();
}
}
Set("slave_notifications", newNotifications);
}

View File

@ -529,17 +529,11 @@ void Service::ApplyCheckResult(const Dictionary::Ptr& cr)
SetLastCheckResult(cr);
// TODO(debug): remove this
SendNotifications();
double now = Utility::GetTime();
if (old_state != GetState()) {
double now = Utility::GetTime();
SetLastStateChange(now);
if (old_stateType != GetStateType())
SetLastHardStateChange(now);
/* remove acknowledgements */
if (GetAcknowledgement() == AcknowledgementNormal ||
(GetAcknowledgement() == AcknowledgementSticky && GetStateType() == StateTypeHard && GetState() == StateOK)) {
@ -560,11 +554,19 @@ void Service::ApplyCheckResult(const Dictionary::Ptr& cr)
service->SetNextCheck(Utility::GetTime());
}
// TODO: notify our child services/hosts that our state has changed
if (GetState() != StateOK)
TriggerDowntimes();
}
if (GetState() != StateOK)
TriggerDowntimes();
if (GetStateType() == StateTypeHard && (old_state != GetState() || old_stateType == StateTypeSoft)) {
SetLastHardStateChange(now);
/* Make sure the notification component sees the updated
* state/state_type attributes. */
DynamicObject::FlushTx();
RequestNotifications(NotificationStateChange);
}
}
ServiceState Service::StateFromString(const String& state)
@ -645,7 +647,7 @@ void Service::OnAttributeChanged(const String& name, const Value& oldValue)
else if (name == "comments")
Service::InvalidateCommentCache();
else if (name == "notifications")
Service::InvalidateNotificationsCache();
UpdateSlaveNotifications();
}
void Service::BeginExecuteCheck(const function<void (void)>& callback)

View File

@ -223,13 +223,16 @@ public:
static void ValidateCommentCache(void);
/* Notifications */
void SendNotifications(void) const;
void RequestNotifications(NotificationType type) const;
void SendNotifications(NotificationType type) const;
static void InvalidateNotificationsCache(void);
static void ValidateNotificationsCache(void);
set<Notification::Ptr> GetNotifications(void) const;
void UpdateSlaveNotifications(void);
protected:
virtual void OnAttributeChanged(const String& name, const Value& oldValue);

View File

@ -236,7 +236,7 @@ void EndpointManager::SendAnycastMessage(const Endpoint::Ptr& sender,
BOOST_FOREACH(tie(tuples::ignore, object), DynamicType::GetByName("Endpoint")->GetObjects()) {
Endpoint::Ptr endpoint = dynamic_pointer_cast<Endpoint>(object);
/* don't forward messages between non-local endpoints */
if (!sender->IsLocal() && !endpoint->IsLocal())
if ((sender && !sender->IsLocal()) && !endpoint->IsLocal())
continue;
if (endpoint->HasSubscription(method))