2019-02-25 08:48:22 -05:00
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
2012-05-10 06:06:41 -04:00
2014-05-25 10:23:35 -04:00
# include "base/object.hpp"
# include "base/value.hpp"
2014-12-12 09:19:23 -05:00
# include "base/dictionary.hpp"
2014-11-03 01:07:54 -05:00
# include "base/primitivetype.hpp"
2014-12-08 03:12:40 -05:00
# include "base/utility.hpp"
2016-01-19 09:24:17 -05:00
# include "base/timer.hpp"
# include "base/logger.hpp"
2016-04-18 05:29:43 -04:00
# include "base/exception.hpp"
2016-04-18 11:21:45 -04:00
# include <boost/lexical_cast.hpp>
2018-01-03 00:01:02 -05:00
# include <boost/thread/recursive_mutex.hpp>
2019-04-17 12:03:40 -04:00
# include <thread>
2012-03-28 07:24:49 -04:00
using namespace icinga ;
2012-05-14 13:14:23 -04:00
2015-11-05 04:29:02 -05:00
DEFINE_TYPE_INSTANCE ( Object ) ;
2014-11-02 18:44:04 -05:00
2016-01-19 10:43:46 -05:00
# ifdef I2_LEAK_DEBUG
2021-02-02 04:16:04 -05:00
static std : : mutex l_ObjectCountLock ;
2016-01-19 09:24:17 -05:00
static std : : map < String , int > l_ObjectCounts ;
static Timer : : Ptr l_ObjectCountTimer ;
2016-01-19 10:43:46 -05:00
# endif /* I2_LEAK_DEBUG */
2016-01-19 09:24:17 -05:00
2019-03-11 07:05:01 -04:00
/**
* Constructor for the Object class .
*/
Object : : Object ( )
{
m_References . store ( 0 ) ;
2019-04-24 06:10:57 -04:00
# ifdef I2_DEBUG
2019-04-17 12:03:40 -04:00
m_LockOwner . store ( decltype ( m_LockOwner . load ( ) ) ( ) ) ;
2019-04-24 06:10:57 -04:00
# endif /* I2_DEBUG */
2019-03-11 07:05:01 -04:00
}
2012-05-14 13:14:23 -04:00
/**
* Destructor for the Object class .
*/
2018-01-03 22:25:35 -05:00
Object : : ~ Object ( )
2016-01-19 09:25:44 -05:00
{
}
2012-06-14 09:16:41 -04:00
2014-12-08 03:12:40 -05:00
/**
* Returns a string representation for the object .
*/
2018-01-03 22:25:35 -05:00
String Object : : ToString ( ) const
2014-12-08 03:12:40 -05:00
{
2015-11-10 01:59:10 -05:00
return " Object of type ' " + GetReflectionType ( ) - > GetName ( ) + " ' " ;
2014-12-08 03:12:40 -05:00
}
2014-12-19 06:19:28 -05:00
# ifdef I2_DEBUG
2012-09-14 08:41:17 -04:00
/**
2013-03-04 09:52:42 -05:00
* Checks if the calling thread owns the lock on this object .
2012-09-14 08:41:17 -04:00
*
2013-03-01 06:07:52 -05:00
* @ returns True if the calling thread owns the lock , false otherwise .
2012-09-14 08:41:17 -04:00
*/
2018-01-03 22:25:35 -05:00
bool Object : : OwnsLock ( ) const
2012-08-03 07:19:55 -04:00
{
2019-04-17 12:03:40 -04:00
return m_LockOwner . load ( ) = = std : : this_thread : : get_id ( ) ;
2012-08-03 07:19:55 -04:00
}
2014-12-19 06:19:28 -05:00
# endif /* I2_DEBUG */
2013-06-13 05:33:00 -04:00
2015-08-04 08:47:44 -04:00
void Object : : SetField ( int id , const Value & , bool , const Value & )
2013-11-04 17:14:34 -05:00
{
2015-07-30 02:23:43 -04:00
if ( id = = 0 )
2015-11-05 04:29:02 -05:00
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Type field cannot be set. " ) ) ;
2015-07-30 02:23:43 -04:00
else
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Invalid field ID. " ) ) ;
2013-11-04 17:14:34 -05:00
}
2015-07-30 02:23:43 -04:00
Value Object : : GetField ( int id ) const
2013-11-04 17:14:34 -05:00
{
2015-07-30 02:23:43 -04:00
if ( id = = 0 )
2015-11-05 04:29:02 -05:00
return GetReflectionType ( ) - > GetName ( ) ;
2015-07-30 02:23:43 -04:00
else
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Invalid field ID. " ) ) ;
2013-11-04 17:14:34 -05:00
}
2014-05-11 00:30:50 -04:00
2016-04-18 05:29:43 -04:00
bool Object : : HasOwnField ( const String & field ) const
{
Type : : Ptr type = GetReflectionType ( ) ;
if ( ! type )
return false ;
return type - > GetFieldId ( field ) ! = - 1 ;
}
2016-09-01 01:41:41 -04: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 05:29:43 -04: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 07:55:41 -04:00
void Object : : SetFieldByName ( const String & field , const Value & value , bool overrideFrozen , const DebugInfo & debugInfo )
2016-04-18 05:29:43 -04: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 07:53:43 -04:00
void Object : : Validate ( int types , const ValidationUtils & utils )
{
/* Nothing to do here. */
}
2018-01-11 01:08:09 -05:00
void Object : : ValidateField ( int id , const Lazy < Value > & lvalue , const ValidationUtils & utils )
2015-08-13 02:52:00 -04:00
{
/* Nothing to do here. */
}
2015-08-04 08:47:44 -04:00
void Object : : NotifyField ( int id , const Value & cookie )
{
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Invalid field ID. " ) ) ;
2015-08-13 02:52:00 -04:00
}
2015-08-17 07:59:49 -04:00
2015-09-22 03:42:30 -04:00
Object : : Ptr Object : : NavigateField ( int id ) const
{
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Invalid field ID. " ) ) ;
}
2018-01-03 22:25:35 -05:00
Object : : Ptr Object : : Clone ( ) const
2015-08-17 07:59:49 -04:00
{
BOOST_THROW_EXCEPTION ( std : : runtime_error ( " Object cannot be cloned. " ) ) ;
}
2015-08-18 01:46:04 -04:00
2018-01-03 22:25:35 -05:00
Type : : Ptr Object : : GetReflectionType ( ) const
2015-08-18 01:46:04 -04:00
{
return Object : : TypeInstance ;
}
2016-04-18 05:29:43 -04: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 10:43:46 -05:00
# ifdef I2_LEAK_DEBUG
2016-01-19 09:24:17 -05:00
void icinga : : TypeAddObject ( Object * object )
{
2021-02-02 04:16:04 -05:00
std : : unique_lock < std : : mutex > lock ( l_ObjectCountLock ) ;
2016-01-19 09:24:17 -05:00
String typeName = Utility : : GetTypeName ( typeid ( * object ) ) ;
l_ObjectCounts [ typeName ] + + ;
}
void icinga : : TypeRemoveObject ( Object * object )
{
2021-02-02 04:16:04 -05:00
std : : unique_lock < std : : mutex > lock ( l_ObjectCountLock ) ;
2016-01-19 09:24:17 -05:00
String typeName = Utility : : GetTypeName ( typeid ( * object ) ) ;
l_ObjectCounts [ typeName ] - - ;
}
2018-01-03 22:25:35 -05:00
static void TypeInfoTimerHandler ( )
2016-01-19 09:24:17 -05:00
{
2021-02-02 04:16:04 -05:00
std : : unique_lock < std : : mutex > lock ( l_ObjectCountLock ) ;
2016-01-19 09:24:17 -05:00
typedef std : : map < String , int > : : value_type kv_pair ;
2016-08-25 00:19:44 -04:00
for ( kv_pair & kv : l_ObjectCounts ) {
2016-01-19 09:24:17 -05:00
if ( kv . second = = 0 )
continue ;
Log ( LogInformation , " TypeInfo " )
2017-12-19 09:50:05 -05:00
< < kv . second < < " " < < kv . first < < " objects " ;
2016-01-19 09:24:17 -05:00
kv . second = 0 ;
}
}
2016-08-27 03:35:08 -04:00
INITIALIZE_ONCE ( [ ] ( ) {
2016-01-19 09:24:17 -05:00
l_ObjectCountTimer = new Timer ( ) ;
l_ObjectCountTimer - > SetInterval ( 10 ) ;
2021-01-18 08:29:05 -05:00
l_ObjectCountTimer - > OnTimerExpired . connect ( [ ] ( const Timer * const & ) { TypeInfoTimerHandler ( ) ; } ) ;
2016-01-19 09:24:17 -05:00
l_ObjectCountTimer - > Start ( ) ;
2016-08-27 03:35:08 -04:00
} ) ;
2016-01-19 10:43:46 -05:00
# endif /* I2_LEAK_DEBUG */
2016-01-19 09:24:17 -05:00
2018-01-04 04:36:35 -05:00
void icinga : : intrusive_ptr_add_ref ( Object * object )
{
# ifdef I2_LEAK_DEBUG
2019-03-11 07:05:01 -04:00
if ( object - > m_References . fetch_add ( 1 ) = = 0u )
2018-01-04 04:36:35 -05:00
TypeAddObject ( object ) ;
2019-03-11 07:05:01 -04:00
# else /* I2_LEAK_DEBUG */
object - > m_References . fetch_add ( 1 ) ;
2018-01-04 04:36:35 -05:00
# endif /* I2_LEAK_DEBUG */
}
void icinga : : intrusive_ptr_release ( Object * object )
{
2019-03-11 07:05:01 -04:00
auto previous ( object - > m_References . fetch_sub ( 1 ) ) ;
2018-01-04 04:36:35 -05:00
2019-03-11 07:05:01 -04:00
if ( previous = = 1u ) {
2018-01-04 04:36:35 -05:00
# ifdef I2_LEAK_DEBUG
TypeRemoveObject ( object ) ;
# endif /* I2_LEAK_DEBUG */
delete object ;
}
}
2018-01-04 12:24:45 -05: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 05:26:07 -05: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 07:42:58 -05:00
}