Files
libreoffice/dbaccess/source/core/dataaccess/datasource.cxx

1160 lines
46 KiB
C++
Raw Normal View History

2000-09-18 23:16:46 +00:00
/*************************************************************************
*
* $RCSfile: datasource.cxx,v $
*
2001-05-03 13:35:27 +00:00
* $Revision: 1.28 $
2000-09-18 23:16:46 +00:00
*
2001-05-03 13:35:27 +00:00
* last change: $Author: vg $ $Date: 2001-05-03 14:35:27 $
2000-09-18 23:16:46 +00:00
*
* The Contents of this file are made available subject to the terms of
* either of the following licenses
*
* - GNU Lesser General Public License Version 2.1
* - Sun Industry Standards Source License Version 1.1
*
* Sun Microsystems Inc., October, 2000
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2000 by Sun Microsystems, Inc.
* 901 San Antonio Road, Palo Alto, CA 94303, USA
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1, as published by the Free Software Foundation.
*
* This library 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 for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
*
*
* Sun Industry Standards Source License Version 1.1
* =================================================
* The contents of this file are subject to the Sun Industry Standards
* Source License Version 1.1 (the "License"); You may not use this file
* except in compliance with the License. You may obtain a copy of the
* License at http://www.openoffice.org/license.html.
*
* Software provided under this License is provided on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
* WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS,
* MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING.
* See the License for the specific provisions governing your rights and
* obligations concerning the Software.
*
* The Initial Developer of the Original Code is: Sun Microsystems, Inc.
*
* Copyright: 2000 by Sun Microsystems, Inc.
*
* All Rights Reserved.
*
* Contributor(s): _______________________________________
*
*
************************************************************************/
#ifndef _DBA_COREDATAACCESS_DATASOURCE_HXX_
#include "datasource.hxx"
#endif
#ifndef _DBA_CORE_USERINFORMATION_HXX_
#include "userinformation.hxx"
#endif
#ifndef _DBA_CORE_REGISTRYHELPER_HXX_
#include "registryhelper.hxx"
#endif
#ifndef _TOOLS_DEBUG_HXX
#include <tools/debug.hxx>
#endif
#ifndef _CPPUHELPER_TYPEPROVIDER_HXX_
#include <cppuhelper/typeprovider.hxx>
#endif
2000-10-25 06:42:28 +00:00
#ifndef DBACCESS_SHARED_DBASTRINGS_HRC
#include "dbastrings.hrc"
2000-09-18 23:16:46 +00:00
#endif
2000-10-11 10:21:40 +00:00
#ifndef _COMPHELPER_SEQUENCE_HXX_
#include <comphelper/sequence.hxx>
2000-09-18 23:16:46 +00:00
#endif
2000-10-11 10:21:40 +00:00
#ifndef _COMPHELPER_PROPERTY_HXX_
#include <comphelper/property.hxx>
2000-09-18 23:16:46 +00:00
#endif
#ifndef _COMPHELPER_EXTRACT_HXX_
#include <comphelper/extract.hxx>
2000-09-18 23:16:46 +00:00
#endif
#ifndef _COM_SUN_STAR_LANG_DISPOSEDEXCEPTION_HPP_
#include <com/sun/star/lang/DisposedException.hpp>
#endif
#ifndef _COM_SUN_STAR_IO_XOBJECTOUTPUTSTREAM_HPP_
#include <com/sun/star/io/XObjectOutputStream.hpp>
#endif
#ifndef _COM_SUN_STAR_IO_XOBJECTINPUTSTREAM_HPP_
#include <com/sun/star/io/XObjectInputStream.hpp>
#endif
#ifndef _COM_SUN_STAR_IO_XPERSISTOBJECT_HPP_
#include <com/sun/star/io/XPersistObject.hpp>
#endif
#ifndef _COM_SUN_STAR_IO_XACTIVEDATASOURCE_HPP_
#include <com/sun/star/io/XActiveDataSource.hpp>
#endif
#ifndef _COM_SUN_STAR_IO_XACTIVEDATASINK_HPP_
#include <com/sun/star/io/XActiveDataSink.hpp>
#endif
#ifndef _COM_SUN_STAR_SDBC_XDRIVERMANAGER_HPP_
#include <com/sun/star/sdbc/XDriverManager.hpp>
#endif
2000-12-12 11:20:31 +00:00
#ifndef _COM_SUN_STAR_SDBCX_XTABLESSUPPLIER_HPP_
#include <com/sun/star/sdbcx/XTablesSupplier.hpp>
#endif
2000-11-08 15:05:02 +00:00
#ifndef _COM_SUN_STAR_UCB_XINTERACTIONSUPPLYAUTHENTICATION_HPP_
#include <com/sun/star/ucb/XInteractionSupplyAuthentication.hpp>
#endif
#ifndef _COM_SUN_STAR_UCB_AUTHENTICATIONREQUEST_HPP_
#include <com/sun/star/ucb/AuthenticationRequest.hpp>
#endif
#ifndef _TYPELIB_TYPEDESCRIPTION_HXX_
#include <typelib/typedescription.hxx>
#endif
#ifndef _DBHELPER_DBEXCEPTION_HXX_
#include <connectivity/dbexception.hxx>
#endif
2000-11-08 15:05:02 +00:00
#ifndef _COMPHELPER_INTERACTION_HXX_
#include <comphelper/interaction.hxx>
#endif
2000-09-18 23:16:46 +00:00
#ifndef _DBA_CORE_CONNECTION_HXX_
#include "connection.hxx"
2000-09-18 23:16:46 +00:00
#endif
#ifndef _COMPHELPER_SEQUENCE_HXX_
#include <comphelper/sequence.hxx>
#endif
2000-09-18 23:16:46 +00:00
using namespace ::com::sun::star::sdbc;
2000-12-12 11:20:31 +00:00
using namespace ::com::sun::star::sdbcx;
2000-09-18 23:16:46 +00:00
using namespace ::com::sun::star::sdb;
using namespace ::com::sun::star::beans;
using namespace ::com::sun::star::uno;
using namespace ::com::sun::star::lang;
using namespace ::com::sun::star::registry;
using namespace ::com::sun::star::container;
using namespace ::com::sun::star::util;
using namespace ::com::sun::star::io;
2000-11-08 15:05:02 +00:00
using namespace ::com::sun::star::task;
using namespace ::com::sun::star::ucb;
2000-09-18 23:16:46 +00:00
using namespace ::cppu;
using namespace ::osl;
using namespace ::vos;
using namespace ::dbtools;
2000-11-08 15:05:02 +00:00
using namespace ::comphelper;
2000-09-18 23:16:46 +00:00
// persistent tokens
#define PT_SVFORMATTER 0x0001
//........................................................................
namespace dbaccess
{
//........................................................................
2000-09-18 23:16:46 +00:00
2000-11-08 15:05:02 +00:00
//============================================================
//= OAuthenticationContinuation
//============================================================
class OAuthenticationContinuation : public OInteraction< XInteractionSupplyAuthentication >
{
sal_Bool m_bDatasourceReadonly : 1; // if sal_True, the data source using this continuation
// is readonly, which means that no user can be set and
// the password can't be remembered
sal_Bool m_bRemberPassword : 1; // remember the password for this session ?
::rtl::OUString m_sUser; // the user
::rtl::OUString m_sPassword; // the user's password
public:
virtual sal_Bool SAL_CALL canSetRealm( ) throw(RuntimeException);
virtual void SAL_CALL setRealm( const ::rtl::OUString& Realm ) throw(RuntimeException);
virtual sal_Bool SAL_CALL canSetUserName( ) throw(RuntimeException);
virtual void SAL_CALL setUserName( const ::rtl::OUString& UserName ) throw(RuntimeException);
virtual sal_Bool SAL_CALL canSetPassword( ) throw(RuntimeException);
virtual void SAL_CALL setPassword( const ::rtl::OUString& Password ) throw(RuntimeException);
virtual Sequence< RememberAuthentication > SAL_CALL getRememberPasswordModes( RememberAuthentication& Default ) throw(RuntimeException);
virtual void SAL_CALL setRememberPassword( RememberAuthentication Remember ) throw(RuntimeException);
virtual sal_Bool SAL_CALL canSetAccount( ) throw(RuntimeException);
virtual void SAL_CALL setAccount( const ::rtl::OUString& Account ) throw(RuntimeException);
virtual Sequence< RememberAuthentication > SAL_CALL getRememberAccountModes( RememberAuthentication& Default ) throw(RuntimeException);
virtual void SAL_CALL setRememberAccount( RememberAuthentication Remember ) throw(RuntimeException);
::rtl::OUString getUser() const { return m_sUser; }
::rtl::OUString getPassword() const { return m_sPassword; }
sal_Bool getRememberPassword() const { return m_bRemberPassword; }
};
//--------------------------------------------------------------------------
sal_Bool SAL_CALL OAuthenticationContinuation::canSetRealm( ) throw(RuntimeException)
{
return sal_False;
}
//--------------------------------------------------------------------------
void SAL_CALL OAuthenticationContinuation::setRealm( const ::rtl::OUString& Realm ) throw(RuntimeException)
{
DBG_ERROR("OAuthenticationContinuation::setRealm: not supported!");
}
//--------------------------------------------------------------------------
sal_Bool SAL_CALL OAuthenticationContinuation::canSetUserName( ) throw(RuntimeException)
{
return !m_bDatasourceReadonly;
}
//--------------------------------------------------------------------------
void SAL_CALL OAuthenticationContinuation::setUserName( const ::rtl::OUString& _rUser ) throw(RuntimeException)
{
m_sUser = _rUser;
}
//--------------------------------------------------------------------------
sal_Bool SAL_CALL OAuthenticationContinuation::canSetPassword( ) throw(RuntimeException)
{
return sal_True;
}
//--------------------------------------------------------------------------
void SAL_CALL OAuthenticationContinuation::setPassword( const ::rtl::OUString& _rPassword ) throw(RuntimeException)
{
m_sPassword = _rPassword;
}
//--------------------------------------------------------------------------
Sequence< RememberAuthentication > SAL_CALL OAuthenticationContinuation::getRememberPasswordModes( RememberAuthentication& _reDefault ) throw(RuntimeException)
2000-11-08 15:05:02 +00:00
{
Sequence< RememberAuthentication > aReturn(1);
_reDefault = aReturn[0] = (m_bDatasourceReadonly ? RememberAuthentication_NO : RememberAuthentication_SESSION);
return aReturn;
2000-11-08 15:05:02 +00:00
}
//--------------------------------------------------------------------------
void SAL_CALL OAuthenticationContinuation::setRememberPassword( RememberAuthentication _eRemember ) throw(RuntimeException)
{
m_bRemberPassword = (RememberAuthentication_NO != _eRemember);
}
//--------------------------------------------------------------------------
sal_Bool SAL_CALL OAuthenticationContinuation::canSetAccount( ) throw(RuntimeException)
{
return sal_False;
}
//--------------------------------------------------------------------------
void SAL_CALL OAuthenticationContinuation::setAccount( const ::rtl::OUString& ) throw(RuntimeException)
{
DBG_ERROR("OAuthenticationContinuation::setAccount: not supported!");
}
//--------------------------------------------------------------------------
Sequence< RememberAuthentication > SAL_CALL OAuthenticationContinuation::getRememberAccountModes( RememberAuthentication& _reDefault ) throw(RuntimeException)
2000-11-08 15:05:02 +00:00
{
Sequence < RememberAuthentication > aReturn(1);
aReturn[0] = RememberAuthentication_NO;
_reDefault = RememberAuthentication_NO;
2000-11-08 15:05:02 +00:00
return aReturn;
}
//--------------------------------------------------------------------------
void SAL_CALL OAuthenticationContinuation::setRememberAccount( RememberAuthentication Remember ) throw(RuntimeException)
{
DBG_ERROR("OAuthenticationContinuation::setRememberAccount: not supported!");
}
//============================================================
//= ODatabaseContext
//============================================================
2000-10-24 14:00:08 +00:00
DBG_NAME(ODatabaseSource)
2000-09-18 23:16:46 +00:00
//--------------------------------------------------------------------------
extern "C" void SAL_CALL createRegistryInfo_ODatabaseSource()
{
static OMultiInstanceAutoRegistration< ODatabaseSource > aAutoRegistration;
}
//--------------------------------------------------------------------------
Reference< XInterface > ODatabaseSource_CreateInstance(const Reference< XMultiServiceFactory >& _rxFactory)
{
return *(new ODatabaseSource(_rxFactory));
}
//--------------------------------------------------------------------------
ODatabaseSource::ODatabaseSource(const Reference< XMultiServiceFactory >& _rxFactory)
:OSubComponent(m_aMutex, Reference< XInterface >())
,OConfigurationFlushable(m_aMutex)
,OPropertySetHelper(OComponentHelper::rBHelper)
,m_nLoginTimeout(0)
,m_xServiceFactory(_rxFactory)
,m_bReadOnly(sal_False) // we're created as service and have to allow the setting of properties
,m_bPasswordRequired(sal_False)
,m_bSuppressVersionColumns(sal_True)
,m_aBookmarks(*this, m_aMutex)
2000-09-18 23:16:46 +00:00
,m_aCommandDefinitions(*this, m_aMutex)
{
2000-10-11 10:21:40 +00:00
// some kind of default
m_sConnectURL = ::rtl::OUString::createFromAscii("jdbc:");
m_aTableFilter.realloc(1);
m_aTableFilter[0] = ::rtl::OUString::createFromAscii("%");
2000-09-18 23:16:46 +00:00
}
//--------------------------------------------------------------------------
ODatabaseSource::ODatabaseSource(
OWeakObject& _rParent,
const OConfigurationNode& _rConfigRoot,
2000-09-18 23:16:46 +00:00
const ::rtl::OUString& _rRegistrationName,
const Reference< XMultiServiceFactory >& _rxFactory)
:OSubComponent(m_aMutex, _rParent)
,OConfigurationFlushable(m_aMutex)
,OPropertySetHelper(OComponentHelper::rBHelper)
,m_nLoginTimeout(0)
,m_sName(_rRegistrationName)
,m_xServiceFactory(_rxFactory)
2000-10-09 11:34:52 +00:00
,m_bReadOnly(sal_True) // assume readonly for the moment, adjusted below
2000-09-18 23:16:46 +00:00
,m_bPasswordRequired(sal_False)
,m_bSuppressVersionColumns(sal_True)
,m_aBookmarks(*this, m_aMutex)
2000-09-18 23:16:46 +00:00
,m_aCommandDefinitions(*this, m_aMutex)
{
m_aConfigurationNode = _rConfigRoot.cloneAsRoot();
2000-09-18 23:16:46 +00:00
DBG_CTOR(ODatabaseSource,NULL);
DBG_ASSERT(m_aConfigurationNode.isValid(), "ODatabaseSource::ODatabaseSource : use ctor 1 if you can't supply a configuration location at the moment !");
if (m_aConfigurationNode.isValid())
2000-09-18 23:16:46 +00:00
initializeFromConfiguration();
2000-10-09 11:34:52 +00:00
// adjust our readonly flag
m_bReadOnly = !m_aConfigurationNode.isValid() || m_aConfigurationNode.isReadonly();
2000-09-18 23:16:46 +00:00
}
//--------------------------------------------------------------------------
ODatabaseSource::~ODatabaseSource()
{
DBG_DTOR(ODatabaseSource,NULL);
}
// com::sun::star::lang::XTypeProvider
//--------------------------------------------------------------------------
Sequence< Type > ODatabaseSource::getTypes() throw (RuntimeException)
{
2001-01-23 07:32:02 +00:00
OTypeCollection aPropertyHelperTypes( ::getCppuType( (const Reference< XFastPropertySet > *)0 ),
::getCppuType( (const Reference< XPropertySet > *)0 ),
::getCppuType( (const Reference< XMultiPropertySet > *)0 ));
return ::comphelper::concatSequences(
::comphelper::concatSequences(
OSubComponent::getTypes(),
OConfigurationFlushable::getTypes(),
aPropertyHelperTypes.getTypes()
),
ODatabaseSource_Base::getTypes()
);
2000-09-18 23:16:46 +00:00
}
//--------------------------------------------------------------------------
Sequence< sal_Int8 > ODatabaseSource::getUnoTunnelImplementationId() throw (RuntimeException)
{
static OImplementationId * pId = 0;
if (! pId)
{
MutexGuard aGuard( Mutex::getGlobalMutex() );
if (! pId)
{
static OImplementationId aId;
pId = &aId;
}
}
return pId->getImplementationId();
}
//--------------------------------------------------------------------------
Sequence< sal_Int8 > ODatabaseSource::getImplementationId() throw (RuntimeException)
{
return getUnoTunnelImplementationId();
}
// com::sun::star::uno::XInterface
//--------------------------------------------------------------------------
Any ODatabaseSource::queryInterface( const Type & rType ) throw (RuntimeException)
{
Any aIface = OSubComponent::queryInterface( rType );
if (!aIface.hasValue())
aIface = ODatabaseSource_Base::queryInterface( rType );
if (!aIface.hasValue())
aIface = OConfigurationFlushable::queryInterface( rType );
if (!aIface.hasValue())
aIface = ::cppu::queryInterface(
rType,
static_cast< XPropertySet* >( this ),
static_cast< XFastPropertySet* >( this ),
static_cast< XMultiPropertySet* >( this ));
return aIface;
}
//--------------------------------------------------------------------------
void ODatabaseSource::acquire() throw (RuntimeException)
{
OSubComponent::acquire();
}
//--------------------------------------------------------------------------
void ODatabaseSource::release() throw (RuntimeException)
{
OSubComponent::release();
}
2000-11-14 12:32:59 +00:00
// -----------------------------------------------------------------------------
void SAL_CALL ODatabaseSource::disposing( const ::com::sun::star::lang::EventObject& Source ) throw(::com::sun::star::uno::RuntimeException)
{
for (OWeakConnectionArray::iterator i = m_aConnections.begin(); m_aConnections.end() != i; i++)
{
if(Source.Source == i->get())
{
*i = OWeakConnection();
}
}
2000-09-18 23:16:46 +00:00
2000-11-14 12:32:59 +00:00
}
2000-09-18 23:16:46 +00:00
// XServiceInfo
//------------------------------------------------------------------------------
rtl::OUString ODatabaseSource::getImplementationName( ) throw(RuntimeException)
{
return getImplementationName_Static();
}
//------------------------------------------------------------------------------
rtl::OUString ODatabaseSource::getImplementationName_Static( ) throw(RuntimeException)
{
2001-01-05 09:49:38 +00:00
return rtl::OUString::createFromAscii("com.sun.star.comp.dba.ODatabaseSource");
2000-09-18 23:16:46 +00:00
}
//------------------------------------------------------------------------------
Sequence< ::rtl::OUString > ODatabaseSource::getSupportedServiceNames( ) throw (RuntimeException)
{
return getSupportedServiceNames_Static();
}
//------------------------------------------------------------------------------
Reference< XInterface > ODatabaseSource::Create(const Reference< XMultiServiceFactory >& _rxFactory)
{
return ODatabaseSource_CreateInstance(_rxFactory);
}
//------------------------------------------------------------------------------
Sequence< ::rtl::OUString > ODatabaseSource::getSupportedServiceNames_Static( ) throw (RuntimeException)
{
Sequence< ::rtl::OUString > aSNS( 1 );
aSNS[0] = SERVICE_SDB_DATASOURCE;
return aSNS;
}
//------------------------------------------------------------------------------
sal_Bool ODatabaseSource::supportsService( const ::rtl::OUString& _rServiceName ) throw (RuntimeException)
{
2000-10-11 10:21:40 +00:00
return ::comphelper::findValue(getSupportedServiceNames(), _rServiceName, sal_True).getLength() != 0;
2000-09-18 23:16:46 +00:00
}
// com::sun::star::lang::XUnoTunnel
//------------------------------------------------------------------
sal_Int64 ODatabaseSource::getSomething( const Sequence< sal_Int8 > & rId ) throw (RuntimeException)
{
if (rId.getLength() != 16)
return 0;
if (0 == rtl_compareMemory(getUnoTunnelImplementationId().getConstArray(), rId.getConstArray(), 16 ) )
return (sal_Int64)this;
if (0 == rtl_compareMemory(OContainerElement::getUnoTunnelImplementationId().getConstArray(), rId.getConstArray(), 16 ) )
return (sal_Int64)static_cast<OContainerElement*>(this);
return 0;
}
// OComponentHelper
//------------------------------------------------------------------------------
void ODatabaseSource::disposing()
{
OPropertySetHelper::disposing();
OConfigurationFlushable::disposing();
MutexGuard aGuard(m_aMutex);
// if (m_aConfigurationNode.isValid())
2000-10-11 10:21:40 +00:00
// flush();
2000-09-18 23:16:46 +00:00
// TODO : we need a mechanism for determining wheter we're modified and need that call or not
Reference< XConnection > xConn;
for (OWeakConnectionArray::iterator i = m_aConnections.begin(); m_aConnections.end() != i; i++)
{
xConn = *i;
if (xConn.is())
xConn->close();
}
m_aConnections.clear();
}
//------------------------------------------------------------------------------
Reference< XConnection > ODatabaseSource::buildLowLevelConnection(const ::rtl::OUString& _rUid, const ::rtl::OUString& _rPwd)
{
Reference< XConnection > xReturn;
2001-04-26 08:18:16 +00:00
Reference< XDriverManager > xManager(m_xServiceFactory->createInstance(
2001-05-03 13:35:27 +00:00
#if SUPD < 631
SERVICE_SDBC_DRIVERMANAGER)
2001-04-26 08:18:16 +00:00
#else
::rtl::OUString::createFromAscii("com.sun.star.sdbc.ConnectionPool") )
#endif
, UNO_QUERY);
2001-04-26 08:09:55 +00:00
// SERVICE_SDBC_DRIVERMANAGER
2000-09-18 23:16:46 +00:00
if (xManager.is())
{
sal_Int32 nAdditionalArgs(0);
if (_rUid.getLength()) ++nAdditionalArgs;
if (_rPwd.getLength()) ++nAdditionalArgs;
Sequence< PropertyValue > aUserPwd(nAdditionalArgs);
if (_rUid.getLength())
{
aUserPwd[0].Name = ::rtl::OUString::createFromAscii("user");
aUserPwd[0].Value <<= _rUid;
}
if (_rPwd.getLength())
{
aUserPwd[1].Name = ::rtl::OUString::createFromAscii("password");
aUserPwd[1].Value <<= _rPwd;
}
if (nAdditionalArgs)
xReturn = xManager->getConnectionWithInfo(m_sConnectURL, ::comphelper::concatSequences(aUserPwd,m_aInfo));
2000-09-18 23:16:46 +00:00
else
xReturn = xManager->getConnectionWithInfo(m_sConnectURL,m_aInfo);
2000-09-18 23:16:46 +00:00
}
return xReturn;
}
//------------------------------------------------------------------------------
const Reference< XNumberFormatsSupplier > & ODatabaseSource::getNumberFormatsSupplier()
{
if (!m_xNumberFormatsSupplier.is())
{
// the arguments : the locale of the current user
UserInformation aUserInfo;
2000-09-18 23:16:46 +00:00
Sequence< Any > aArguments(1);
aArguments.getArray()[0] <<= aUserInfo.getUserLanguage();
m_xNumberFormatsSupplier = Reference< XNumberFormatsSupplier >(m_xServiceFactory->createInstanceWithArguments(
::rtl::OUString::createFromAscii("com.sun.star.util.NumberFormatsSupplier"),
aArguments),
UNO_QUERY);
DBG_ASSERT(m_xNumberFormatsSupplier.is(), "ODatabaseSource::getNumberFormatsSupplier : could not instantiate the formats supplier !");
}
return m_xNumberFormatsSupplier;
}
// OPropertySetHelper
//------------------------------------------------------------------------------
Reference< XPropertySetInfo > ODatabaseSource::getPropertySetInfo() throw (RuntimeException)
{
return createPropertySetInfo( getInfoHelper() ) ;
}
2000-10-11 10:21:40 +00:00
// comphelper::OPropertyArrayUsageHelper
2000-09-18 23:16:46 +00:00
//------------------------------------------------------------------------------
::cppu::IPropertyArrayHelper* ODatabaseSource::createArrayHelper( ) const
{
BEGIN_PROPERTY_HELPER(12)
2000-10-11 10:21:40 +00:00
DECL_PROP0(INFO, Sequence< PropertyValue >);
2000-09-18 23:16:46 +00:00
DECL_PROP1_BOOL(ISPASSWORDREQUIRED, BOUND);
DECL_PROP1_BOOL(ISREADONLY, READONLY);
DECL_PROP0(LAYOUTINFORMATION, Sequence< sal_Int8 >);
2000-09-18 23:16:46 +00:00
DECL_PROP1(NAME, ::rtl::OUString, READONLY);
DECL_PROP1_IFACE(NUMBERFORMATSSUPPLIER, XNumberFormatsSupplier, READONLY);
DECL_PROP1(PASSWORD, ::rtl::OUString, TRANSIENT);
DECL_PROP1_BOOL(SUPPRESSVERSIONCL, BOUND);
2000-09-18 23:16:46 +00:00
DECL_PROP1(TABLEFILTER, Sequence< ::rtl::OUString >,BOUND);
DECL_PROP1(TABLETYPEFILTER, Sequence< ::rtl::OUString >,BOUND);
2000-10-11 10:21:40 +00:00
DECL_PROP0(URL, ::rtl::OUString);
2000-09-18 23:16:46 +00:00
DECL_PROP1(USER, ::rtl::OUString, BOUND);
END_PROPERTY_HELPER();
}
// cppu::OPropertySetHelper
//------------------------------------------------------------------------------
::cppu::IPropertyArrayHelper& ODatabaseSource::getInfoHelper()
{
return *getArrayHelper();
}
//------------------------------------------------------------------------------
sal_Bool ODatabaseSource::convertFastPropertyValue(Any & rConvertedValue, Any & rOldValue, sal_Int32 nHandle, const Any& rValue ) throw( IllegalArgumentException )
{
if (m_bReadOnly)
throw IllegalArgumentException();
sal_Bool bModified(sal_False);
switch (nHandle)
{
case PROPERTY_ID_TABLEFILTER:
2000-10-11 10:21:40 +00:00
bModified = ::comphelper::tryPropertyValue(rConvertedValue, rOldValue, rValue, m_aTableFilter);
2000-09-18 23:16:46 +00:00
break;
case PROPERTY_ID_TABLETYPEFILTER:
2000-10-11 10:21:40 +00:00
bModified = ::comphelper::tryPropertyValue(rConvertedValue, rOldValue, rValue, m_aTableTypeFilter);
2000-09-18 23:16:46 +00:00
break;
case PROPERTY_ID_USER:
2000-10-11 10:21:40 +00:00
bModified = ::comphelper::tryPropertyValue(rConvertedValue, rOldValue, rValue, m_sUser);
2000-09-18 23:16:46 +00:00
break;
case PROPERTY_ID_PASSWORD:
2000-10-11 10:21:40 +00:00
bModified = ::comphelper::tryPropertyValue(rConvertedValue, rOldValue, rValue, m_aPassword);
2000-09-18 23:16:46 +00:00
break;
case PROPERTY_ID_ISPASSWORDREQUIRED:
2000-10-11 10:21:40 +00:00
bModified = ::comphelper::tryPropertyValue(rConvertedValue, rOldValue, rValue, m_bPasswordRequired);
2000-09-18 23:16:46 +00:00
break;
case PROPERTY_ID_SUPPRESSVERSIONCL:
bModified = ::comphelper::tryPropertyValue(rConvertedValue, rOldValue, rValue, m_bSuppressVersionColumns);
break;
case PROPERTY_ID_LAYOUTINFORMATION:
bModified = ::comphelper::tryPropertyValue(rConvertedValue, rOldValue, rValue, m_aLayoutInformation);
break;
2000-09-18 23:16:46 +00:00
case PROPERTY_ID_URL:
{
2000-10-11 10:21:40 +00:00
bModified = ::comphelper::tryPropertyValue(rConvertedValue, rOldValue, rValue, m_sConnectURL);
2000-09-18 23:16:46 +00:00
} break;
case PROPERTY_ID_INFO:
{
Sequence<PropertyValue> aValues;
if (!(rValue >>= aValues))
throw IllegalArgumentException();
bModified = sal_True;
// don't wan't to check the properties, it's seems more expensiv than just to set the same props again
rConvertedValue = rValue;
rOldValue <<= m_aInfo;
2000-09-18 23:16:46 +00:00
} break;
default:
DBG_ERROR("unknown Property");
}
return bModified;
}
//------------------------------------------------------------------------------
void ODatabaseSource::setFastPropertyValue_NoBroadcast( sal_Int32 nHandle, const Any& rValue ) throw (Exception)
{
switch(nHandle)
{
case PROPERTY_ID_TABLEFILTER:
rValue >>= m_aTableFilter;
break;
case PROPERTY_ID_TABLETYPEFILTER:
rValue >>= m_aTableTypeFilter;
break;
case PROPERTY_ID_USER:
rValue >>= m_sUser;
// if the user name changed, reset the password
m_aPassword = ::rtl::OUString();
2000-09-18 23:16:46 +00:00
break;
case PROPERTY_ID_PASSWORD:
rValue >>= m_aPassword;
break;
case PROPERTY_ID_ISPASSWORDREQUIRED:
m_bPasswordRequired = any2bool(rValue);
break;
case PROPERTY_ID_SUPPRESSVERSIONCL:
m_bSuppressVersionColumns = any2bool(rValue);
break;
2000-09-18 23:16:46 +00:00
case PROPERTY_ID_URL:
rValue >>= m_sConnectURL;
break;
case PROPERTY_ID_INFO:
rValue >>= m_aInfo;
break;
case PROPERTY_ID_LAYOUTINFORMATION:
rValue >>= m_aLayoutInformation;
break;
2000-09-18 23:16:46 +00:00
}
}
//------------------------------------------------------------------------------
void ODatabaseSource::getFastPropertyValue( Any& rValue, sal_Int32 nHandle ) const
{
switch (nHandle)
{
case PROPERTY_ID_TABLEFILTER:
rValue <<= m_aTableFilter;
break;
case PROPERTY_ID_TABLETYPEFILTER:
rValue <<= m_aTableTypeFilter;
break;
case PROPERTY_ID_USER:
rValue <<= m_sUser;
break;
case PROPERTY_ID_PASSWORD:
rValue <<= m_aPassword;
break;
case PROPERTY_ID_ISPASSWORDREQUIRED:
rValue = bool2any(m_bPasswordRequired);
break;
case PROPERTY_ID_SUPPRESSVERSIONCL:
rValue = bool2any(m_bSuppressVersionColumns);
break;
2000-09-18 23:16:46 +00:00
case PROPERTY_ID_ISREADONLY:
rValue = bool2any(m_bReadOnly);
break;
case PROPERTY_ID_INFO:
rValue <<= m_aInfo;
2000-09-18 23:16:46 +00:00
break;
case PROPERTY_ID_URL:
rValue <<= m_sConnectURL;
break;
case PROPERTY_ID_NUMBERFORMATSSUPPLIER:
rValue <<= const_cast<ODatabaseSource*>(this)->getNumberFormatsSupplier();
break;
case PROPERTY_ID_NAME:
rValue <<= m_sName;
break;
case PROPERTY_ID_LAYOUTINFORMATION:
rValue <<= m_aLayoutInformation;
break;
2000-09-18 23:16:46 +00:00
default:
DBG_ERROR("unknown Property");
}
}
// XDataSource
//------------------------------------------------------------------------------
void ODatabaseSource::setLoginTimeout(sal_Int32 seconds) throw( SQLException, RuntimeException )
{
MutexGuard aGuard(m_aMutex);
m_nLoginTimeout = seconds;
}
//------------------------------------------------------------------------------
sal_Int32 ODatabaseSource::getLoginTimeout(void) throw( SQLException, RuntimeException )
{
return m_nLoginTimeout;
}
2000-11-08 15:05:02 +00:00
// XCompletedConnection
//------------------------------------------------------------------------------
Reference< XConnection > SAL_CALL ODatabaseSource::connectWithCompletion( const Reference< XInteractionHandler >& _rxHandler ) throw(SQLException, RuntimeException)
{
MutexGuard aGuard(m_aMutex);
if (OComponentHelper::rBHelper.bDisposed)
throw DisposedException();
if (!_rxHandler.is())
{
DBG_ERROR("ODatabaseSource::connectWithCompletion: invalid interaction handler!");
return getConnection(m_sUser, m_aPassword);
}
::rtl::OUString sUser(m_sUser), sPassword(m_aPassword);
sal_Bool bNewPasswordGiven = sal_False;
2000-11-08 15:05:02 +00:00
if (m_bPasswordRequired && (0 == sPassword.getLength()))
{ // we need a password, but don't have one yet.
// -> ask the user
// build an interaction request
// two continuations (Ok and Cancel)
OInteractionAbort* pAbort = new OInteractionAbort;
OAuthenticationContinuation* pAuthenticate = new OAuthenticationContinuation;
// the request
AuthenticationRequest aRequest;
aRequest.ServerName = m_sName;
aRequest.HasRealm = aRequest.HasAccount = sal_False;
aRequest.HasUserName = aRequest.HasPassword = sal_True;
aRequest.UserName = m_sUser;
aRequest.Password = m_aPassword;
OInteractionRequest* pRequest = new OInteractionRequest(makeAny(aRequest));
Reference< XInteractionRequest > xRequest(pRequest);
// some knittings
pRequest->addContinuation(pAbort);
pRequest->addContinuation(pAuthenticate);
// handle the request
try
{
_rxHandler->handle(xRequest);
}
catch(Exception&)
{
DBG_ERROR("ODatabaseSource::connectWithCompletion: caught an exception while calling the handler!");
}
if (!pAuthenticate->wasSelected())
return Reference< XConnection >();
// get the result
sUser = m_sUser = pAuthenticate->getUser();
sPassword = pAuthenticate->getPassword();
if (pAuthenticate->getRememberPassword())
{
2000-11-08 15:05:02 +00:00
m_aPassword = pAuthenticate->getPassword();
bNewPasswordGiven = sal_True;
}
}
try
{
return getConnection(sUser, sPassword);
}
catch(Exception&)
{
if (bNewPasswordGiven)
// assume that we had an authentication problem. Without this we may, after an unsucessfull connect, while
// the user gave us a password an the order to remember it, never allow an password input again (at least
// not without restarting the session)
m_aPassword = ::rtl::OUString();
throw;
2000-11-08 15:05:02 +00:00
}
DBG_ERROR("ODatabaseSource::connectWithCompletion: reached the unreacable!");
return Reference< XConnection >();
2000-11-08 15:05:02 +00:00
}
2000-09-18 23:16:46 +00:00
//------------------------------------------------------------------------------
Reference< XConnection > ODatabaseSource::getConnection(const rtl::OUString& user, const rtl::OUString& password) throw( SQLException, RuntimeException )
{
MutexGuard aGuard(m_aMutex);
if (OComponentHelper::rBHelper.bDisposed)
throw DisposedException();
Reference< XConnection > xSdbcConn = buildLowLevelConnection(user, password);
if (!xSdbcConn.is())
// something went heavily wrong, for instance the driver manager could not be instantiated
2000-11-08 15:05:02 +00:00
throwGenericSQLException(::rtl::OUString::createFromAscii("Could not connect to the data source. An unspecified error occured."), static_cast< XDataSource* >(this));
2000-09-18 23:16:46 +00:00
Reference< XConnection > xConn;
// build a connection server and return it (no stubs)
2000-12-12 11:20:31 +00:00
xConn = new OConnection(*this, m_aConfigurationNode.openNode(CONFIGKEY_DBLINK_TABLES),m_aConfigurationNode,xSdbcConn, m_xServiceFactory);
2000-11-14 12:32:59 +00:00
Reference< XComponent> xComp(xConn,UNO_QUERY);
if(xComp.is())
{
xComp->addEventListener(this);
}
2000-09-18 23:16:46 +00:00
m_aConnections.push_back(OWeakConnection(xConn));
return xConn;
}
//------------------------------------------------------------------------------
void ODatabaseSource::flush_NoBroadcast_NoCommit()
2000-09-18 23:16:46 +00:00
{
flushToConfiguration();
}
//------------------------------------------------------------------------------
Reference< XNameAccess > SAL_CALL ODatabaseSource::getBookmarks( ) throw (RuntimeException)
2000-09-18 23:16:46 +00:00
{
return static_cast< XNameContainer* >(&m_aBookmarks);
2000-09-18 23:16:46 +00:00
}
//------------------------------------------------------------------------------
Reference< XNameAccess > SAL_CALL ODatabaseSource::getQueryDefinitions( ) throw(RuntimeException)
{
return static_cast< XNameContainer* >(&m_aCommandDefinitions);
}
//------------------------------------------------------------------------------
void ODatabaseSource::inserted(const Reference< XInterface >& _rxContainer, const ::rtl::OUString& _rRegistrationName, const OConfigurationTreeRoot& _rConfigRoot)
2000-09-18 23:16:46 +00:00
{
MutexGuard aGuard(m_aMutex);
DBG_ASSERT(!m_aConfigurationNode.isValid(), "ODatabaseSource::inserted : you're not allowed to change the location if the current one is valid !");
DBG_ASSERT(_rConfigRoot.isValid(), "ODatabaseSource::inserted : invalid argument (the configuration root) !");
2000-09-18 23:16:46 +00:00
DBG_ASSERT(_rRegistrationName.getLength() != 0, "ODatabaseSource::inserted : invalid argument (the name) !");
DBG_ASSERT(!m_xParent.is(), "ODatabaseSource::inserted : already connected to a parent !");
m_aConfigurationNode = _rConfigRoot;
2000-09-18 23:16:46 +00:00
m_xParent = _rxContainer;
m_sName = _rRegistrationName;
if (m_aConfigurationNode.isValid())
2000-09-18 23:16:46 +00:00
{
// propagate the new location to our documents
// (Usually, we do this from within the ctor which gets a config node, but if we're here, we have been
// instantiated as service, so we didn't have any config location before, so the documents haven't any, too.)
initializeDocuments(sal_False);
flushToConfiguration();
2000-09-18 23:16:46 +00:00
}
// adjust our readonly flag
m_bReadOnly = !m_aConfigurationNode.isValid() || m_aConfigurationNode.isReadonly();
2000-09-18 23:16:46 +00:00
}
//------------------------------------------------------------------------------
void ODatabaseSource::removed()
{
MutexGuard aGuard(m_aMutex);
DBG_ASSERT(m_xParent.is(), "ODatabaseSource::removed : not connected to a parent !");
// dispose the document containers so they release the documents and the configuration resources
m_aBookmarks.dispose();
2000-09-18 23:16:46 +00:00
m_aCommandDefinitions.dispose();
m_xParent = NULL;
m_aConfigurationNode.clear();
2000-09-18 23:16:46 +00:00
m_sName = ::rtl::OUString();
m_bReadOnly = sal_False;
}
//------------------------------------------------------------------------------
void ODatabaseSource::initializeDocuments(sal_Bool _bRead)
2000-09-18 23:16:46 +00:00
{
// initialize the document containers
m_aBookmarks.initialize(m_aConfigurationNode.openNode(CONFIGKEY_DBLINK_BOOKMARKS).cloneAsRoot(), _bRead);
m_aCommandDefinitions.initialize(m_aConfigurationNode.openNode(CONFIGKEY_DBLINK_QUERYDOCUMENTS).cloneAsRoot(), _bRead);
2000-09-18 23:16:46 +00:00
}
//------------------------------------------------------------------------------
void ODatabaseSource::initializeFromConfiguration()
{
if (!m_aConfigurationNode.isValid())
2000-09-18 23:16:46 +00:00
{
DBG_ERROR("ODatabaseSource::initializeFromConfiguration : invalid configuration key !");
return;
}
m_aConfigurationNode.getNodeValue(CONFIGKEY_DBLINK_CONNECTURL) >>= m_sConnectURL;
m_aConfigurationNode.getNodeValue(CONFIGKEY_DBLINK_USER) >>= m_sUser;
m_aConfigurationNode.getNodeValue(CONFIGKEY_DBLINK_TABLEFILTER) >>= m_aTableFilter;
m_aConfigurationNode.getNodeValue(CONFIGKEY_DBLINK_TABLETYEFILTER) >>= m_aTableTypeFilter;
m_aConfigurationNode.getNodeValue(CONFIGKEY_DBLINK_LOGINTIMEOUT) >>= m_nLoginTimeout;
m_bPasswordRequired = ::cppu::any2bool(m_aConfigurationNode.getNodeValue(CONFIGKEY_DBLINK_PASSWORDREQUIRED));
m_bSuppressVersionColumns = ::cppu::any2bool(m_aConfigurationNode.getNodeValue(CONFIGKEY_DBLINK_SUPPRESSVERSIONCL));
m_aConfigurationNode.getNodeValue(CONFIGKEY_LAYOUTINFORMATION) >>= m_aLayoutInformation;
2000-09-18 23:16:46 +00:00
// the property sequence in m_aInfo
OConfigurationNode aInfoNode = m_aConfigurationNode.openNode(CONFIGKEY_DBLINK_INFO);
if (aInfoNode.isValid())
{
Sequence< ::rtl::OUString > aNodeNames = aInfoNode.getNodeNames();
m_aInfo.realloc(aNodeNames.getLength());
PropertyValue* pInfos = m_aInfo.getArray();
for ( const ::rtl::OUString* pNodeNames = aNodeNames.getConstArray() + aNodeNames.getLength() - 1;
pNodeNames >= aNodeNames.getConstArray();
--pNodeNames, ++pInfos
)
{
OConfigurationNode aItemSubNode = aInfoNode.openNode(*pNodeNames);
pInfos->Name = *pNodeNames;
pInfos->Value = aItemSubNode.getNodeValue(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("Value")));
}
}
2000-09-18 23:16:46 +00:00
initializeDocuments();
}
//------------------------------------------------------------------------------
void ODatabaseSource::flushDocuments()
{
m_aBookmarks.flush();
2000-09-18 23:16:46 +00:00
m_aCommandDefinitions.flush();
}
2000-12-12 11:20:31 +00:00
// -----------------------------------------------------------------------------
void ODatabaseSource::flushTables()
{
}
2000-09-18 23:16:46 +00:00
//------------------------------------------------------------------------------
void ODatabaseSource::flushToConfiguration()
{
if (!m_aConfigurationNode.isValid())
2000-09-18 23:16:46 +00:00
{
DBG_ERROR("ODatabaseSource::flushToConfiguration : invalid configuration key !");
return;
}
m_aConfigurationNode.setNodeValue(CONFIGKEY_DBLINK_CONNECTURL, makeAny(m_sConnectURL));
m_aConfigurationNode.setNodeValue(CONFIGKEY_DBLINK_USER, makeAny(m_sUser));
m_aConfigurationNode.setNodeValue(CONFIGKEY_DBLINK_TABLEFILTER, makeAny(m_aTableFilter));
m_aConfigurationNode.setNodeValue(CONFIGKEY_DBLINK_TABLETYEFILTER, makeAny(m_aTableTypeFilter));
m_aConfigurationNode.setNodeValue(CONFIGKEY_DBLINK_LOGINTIMEOUT, makeAny(m_nLoginTimeout));
m_aConfigurationNode.setNodeValue(CONFIGKEY_DBLINK_PASSWORDREQUIRED, ::cppu::bool2any(m_bPasswordRequired));
m_aConfigurationNode.setNodeValue(CONFIGKEY_DBLINK_SUPPRESSVERSIONCL, ::cppu::bool2any(m_bSuppressVersionColumns));
m_aConfigurationNode.setNodeValue(CONFIGKEY_LAYOUTINFORMATION, makeAny(m_aLayoutInformation));
2000-09-18 23:16:46 +00:00
// write the additional info tags
// unfortunately, the same as always applies here: the configuration does not support different
// operations in one transaction, so we have to open an separate sub tree for the info node, so we can
// do separate transactions
OConfigurationTreeRoot aInfoNode = m_aConfigurationNode.openNode(CONFIGKEY_DBLINK_INFO).cloneAsRoot();
if (aInfoNode.isValid())
{
// stage 0: collect all names under the info node which currently exist
::std::set< rtl::OUString > aExistentKeys;
Sequence< ::rtl::OUString > aNodeNames = aInfoNode.getNodeNames();
for ( const ::rtl::OUString* pNodeNames = aNodeNames.getConstArray() + aNodeNames.getLength() - 1;
pNodeNames >= aNodeNames.getConstArray();
--pNodeNames
)
{
aExistentKeys.insert(*pNodeNames);
aInfoNode.commit();
}
// stage 1: write all currently set info values
static ::rtl::OUString s_sValueConfigKey(RTL_CONSTASCII_USTRINGPARAM("Value"));
::std::set< rtl::OUString > aUsedKeys;
const PropertyValue* pInfoValues = m_aInfo.getConstArray();
for (sal_Int32 i=0; i<m_aInfo.getLength(); ++i, ++pInfoValues)
{
OConfigurationNode aSettingsItem;
if (aExistentKeys.end() == aExistentKeys.find(pInfoValues->Name))
{
// we do not have such a key -> create it
aSettingsItem = aInfoNode.createNode(pInfoValues->Name);
}
else
{
aSettingsItem = aInfoNode.openNode(pInfoValues->Name);
aExistentKeys.erase(pInfoValues->Name);
// no need to delete this previously-existing node afterwards ....
}
aSettingsItem.setNodeValue(s_sValueConfigKey, pInfoValues->Value);
aInfoNode.commit();
}
// stage 2: delete all info values which may be present in the registry, but not used by the current values
for ( ::std::set< rtl::OUString >::const_iterator aErase = aExistentKeys.begin();
aErase != aExistentKeys.end();
++aErase
)
{
aInfoNode.removeNode(*aErase);
aInfoNode.commit();
}
}
2000-09-18 23:16:46 +00:00
flushDocuments();
2000-12-12 11:20:31 +00:00
flushTables();
2000-09-18 23:16:46 +00:00
// TODO : flushing of queries/tables ?
}
#if 0
//------------------------------------------------------------------------------
void ODatabaseSource::readUIAspects(const ::vos::ORef< ::store::OStream >& _rStream)
{
DBG_ASSERT(_rStream.isValid() && _rStream->isOpen(), "ODatabaseSource::readUIAspects : you gave me garbage !");
OCompatStreamSector aCompat(_rStream, OCompatStreamSector::mode_Read);
sal_Int32 nTokensUsed(0);
*_rStream >> nTokensUsed;
if ((nTokensUsed & PT_SVFORMATTER) != 0)
{
OCompatStreamSector aCompat(_rStream, OCompatStreamSector::mode_Read);
// a object which wraps the storage stream into an SvStream
OStorageStreamWrapper aSvStreamWrapper(_rStream);
// an ObjectInputStream
Reference< XObjectInputStream > xInStream(m_xServiceFactory->createInstance(
::rtl::OUString::createFromAscii("com.sun.star.io.ObjectInputStream")),
UNO_QUERY
);
Reference< XActiveDataSink > xInDataSink(xInStream, UNO_QUERY);
DBG_ASSERT(xInDataSink.is(), "ODatabaseSource::readUIAspects : could not instantiate an input stream !");
if (xInDataSink.is())
{ // can't load the formatter without that service
// an uno wrapper for the SvStream
2000-10-11 10:21:40 +00:00
Reference< XInputStream > xUnoStream = new ::comphelper::OInputStreamWrapper(aSvStreamWrapper);
2000-09-18 23:16:46 +00:00
xInDataSink->setInputStream(xUnoStream);
try
{
Reference< XPersistObject > xPersistentSupplier = xInStream->readObject();
Reference< XNumberFormatsSupplier > xSupplier(xPersistentSupplier, UNO_QUERY);
DBG_ASSERT(xSupplier.is(), "ODatabaseSource::readUIAspects : the object read is no NumberFormatsSupplier !");
if (xSupplier.is())
m_xNumberFormatsSupplier = xSupplier;
}
2001-04-02 05:15:56 +00:00
catch(Exception&)
2000-09-18 23:16:46 +00:00
{
DBG_ERROR("ODatabaseSource::readUIAspects : could not read the formatter !");
}
}
}
}
//------------------------------------------------------------------------------
void ODatabaseSource::writeUIAspects(const ::vos::ORef< ::store::OStream >& _rStream)
{
DBG_ASSERT(_rStream.isValid() && _rStream->isOpen(), "ODatabaseSource::writeUIAspects : you gave me garbage !");
OCompatStreamSector aCompat(_rStream, OCompatStreamSector::mode_Write);
sal_Int32 nTokensUsed(0);
Reference< XPersistObject > xPersistentSupplier(m_xNumberFormatsSupplier, UNO_QUERY);
DBG_ASSERT(xPersistentSupplier.is() || !m_xNumberFormatsSupplier.is(),
"ODatabaseSource::writeUIAspects : have a formats supplier, but it has no XPersistObject interface !");
// create an SvStream based on _rStream (we'll need this below)
OStorageStreamWrapper aSvStreamWrapper(_rStream);
Reference< XObjectOutputStream > xOutStream;
Reference< XOutputStream > xUnoStream;
// will be filled below
if (xPersistentSupplier.is())
{ // no need to store the formats supplier if we don't have one, then it can be defaulted on loading
// first, we need an XObjectOutputStream based on _rStream
xOutStream = Reference< XObjectOutputStream >(m_xServiceFactory->createInstance(
::rtl::OUString::createFromAscii("com.sun.star.io.ObjectOutputStream")),
UNO_QUERY
);
Reference< XActiveDataSource > xOutDataSource(xOutStream, UNO_QUERY);
DBG_ASSERT(xOutDataSource.is(), "ODatabaseSource::writeUIAspects : could not instantiate an output stream !");
if (xOutDataSource.is())
{ // can't store the formatter without that service
// an uno wrapper for the SvStream
2000-10-11 10:21:40 +00:00
xUnoStream = new ::comphelper::OOutputStreamWrapper(aSvStreamWrapper);
2000-09-18 23:16:46 +00:00
xOutDataSource->setOutputStream(xUnoStream);
nTokensUsed |= PT_SVFORMATTER;
// now we have all items for storing the formats supplier
}
}
*_rStream << nTokensUsed;
if ((nTokensUsed & PT_SVFORMATTER) != 0)
{
OCompatStreamSector aCompat(_rStream, OCompatStreamSector::mode_Write);
// as the formatter is a rather complex object and as it's store and load depend on the
// availableness of some services, we put this into an own "skippable sector"
try
{
xOutStream->writeObject(xPersistentSupplier);
}
2001-04-02 05:15:56 +00:00
catch(Exception&)
2000-09-18 23:16:46 +00:00
{
DBG_ERROR("ODatabaseSource::writeUIAspects : could not store the formatter");
}
}
}
2000-09-18 23:16:46 +00:00
#endif
//........................................................................
} // namespace dbaccess
//........................................................................