2019-02-25 14:48:22 +01:00
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
2012-05-10 12:06:41 +02:00
2014-05-25 16:23:35 +02:00
# include "base/object.hpp"
# include "base/value.hpp"
2014-12-12 15:19:23 +01:00
# include "base/dictionary.hpp"
2014-11-03 07:07:54 +01:00
# include "base/primitivetype.hpp"
2014-12-08 09:12:40 +01:00
# include "base/utility.hpp"
2016-01-19 15:24:17 +01:00
# include "base/timer.hpp"
# include "base/logger.hpp"
2016-04-18 11:29:43 +02:00
# include "base/exception.hpp"
2016-04-18 17:21:45 +02:00
# include <boost/lexical_cast.hpp>
2018-01-03 06:01:02 +01:00
# include <boost/thread/recursive_mutex.hpp>
2019-04-17 18:03:40 +02:00
# include <thread>
2012-03-28 13:24:49 +02:00
using namespace icinga ;
2012-05-14 19:14:23 +02:00
2015-11-05 10:29:02 +01:00
DEFINE_TYPE_INSTANCE ( Object ) ;
2014-11-03 00:44:04 +01:00
2016-01-19 16:43:46 +01:00
# ifdef I2_LEAK_DEBUG
2021-02-02 10:16:04 +01:00
static std : : mutex l_ObjectCountLock ;
2016-01-19 15:24:17 +01:00
static std : : map < String , int > l_ObjectCounts ;
static Timer : : Ptr l_ObjectCountTimer ;
2016-01-19 16:43:46 +01:00
# endif /* I2_LEAK_DEBUG */
2016-01-19 15:24:17 +01:00
2019-03-11 12:05:01 +01:00
/**
* Constructor for the Object class .
*/
Object : : Object ( )
{
m_References . store ( 0 ) ;
2019-04-24 12:10:57 +02:00
# ifdef I2_DEBUG
2019-04-17 18:03:40 +02:00
m_LockOwner . store ( decltype ( m_LockOwner . load ( ) ) ( ) ) ;
2019-04-24 12:10:57 +02:00
# endif /* I2_DEBUG */
2019-03-11 12:05:01 +01:00
}
2012-05-14 19:14:23 +02:00
/**
* Destructor for the Object class .
*/
2018-01-04 04:25:35 +01:00
Object : : ~ Object ( )
2016-01-19 15:25:44 +01:00
{
}
2012-06-14 15:16:41 +02:00
2014-12-08 09:12:40 +01:00
/**
* Returns a string representation for the object .
*/
2018-01-04 04:25:35 +01:00
String Object : : ToString ( ) const
2014-12-08 09:12:40 +01:00
{
2015-11-10 07:59:10 +01:00
return " Object of type ' " + GetReflectionType ( ) - > GetName ( ) + " ' " ;
2014-12-08 09:12:40 +01:00
}
2014-12-19 12:19:28 +01:00
# ifdef I2_DEBUG
2012-09-14 14:41:17 +02:00
/**
2013-03-04 15:52:42 +01:00
* Checks if the calling thread owns the lock on this object .
2012-09-14 14:41:17 +02:00
*
2013-03-01 12:07:52 +01:00
* @ returns True if the calling thread owns the lock , false otherwise .
2012-09-14 14:41:17 +02:00
*/
2018-01-04 04:25:35 +01:00
bool Object : : OwnsLock ( ) const
2012-08-03 13:19:55 +02:00
{
2019-04-17 18:03:40 +02:00
return m_LockOwner . load ( ) = = std : : this_thread : : get_id ( ) ;
2012-08-03 13:19:55 +02:00
}
2014-12-19 12:19:28 +01:00
# endif /* I2_DEBUG */
2013-06-13 11:33:00 +02:00
2015-08-04 14:47:44 +02:00
void Object : : SetField ( int id , const Value & , bool , const Value & )
2013-11-04 23:14:34 +01:00
{
2015-07-30 08:23:43 +02:00
if ( id = = 0 )
2015-11-05 10:29:02 +01:00
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Type field cannot be set. " ) ) ;
2015-07-30 08:23:43 +02:00
else
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Invalid field ID. " ) ) ;
2013-11-04 23:14:34 +01:00
}
2015-07-30 08:23:43 +02:00
Value Object : : GetField ( int id ) const
2013-11-04 23:14:34 +01:00
{
2015-07-30 08:23:43 +02:00
if ( id = = 0 )
2015-11-05 10:29:02 +01:00
return GetReflectionType ( ) - > GetName ( ) ;
2015-07-30 08:23:43 +02:00
else
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Invalid field ID. " ) ) ;
2013-11-04 23:14:34 +01:00
}
2014-05-11 06:30:50 +02:00
2016-04-18 11:29:43 +02:00
bool Object : : HasOwnField ( const String & field ) const
{
Type : : Ptr type = GetReflectionType ( ) ;
if ( ! type )
return false ;
return type - > GetFieldId ( field ) ! = - 1 ;
}
2016-09-01 07:41:41 +02:00
bool Object : : GetOwnField ( const String & field , Value * result ) const
{
Type : : Ptr type = GetReflectionType ( ) ;
if ( ! type )
return false ;
int tid = type - > GetFieldId ( field ) ;
if ( tid = = - 1 )
return false ;
* result = GetField ( tid ) ;
return true ;
}
2016-04-18 11:29:43 +02:00
Value Object : : GetFieldByName ( const String & field , bool sandboxed , const DebugInfo & debugInfo ) const
{
Type : : Ptr type = GetReflectionType ( ) ;
if ( ! type )
return Empty ;
int fid = type - > GetFieldId ( field ) ;
if ( fid = = - 1 )
return GetPrototypeField ( const_cast < Object * > ( this ) , field , true , debugInfo ) ;
if ( sandboxed ) {
Field fieldInfo = type - > GetFieldInfo ( fid ) ;
if ( fieldInfo . Attributes & FANoUserView )
BOOST_THROW_EXCEPTION ( ScriptError ( " Accessing the field ' " + field + " ' for type ' " + type - > GetName ( ) + " ' is not allowed in sandbox mode. " , debugInfo ) ) ;
}
return GetField ( fid ) ;
}
2018-08-07 13:55:41 +02:00
void Object : : SetFieldByName ( const String & field , const Value & value , bool overrideFrozen , const DebugInfo & debugInfo )
2016-04-18 11:29:43 +02:00
{
Type : : Ptr type = GetReflectionType ( ) ;
if ( ! type )
BOOST_THROW_EXCEPTION ( ScriptError ( " Cannot set field on object. " , debugInfo ) ) ;
int fid = type - > GetFieldId ( field ) ;
if ( fid = = - 1 )
BOOST_THROW_EXCEPTION ( ScriptError ( " Attribute ' " + field + " ' does not exist. " , debugInfo ) ) ;
try {
SetField ( fid , value ) ;
} catch ( const boost : : bad_lexical_cast & ) {
Field fieldInfo = type - > GetFieldInfo ( fid ) ;
Type : : Ptr ftype = Type : : GetByName ( fieldInfo . TypeName ) ;
BOOST_THROW_EXCEPTION ( ScriptError ( " Attribute ' " + field + " ' cannot be set to value of type ' " + value . GetTypeName ( ) + " ', expected ' " + ftype - > GetName ( ) + " ' " , debugInfo ) ) ;
} catch ( const std : : bad_cast & ) {
Field fieldInfo = type - > GetFieldInfo ( fid ) ;
Type : : Ptr ftype = Type : : GetByName ( fieldInfo . TypeName ) ;
BOOST_THROW_EXCEPTION ( ScriptError ( " Attribute ' " + field + " ' cannot be set to value of type ' " + value . GetTypeName ( ) + " ', expected ' " + ftype - > GetName ( ) + " ' " , debugInfo ) ) ;
}
}
2015-08-25 13:53:43 +02:00
void Object : : Validate ( int types , const ValidationUtils & utils )
{
/* Nothing to do here. */
}
2018-01-11 07:08:09 +01:00
void Object : : ValidateField ( int id , const Lazy < Value > & lvalue , const ValidationUtils & utils )
2015-08-13 08:52:00 +02:00
{
/* Nothing to do here. */
}
2015-08-04 14:47:44 +02:00
void Object : : NotifyField ( int id , const Value & cookie )
{
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Invalid field ID. " ) ) ;
2015-08-13 08:52:00 +02:00
}
2015-08-17 13:59:49 +02:00
2015-09-22 09:42:30 +02:00
Object : : Ptr Object : : NavigateField ( int id ) const
{
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Invalid field ID. " ) ) ;
}
2018-01-04 04:25:35 +01:00
Object : : Ptr Object : : Clone ( ) const
2015-08-17 13:59:49 +02:00
{
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Object cannot be cloned. " ) ) ;
}
2015-08-18 07:46:04 +02:00
2018-01-04 04:25:35 +01:00
Type : : Ptr Object : : GetReflectionType ( ) const
2015-08-18 07:46:04 +02:00
{
return Object : : TypeInstance ;
}
2016-04-18 11:29:43 +02:00
Value icinga : : GetPrototypeField ( const Value & context , const String & field , bool not_found_error , const DebugInfo & debugInfo )
{
Type : : Ptr ctype = context . GetReflectionType ( ) ;
Type : : Ptr type = ctype ;
do {
Object : : Ptr object = type - > GetPrototype ( ) ;
if ( object & & object - > HasOwnField ( field ) )
return object - > GetFieldByName ( field , false , debugInfo ) ;
type = type - > GetBaseType ( ) ;
} while ( type ) ;
if ( not_found_error )
BOOST_THROW_EXCEPTION ( ScriptError ( " Invalid field access (for value of type ' " + ctype - > GetName ( ) + " '): ' " + field + " ' " , debugInfo ) ) ;
else
return Empty ;
}
2016-01-19 16:43:46 +01:00
# ifdef I2_LEAK_DEBUG
2016-01-19 15:24:17 +01:00
void icinga : : TypeAddObject ( Object * object )
{
2021-02-02 10:16:04 +01:00
std : : unique_lock < std : : mutex > lock ( l_ObjectCountLock ) ;
2016-01-19 15:24:17 +01:00
String typeName = Utility : : GetTypeName ( typeid ( * object ) ) ;
l_ObjectCounts [ typeName ] + + ;
}
void icinga : : TypeRemoveObject ( Object * object )
{
2021-02-02 10:16:04 +01:00
std : : unique_lock < std : : mutex > lock ( l_ObjectCountLock ) ;
2016-01-19 15:24:17 +01:00
String typeName = Utility : : GetTypeName ( typeid ( * object ) ) ;
l_ObjectCounts [ typeName ] - - ;
}
2018-01-04 04:25:35 +01:00
static void TypeInfoTimerHandler ( )
2016-01-19 15:24:17 +01:00
{
2021-02-02 10:16:04 +01:00
std : : unique_lock < std : : mutex > lock ( l_ObjectCountLock ) ;
2016-01-19 15:24:17 +01:00
typedef std : : map < String , int > : : value_type kv_pair ;
2016-08-25 06:19:44 +02:00
for ( kv_pair & kv : l_ObjectCounts ) {
2016-01-19 15:24:17 +01:00
if ( kv . second = = 0 )
continue ;
Log ( LogInformation , " TypeInfo " )
2017-12-19 15:50:05 +01:00
< < kv . second < < " " < < kv . first < < " objects " ;
2016-01-19 15:24:17 +01:00
kv . second = 0 ;
}
}
2016-08-27 09:35:08 +02:00
INITIALIZE_ONCE ( [ ] ( ) {
2016-01-19 15:24:17 +01:00
l_ObjectCountTimer = new Timer ( ) ;
l_ObjectCountTimer - > SetInterval ( 10 ) ;
2021-01-18 14:29:05 +01:00
l_ObjectCountTimer - > OnTimerExpired . connect ( [ ] ( const Timer * const & ) { TypeInfoTimerHandler ( ) ; } ) ;
2016-01-19 15:24:17 +01:00
l_ObjectCountTimer - > Start ( ) ;
2016-08-27 09:35:08 +02:00
} ) ;
2016-01-19 16:43:46 +01:00
# endif /* I2_LEAK_DEBUG */
2016-01-19 15:24:17 +01:00
2018-01-04 10:36:35 +01:00
void icinga : : intrusive_ptr_add_ref ( Object * object )
{
# ifdef I2_LEAK_DEBUG
2019-03-11 12:05:01 +01:00
if ( object - > m_References . fetch_add ( 1 ) = = 0u )
2018-01-04 10:36:35 +01:00
TypeAddObject ( object ) ;
2019-03-11 12:05:01 +01:00
# else /* I2_LEAK_DEBUG */
object - > m_References . fetch_add ( 1 ) ;
2018-01-04 10:36:35 +01:00
# endif /* I2_LEAK_DEBUG */
}
void icinga : : intrusive_ptr_release ( Object * object )
{
2019-03-11 12:05:01 +01:00
auto previous ( object - > m_References . fetch_sub ( 1 ) ) ;
2018-01-04 10:36:35 +01:00
2019-03-11 12:05:01 +01:00
if ( previous = = 1u ) {
2018-01-04 10:36:35 +01:00
# ifdef I2_LEAK_DEBUG
TypeRemoveObject ( object ) ;
# endif /* I2_LEAK_DEBUG */
delete object ;
}
}
2018-01-04 18:24:45 +01:00
void icinga : : DefaultObjectFactoryCheckArgs ( const std : : vector < Value > & args )
{
if ( ! args . empty ( ) )
BOOST_THROW_EXCEPTION ( std : : invalid_argument ( " Constructor does not take any arguments. " ) ) ;
}
2018-01-30 11:26:07 +01:00
void icinga : : RequireNotNullInternal ( const intrusive_ptr < Object > & object , const char * description )
{
if ( ! object )
BOOST_THROW_EXCEPTION ( std : : invalid_argument ( " Pointer must not be null: " + String ( description ) ) ) ;
2018-02-21 13:42:58 +01:00
}