2012-05-10 12:06:41 +02:00
/******************************************************************************
* Icinga 2 *
2018-10-18 09:27:04 +02:00
* Copyright ( C ) 2012 - 2018 Icinga Development Team ( https : //icinga.com/) *
2012-05-10 12:06:41 +02:00
* *
* 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 *
2012-05-11 13:33:57 +02:00
* Inc . , 51 Franklin St , Fifth Floor , Boston , MA 02110 - 1301 , USA . *
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>
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
2016-01-19 15:24:17 +01:00
static boost : : mutex l_ObjectCountLock ;
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
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
{
delete reinterpret_cast < boost : : recursive_mutex * > ( m_Mutex ) ;
}
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
{
2014-11-12 12:32:14 +01:00
# ifdef _WIN32
DWORD tid = InterlockedExchangeAdd ( & m_LockOwner , 0 ) ;
return ( tid = = GetCurrentThreadId ( ) ) ;
# else /* _WIN32 */
2018-01-04 10:46:35 +01:00
pthread_t tid = __sync_fetch_and_add ( & m_LockOwner , 0 ) ;
2014-11-12 12:32:14 +01:00
return ( tid = = pthread_self ( ) ) ;
# endif /* _WIN32 */
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 )
{
boost : : mutex : : scoped_lock lock ( l_ObjectCountLock ) ;
String typeName = Utility : : GetTypeName ( typeid ( * object ) ) ;
l_ObjectCounts [ typeName ] + + ;
}
void icinga : : TypeRemoveObject ( Object * object )
{
boost : : mutex : : scoped_lock lock ( l_ObjectCountLock ) ;
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
{
boost : : mutex : : scoped_lock lock ( l_ObjectCountLock ) ;
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 ) ;
2017-11-21 11:52:55 +01:00
l_ObjectCountTimer - > OnTimerExpired . connect ( std : : bind ( 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
if ( object - > m_References = = 0 )
TypeAddObject ( object ) ;
# endif /* I2_LEAK_DEBUG */
# ifdef _WIN32
InterlockedIncrement ( & object - > m_References ) ;
# else /* _WIN32 */
__sync_add_and_fetch ( & object - > m_References , 1 ) ;
# endif /* _WIN32 */
}
void icinga : : intrusive_ptr_release ( Object * object )
{
uintptr_t refs ;
# ifdef _WIN32
refs = InterlockedDecrement ( & object - > m_References ) ;
# else /* _WIN32 */
refs = __sync_sub_and_fetch ( & object - > m_References , 1 ) ;
# endif /* _WIN32 */
if ( unlikely ( refs = = 0 ) ) {
# 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
}