/****************************************************************************** * Icinga 2 * * Copyright (C) 2012-2017 Icinga Development Team (https://www.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. * ******************************************************************************/ #ifndef VALUE_H #define VALUE_H #include "base/object.hpp" #include "base/string.hpp" #include #include namespace icinga { typedef double Timestamp; /** * The type of a Value. * * @ingroup base */ enum ValueType { ValueEmpty = 0, ValueNumber = 1, ValueBoolean = 2, ValueString = 3, ValueObject = 4 }; /** * A type that can hold an arbitrary value. * * @ingroup base */ class I2_BASE_API Value { public: Value(void) { } Value(std::nullptr_t) { } Value(int value) : m_Value(double(value)) { } Value(unsigned int value) : m_Value(double(value)) { } Value(long value) : m_Value(double(value)) { } Value(unsigned long value) : m_Value(double(value)) { } Value(long long value) : m_Value(double(value)) { } Value(unsigned long long value) : m_Value(double(value)) { } Value(double value) : m_Value(value) { } Value(bool value) : m_Value(value) { } Value(const String& value) : m_Value(value) { } Value(String&& value) : m_Value(value) { } Value(const char *value) : m_Value(String(value)) { } Value(const Value& other) : m_Value(other.m_Value) { } Value(Value&& other) { #if BOOST_VERSION >= 105400 m_Value = std::move(other.m_Value); #else /* BOOST_VERSION */ m_Value.swap(other.m_Value); #endif /* BOOST_VERSION */ } Value(Object *value) { if (!value) return; m_Value = Object::Ptr(value); } template Value(const intrusive_ptr& value) { if (!value) return; m_Value = static_pointer_cast(value); } bool ToBool(void) const; operator double(void) const; operator String(void) const; Value& operator=(const Value& other) { m_Value = other.m_Value; return *this; } Value& operator=(Value&& other) { #if BOOST_VERSION >= 105400 m_Value = std::move(other.m_Value); #else /* BOOST_VERSION */ m_Value.swap(other.m_Value); #endif /* BOOST_VERSION */ return *this; } bool operator==(bool rhs) const; bool operator!=(bool rhs) const; bool operator==(int rhs) const; bool operator!=(int rhs) const; bool operator==(double rhs) const; bool operator!=(double rhs) const; bool operator==(const char *rhs) const; bool operator!=(const char *rhs) const; bool operator==(const String& rhs) const; bool operator!=(const String& rhs) const; bool operator==(const Value& rhs) const; bool operator!=(const Value& rhs) const; template operator intrusive_ptr(void) const { if (IsEmpty() && !IsString()) return intrusive_ptr(); if (!IsObject()) BOOST_THROW_EXCEPTION(std::runtime_error("Cannot convert value of type '" + GetTypeName() + "' to an object.")); const Object::Ptr& object = boost::get(m_Value); ASSERT(object); intrusive_ptr tobject = dynamic_pointer_cast(object); if (!tobject) BOOST_THROW_EXCEPTION(std::bad_cast()); return tobject; } /** * Checks whether the variant is empty. * * @returns true if the variant is empty, false otherwise. */ bool IsEmpty(void) const { return (GetType() == ValueEmpty || (IsString() && boost::get(m_Value).IsEmpty())); } /** * Checks whether the variant is scalar (i.e. not an object and not empty). * * @returns true if the variant is scalar, false otherwise. */ bool IsScalar(void) const { return !IsEmpty() && !IsObject(); } /** * Checks whether the variant is a number. * * @returns true if the variant is a number. */ bool IsNumber(void) const { return (GetType() == ValueNumber); } /** * Checks whether the variant is a boolean. * * @returns true if the variant is a boolean. */ bool IsBoolean(void) const { return (GetType() == ValueBoolean); } /** * Checks whether the variant is a string. * * @returns true if the variant is a string. */ bool IsString(void) const { return (GetType() == ValueString); } /** * Checks whether the variant is a non-null object. * * @returns true if the variant is a non-null object, false otherwise. */ bool IsObject(void) const { return (GetType() == ValueObject); } template bool IsObjectType(void) const { if (!IsObject()) return false; return dynamic_cast(boost::get(m_Value).get()); } /** * Returns the type of the value. * * @returns The type. */ ValueType GetType(void) const { return static_cast(m_Value.which()); } void Swap(Value& other) { m_Value.swap(other.m_Value); } String GetTypeName(void) const; Type::Ptr GetReflectionType(void) const; Value Clone(void) const; template const T& Get(void) const { return boost::get(m_Value); } private: boost::variant m_Value; }; extern I2_BASE_API Value Empty; I2_BASE_API Value operator+(const Value& lhs, const char *rhs); I2_BASE_API Value operator+(const char *lhs, const Value& rhs); I2_BASE_API Value operator+(const Value& lhs, const String& rhs); I2_BASE_API Value operator+(const String& lhs, const Value& rhs); I2_BASE_API Value operator+(const Value& lhs, const Value& rhs); I2_BASE_API Value operator+(const Value& lhs, double rhs); I2_BASE_API Value operator+(double lhs, const Value& rhs); I2_BASE_API Value operator+(const Value& lhs, int rhs); I2_BASE_API Value operator+(int lhs, const Value& rhs); I2_BASE_API Value operator-(const Value& lhs, const Value& rhs); I2_BASE_API Value operator-(const Value& lhs, double rhs); I2_BASE_API Value operator-(double lhs, const Value& rhs); I2_BASE_API Value operator-(const Value& lhs, int rhs); I2_BASE_API Value operator-(int lhs, const Value& rhs); I2_BASE_API Value operator*(const Value& lhs, const Value& rhs); I2_BASE_API Value operator*(const Value& lhs, double rhs); I2_BASE_API Value operator*(double lhs, const Value& rhs); I2_BASE_API Value operator*(const Value& lhs, int rhs); I2_BASE_API Value operator*(int lhs, const Value& rhs); I2_BASE_API Value operator/(const Value& lhs, const Value& rhs); I2_BASE_API Value operator/(const Value& lhs, double rhs); I2_BASE_API Value operator/(double lhs, const Value& rhs); I2_BASE_API Value operator/(const Value& lhs, int rhs); I2_BASE_API Value operator/(int lhs, const Value& rhs); I2_BASE_API Value operator%(const Value& lhs, const Value& rhs); I2_BASE_API Value operator%(const Value& lhs, double rhs); I2_BASE_API Value operator%(double lhs, const Value& rhs); I2_BASE_API Value operator%(const Value& lhs, int rhs); I2_BASE_API Value operator%(int lhs, const Value& rhs); I2_BASE_API Value operator^(const Value& lhs, const Value& rhs); I2_BASE_API Value operator^(const Value& lhs, double rhs); I2_BASE_API Value operator^(double lhs, const Value& rhs); I2_BASE_API Value operator^(const Value& lhs, int rhs); I2_BASE_API Value operator^(int lhs, const Value& rhs); I2_BASE_API Value operator&(const Value& lhs, const Value& rhs); I2_BASE_API Value operator&(const Value& lhs, double rhs); I2_BASE_API Value operator&(double lhs, const Value& rhs); I2_BASE_API Value operator&(const Value& lhs, int rhs); I2_BASE_API Value operator&(int lhs, const Value& rhs); I2_BASE_API Value operator|(const Value& lhs, const Value& rhs); I2_BASE_API Value operator|(const Value& lhs, double rhs); I2_BASE_API Value operator|(double lhs, const Value& rhs); I2_BASE_API Value operator|(const Value& lhs, int rhs); I2_BASE_API Value operator|(int lhs, const Value& rhs); I2_BASE_API Value operator<<(const Value& lhs, const Value& rhs); I2_BASE_API Value operator<<(const Value& lhs, double rhs); I2_BASE_API Value operator<<(double lhs, const Value& rhs); I2_BASE_API Value operator<<(const Value& lhs, int rhs); I2_BASE_API Value operator<<(int lhs, const Value& rhs); I2_BASE_API Value operator>>(const Value& lhs, const Value& rhs); I2_BASE_API Value operator>>(const Value& lhs, double rhs); I2_BASE_API Value operator>>(double lhs, const Value& rhs); I2_BASE_API Value operator>>(const Value& lhs, int rhs); I2_BASE_API Value operator>>(int lhs, const Value& rhs); I2_BASE_API bool operator<(const Value& lhs, const Value& rhs); I2_BASE_API bool operator<(const Value& lhs, double rhs); I2_BASE_API bool operator<(double lhs, const Value& rhs); I2_BASE_API bool operator<(const Value& lhs, int rhs); I2_BASE_API bool operator<(int lhs, const Value& rhs); I2_BASE_API bool operator>(const Value& lhs, const Value& rhs); I2_BASE_API bool operator>(const Value& lhs, double rhs); I2_BASE_API bool operator>(double lhs, const Value& rhs); I2_BASE_API bool operator>(const Value& lhs, int rhs); I2_BASE_API bool operator>(int lhs, const Value& rhs); I2_BASE_API bool operator<=(const Value& lhs, const Value& rhs); I2_BASE_API bool operator<=(const Value& lhs, double rhs); I2_BASE_API bool operator<=(double lhs, const Value& rhs); I2_BASE_API bool operator<=(const Value& lhs, int rhs); I2_BASE_API bool operator<=(int lhs, const Value& rhs); I2_BASE_API bool operator>=(const Value& lhs, const Value& rhs); I2_BASE_API bool operator>=(const Value& lhs, double rhs); I2_BASE_API bool operator>=(double lhs, const Value& rhs); I2_BASE_API bool operator>=(const Value& lhs, int rhs); I2_BASE_API bool operator>=(int lhs, const Value& rhs); I2_BASE_API std::ostream& operator<<(std::ostream& stream, const Value& value); I2_BASE_API std::istream& operator>>(std::istream& stream, Value& value); } #endif /* VALUE_H */