2019-02-25 14:48:22 +01:00
|
|
|
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
2013-11-08 11:17:46 +01:00
|
|
|
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "base/type.hpp"
|
2024-09-02 17:34:47 +02:00
|
|
|
#include "base/atomic.hpp"
|
|
|
|
#include "base/configobject.hpp"
|
|
|
|
#include "base/debug.hpp"
|
2014-12-14 11:33:45 +01:00
|
|
|
#include "base/scriptglobal.hpp"
|
2018-08-07 13:55:41 +02:00
|
|
|
#include "base/namespace.hpp"
|
2016-08-12 16:53:44 +02:00
|
|
|
#include "base/objectlock.hpp"
|
2024-09-02 17:34:47 +02:00
|
|
|
#include <algorithm>
|
|
|
|
#include <functional>
|
|
|
|
#include <unordered_map>
|
2013-11-08 11:17:46 +01:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
2015-08-04 14:47:44 +02:00
|
|
|
Type::Ptr Type::TypeInstance;
|
|
|
|
|
2022-12-13 15:28:24 +01:00
|
|
|
static Namespace::Ptr l_TypesNS = new Namespace(true);
|
|
|
|
|
|
|
|
INITIALIZE_ONCE_WITH_PRIORITY([]() {
|
|
|
|
ScriptGlobal::GetGlobals()->Set("Types", l_TypesNS, true);
|
|
|
|
}, InitializePriority::CreateNamespaces);
|
|
|
|
|
|
|
|
INITIALIZE_ONCE_WITH_PRIORITY([]() {
|
|
|
|
l_TypesNS->Freeze();
|
|
|
|
|
|
|
|
ObjectLock olock (l_TypesNS);
|
|
|
|
for (const auto& t : l_TypesNS) {
|
|
|
|
VERIFY(t.second.Val.IsObjectType<Type>());
|
|
|
|
}
|
|
|
|
}, InitializePriority::FreezeNamespaces);
|
|
|
|
|
2018-09-04 15:17:34 +02:00
|
|
|
/* Ensure that the priority is lower than the basic namespace initialization in scriptframe.cpp. */
|
2016-08-27 09:35:08 +02:00
|
|
|
INITIALIZE_ONCE_WITH_PRIORITY([]() {
|
2015-01-14 09:51:44 +01:00
|
|
|
Type::Ptr type = new TypeType();
|
2015-08-04 14:47:44 +02:00
|
|
|
type->SetPrototype(TypeType::GetPrototype());
|
2015-01-14 09:51:44 +01:00
|
|
|
Type::TypeInstance = type;
|
|
|
|
Type::Register(type);
|
2022-12-12 12:32:59 +01:00
|
|
|
}, InitializePriority::RegisterTypeType);
|
2015-01-14 09:51:44 +01:00
|
|
|
|
2024-09-02 17:34:47 +02:00
|
|
|
static std::vector<Type::Ptr> l_SortedByLoadDependencies;
|
|
|
|
static Atomic l_SortingByLoadDependenciesDone (false);
|
|
|
|
|
|
|
|
typedef std::unordered_map<Type*, bool> Visited; // https://stackoverflow.com/a/8942986
|
|
|
|
|
|
|
|
INITIALIZE_ONCE_WITH_PRIORITY([] {
|
|
|
|
auto types (Type::GetAllTypes());
|
|
|
|
|
|
|
|
types.erase(std::remove_if(types.begin(), types.end(), [](auto& type) {
|
|
|
|
return !ConfigObject::TypeInstance->IsAssignableFrom(type);
|
|
|
|
}), types.end());
|
|
|
|
|
|
|
|
// Depth-first search
|
|
|
|
std::unordered_set<Type*> unsorted;
|
|
|
|
Visited visited;
|
|
|
|
std::vector<Type::Ptr> sorted;
|
|
|
|
|
|
|
|
for (auto type : types) {
|
|
|
|
unsorted.emplace(type.get());
|
|
|
|
}
|
|
|
|
|
|
|
|
std::function<void(Type*)> visit ([&visit, &unsorted, &visited, &sorted](Type* type) {
|
|
|
|
if (unsorted.find(type) == unsorted.end()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool& alreadyVisited (visited.at(type));
|
|
|
|
VERIFY(!alreadyVisited);
|
|
|
|
alreadyVisited = true;
|
|
|
|
|
|
|
|
for (auto dep : type->GetLoadDependencies()) {
|
|
|
|
visit(dep);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsorted.erase(type);
|
|
|
|
sorted.emplace_back(type);
|
|
|
|
});
|
|
|
|
|
|
|
|
while (!unsorted.empty()) {
|
|
|
|
for (auto& type : types) {
|
|
|
|
visited[type.get()] = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
visit(*unsorted.begin());
|
|
|
|
}
|
|
|
|
|
|
|
|
VERIFY(sorted.size() == types.size());
|
|
|
|
VERIFY(sorted[0]->GetLoadDependencies().empty());
|
|
|
|
|
|
|
|
std::swap(sorted, l_SortedByLoadDependencies);
|
|
|
|
l_SortingByLoadDependenciesDone.store(true);
|
|
|
|
}, InitializePriority::SortTypes);
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String Type::ToString() const
|
2014-12-08 09:12:40 +01:00
|
|
|
{
|
|
|
|
return "type '" + GetName() + "'";
|
|
|
|
}
|
|
|
|
|
2014-11-03 00:44:04 +01:00
|
|
|
void Type::Register(const Type::Ptr& type)
|
2013-11-08 11:17:46 +01:00
|
|
|
{
|
2023-01-09 17:09:46 +01:00
|
|
|
ScriptGlobal::Set("Types." + type->GetName(), type);
|
2013-11-08 11:17:46 +01:00
|
|
|
}
|
|
|
|
|
2014-11-03 00:44:04 +01:00
|
|
|
Type::Ptr Type::GetByName(const String& name)
|
2013-11-08 11:17:46 +01:00
|
|
|
{
|
2018-08-07 13:55:41 +02:00
|
|
|
Value ptype;
|
2013-11-08 11:17:46 +01:00
|
|
|
|
2022-12-13 15:28:24 +01:00
|
|
|
if (!l_TypesNS->Get(name, &ptype))
|
2017-11-30 08:36:35 +01:00
|
|
|
return nullptr;
|
2013-11-08 11:17:46 +01:00
|
|
|
|
2014-12-08 08:49:32 +01:00
|
|
|
return ptype;
|
2013-11-08 11:17:46 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
std::vector<Type::Ptr> Type::GetAllTypes()
|
2016-08-12 16:53:44 +02:00
|
|
|
{
|
|
|
|
std::vector<Type::Ptr> types;
|
|
|
|
|
2018-08-07 13:55:41 +02:00
|
|
|
Namespace::Ptr typesNS = ScriptGlobal::Get("Types", &Empty);
|
2016-08-12 16:53:44 +02:00
|
|
|
|
|
|
|
if (typesNS) {
|
|
|
|
ObjectLock olock(typesNS);
|
|
|
|
|
2018-08-07 13:55:41 +02:00
|
|
|
for (const Namespace::Pair& kv : typesNS) {
|
2023-01-03 12:02:51 +01:00
|
|
|
Value value = kv.second.Val;
|
2018-08-07 13:55:41 +02:00
|
|
|
|
|
|
|
if (value.IsObjectType<Type>())
|
|
|
|
types.push_back(value);
|
2016-08-25 06:19:44 +02:00
|
|
|
}
|
2016-08-12 16:53:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return types;
|
|
|
|
}
|
|
|
|
|
2024-09-02 17:34:47 +02:00
|
|
|
const std::vector<Type::Ptr>& Type::GetConfigTypesSortedByLoadDependencies()
|
|
|
|
{
|
|
|
|
VERIFY(l_SortingByLoadDependenciesDone.load());
|
|
|
|
return l_SortedByLoadDependencies;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String Type::GetPluralName() const
|
2015-08-13 09:02:52 +02:00
|
|
|
{
|
|
|
|
String name = GetName();
|
|
|
|
|
2015-11-01 10:35:08 +01:00
|
|
|
if (name.GetLength() >= 2 && name[name.GetLength() - 1] == 'y' &&
|
2017-12-19 15:50:05 +01:00
|
|
|
name.SubStr(name.GetLength() - 2, 1).FindFirstOf("aeiou") == String::NPos)
|
2015-08-13 09:02:52 +02:00
|
|
|
return name.SubStr(0, name.GetLength() - 1) + "ies";
|
|
|
|
else
|
|
|
|
return name + "s";
|
|
|
|
}
|
|
|
|
|
2016-03-29 12:45:22 +02:00
|
|
|
Object::Ptr Type::Instantiate(const std::vector<Value>& args) const
|
2013-11-08 11:17:46 +01:00
|
|
|
{
|
2014-11-07 12:32:25 +01:00
|
|
|
ObjectFactory factory = GetFactory();
|
|
|
|
|
|
|
|
if (!factory)
|
2015-08-17 07:59:44 +02:00
|
|
|
BOOST_THROW_EXCEPTION(std::runtime_error("Type does not have a factory function."));
|
2014-11-07 12:32:25 +01:00
|
|
|
|
2016-03-29 12:45:22 +02:00
|
|
|
return factory(args);
|
2013-11-08 11:17:46 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
bool Type::IsAbstract() const
|
2013-12-18 10:18:57 +01:00
|
|
|
{
|
2014-05-11 06:00:34 +02:00
|
|
|
return ((GetAttributes() & TAAbstract) != 0);
|
2013-12-18 10:18:57 +01:00
|
|
|
}
|
|
|
|
|
2014-11-03 00:44:04 +01:00
|
|
|
bool Type::IsAssignableFrom(const Type::Ptr& other) const
|
2013-11-08 11:17:46 +01:00
|
|
|
{
|
2014-11-03 00:44:04 +01:00
|
|
|
for (Type::Ptr t = other; t; t = t->GetBaseType()) {
|
|
|
|
if (t.get() == this)
|
2013-11-08 11:17:46 +01:00
|
|
|
return true;
|
2013-11-08 21:37:21 +01:00
|
|
|
}
|
2013-11-08 11:17:46 +01:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
Object::Ptr Type::GetPrototype() const
|
2014-12-12 15:19:23 +01:00
|
|
|
{
|
|
|
|
return m_Prototype;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Type::SetPrototype(const Object::Ptr& object)
|
|
|
|
{
|
|
|
|
m_Prototype = object;
|
|
|
|
}
|
|
|
|
|
2015-08-17 09:34:32 +02:00
|
|
|
void Type::SetField(int id, const Value& value, bool suppress_events, const Value& cookie)
|
2015-01-14 13:32:40 +01:00
|
|
|
{
|
2015-08-26 13:05:09 +02:00
|
|
|
if (id == 1) {
|
2015-01-14 13:32:40 +01:00
|
|
|
SetPrototype(value);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 09:34:32 +02:00
|
|
|
Object::SetField(id, value, suppress_events, cookie);
|
2015-01-14 13:32:40 +01:00
|
|
|
}
|
|
|
|
|
2015-01-14 09:51:44 +01:00
|
|
|
Value Type::GetField(int id) const
|
|
|
|
{
|
2015-11-05 10:58:09 +01:00
|
|
|
int real_id = id - Object::TypeInstance->GetFieldCount();
|
|
|
|
if (real_id < 0)
|
|
|
|
return Object::GetField(id);
|
|
|
|
|
|
|
|
if (real_id == 0)
|
2015-08-26 13:05:09 +02:00
|
|
|
return GetName();
|
2015-11-05 10:58:09 +01:00
|
|
|
else if (real_id == 1)
|
2015-08-26 13:05:09 +02:00
|
|
|
return GetPrototype();
|
2015-11-05 10:58:09 +01:00
|
|
|
else if (real_id == 2)
|
2015-08-17 07:59:44 +02:00
|
|
|
return GetBaseType();
|
2015-01-14 09:51:44 +01:00
|
|
|
|
2015-11-05 10:58:09 +01:00
|
|
|
BOOST_THROW_EXCEPTION(std::runtime_error("Invalid field ID."));
|
2015-01-14 09:51:44 +01:00
|
|
|
}
|
|
|
|
|
2022-10-28 13:11:58 +02:00
|
|
|
const std::unordered_set<Type*>& Type::GetLoadDependencies() const
|
2015-02-25 12:43:03 +01:00
|
|
|
{
|
2022-10-28 13:11:58 +02:00
|
|
|
static const std::unordered_set<Type*> noDeps;
|
|
|
|
return noDeps;
|
2015-02-25 12:43:03 +01:00
|
|
|
}
|
|
|
|
|
2018-04-26 15:53:11 +02:00
|
|
|
int Type::GetActivationPriority() const
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-08-04 14:47:44 +02:00
|
|
|
void Type::RegisterAttributeHandler(int fieldId, const AttributeHandler& callback)
|
|
|
|
{
|
|
|
|
throw std::runtime_error("Invalid field ID.");
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String TypeType::GetName() const
|
2015-01-14 09:51:44 +01:00
|
|
|
{
|
|
|
|
return "Type";
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
Type::Ptr TypeType::GetBaseType() const
|
2015-01-14 09:51:44 +01:00
|
|
|
{
|
2015-08-26 10:58:03 +02:00
|
|
|
return Object::TypeInstance;
|
2015-01-14 09:51:44 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
int TypeType::GetAttributes() const
|
2015-01-14 09:51:44 +01:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int TypeType::GetFieldId(const String& name) const
|
|
|
|
{
|
2015-08-26 13:05:09 +02:00
|
|
|
int base_field_count = GetBaseType()->GetFieldCount();
|
|
|
|
|
|
|
|
if (name == "name")
|
|
|
|
return base_field_count + 0;
|
|
|
|
else if (name == "prototype")
|
|
|
|
return base_field_count + 1;
|
2015-08-17 07:59:44 +02:00
|
|
|
else if (name == "base")
|
2015-08-26 13:05:09 +02:00
|
|
|
return base_field_count + 2;
|
2015-01-14 09:51:44 +01:00
|
|
|
|
2015-08-26 10:58:03 +02:00
|
|
|
return GetBaseType()->GetFieldId(name);
|
2015-01-14 09:51:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Field TypeType::GetFieldInfo(int id) const
|
|
|
|
{
|
2015-08-26 10:58:03 +02:00
|
|
|
int real_id = id - GetBaseType()->GetFieldCount();
|
|
|
|
if (real_id < 0)
|
|
|
|
return GetBaseType()->GetFieldInfo(id);
|
|
|
|
|
2015-11-05 10:52:25 +01:00
|
|
|
if (real_id == 0)
|
2018-01-04 09:28:24 +01:00
|
|
|
return {0, "String", "name", "", nullptr, 0, 0};
|
2015-11-05 10:52:25 +01:00
|
|
|
else if (real_id == 1)
|
2017-12-14 15:37:20 +01:00
|
|
|
return Field(1, "Object", "prototype", "", nullptr, 0, 0);
|
2015-11-05 10:52:25 +01:00
|
|
|
else if (real_id == 2)
|
2017-12-14 15:37:20 +01:00
|
|
|
return Field(2, "Type", "base", "", nullptr, 0, 0);
|
2015-01-14 09:51:44 +01:00
|
|
|
|
|
|
|
throw std::runtime_error("Invalid field ID.");
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
int TypeType::GetFieldCount() const
|
2015-01-14 09:51:44 +01:00
|
|
|
{
|
2015-08-26 13:05:09 +02:00
|
|
|
return GetBaseType()->GetFieldCount() + 3;
|
2015-01-14 09:51:44 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
ObjectFactory TypeType::GetFactory() const
|
2015-01-14 09:51:44 +01:00
|
|
|
{
|
2017-12-14 15:37:20 +01:00
|
|
|
return nullptr;
|
2015-01-14 09:51:44 +01:00
|
|
|
}
|