2012-05-10 12:06:41 +02:00
|
|
|
/******************************************************************************
|
|
|
|
* Icinga 2 *
|
2014-03-19 01:02:29 +01:00
|
|
|
* Copyright (C) 2012-2014 Icinga Development Team (http://www.icinga.org) *
|
2012-05-10 12:06:41 +02:00
|
|
|
* *
|
|
|
|
* 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. *
|
2012-05-10 12:06:41 +02:00
|
|
|
******************************************************************************/
|
|
|
|
|
2013-03-16 21:18:53 +01:00
|
|
|
#include "base/dynamicobject.h"
|
|
|
|
#include "base/dynamictype.h"
|
2013-11-04 23:14:34 +01:00
|
|
|
#include "base/serializer.h"
|
2013-03-16 21:18:53 +01:00
|
|
|
#include "base/netstring.h"
|
|
|
|
#include "base/registry.h"
|
|
|
|
#include "base/stdiostream.h"
|
2013-08-28 08:18:58 +02:00
|
|
|
#include "base/debug.h"
|
2013-03-16 21:18:53 +01:00
|
|
|
#include "base/objectlock.h"
|
|
|
|
#include "base/logger_fwd.h"
|
|
|
|
#include "base/exception.h"
|
2013-03-25 18:36:15 +01:00
|
|
|
#include "base/scriptfunction.h"
|
2013-12-17 14:24:29 +01:00
|
|
|
#include "base/initialize.h"
|
|
|
|
#include "base/scriptvariable.h"
|
2013-03-16 21:18:53 +01:00
|
|
|
#include <fstream>
|
|
|
|
#include <boost/make_shared.hpp>
|
|
|
|
#include <boost/foreach.hpp>
|
2013-03-18 17:04:22 +01:00
|
|
|
#include <boost/exception/errinfo_api_function.hpp>
|
|
|
|
#include <boost/exception/errinfo_errno.hpp>
|
|
|
|
#include <boost/exception/errinfo_file_name.hpp>
|
2012-03-31 15:18:09 +02:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
2013-11-08 16:07:21 +01:00
|
|
|
REGISTER_TYPE(DynamicObject);
|
2013-12-17 14:24:29 +01:00
|
|
|
INITIALIZE_ONCE(&DynamicObject::StaticInitialize);
|
2013-11-08 11:17:46 +01:00
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
boost::signals2::signal<void (const DynamicObject::Ptr&)> DynamicObject::OnStarted;
|
|
|
|
boost::signals2::signal<void (const DynamicObject::Ptr&)> DynamicObject::OnStopped;
|
2014-02-21 17:26:56 +01:00
|
|
|
boost::signals2::signal<void (const DynamicObject::Ptr&, const String&)> DynamicObject::OnStateChanged;
|
2013-09-13 07:49:12 +02:00
|
|
|
boost::signals2::signal<void (const DynamicObject::Ptr&, const String&, bool)> DynamicObject::OnAuthorityChanged;
|
2012-08-02 09:38:08 +02:00
|
|
|
|
2013-12-17 14:24:29 +01:00
|
|
|
void DynamicObject::StaticInitialize(void)
|
|
|
|
{
|
|
|
|
ScriptVariable::Set("DomainPrivRead", DomainPrivRead, true, true);
|
|
|
|
ScriptVariable::Set("DomainPrivCheckResult", DomainPrivCheckResult, true, true);
|
|
|
|
ScriptVariable::Set("DomainPrivCommand", DomainPrivCommand, true, true);
|
|
|
|
|
2014-03-07 15:21:57 +01:00
|
|
|
ScriptVariable::Set("DomainPrivReadOnly", DomainPrivRead, true, true);
|
2013-12-17 14:24:29 +01:00
|
|
|
ScriptVariable::Set("DomainPrivReadWrite", DomainPrivRead | DomainPrivCheckResult | DomainPrivCommand, true, true);
|
|
|
|
}
|
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
DynamicObject::DynamicObject(void)
|
2013-10-26 09:41:45 +02:00
|
|
|
{ }
|
2012-04-20 14:20:25 +02:00
|
|
|
|
2012-12-04 08:42:24 +01:00
|
|
|
DynamicType::Ptr DynamicObject::GetType(void) const
|
2012-08-02 09:38:08 +02:00
|
|
|
{
|
2013-10-26 09:41:45 +02:00
|
|
|
return DynamicType::GetByName(GetTypeName());
|
2012-08-02 09:38:08 +02:00
|
|
|
}
|
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
bool DynamicObject::IsActive(void) const
|
2012-08-02 09:38:08 +02:00
|
|
|
{
|
2013-10-26 09:41:45 +02:00
|
|
|
return GetActive();
|
2013-09-12 10:03:48 +02:00
|
|
|
}
|
|
|
|
|
2013-09-13 07:49:12 +02:00
|
|
|
void DynamicObject::SetAuthority(const String& type, bool value)
|
2013-09-12 10:03:48 +02:00
|
|
|
{
|
2013-09-13 07:49:12 +02:00
|
|
|
ASSERT(!OwnsLock());
|
|
|
|
|
|
|
|
{
|
|
|
|
ObjectLock olock(this);
|
|
|
|
|
|
|
|
bool old_value = HasAuthority(type);
|
|
|
|
|
|
|
|
if (old_value == value)
|
|
|
|
return;
|
|
|
|
|
2013-11-11 08:22:58 +01:00
|
|
|
if (GetAuthorityInfo() == NULL)
|
|
|
|
SetAuthorityInfo(make_shared<Dictionary>());
|
|
|
|
|
2013-10-26 09:41:45 +02:00
|
|
|
GetAuthorityInfo()->Set(type, value);
|
2013-09-13 07:49:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
OnAuthorityChanged(GetSelf(), type, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DynamicObject::HasAuthority(const String& type) const
|
|
|
|
{
|
2013-10-30 15:59:14 +01:00
|
|
|
Dictionary::Ptr authorityInfo = GetAuthorityInfo();
|
|
|
|
|
2013-11-11 08:22:58 +01:00
|
|
|
if (!authorityInfo || !authorityInfo->Contains(type))
|
2013-10-30 15:59:14 +01:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return authorityInfo->Get(type);
|
2013-09-17 13:18:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void DynamicObject::SetPrivileges(const String& instance, int privs)
|
|
|
|
{
|
|
|
|
m_Privileges[instance] = privs;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DynamicObject::HasPrivileges(const String& instance, int privs) const
|
|
|
|
{
|
2013-09-19 14:47:17 +02:00
|
|
|
if (privs == 0)
|
|
|
|
return true;
|
|
|
|
|
2013-09-17 13:18:26 +02:00
|
|
|
std::map<String, int>::const_iterator it;
|
|
|
|
it = m_Privileges.find(instance);
|
|
|
|
|
|
|
|
if (it == m_Privileges.end())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return (it->second & privs) == privs;
|
|
|
|
}
|
|
|
|
|
2013-07-05 09:37:04 +02:00
|
|
|
void DynamicObject::SetExtension(const String& key, const Object::Ptr& object)
|
|
|
|
{
|
2013-10-26 09:41:45 +02:00
|
|
|
Dictionary::Ptr extensions = GetExtensions();
|
2013-07-05 09:37:04 +02:00
|
|
|
|
|
|
|
if (!extensions) {
|
2013-11-06 08:51:56 +01:00
|
|
|
extensions = make_shared<Dictionary>();
|
2013-10-26 09:41:45 +02:00
|
|
|
SetExtensions(extensions);
|
2013-07-05 09:37:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
extensions->Set(key, object);
|
|
|
|
}
|
|
|
|
|
|
|
|
Object::Ptr DynamicObject::GetExtension(const String& key)
|
|
|
|
{
|
2013-10-26 09:41:45 +02:00
|
|
|
Dictionary::Ptr extensions = GetExtensions();
|
2013-07-05 09:37:04 +02:00
|
|
|
|
|
|
|
if (!extensions)
|
|
|
|
return Object::Ptr();
|
|
|
|
|
|
|
|
return extensions->Get(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DynamicObject::ClearExtension(const String& key)
|
|
|
|
{
|
2013-10-26 09:41:45 +02:00
|
|
|
Dictionary::Ptr extensions = GetExtensions();
|
2013-07-05 09:37:04 +02:00
|
|
|
|
|
|
|
if (!extensions)
|
|
|
|
return;
|
|
|
|
|
|
|
|
extensions->Remove(key);
|
|
|
|
}
|
|
|
|
|
2012-08-02 09:38:08 +02:00
|
|
|
void DynamicObject::Register(void)
|
2012-03-31 15:18:09 +02:00
|
|
|
{
|
2013-03-07 16:00:10 +01:00
|
|
|
ASSERT(!OwnsLock());
|
2013-03-01 12:07:52 +01:00
|
|
|
|
2013-03-04 15:52:42 +01:00
|
|
|
DynamicType::Ptr dtype = GetType();
|
|
|
|
dtype->RegisterObject(GetSelf());
|
2013-02-27 12:44:51 +01:00
|
|
|
}
|
|
|
|
|
2012-09-28 10:39:28 +02:00
|
|
|
void DynamicObject::Start(void)
|
|
|
|
{
|
2013-03-07 16:00:10 +01:00
|
|
|
ASSERT(!OwnsLock());
|
2013-12-14 07:36:49 +01:00
|
|
|
ObjectLock olock(this);
|
2013-03-01 12:07:52 +01:00
|
|
|
|
2013-12-14 07:36:49 +01:00
|
|
|
SetStartCalled(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DynamicObject::Activate(void)
|
|
|
|
{
|
|
|
|
ASSERT(!OwnsLock());
|
|
|
|
|
|
|
|
Start();
|
|
|
|
|
|
|
|
ASSERT(GetStartCalled());
|
|
|
|
|
|
|
|
{
|
|
|
|
ObjectLock olock(this);
|
|
|
|
ASSERT(!IsActive());
|
|
|
|
SetActive(true);
|
|
|
|
}
|
2013-03-12 14:01:11 +01:00
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
OnStarted(GetSelf());
|
2013-03-12 14:01:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
void DynamicObject::Stop(void)
|
2012-06-12 09:36:18 +02:00
|
|
|
{
|
2013-03-07 16:00:10 +01:00
|
|
|
ASSERT(!OwnsLock());
|
2013-12-14 07:36:49 +01:00
|
|
|
ObjectLock olock(this);
|
|
|
|
|
|
|
|
SetStopCalled(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DynamicObject::Deactivate(void)
|
|
|
|
{
|
|
|
|
ASSERT(!OwnsLock());
|
|
|
|
|
|
|
|
{
|
|
|
|
ObjectLock olock(this);
|
|
|
|
|
|
|
|
if (!IsActive())
|
|
|
|
return;
|
|
|
|
|
|
|
|
SetActive(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
Stop();
|
2013-03-02 09:07:47 +01:00
|
|
|
|
2013-12-14 07:36:49 +01:00
|
|
|
ASSERT(GetStopCalled());
|
2012-06-12 09:36:18 +02:00
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
OnStopped(GetSelf());
|
2012-06-12 09:36:18 +02:00
|
|
|
}
|
|
|
|
|
2013-08-29 16:53:57 +02:00
|
|
|
void DynamicObject::OnConfigLoaded(void)
|
|
|
|
{
|
|
|
|
/* Nothing to do here. */
|
|
|
|
}
|
|
|
|
|
|
|
|
void DynamicObject::OnStateLoaded(void)
|
|
|
|
{
|
|
|
|
/* Nothing to do here. */
|
|
|
|
}
|
|
|
|
|
2013-03-25 18:36:15 +01:00
|
|
|
Value DynamicObject::InvokeMethod(const String& method,
|
2013-03-16 21:18:53 +01:00
|
|
|
const std::vector<Value>& arguments)
|
2012-07-14 15:59:59 +02:00
|
|
|
{
|
2013-03-02 09:07:47 +01:00
|
|
|
Dictionary::Ptr methods;
|
2013-02-26 10:13:54 +01:00
|
|
|
|
2013-10-26 09:41:45 +02:00
|
|
|
methods = GetMethods();
|
2012-07-14 15:59:59 +02:00
|
|
|
|
2013-03-08 16:02:33 +01:00
|
|
|
if (!methods)
|
2013-03-25 18:36:15 +01:00
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Method '" + method + "' does not exist."));
|
2013-03-08 16:02:33 +01:00
|
|
|
|
2014-04-02 09:04:17 +02:00
|
|
|
Value funcName = methods->Get(method);
|
2013-02-18 14:40:24 +01:00
|
|
|
|
2013-03-01 12:07:52 +01:00
|
|
|
if (funcName.IsEmpty())
|
2013-03-25 18:36:15 +01:00
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Method '" + method + "' does not exist."));
|
2012-08-02 09:38:08 +02:00
|
|
|
|
2014-04-02 09:04:17 +02:00
|
|
|
ScriptFunction::Ptr func;
|
2012-07-14 15:59:59 +02:00
|
|
|
|
2014-04-02 09:04:17 +02:00
|
|
|
if (funcName.IsObjectType<ScriptFunction>()) {
|
|
|
|
func = funcName;
|
|
|
|
} else {
|
|
|
|
func = ScriptFunction::GetByName(funcName);
|
|
|
|
|
|
|
|
if (!func)
|
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Function '" + String(funcName) + "' does not exist."));
|
|
|
|
}
|
2012-07-14 15:59:59 +02:00
|
|
|
|
2013-03-25 18:36:15 +01:00
|
|
|
return func->Invoke(arguments);
|
2012-07-14 15:59:59 +02:00
|
|
|
}
|
2012-07-24 13:13:02 +02:00
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
void DynamicObject::DumpObjects(const String& filename, int attributeTypes)
|
2012-07-24 13:13:02 +02:00
|
|
|
{
|
2013-03-16 21:18:53 +01:00
|
|
|
Log(LogInformation, "base", "Dumping program state to file '" + filename + "'");
|
2012-07-24 13:13:02 +02:00
|
|
|
|
2012-08-07 14:17:36 +02:00
|
|
|
String tempFilename = filename + ".tmp";
|
|
|
|
|
2013-03-16 21:18:53 +01:00
|
|
|
std::fstream fp;
|
2012-11-23 11:02:34 +01:00
|
|
|
fp.open(tempFilename.CStr(), std::ios_base::out);
|
2012-07-24 13:13:02 +02:00
|
|
|
|
|
|
|
if (!fp)
|
2013-03-16 21:18:53 +01:00
|
|
|
BOOST_THROW_EXCEPTION(std::runtime_error("Could not open '" + filename + "' file"));
|
2012-07-24 13:13:02 +02:00
|
|
|
|
2013-11-06 08:51:56 +01:00
|
|
|
StdioStream::Ptr sfp = make_shared<StdioStream>(&fp, false);
|
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()) {
|
2013-11-06 08:51:56 +01:00
|
|
|
Dictionary::Ptr persistentObject = make_shared<Dictionary>();
|
2012-07-24 13:13:02 +02:00
|
|
|
|
2013-03-02 09:07:47 +01:00
|
|
|
persistentObject->Set("type", type->GetName());
|
2012-07-27 16:05:02 +02:00
|
|
|
persistentObject->Set("name", object->GetName());
|
2012-07-24 14:18:33 +02:00
|
|
|
|
2013-11-08 11:17:46 +01:00
|
|
|
Dictionary::Ptr update = Serialize(object, attributeTypes);
|
2012-08-02 09:38:08 +02:00
|
|
|
|
|
|
|
if (!update)
|
|
|
|
continue;
|
2012-07-24 14:18:33 +02:00
|
|
|
|
2012-08-02 09:38:08 +02:00
|
|
|
persistentObject->Set("update", update);
|
2012-07-24 13:13:02 +02:00
|
|
|
|
2013-11-08 11:17:46 +01:00
|
|
|
String json = JsonSerialize(persistentObject);
|
2012-07-24 13:13:02 +02:00
|
|
|
|
2012-11-23 11:02:34 +01:00
|
|
|
NetString::WriteStringToStream(sfp, json);
|
2012-07-24 13:13:02 +02:00
|
|
|
}
|
|
|
|
}
|
2012-08-07 14:17:36 +02:00
|
|
|
|
2012-11-23 11:02:34 +01:00
|
|
|
sfp->Close();
|
2012-11-22 12:04:32 +01:00
|
|
|
|
2012-10-17 11:35:06 +02:00
|
|
|
fp.close();
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
_unlink(filename.CStr());
|
|
|
|
#endif /* _WIN32 */
|
|
|
|
|
2013-03-11 13:45:08 +01:00
|
|
|
if (rename(tempFilename.CStr(), filename.CStr()) < 0) {
|
|
|
|
BOOST_THROW_EXCEPTION(posix_error()
|
2013-03-18 17:04:22 +01:00
|
|
|
<< boost::errinfo_api_function("rename")
|
|
|
|
<< boost::errinfo_errno(errno)
|
|
|
|
<< boost::errinfo_file_name(tempFilename));
|
2013-03-11 13:45:08 +01:00
|
|
|
}
|
2012-07-24 13:13:02 +02:00
|
|
|
}
|
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
void DynamicObject::RestoreObjects(const String& filename, int attributeTypes)
|
2012-07-24 13:13:02 +02:00
|
|
|
{
|
2013-03-16 21:18:53 +01:00
|
|
|
Log(LogInformation, "base", "Restoring program state from file '" + filename + "'");
|
2012-07-24 13:13:02 +02:00
|
|
|
|
2012-11-23 11:02:34 +01:00
|
|
|
std::fstream fp;
|
|
|
|
fp.open(filename.CStr(), std::ios_base::in);
|
2012-07-24 13:13:02 +02:00
|
|
|
|
2013-11-06 08:51:56 +01:00
|
|
|
StdioStream::Ptr sfp = make_shared<StdioStream>(&fp, false);
|
2012-07-24 13:13:02 +02:00
|
|
|
|
2013-01-30 10:52:52 +01:00
|
|
|
unsigned long restored = 0;
|
|
|
|
|
2012-08-02 09:38:08 +02:00
|
|
|
String message;
|
2012-11-23 11:02:34 +01:00
|
|
|
while (NetString::ReadStringFromStream(sfp, &message)) {
|
2013-11-08 11:17:46 +01:00
|
|
|
Dictionary::Ptr persistentObject = JsonDeserialize(message);
|
2012-07-24 13:13:02 +02:00
|
|
|
|
2012-08-02 09:38:08 +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-12-04 08:42:24 +01:00
|
|
|
DynamicType::Ptr dt = DynamicType::GetByName(type);
|
|
|
|
|
|
|
|
if (!dt)
|
2013-08-27 12:21:41 +02:00
|
|
|
continue;
|
2012-12-04 08:42:24 +01:00
|
|
|
|
|
|
|
DynamicObject::Ptr object = dt->GetObject(name);
|
2012-08-05 03:10:53 +02:00
|
|
|
|
2013-08-29 10:17:12 +02:00
|
|
|
if (object) {
|
|
|
|
ASSERT(!object->IsActive());
|
2013-11-13 14:56:31 +01:00
|
|
|
#ifdef _DEBUG
|
2013-08-29 16:09:11 +02:00
|
|
|
Log(LogDebug, "base", "Restoring object '" + name + "' of type '" + type + "'.");
|
2013-11-13 14:56:31 +01:00
|
|
|
#endif /* _DEBUG */
|
2013-12-18 10:18:57 +01:00
|
|
|
Deserialize(object, update, false, attributeTypes);
|
2013-08-29 19:25:34 +02:00
|
|
|
object->OnStateLoaded();
|
2013-08-29 10:17:12 +02:00
|
|
|
}
|
2013-01-30 10:52:52 +01:00
|
|
|
|
|
|
|
restored++;
|
2012-07-24 13:13:02 +02:00
|
|
|
}
|
2012-11-22 12:04:32 +01:00
|
|
|
|
2012-11-23 11:02:34 +01:00
|
|
|
sfp->Close();
|
2013-01-30 10:52:52 +01:00
|
|
|
|
2013-03-16 21:18:53 +01:00
|
|
|
std::ostringstream msgbuf;
|
2013-01-30 10:52:52 +01:00
|
|
|
msgbuf << "Restored " << restored << " objects";
|
2013-03-16 21:18:53 +01:00
|
|
|
Log(LogInformation, "base", msgbuf.str());
|
2012-07-24 13:13:02 +02:00
|
|
|
}
|
2012-07-27 16:05:02 +02:00
|
|
|
|
2013-08-20 11:06:04 +02:00
|
|
|
void DynamicObject::StopObjects(void)
|
2012-08-07 13:08:14 +02:00
|
|
|
{
|
2013-02-18 23:44:24 +01:00
|
|
|
BOOST_FOREACH(const DynamicType::Ptr& dt, DynamicType::GetTypes()) {
|
2013-03-01 12:07:52 +01:00
|
|
|
BOOST_FOREACH(const DynamicObject::Ptr& object, dt->GetObjects()) {
|
2013-12-14 07:36:49 +01:00
|
|
|
object->Deactivate();
|
2012-08-07 13:08:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-04 08:42:24 +01:00
|
|
|
DynamicObject::Ptr DynamicObject::GetObject(const String& type, const String& name)
|
|
|
|
{
|
|
|
|
DynamicType::Ptr dtype = DynamicType::GetByName(type);
|
2013-03-01 12:07:52 +01:00
|
|
|
return dtype->GetObject(name);
|
2012-12-04 08:42:24 +01:00
|
|
|
}
|