/****************************************************************************** * Icinga 2 * * Copyright (C) 2012-2018 Icinga Development Team (https://icinga.com/) * * * * 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 "base/type.hpp" #include "base/scriptglobal.hpp" #include "base/namespace.hpp" #include "base/objectlock.hpp" using namespace icinga; Type::Ptr Type::TypeInstance; /* Ensure that the priority is lower than the basic namespace initialization in scriptframe.cpp. */ INITIALIZE_ONCE_WITH_PRIORITY([]() { Type::Ptr type = new TypeType(); type->SetPrototype(TypeType::GetPrototype()); Type::TypeInstance = type; Type::Register(type); }, 20); String Type::ToString() const { return "type '" + GetName() + "'"; } void Type::Register(const Type::Ptr& type) { ScriptGlobal::Set("Types." + type->GetName(), type, true); } Type::Ptr Type::GetByName(const String& name) { Namespace::Ptr typesNS = ScriptGlobal::Get("Types", &Empty); if (!typesNS) return nullptr; Value ptype; if (!typesNS->Get(name, &ptype)) return nullptr; if (!ptype.IsObjectType()) return nullptr; return ptype; } std::vector Type::GetAllTypes() { std::vector types; Namespace::Ptr typesNS = ScriptGlobal::Get("Types", &Empty); if (typesNS) { ObjectLock olock(typesNS); for (const Namespace::Pair& kv : typesNS) { Value value = kv.second->Get(); if (value.IsObjectType()) types.push_back(value); } } return types; } String Type::GetPluralName() const { String name = GetName(); if (name.GetLength() >= 2 && name[name.GetLength() - 1] == 'y' && name.SubStr(name.GetLength() - 2, 1).FindFirstOf("aeiou") == String::NPos) return name.SubStr(0, name.GetLength() - 1) + "ies"; else return name + "s"; } Object::Ptr Type::Instantiate(const std::vector& args) const { ObjectFactory factory = GetFactory(); if (!factory) BOOST_THROW_EXCEPTION(std::runtime_error("Type does not have a factory function.")); return factory(args); } bool Type::IsAbstract() const { return ((GetAttributes() & TAAbstract) != 0); } bool Type::IsAssignableFrom(const Type::Ptr& other) const { for (Type::Ptr t = other; t; t = t->GetBaseType()) { if (t.get() == this) return true; } return false; } Object::Ptr Type::GetPrototype() const { return m_Prototype; } void Type::SetPrototype(const Object::Ptr& object) { m_Prototype = object; } void Type::SetField(int id, const Value& value, bool suppress_events, const Value& cookie) { if (id == 1) { SetPrototype(value); return; } Object::SetField(id, value, suppress_events, cookie); } Value Type::GetField(int id) const { int real_id = id - Object::TypeInstance->GetFieldCount(); if (real_id < 0) return Object::GetField(id); if (real_id == 0) return GetName(); else if (real_id == 1) return GetPrototype(); else if (real_id == 2) return GetBaseType(); BOOST_THROW_EXCEPTION(std::runtime_error("Invalid field ID.")); } std::vector Type::GetLoadDependencies() const { return std::vector(); } int Type::GetActivationPriority() const { return 0; } void Type::RegisterAttributeHandler(int fieldId, const AttributeHandler& callback) { throw std::runtime_error("Invalid field ID."); } String TypeType::GetName() const { return "Type"; } Type::Ptr TypeType::GetBaseType() const { return Object::TypeInstance; } int TypeType::GetAttributes() const { return 0; } int TypeType::GetFieldId(const String& name) const { int base_field_count = GetBaseType()->GetFieldCount(); if (name == "name") return base_field_count + 0; else if (name == "prototype") return base_field_count + 1; else if (name == "base") return base_field_count + 2; return GetBaseType()->GetFieldId(name); } Field TypeType::GetFieldInfo(int id) const { int real_id = id - GetBaseType()->GetFieldCount(); if (real_id < 0) return GetBaseType()->GetFieldInfo(id); if (real_id == 0) return {0, "String", "name", "", nullptr, 0, 0}; else if (real_id == 1) return Field(1, "Object", "prototype", "", nullptr, 0, 0); else if (real_id == 2) return Field(2, "Type", "base", "", nullptr, 0, 0); throw std::runtime_error("Invalid field ID."); } int TypeType::GetFieldCount() const { return GetBaseType()->GetFieldCount() + 3; } ObjectFactory TypeType::GetFactory() const { return nullptr; }