/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org 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 Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ #include #include /** === begin UNO includes === **/ #include #include #include #include /** === end UNO includes === **/ #include //........................................................................ namespace comphelper { //........................................................................ /** === begin UNO using === **/ using ::com::sun::star::uno::Reference; using ::com::sun::star::uno::XComponentContext; using ::com::sun::star::logging::XLoggerPool; using ::com::sun::star::logging::LoggerPool; using ::com::sun::star::logging::XLogger; using ::com::sun::star::uno::UNO_QUERY_THROW; using ::com::sun::star::uno::Exception; using ::com::sun::star::logging::XLogHandler; using ::com::sun::star::resource::XResourceBundle; using ::com::sun::star::resource::XResourceBundleLoader; /** === end UNO using === **/ namespace LogLevel = ::com::sun::star::logging::LogLevel; //==================================================================== //= EventLogger_Impl - declaration //==================================================================== class EventLogger_Impl { private: ::comphelper::ComponentContext m_aContext; ::rtl::OUString m_sLoggerName; Reference< XLogger > m_xLogger; public: EventLogger_Impl( const Reference< XComponentContext >& _rxContext, const ::rtl::OUString& _rLoggerName ) :m_aContext( _rxContext ) ,m_sLoggerName( _rLoggerName ) { impl_createLogger_nothrow(); } inline bool isValid() const { return m_xLogger.is(); } inline const ::rtl::OUString& getName() const { return m_sLoggerName; } inline const Reference< XLogger >& getLogger() const { return m_xLogger; } inline const ::comphelper::ComponentContext& getContext() const { return m_aContext; } private: void impl_createLogger_nothrow(); }; //==================================================================== //= EventLogger_Impl - implementation //==================================================================== //-------------------------------------------------------------------- void EventLogger_Impl::impl_createLogger_nothrow() { try { Reference< XLoggerPool > xPool( LoggerPool::get( m_aContext.getUNOContext() ), UNO_QUERY_THROW ); if ( !m_sLoggerName.isEmpty() ) m_xLogger = xPool->getNamedLogger( m_sLoggerName ); else m_xLogger = xPool->getDefaultLogger(); } catch( const Exception& e ) { (void)e; OSL_FAIL( "EventLogger_Impl::impl_createLogger_nothrow: caught an exception!" ); } } //==================================================================== //= EventLogger //==================================================================== //-------------------------------------------------------------------- EventLogger::EventLogger( const Reference< XComponentContext >& _rxContext, const ::rtl::OUString& _rLoggerName ) :m_pImpl( new EventLogger_Impl( _rxContext, _rLoggerName ) ) { } //-------------------------------------------------------------------- EventLogger::EventLogger( const Reference< XComponentContext >& _rxContext, const sal_Char* _pAsciiLoggerName ) :m_pImpl( new EventLogger_Impl( _rxContext, ::rtl::OUString::createFromAscii( _pAsciiLoggerName ) ) ) { } //-------------------------------------------------------------------- EventLogger::~EventLogger() { } //-------------------------------------------------------------------- bool EventLogger::isLoggable( const sal_Int32 _nLogLevel ) const { if ( !m_pImpl->isValid() ) return false; try { return m_pImpl->getLogger()->isLoggable( _nLogLevel ); } catch( const Exception& e ) { (void)e; OSL_FAIL( "EventLogger::isLoggable: caught an exception!" ); } return false; } //-------------------------------------------------------------------- namespace { void lcl_replaceParameter( ::rtl::OUString& _inout_Message, const ::rtl::OUString& _rPlaceHolder, const ::rtl::OUString& _rReplacement ) { sal_Int32 nPlaceholderPosition = _inout_Message.indexOf( _rPlaceHolder ); OSL_ENSURE( nPlaceholderPosition >= 0, "lcl_replaceParameter: placeholder not found!" ); if ( nPlaceholderPosition < 0 ) return; _inout_Message = _inout_Message.replaceAt( nPlaceholderPosition, _rPlaceHolder.getLength(), _rReplacement ); } } //-------------------------------------------------------------------- bool EventLogger::impl_log( const sal_Int32 _nLogLevel, const sal_Char* _pSourceClass, const sal_Char* _pSourceMethod, const ::rtl::OUString& _rMessage, const OptionalString& _rArgument1, const OptionalString& _rArgument2, const OptionalString& _rArgument3, const OptionalString& _rArgument4, const OptionalString& _rArgument5, const OptionalString& _rArgument6 ) const { // (if ::rtl::OUString had an indexOfAscii, we could save those ugly statics ...) static ::rtl::OUString sPH1( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "$1$" ) ) ); static ::rtl::OUString sPH2( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "$2$" ) ) ); static ::rtl::OUString sPH3( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "$3$" ) ) ); static ::rtl::OUString sPH4( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "$4$" ) ) ); static ::rtl::OUString sPH5( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "$5$" ) ) ); static ::rtl::OUString sPH6( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "$6$" ) ) ); ::rtl::OUString sMessage( _rMessage ); if ( !!_rArgument1 ) lcl_replaceParameter( sMessage, sPH1, *_rArgument1 ); if ( !!_rArgument2 ) lcl_replaceParameter( sMessage, sPH2, *_rArgument2 ); if ( !!_rArgument3 ) lcl_replaceParameter( sMessage, sPH3, *_rArgument3 ); if ( !!_rArgument4 ) lcl_replaceParameter( sMessage, sPH4, *_rArgument4 ); if ( !!_rArgument5 ) lcl_replaceParameter( sMessage, sPH5, *_rArgument5 ); if ( !!_rArgument6 ) lcl_replaceParameter( sMessage, sPH6, *_rArgument6 ); try { Reference< XLogger > xLogger( m_pImpl->getLogger() ); OSL_PRECOND( xLogger.is(), "EventLogger::impl_log: should never be called without a logger!" ); if ( _pSourceClass && _pSourceMethod ) { xLogger->logp( _nLogLevel, ::rtl::OUString::createFromAscii( _pSourceClass ), ::rtl::OUString::createFromAscii( _pSourceMethod ), sMessage ); } else { xLogger->log( _nLogLevel, sMessage ); } } catch( const Exception& e ) { (void)e; OSL_FAIL( "EventLogger::impl_log: caught an exception!" ); } return false; } //==================================================================== //= ResourceBasedEventLogger_Data //==================================================================== struct ResourceBasedEventLogger_Data { /// the base name of the resource bundle ::rtl::OUString sBundleBaseName; /// did we already attempt to load the bundle? bool bBundleLoaded; /// the lazily loaded bundle Reference< XResourceBundle > xBundle; ResourceBasedEventLogger_Data() :sBundleBaseName() ,bBundleLoaded( false ) ,xBundle() { } }; //-------------------------------------------------------------------- bool lcl_loadBundle_nothrow( const ComponentContext& _rContext, ResourceBasedEventLogger_Data& _rLoggerData ) { if ( _rLoggerData.bBundleLoaded ) return _rLoggerData.xBundle.is(); // no matter what happens below, don't attempt creation ever again _rLoggerData.bBundleLoaded = true; try { Reference< XResourceBundleLoader > xLoader( _rContext.getSingleton( "com.sun.star.resource.OfficeResourceLoader" ), UNO_QUERY_THROW ); _rLoggerData.xBundle = Reference< XResourceBundle >( xLoader->loadBundle_Default( _rLoggerData.sBundleBaseName ), UNO_QUERY_THROW ); } catch( const Exception& e ) { (void)e; OSL_FAIL( "lcl_loadBundle_nothrow: caught an exception!" ); } return _rLoggerData.xBundle.is(); } //-------------------------------------------------------------------- ::rtl::OUString lcl_loadString_nothrow( const Reference< XResourceBundle >& _rxBundle, const sal_Int32 _nMessageResID ) { OSL_PRECOND( _rxBundle.is(), "lcl_loadString_nothrow: this will crash!" ); ::rtl::OUString sMessage; try { ::rtl::OUStringBuffer aBuffer; aBuffer.appendAscii( "string:" ); aBuffer.append( _nMessageResID ); OSL_VERIFY( _rxBundle->getDirectElement( aBuffer.makeStringAndClear() ) >>= sMessage ); } catch( const Exception& e ) { (void)e; OSL_FAIL( "lcl_loadString_nothrow: caught an exception!" ); } return sMessage; } //==================================================================== //= ResourceBasedEventLogger //==================================================================== //-------------------------------------------------------------------- ResourceBasedEventLogger::ResourceBasedEventLogger( const Reference< XComponentContext >& _rxContext, const sal_Char* _pResourceBundleBaseName, const sal_Char* _pAsciiLoggerName ) :EventLogger( _rxContext, _pAsciiLoggerName ) ,m_pData( new ResourceBasedEventLogger_Data ) { m_pData->sBundleBaseName = ::rtl::OUString::createFromAscii( _pResourceBundleBaseName ); } //-------------------------------------------------------------------- ::rtl::OUString ResourceBasedEventLogger::impl_loadStringMessage_nothrow( const sal_Int32 _nMessageResID ) const { ::rtl::OUString sMessage; if ( lcl_loadBundle_nothrow( m_pImpl->getContext(), *m_pData ) ) sMessage = lcl_loadString_nothrow( m_pData->xBundle, _nMessageResID ); if ( sMessage.isEmpty() ) { ::rtl::OUStringBuffer aBuffer; aBuffer.appendAscii( "sBundleBaseName ); aBuffer.appendAscii( ":" ); aBuffer.append( _nMessageResID ); aBuffer.appendAscii( "'>" ); sMessage = aBuffer.makeStringAndClear(); } return sMessage; } //........................................................................ } // namespace comphelper //........................................................................ /* vim:set shiftwidth=4 softtabstop=4 expandtab: */