/************************************************************** * * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. * *************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_connectivity.hxx" #include "hsqldb/HStorageMap.hxx" #include #include #include #include #include #include "diagnose_ex.h" #include //........................................................................ namespace connectivity { //........................................................................ namespace hsqldb { //........................................................................ using namespace ::com::sun::star::uno; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::embed; using namespace ::com::sun::star::io; #define ThrowException(env, type, msg) { \ env->ThrowNew(env->FindClass(type), msg); } StreamHelper::StreamHelper(const Reference< XStream>& _xStream) : m_xStream(_xStream) { } // ----------------------------------------------------------------------------- StreamHelper::~StreamHelper() { try { m_xStream.clear(); m_xSeek.clear(); if ( m_xInputStream.is() ) { m_xInputStream->closeInput(); m_xInputStream.clear(); } // this is done implicity by the closing of the input stream else if ( m_xOutputStream.is() ) { m_xOutputStream->closeOutput(); try { ::comphelper::disposeComponent(m_xOutputStream); } catch(DisposedException&) { } catch(const Exception& e) { OSL_UNUSED( e ); OSL_ENSURE(0,"Could not dispose OutputStream"); } m_xOutputStream.clear(); } } catch(Exception& ex) { OSL_UNUSED( ex ); OSL_ENSURE(0,"Exception catched!"); } } // ----------------------------------------------------------------------------- Reference< XInputStream> StreamHelper::getInputStream() { if ( !m_xInputStream.is() ) m_xInputStream = m_xStream->getInputStream(); return m_xInputStream; } // ----------------------------------------------------------------------------- Reference< XOutputStream> StreamHelper::getOutputStream() { if ( !m_xOutputStream.is() ) m_xOutputStream = m_xStream->getOutputStream(); return m_xOutputStream; } // ----------------------------------------------------------------------------- Reference< XSeekable> StreamHelper::getSeek() { if ( !m_xSeek.is() ) m_xSeek.set(m_xStream,UNO_QUERY); return m_xSeek; } // ----------------------------------------------------------------------------- TStorages& lcl_getStorageMap() { static TStorages s_aMap; return s_aMap; } // ----------------------------------------------------------------------------- ::rtl::OUString lcl_getNextCount() { static sal_Int32 s_nCount = 0; return ::rtl::OUString::valueOf(s_nCount++); } // ----------------------------------------------------------------------------- ::rtl::OUString StorageContainer::removeURLPrefix(const ::rtl::OUString& _sURL,const ::rtl::OUString& _sFileURL) { return _sURL.copy(_sFileURL.getLength()+1); } // ----------------------------------------------------------------------------- ::rtl::OUString StorageContainer::removeOldURLPrefix(const ::rtl::OUString& _sURL) { ::rtl::OUString sRet = _sURL; #if defined(WNT) sal_Int32 nIndex = sRet.lastIndexOf('\\'); #else sal_Int32 nIndex = sRet.lastIndexOf('/'); #endif if ( nIndex != -1 ) { sRet = _sURL.copy(nIndex+1); } return sRet; } /*****************************************************************************/ /* convert jstring to rtl_uString */ ::rtl::OUString StorageContainer::jstring2ustring(JNIEnv * env, jstring jstr) { if (JNI_FALSE != env->ExceptionCheck()) { env->ExceptionClear(); OSL_ENSURE(0,"ExceptionClear"); } ::rtl::OUString aStr; if ( jstr ) { jboolean bCopy(sal_True); const jchar* pChar = env->GetStringChars(jstr,&bCopy); jsize len = env->GetStringLength(jstr); aStr = ::rtl::OUString(pChar,len); if(bCopy) env->ReleaseStringChars(jstr,pChar); } if (JNI_FALSE != env->ExceptionCheck()) { env->ExceptionClear(); OSL_ENSURE(0,"ExceptionClear"); } return aStr; } // ----------------------------------------------------------------------------- ::rtl::OUString StorageContainer::registerStorage(const Reference< XStorage>& _xStorage,const ::rtl::OUString& _sURL) { OSL_ENSURE(_xStorage.is(),"Storage is NULL!"); TStorages& rMap = lcl_getStorageMap(); // check if the storage is already in our map TStorages::iterator aFind = ::std::find_if(rMap.begin(),rMap.end(), ::std::compose1( ::std::bind2nd(::std::equal_to >(),_xStorage) ,::std::compose1(::std::select1st(),::std::compose1(::std::select1st(),::std::select2nd()))) ); if ( aFind == rMap.end() ) { aFind = rMap.insert(TStorages::value_type(lcl_getNextCount(),TStorages::mapped_type(TStorageURLPair(_xStorage,_sURL),TStreamMap()))).first; } return aFind->first; } // ----------------------------------------------------------------------------- TStorages::mapped_type StorageContainer::getRegisteredStorage(const ::rtl::OUString& _sKey) { TStorages::mapped_type aRet; TStorages& rMap = lcl_getStorageMap(); TStorages::iterator aFind = rMap.find(_sKey); OSL_ENSURE(aFind != rMap.end(),"Storage could not be found in list!"); if ( aFind != rMap.end() ) aRet = aFind->second; return aRet; } // ----------------------------------------------------------------------------- ::rtl::OUString StorageContainer::getRegisteredKey(const Reference< XStorage>& _xStorage) { ::rtl::OUString sKey; OSL_ENSURE(_xStorage.is(),"Storage is NULL!"); TStorages& rMap = lcl_getStorageMap(); // check if the storage is already in our map TStorages::iterator aFind = ::std::find_if(rMap.begin(),rMap.end(), ::std::compose1( ::std::bind2nd(::std::equal_to >(),_xStorage) ,::std::compose1(::std::select1st(),::std::compose1(::std::select1st(),::std::select2nd()))) ); if ( aFind != rMap.end() ) sKey = aFind->first; return sKey; } // ----------------------------------------------------------------------------- void StorageContainer::revokeStorage(const ::rtl::OUString& _sKey,const Reference& _xListener) { TStorages& rMap = lcl_getStorageMap(); TStorages::iterator aFind = rMap.find(_sKey); if ( aFind != rMap.end() ) { try { if ( _xListener.is() ) { Reference xBroad(aFind->second.first.first,UNO_QUERY); if ( xBroad.is() ) xBroad->removeTransactionListener(_xListener); Reference xTrans(aFind->second.first.first,UNO_QUERY); if ( xTrans.is() ) xTrans->commit(); } } catch(Exception&) { } rMap.erase(aFind); } } // ----------------------------------------------------------------------------- TStreamMap::mapped_type StorageContainer::registerStream(JNIEnv * env,jstring name, jstring key,sal_Int32 _nMode) { TStreamMap::mapped_type pHelper; TStorages& rMap = lcl_getStorageMap(); ::rtl::OUString sKey = jstring2ustring(env,key); TStorages::iterator aFind = rMap.find(sKey); OSL_ENSURE(aFind != rMap.end(),"Storage could not be found in list!"); if ( aFind != rMap.end() ) { TStorages::mapped_type aStoragePair = StorageContainer::getRegisteredStorage(sKey); OSL_ENSURE(aStoragePair.first.first.is(),"No Storage available!"); if ( aStoragePair.first.first.is() ) { ::rtl::OUString sOrgName = StorageContainer::jstring2ustring(env,name); ::rtl::OUString sName = removeURLPrefix(sOrgName,aStoragePair.first.second); TStreamMap::iterator aStreamFind = aFind->second.second.find(sName); OSL_ENSURE( aStreamFind == aFind->second.second.end(),"A Stream was already registered for this object!"); if ( aStreamFind != aFind->second.second.end() ) { pHelper = aStreamFind->second; } else { try { try { pHelper.reset(new StreamHelper(aStoragePair.first.first->openStreamElement(sName,_nMode))); } catch(Exception& ) { ::rtl::OUString sStrippedName = removeOldURLPrefix(sOrgName); if ( ((_nMode & ElementModes::WRITE) != ElementModes::WRITE ) ) { sal_Bool bIsStream = sal_True; try { bIsStream = aStoragePair.first.first->isStreamElement(sStrippedName); } catch(Exception& ) { bIsStream = sal_False; } if ( !bIsStream ) return pHelper; // readonly file without data stream } pHelper.reset( new StreamHelper(aStoragePair.first.first->openStreamElement( sStrippedName, _nMode ) ) ); } aFind->second.second.insert(TStreamMap::value_type(sName,pHelper)); } catch(Exception& e) { #if OSL_DEBUG_LEVEL > 0 ::rtl::OString sMessage( "[HSQLDB-SDBC] caught an exception while opening a stream\n" ); sMessage += "Name: "; sMessage += ::rtl::OString( sName.getStr(), sName.getLength(), osl_getThreadTextEncoding() ); sMessage += "\nMode: 0x"; if ( _nMode < 16 ) sMessage += "0"; sMessage += ::rtl::OString::valueOf( _nMode, 16 ).toAsciiUpperCase(); OSL_ENSURE( false, sMessage.getStr() ); #endif StorageContainer::throwJavaException(e,env); } } } } return pHelper; } // ----------------------------------------------------------------------------- void StorageContainer::revokeStream( JNIEnv * env,jstring name, jstring key) { TStorages& rMap = lcl_getStorageMap(); TStorages::iterator aFind = rMap.find(jstring2ustring(env,key)); OSL_ENSURE(aFind != rMap.end(),"Storage could not be found in list!"); if ( aFind != rMap.end() ) aFind->second.second.erase(removeURLPrefix(jstring2ustring(env,name),aFind->second.first.second)); } // ----------------------------------------------------------------------------- TStreamMap::mapped_type StorageContainer::getRegisteredStream( JNIEnv * env,jstring name, jstring key) { TStreamMap::mapped_type pRet; TStorages& rMap = lcl_getStorageMap(); TStorages::iterator aFind = rMap.find(jstring2ustring(env,key)); OSL_ENSURE(aFind != rMap.end(),"Storage could not be found in list!"); if ( aFind != rMap.end() ) { TStreamMap::iterator aStreamFind = aFind->second.second.find(removeURLPrefix(jstring2ustring(env,name),aFind->second.first.second)); if ( aStreamFind != aFind->second.second.end() ) pRet = aStreamFind->second; } return pRet; } // ----------------------------------------------------------------------------- void StorageContainer::throwJavaException(const Exception& _aException,JNIEnv * env) { if (JNI_FALSE != env->ExceptionCheck()) env->ExceptionClear(); ::rtl::OString cstr( ::rtl::OUStringToOString(_aException.Message, RTL_TEXTENCODING_JAVA_UTF8 ) ); OSL_TRACE( __FILE__": forwarding Exception: %s", cstr.getStr() ); env->ThrowNew(env->FindClass("java/io/IOException"), cstr.getStr()); } //........................................................................ } // namespace hsqldb //........................................................................ //........................................................................ } // namespace connectivity //........................................................................