icinga2/lib/base/dynamicobject.cpp

607 lines
15 KiB
C++
Raw Normal View History

/******************************************************************************
* 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 *
2012-05-11 13:33:57 +02:00
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
******************************************************************************/
#include "i2-base.h"
using namespace icinga;
double DynamicObject::m_CurrentTx = 0;
2013-02-19 12:17:31 +01:00
set<DynamicObject::WeakPtr> DynamicObject::m_ModifiedObjects;
2013-02-18 14:40:24 +01:00
boost::mutex DynamicObject::m_TransactionMutex;
boost::once_flag DynamicObject::m_TransactionOnce = BOOST_ONCE_INIT;
2013-02-18 14:40:24 +01:00
Timer::Ptr DynamicObject::m_TransactionTimer;
2013-02-17 19:14:34 +01:00
signals2::signal<void (const DynamicObject::Ptr&)> DynamicObject::OnRegistered;
signals2::signal<void (const DynamicObject::Ptr&)> DynamicObject::OnUnregistered;
2013-02-19 12:17:31 +01:00
signals2::signal<void (double, const set<DynamicObject::WeakPtr>&)> DynamicObject::OnTransactionClosing;
2013-02-19 23:02:08 +01:00
signals2::signal<void (const DynamicObject::Ptr&)> DynamicObject::OnFlushObject;
DynamicObject::DynamicObject(const Dictionary::Ptr& serializedObject)
2013-02-20 19:52:25 +01:00
: m_EventSafe(false), m_ConfigTx(0)
{
2013-02-26 10:13:54 +01:00
RegisterAttribute("__name", Attribute_Config, &m_Name);
RegisterAttribute("__type", Attribute_Config, &m_Type);
RegisterAttribute("__local", Attribute_Config, &m_Local);
RegisterAttribute("__abstract", Attribute_Config, &m_Abstract);
RegisterAttribute("__source", Attribute_Local, &m_Source);
RegisterAttribute("methods", Attribute_Config, &m_Methods);
2013-02-18 23:44:24 +01:00
{
ObjectLock olock(serializedObject);
if (!serializedObject->Contains("configTx"))
BOOST_THROW_EXCEPTION(invalid_argument("Serialized object must contain a config snapshot."));
}
2012-08-02 12:12:59 +02:00
/* apply config state from the config item/remote update;
2013-02-26 10:13:54 +01:00
* The DynamicType::CreateObject function takes care of restoring
* non-config state after the object has been fully constructed */
2013-02-18 23:44:24 +01:00
{
ObjectLock olock(this);
ApplyUpdate(serializedObject, Attribute_Config);
}
2013-02-18 14:40:24 +01:00
boost::call_once(m_TransactionOnce, &DynamicObject::Initialize);
}
2013-02-17 19:14:34 +01:00
/*
* @threadsafety Always.
*/
DynamicObject::~DynamicObject(void)
2013-02-19 12:17:31 +01:00
{ }
2013-02-18 14:40:24 +01:00
void DynamicObject::Initialize(void)
{
/* Set up a timer to periodically create a new transaction. */
m_TransactionTimer = boost::make_shared<Timer>();
m_TransactionTimer->SetInterval(0.5);
m_TransactionTimer->OnTimerExpired.connect(boost::bind(&DynamicObject::NewTx));
m_TransactionTimer->Start();
}
/**
* @threadsafety Always.
*/
void DynamicObject::SendLocalUpdateEvents(void)
{
2013-02-19 12:17:31 +01:00
map<String, Value, string_iless> attrs;
{
ObjectLock olock(this);
attrs.swap(m_ModifiedAttributes);
}
2013-02-18 23:44:24 +01:00
/* Check if it's safe to send events. */
2013-02-20 19:52:25 +01:00
if (GetEventSafe()) {
2013-02-18 23:44:24 +01:00
map<String, Value, string_iless>::iterator it;
2013-02-19 12:17:31 +01:00
for (it = attrs.begin(); it != attrs.end(); it++)
2013-02-18 23:44:24 +01:00
OnAttributeChanged(it->first, it->second);
}
2012-07-24 13:13:02 +02:00
}
Dictionary::Ptr DynamicObject::BuildUpdate(double sinceTx, int attributeTypes) const
{
DynamicObject::AttributeConstIterator it;
Dictionary::Ptr attrs = boost::make_shared<Dictionary>();
for (it = m_Attributes.begin(); it != m_Attributes.end(); it++) {
2013-02-26 10:13:54 +01:00
if (it->second.GetType() == Attribute_Transient)
continue;
2013-02-26 10:13:54 +01:00
if ((it->second.GetType() & attributeTypes) == 0)
continue;
2013-02-26 10:13:54 +01:00
if (it->second.GetTx() == 0)
continue;
2013-02-26 10:13:54 +01:00
if (it->second.GetTx() < sinceTx && !(it->second.GetType() == Attribute_Config && m_ConfigTx >= sinceTx))
continue;
Dictionary::Ptr attr = boost::make_shared<Dictionary>();
2013-02-26 10:13:54 +01:00
attr->Set("data", it->second.GetValue());
attr->Set("type", it->second.GetType());
attr->Set("tx", it->second.GetTx());
attrs->Set(it->first, attr);
}
2013-02-24 01:10:34 +01:00
attrs->Seal();
Dictionary::Ptr update = boost::make_shared<Dictionary>();
update->Set("attrs", attrs);
if (m_ConfigTx >= sinceTx && attributeTypes & Attribute_Config)
update->Set("configTx", m_ConfigTx);
else if (attrs->GetLength() == 0)
return Dictionary::Ptr();
2013-02-24 01:10:34 +01:00
update->Seal();
return update;
}
2012-04-20 13:49:04 +02:00
2012-09-21 09:43:06 +02:00
void DynamicObject::ApplyUpdate(const Dictionary::Ptr& serializedUpdate,
int allowedTypes)
{
2013-02-18 23:44:24 +01:00
Dictionary::Ptr attrs;
{
ObjectLock olock(serializedUpdate);
2013-02-18 23:44:24 +01:00
double configTx = 0;
if ((allowedTypes & Attribute_Config) != 0 &&
serializedUpdate->Contains("configTx")) {
configTx = serializedUpdate->Get("configTx");
if (configTx > m_ConfigTx)
ClearAttributesByType(Attribute_Config);
}
attrs = serializedUpdate->Get("attrs");
}
2013-02-18 23:44:24 +01:00
{
ObjectLock olock(attrs);
2013-02-18 23:44:24 +01:00
Dictionary::Iterator it;
for (it = attrs->Begin(); it != attrs->End(); it++) {
if (!it->second.IsObjectType<Dictionary>())
continue;
2013-02-18 23:44:24 +01:00
Dictionary::Ptr attr = it->second;
ObjectLock alock(attr);
2013-02-18 23:44:24 +01:00
int type = attr->Get("type");
2013-02-18 23:44:24 +01:00
if ((type & ~allowedTypes) != 0)
continue;
2013-02-18 23:44:24 +01:00
Value data = attr->Get("data");
double tx = attr->Get("tx");
2013-02-18 23:44:24 +01:00
if (type & Attribute_Config)
RegisterAttribute(it->first, Attribute_Config);
2013-02-18 23:44:24 +01:00
if (!HasAttribute(it->first))
2013-02-26 10:13:54 +01:00
RegisterAttribute(it->first, static_cast<AttributeType>(type));
2013-02-18 23:44:24 +01:00
InternalSetAttribute(it->first, data, tx, true);
}
}
}
2012-09-21 09:43:06 +02:00
void DynamicObject::RegisterAttribute(const String& name,
2013-02-26 10:13:54 +01:00
AttributeType type, AttributeBase *boundAttribute)
{
2013-02-26 10:13:54 +01:00
AttributeHolder attr(type, boundAttribute);
pair<DynamicObject::AttributeIterator, bool> tt;
tt = m_Attributes.insert(make_pair(name, attr));
2013-02-26 10:13:54 +01:00
if (!tt.second) {
tt.first->second.SetType(type);
if (boundAttribute)
tt.first->second.Bind(boundAttribute);
}
}
2012-08-03 13:19:55 +02:00
void DynamicObject::Set(const String& name, const Value& data)
{
InternalSetAttribute(name, data, GetCurrentTx());
}
void DynamicObject::Touch(const String& name)
{
InternalSetAttribute(name, InternalGetAttribute(name), GetCurrentTx());
}
2012-08-03 13:19:55 +02:00
Value DynamicObject::Get(const String& name) const
{
return InternalGetAttribute(name);
}
2012-09-21 09:43:06 +02:00
void DynamicObject::InternalSetAttribute(const String& name, const Value& data,
double tx, bool allowEditConfig)
{
2013-02-26 10:13:54 +01:00
DynamicObject::AttributeIterator it;
it = m_Attributes.find(name);
Value oldValue;
2013-02-26 10:13:54 +01:00
if (it == m_Attributes.end()) {
AttributeHolder attr(Attribute_Transient);
attr.SetValue(tx, data);
2013-02-26 10:13:54 +01:00
m_Attributes.insert(make_pair(name, attr));
} else {
if (!allowEditConfig && (it->second.GetType() & Attribute_Config))
BOOST_THROW_EXCEPTION(runtime_error("Config properties are immutable: '" + name + "'."));
oldValue = it->second.GetValue();
it->second.SetValue(tx, data);
2013-02-26 10:13:54 +01:00
if (it->second.GetType() & Attribute_Config)
m_ConfigTx = tx;
}
2013-02-20 19:52:25 +01:00
if (GetEventSafe()) {
/* We can't call GetSelf() in the constructor or destructor.
* The OnConstructionCompleted() function will take care of adding this
* object to the list of modified objects later on if we can't
* do it here. */
2013-02-19 12:17:31 +01:00
2013-02-24 01:10:34 +01:00
DynamicObject::Ptr self = GetSelf();
{
boost::mutex::scoped_lock lock(m_TransactionMutex);
m_ModifiedObjects.insert(self);
}
2013-02-17 19:14:34 +01:00
}
/* Use insert() rather than [] so we don't overwrite
* an existing oldValue if the attribute was previously
* changed in the same transaction */
m_ModifiedAttributes.insert(make_pair(name, oldValue));
}
Value DynamicObject::InternalGetAttribute(const String& name) const
{
DynamicObject::AttributeConstIterator it;
it = m_Attributes.find(name);
if (it == m_Attributes.end())
2012-08-03 13:19:55 +02:00
return Empty;
2013-02-26 10:13:54 +01:00
return it->second.GetValue();
}
bool DynamicObject::HasAttribute(const String& name) const
2012-07-02 14:38:37 +02:00
{
return (m_Attributes.find(name) != m_Attributes.end());
2012-07-02 14:38:37 +02:00
}
2013-02-26 10:13:54 +01:00
void DynamicObject::ClearAttributesByType(AttributeType type)
2012-07-02 14:38:37 +02:00
{
DynamicObject::AttributeIterator at;
for (at = m_Attributes.begin(); at != m_Attributes.end(); at++) {
2013-02-26 10:13:54 +01:00
if (at->second.GetType() != type)
continue;
2013-02-26 10:13:54 +01:00
at->second.SetValue(0, Empty);
}
}
DynamicType::Ptr DynamicObject::GetType(void) const
{
2013-02-26 10:13:54 +01:00
return DynamicType::GetByName(m_Type);
}
String DynamicObject::GetName(void) const
{
2013-02-26 10:13:54 +01:00
return m_Name;
}
bool DynamicObject::IsLocal(void) const
{
2013-02-26 10:13:54 +01:00
return m_Local;
}
bool DynamicObject::IsAbstract(void) const
{
2013-02-26 10:13:54 +01:00
return m_Abstract;
2012-07-02 14:38:37 +02:00
}
void DynamicObject::SetSource(const String& value)
{
2013-02-26 10:13:54 +01:00
m_Source = value;
Touch("__source");
}
String DynamicObject::GetSource(void) const
{
2013-02-26 10:13:54 +01:00
return m_Source;
}
void DynamicObject::Register(void)
{
2013-02-18 23:44:24 +01:00
{
DynamicType::Ptr dtype = GetType();
ObjectLock olock(dtype);
DynamicObject::Ptr dobj = dtype->GetObject(GetName());
2013-02-18 23:44:24 +01:00
DynamicObject::Ptr self = GetSelf();
assert(!dobj || dobj == self);
2013-02-18 23:44:24 +01:00
if (!dobj)
dtype->RegisterObject(self);
}
OnRegistered(GetSelf());
Start();
}
void DynamicObject::Start(void)
{
/* Nothing to do here. */
}
2012-07-30 10:17:29 +02:00
void DynamicObject::Unregister(void)
{
DynamicType::Ptr dtype = GetType();
2013-02-18 14:40:24 +01:00
ObjectLock olock(dtype);
if (!dtype || !dtype->GetObject(GetName()))
return;
dtype->UnregisterObject(GetSelf());
OnUnregistered(GetSelf());
}
2013-02-19 07:26:52 +01:00
ScriptTask::Ptr DynamicObject::MakeMethodTask(const String& method,
const vector<Value>& arguments)
2012-07-14 15:59:59 +02:00
{
2013-02-18 14:40:24 +01:00
String funcName;
2013-02-26 10:13:54 +01:00
Dictionary::Ptr methods = m_Methods;
2012-07-14 15:59:59 +02:00
2013-02-18 14:40:24 +01:00
{
ObjectLock olock(methods);
if (!methods->Contains(method))
return ScriptTask::Ptr();
funcName = methods->Get(method);
}
2012-07-14 15:59:59 +02:00
ScriptFunction::Ptr func = ScriptFunction::GetByName(funcName);
if (!func)
BOOST_THROW_EXCEPTION(invalid_argument("Function '" + funcName + "' does not exist."));
2012-07-14 15:59:59 +02:00
2013-02-19 07:26:52 +01:00
return boost::make_shared<ScriptTask>(func, arguments);
2012-07-14 15:59:59 +02:00
}
2012-07-24 13:13:02 +02:00
2013-02-17 19:14:34 +01:00
/*
* @threadsafety Always.
*/
void DynamicObject::DumpObjects(const String& filename)
2012-07-24 13:13:02 +02:00
{
Logger::Write(LogInformation, "base", "Dumping program state to file '" + filename + "'");
String tempFilename = filename + ".tmp";
fstream fp;
fp.open(tempFilename.CStr(), std::ios_base::out);
2012-07-24 13:13:02 +02:00
if (!fp)
BOOST_THROW_EXCEPTION(runtime_error("Could not open '" + filename + "' file"));
2012-07-24 13:13:02 +02:00
StdioStream::Ptr sfp = boost::make_shared<StdioStream>(&fp, false);
sfp->Start();
2012-07-24 13:13:02 +02:00
2013-02-18 23:44:24 +01:00
;
BOOST_FOREACH(const DynamicType::Ptr& type, DynamicType::GetTypes()) {
BOOST_FOREACH(const DynamicObject::Ptr& object, type->GetObjects()) {
if (object->IsLocal())
continue;
Dictionary::Ptr persistentObject = boost::make_shared<Dictionary>();
2012-07-24 13:13:02 +02:00
persistentObject->Set("type", object->GetType()->GetName());
persistentObject->Set("name", object->GetName());
int types = Attribute_Local | Attribute_Replicated;
/* only persist properties for replicated objects or for objects
* that are marked as persistent */
2012-08-02 12:12:59 +02:00
if (!object->GetSource().IsEmpty() /*|| object->IsPersistent()*/)
types |= Attribute_Config;
Dictionary::Ptr update = object->BuildUpdate(0, types);
if (!update)
continue;
persistentObject->Set("update", update);
2012-07-24 13:13:02 +02:00
Value value = persistentObject;
String json = value.Serialize();
2012-07-24 13:13:02 +02:00
NetString::WriteStringToStream(sfp, json);
2012-07-24 13:13:02 +02:00
}
}
sfp->Close();
2012-11-22 12:04:32 +01:00
fp.close();
#ifdef _WIN32
_unlink(filename.CStr());
#endif /* _WIN32 */
2012-08-14 09:51:11 +02:00
if (rename(tempFilename.CStr(), filename.CStr()) < 0)
BOOST_THROW_EXCEPTION(PosixException("rename() failed", errno));
2012-07-24 13:13:02 +02:00
}
2013-02-17 19:14:34 +01:00
/*
* @threadsafety Always.
*/
void DynamicObject::RestoreObjects(const String& filename)
2012-07-24 13:13:02 +02:00
{
Logger::Write(LogInformation, "base", "Restoring program state from file '" + filename + "'");
std::fstream fp;
fp.open(filename.CStr(), std::ios_base::in);
2012-07-24 13:13:02 +02:00
StdioStream::Ptr sfp = boost::make_shared<StdioStream>(&fp, false);
sfp->Start();
2012-07-24 13:13:02 +02:00
unsigned long restored = 0;
String message;
while (NetString::ReadStringFromStream(sfp, &message)) {
2012-08-05 03:10:53 +02:00
Dictionary::Ptr persistentObject = Value::Deserialize(message);
2012-07-24 13:13:02 +02:00
String type = persistentObject->Get("type");
String name = persistentObject->Get("name");
Dictionary::Ptr update = persistentObject->Get("update");
2012-07-24 13:13:02 +02:00
2012-08-05 03:10:53 +02:00
bool hasConfig = update->Contains("configTx");
DynamicType::Ptr dt = DynamicType::GetByName(type);
2013-02-18 14:40:24 +01:00
ObjectLock dlock(dt);
if (!dt)
BOOST_THROW_EXCEPTION(invalid_argument("Invalid type: " + type));
DynamicObject::Ptr object = dt->GetObject(name);
2012-08-05 03:10:53 +02:00
if (hasConfig && !object) {
object = dt->CreateObject(update);
object->Register();
2012-08-05 03:10:53 +02:00
} else if (object) {
object->ApplyUpdate(update, Attribute_All);
2012-07-24 13:13:02 +02:00
}
restored++;
2012-07-24 13:13:02 +02:00
}
2012-11-22 12:04:32 +01:00
sfp->Close();
stringstream msgbuf;
msgbuf << "Restored " << restored << " objects";
2013-02-24 01:10:34 +01:00
Logger::Write(LogInformation, "base", msgbuf.str());
2012-07-24 13:13:02 +02:00
}
void DynamicObject::DeactivateObjects(void)
{
2013-02-18 23:44:24 +01:00
BOOST_FOREACH(const DynamicType::Ptr& dt, DynamicType::GetTypes()) {
set<DynamicObject::Ptr> objects;
2013-02-18 23:44:24 +01:00
{
ObjectLock olock(dt);
objects = dt->GetObjects();
}
2013-02-18 23:44:24 +01:00
BOOST_FOREACH(const DynamicObject::Ptr& object, objects) {
ObjectLock olock(object);
object->Unregister();
}
}
}
2013-02-17 19:14:34 +01:00
/*
* @threadsafety Always.
*/
double DynamicObject::GetCurrentTx(void)
{
2013-02-18 14:40:24 +01:00
boost::mutex::scoped_lock lock(m_TransactionMutex);
2013-02-17 19:14:34 +01:00
2013-02-19 23:02:08 +01:00
if (m_CurrentTx == 0) {
/* Set the initial transaction ID. */
m_CurrentTx = Utility::GetTime();
}
return m_CurrentTx;
}
2013-02-19 23:02:08 +01:00
void DynamicObject::Flush(void)
{
SendLocalUpdateEvents();
OnFlushObject(GetSelf());
}
2013-02-17 19:14:34 +01:00
/*
2013-02-18 14:40:24 +01:00
* @threadsafety Always. Caller must not hold any Object locks.
2013-02-17 19:14:34 +01:00
*/
void DynamicObject::NewTx(void)
{
2013-02-18 14:40:24 +01:00
double tx;
2013-02-19 12:17:31 +01:00
set<DynamicObject::WeakPtr> objects;
2013-02-17 19:14:34 +01:00
{
2013-02-18 14:40:24 +01:00
boost::mutex::scoped_lock lock(m_TransactionMutex);
2013-02-17 19:14:34 +01:00
2013-02-18 14:40:24 +01:00
tx = m_CurrentTx;
2013-02-17 19:14:34 +01:00
m_ModifiedObjects.swap(objects);
2013-02-18 14:40:24 +01:00
m_CurrentTx = Utility::GetTime();
}
2013-02-19 12:17:31 +01:00
BOOST_FOREACH(const DynamicObject::WeakPtr& wobject, objects) {
DynamicObject::Ptr object = wobject.lock();
if (!object)
continue;
2013-02-17 19:14:34 +01:00
object->SendLocalUpdateEvents();
}
2013-02-18 14:40:24 +01:00
OnTransactionClosing(tx, objects);
}
2013-02-20 19:52:25 +01:00
void DynamicObject::OnConstructionCompleted(void)
2013-02-19 12:17:31 +01:00
{
2013-02-20 19:52:25 +01:00
/* It's now safe to send us attribute events. */
SetEventSafe(true);
2013-02-19 12:17:31 +01:00
/* Add this new object to the list of modified objects.
* We're doing this here because we can't construct
* a while WeakPtr from within the object's constructor. */
boost::mutex::scoped_lock lock(m_TransactionMutex);
m_ModifiedObjects.insert(GetSelf());
}
2013-02-14 14:58:26 +01:00
2013-02-20 19:52:25 +01:00
void DynamicObject::OnRegistrationCompleted(void)
{ }
2012-08-07 21:02:12 +02:00
void DynamicObject::OnAttributeChanged(const String&, const Value&)
{ }
2012-08-07 21:02:12 +02:00
2013-02-17 19:14:34 +01:00
/*
* @threadsafety Always.
*/
DynamicObject::Ptr DynamicObject::GetObject(const String& type, const String& name)
{
DynamicType::Ptr dtype = DynamicType::GetByName(type);
2013-02-18 14:40:24 +01:00
{
ObjectLock olock(dtype);
return dtype->GetObject(name);
}
}
const DynamicObject::AttributeMap& DynamicObject::GetAttributes(void) const
{
return m_Attributes;
2013-02-02 20:00:02 +01:00
}
2013-02-18 23:44:24 +01:00
2013-02-20 19:52:25 +01:00
void DynamicObject::SetEventSafe(bool safe)
2013-02-18 23:44:24 +01:00
{
2013-02-20 19:52:25 +01:00
m_EventSafe = safe;
2013-02-18 23:44:24 +01:00
}
2013-02-20 19:52:25 +01:00
bool DynamicObject::GetEventSafe(void) const
2013-02-18 23:44:24 +01:00
{
2013-02-20 19:52:25 +01:00
return m_EventSafe;
2013-02-18 23:44:24 +01:00
}