icinga2/base/dynamicobject.cpp

525 lines
14 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;
map<pair<String, String>, Dictionary::Ptr> DynamicObject::m_PersistentUpdates;
double DynamicObject::m_CurrentTx = 0;
set<DynamicObject::Ptr> DynamicObject::m_ModifiedObjects;
boost::signal<void (const DynamicObject::Ptr&, const String& name)> DynamicObject::OnAttributeChanged;
boost::signal<void (const DynamicObject::Ptr&)> DynamicObject::OnRegistered;
boost::signal<void (const DynamicObject::Ptr&)> DynamicObject::OnUnregistered;
boost::signal<void (const set<DynamicObject::Ptr>&)> DynamicObject::OnTransactionClosing;
DynamicObject::DynamicObject(const Dictionary::Ptr& serializedObject)
: m_ConfigTx(0)
{
RegisterAttribute("__name", Attribute_Config);
RegisterAttribute("__type", Attribute_Config);
RegisterAttribute("__local", Attribute_Config);
RegisterAttribute("__abstract", Attribute_Config);
RegisterAttribute("__source", Attribute_Local);
RegisterAttribute("methods", Attribute_Config);
2012-08-02 12:12:59 +02:00
if (!serializedObject->Contains("configTx"))
throw invalid_argument("Serialized object must contain a config snapshot.");
/* apply state from the config item/remote update */
ApplyUpdate(serializedObject, true);
/* restore the object's persistent state */
map<pair<String, String>, Dictionary::Ptr>::iterator it;
it = m_PersistentUpdates.find(make_pair(GetType(), GetName()));
if (it != m_PersistentUpdates.end()) {
Logger::Write(LogDebug, "base", "Restoring persistent state "
"for object " + GetType() + ":" + GetName());
ApplyUpdate(it->second, true);
m_PersistentUpdates.erase(it);
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++) {
if (it->second.Type == Attribute_Transient)
continue;
if ((it->second.Type & attributeTypes) == 0)
continue;
if (it->second.Tx == 0)
continue;
if (it->second.Tx < sinceTx && !(it->second.Type == Attribute_Config && m_ConfigTx >= sinceTx))
continue;
Dictionary::Ptr attr = boost::make_shared<Dictionary>();
attr->Set("data", it->second.Data);
attr->Set("type", it->second.Type);
attr->Set("tx", it->second.Tx);
attrs->Set(it->first, attr);
}
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();
return update;
}
2012-04-20 13:49:04 +02:00
void DynamicObject::ApplyUpdate(const Dictionary::Ptr& serializedUpdate, bool suppressEvents)
{
double configTx = 0;
if (serializedUpdate->Contains("configTx")) {
configTx = serializedUpdate->Get("configTx");
if (configTx > m_ConfigTx)
ClearAttributesByType(Attribute_Config);
}
Dictionary::Ptr attrs = serializedUpdate->Get("attrs");
Dictionary::Iterator it;
for (it = attrs->Begin(); it != attrs->End(); it++) {
if (!it->second.IsObjectType<Dictionary>())
continue;
Dictionary::Ptr attr = it->second;
Value data = attr->Get("data");
int type = attr->Get("type");
double tx = attr->Get("tx");
if (type & Attribute_Config)
RegisterAttribute(it->first, Attribute_Config);
if (!HasAttribute(it->first))
RegisterAttribute(it->first, static_cast<DynamicAttributeType>(type));
InternalSetAttribute(it->first, data, tx, suppressEvents);
}
}
void DynamicObject::SanitizeUpdate(const Dictionary::Ptr& serializedUpdate, int allowedTypes)
2012-07-24 13:13:02 +02:00
{
if ((allowedTypes & Attribute_Config) == 0)
serializedUpdate->Remove("configTx");
Dictionary::Ptr attrs = serializedUpdate->Get("attrs");
Dictionary::Iterator prev, it;
for (it = attrs->Begin(); it != attrs->End(); ) {
if (!it->second.IsObjectType<Dictionary>())
continue;
Dictionary::Ptr attr = it->second;
int type = attr->Get("type");
if (type == 0 || type & ~allowedTypes) {
prev = it;
it++;
attrs->Remove(prev);
continue;
}
it++;
}
2012-07-24 13:13:02 +02:00
}
void DynamicObject::RegisterAttribute(const String& name, DynamicAttributeType type)
{
DynamicAttribute attr;
attr.Type = type;
attr.Tx = 0;
pair<DynamicObject::AttributeIterator, bool> tt;
tt = m_Attributes.insert(make_pair(name, attr));
if (!tt.second)
tt.first->second.Type = type;
}
2012-08-03 13:19:55 +02:00
void DynamicObject::Set(const String& name, const Value& data)
{
InternalSetAttribute(name, data, GetCurrentTx());
}
2012-08-03 13:19:55 +02:00
Value DynamicObject::Get(const String& name) const
{
return InternalGetAttribute(name);
}
void DynamicObject::InternalSetAttribute(const String& name, const Value& data, double tx, bool suppressEvent)
{
DynamicAttribute attr;
attr.Type = Attribute_Transient;
attr.Data = data;
attr.Tx = tx;
pair<DynamicObject::AttributeIterator, bool> tt;
tt = m_Attributes.insert(make_pair(name, attr));
if (!tt.second && tx >= tt.first->second.Tx) {
tt.first->second.Data = data;
tt.first->second.Tx = tx;
}
if (tt.first->second.Type & Attribute_Config)
m_ConfigTx = tx;
if (!suppressEvent) {
m_ModifiedObjects.insert(GetSelf());
DynamicObject::OnAttributeChanged(GetSelf(), name);
}
}
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;
2012-08-03 13:19:55 +02:00
return it->second.Data;
}
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
}
void DynamicObject::ClearAttributesByType(DynamicAttributeType type)
2012-07-02 14:38:37 +02:00
{
DynamicObject::AttributeIterator prev, at;
for (at = m_Attributes.begin(); at != m_Attributes.end(); ) {
if (at->second.Type == type) {
prev = at;
at++;
m_Attributes.erase(prev);
continue;
}
at++;
}
}
DynamicObject::AttributeConstIterator DynamicObject::AttributeBegin(void) const
{
return m_Attributes.begin();
}
DynamicObject::AttributeConstIterator DynamicObject::AttributeEnd(void) const
{
return m_Attributes.end();
}
String DynamicObject::GetType(void) const
{
2012-08-03 13:19:55 +02:00
return Get("__type");
}
String DynamicObject::GetName(void) const
{
2012-08-03 13:19:55 +02:00
return Get("__name");
}
bool DynamicObject::IsLocal(void) const
{
2012-08-03 13:19:55 +02:00
Value value = Get("__local");
if (value.IsEmpty())
return false;
return (value != 0);
}
bool DynamicObject::IsAbstract(void) const
{
2012-08-03 13:19:55 +02:00
Value value = Get("__abstract");
if (value.IsEmpty())
return false;
return (value != 0);
2012-07-02 14:38:37 +02:00
}
void DynamicObject::SetSource(const String& value)
{
2012-08-03 13:19:55 +02:00
Set("__source", value);
}
String DynamicObject::GetSource(void) const
{
2012-08-03 13:19:55 +02:00
return Get("__source");
}
void DynamicObject::Register(void)
{
assert(Application::IsMainThread());
2012-07-30 10:17:29 +02:00
DynamicObject::Ptr dobj = GetObject(GetType(), GetName());
DynamicObject::Ptr self = GetSelf();
assert(!dobj || dobj == self);
2012-07-30 10:17:29 +02:00
pair<DynamicObject::TypeMap::iterator, bool> ti;
ti = GetAllObjects().insert(make_pair(GetType(), DynamicObject::NameMap()));
ti.first->second.insert(make_pair(GetName(), GetSelf()));
OnRegistered(GetSelf());
}
2012-07-30 10:17:29 +02:00
void DynamicObject::Unregister(void)
{
assert(Application::IsMainThread());
2012-07-30 10:17:29 +02:00
DynamicObject::TypeMap::iterator tt;
tt = GetAllObjects().find(GetType());
if (tt == GetAllObjects().end())
return;
2012-07-30 10:17:29 +02:00
DynamicObject::NameMap::iterator nt = tt->second.find(GetName());
if (nt == tt->second.end())
return;
tt->second.erase(nt);
OnUnregistered(GetSelf());
}
DynamicObject::Ptr DynamicObject::GetObject(const String& type, const String& name)
{
2012-07-30 10:17:29 +02:00
DynamicObject::TypeMap::iterator tt;
tt = GetAllObjects().find(type);
if (tt == GetAllObjects().end())
2012-07-30 10:17:29 +02:00
return DynamicObject::Ptr();
2012-07-30 10:17:29 +02:00
DynamicObject::NameMap::iterator nt = tt->second.find(name);
if (nt == tt->second.end())
2012-07-30 10:17:29 +02:00
return DynamicObject::Ptr();
return nt->second;
}
2012-07-30 10:17:29 +02:00
pair<DynamicObject::TypeMap::iterator, DynamicObject::TypeMap::iterator> DynamicObject::GetTypes(void)
{
return make_pair(GetAllObjects().begin(), GetAllObjects().end());
}
pair<DynamicObject::NameMap::iterator, DynamicObject::NameMap::iterator> DynamicObject::GetObjects(const String& type)
{
2012-07-30 10:17:29 +02:00
pair<DynamicObject::TypeMap::iterator, bool> ti;
ti = GetAllObjects().insert(make_pair(type, DynamicObject::NameMap()));
return make_pair(ti.first->second.begin(), ti.first->second.end());
}
ScriptTask::Ptr DynamicObject::InvokeMethod(const String& method,
const vector<Value>& arguments, ScriptTask::CompletionCallback callback)
2012-07-14 15:59:59 +02:00
{
2012-08-03 13:19:55 +02:00
Value value = Get("methods");
if (!value.IsObjectType<Dictionary>())
return ScriptTask::Ptr();
Dictionary::Ptr methods = value;
if (!methods->Contains(method))
2012-07-14 15:59:59 +02:00
return ScriptTask::Ptr();
String funcName = methods->Get(method);
2012-07-14 15:59:59 +02:00
ScriptFunction::Ptr func = ScriptFunction::GetByName(funcName);
if (!func)
2012-07-17 20:41:06 +02:00
throw_exception(invalid_argument("Function '" + funcName + "' does not exist."));
2012-07-14 15:59:59 +02:00
2012-07-15 17:15:49 +02:00
ScriptTask::Ptr task = boost::make_shared<ScriptTask>(func, arguments);
task->Start(callback);
2012-07-14 15:59:59 +02:00
return task;
}
2012-07-24 13:13:02 +02:00
void DynamicObject::DumpObjects(const String& filename)
2012-07-24 13:13:02 +02:00
{
Logger::Write(LogInformation, "base", "Dumping program state to file '" + filename + "'");
ofstream fp;
fp.open(filename.CStr());
2012-07-24 13:13:02 +02:00
if (!fp)
throw_exception(runtime_error("Could not open '" + filename + "' file"));
2012-07-24 13:13:02 +02:00
FIFO::Ptr fifo = boost::make_shared<FIFO>();
2012-07-30 10:17:29 +02:00
DynamicObject::TypeMap::iterator tt;
for (tt = GetAllObjects().begin(); tt != GetAllObjects().end(); tt++) {
2012-07-30 10:17:29 +02:00
DynamicObject::NameMap::iterator nt;
for (nt = tt->second.begin(); nt != tt->second.end(); nt++) {
2012-07-30 10:17:29 +02:00
DynamicObject::Ptr object = nt->second;
Dictionary::Ptr persistentObject = boost::make_shared<Dictionary>();
2012-07-24 13:13:02 +02:00
persistentObject->Set("type", object->GetType());
persistentObject->Set("name", object->GetName());
int types = 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
/* This is quite ugly, unfortunatelly NetString requires an IOQueue object */
NetString::WriteStringToIOQueue(fifo.get(), json);
2012-07-24 13:13:02 +02:00
size_t count;
while ((count = fifo->GetAvailableBytes()) > 0) {
char buffer[1024];
2012-07-24 13:13:02 +02:00
if (count > sizeof(buffer))
count = sizeof(buffer);
2012-07-24 13:13:02 +02:00
fifo->Read(buffer, count);
fp.write(buffer, count);
}
2012-07-24 13:13:02 +02:00
}
}
}
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::ifstream fp;
fp.open(filename.CStr());
2012-07-24 13:13:02 +02:00
/* TODO: Fix this horrible mess. */
FIFO::Ptr fifo = boost::make_shared<FIFO>();
while (fp) {
char buffer[1024];
fp.read(buffer, sizeof(buffer));
fifo->Write(buffer, fp.gcount());
}
String message;
while (NetString::ReadStringFromIOQueue(fifo.get(), &message)) {
Value value = Value::Deserialize(message);
2012-07-24 13:13:02 +02:00
if (!value.IsObjectType<Dictionary>())
throw_exception(runtime_error("JSON objects in the program state file must be dictionaries."));
2012-07-24 13:13:02 +02:00
Dictionary::Ptr persistentObject = value;
String type = persistentObject->Get("type");
String name = persistentObject->Get("name");
2012-08-02 12:12:59 +02:00
bool hasConfig = persistentObject->Contains("configTx");
Dictionary::Ptr update = persistentObject->Get("update");
2012-07-24 13:13:02 +02:00
2012-08-02 12:12:59 +02:00
if (hasConfig) {
DynamicObject::Ptr object = Create(type, update);
object->Register();
2012-07-24 13:13:02 +02:00
} else {
/* keep non-replicated objects until another config object with
* the same name is created (which is when we restore its tags) */
m_PersistentUpdates[make_pair(type, name)] = update;
2012-07-24 13:13:02 +02:00
}
}
}
2012-07-30 10:17:29 +02:00
DynamicObject::TypeMap& DynamicObject::GetAllObjects(void)
{
static TypeMap objects;
return objects;
}
2012-07-30 10:17:29 +02:00
DynamicObject::ClassMap& DynamicObject::GetClasses(void)
{
static ClassMap classes;
return classes;
}
void DynamicObject::RegisterClass(const String& type, DynamicObject::Factory factory)
{
2012-07-30 10:17:29 +02:00
if (GetObjects(type).first != GetObjects(type).second)
throw_exception(runtime_error("Cannot register class for type '" +
type + "': Objects of this type already exist."));
2012-07-30 10:17:29 +02:00
GetClasses()[type] = factory;
}
DynamicObject::Ptr DynamicObject::Create(const String& type, const Dictionary::Ptr& properties)
{
2012-07-30 10:17:29 +02:00
DynamicObject::ClassMap::iterator it;
it = GetClasses().find(type);
if (it != GetClasses().end())
return it->second(properties);
else
2012-07-30 10:17:29 +02:00
return boost::make_shared<DynamicObject>(properties);
}
double DynamicObject::GetCurrentTx(void)
{
assert(m_CurrentTx != 0);
return m_CurrentTx;
}
void DynamicObject::BeginTx(void)
{
m_CurrentTx = Utility::GetTime();
}
void DynamicObject::FinishTx(void)
{
OnTransactionClosing(m_ModifiedObjects);
m_ModifiedObjects.clear();
m_CurrentTx = 0;
}