1 /*************************************************************************
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
5 * Copyright 2008 by Sun Microsystems, Inc.
7 * OpenOffice.org - a multi-platform office productivity suite
9 * $RCSfile: HStorageMap.cxx,v $
12 * This file is part of OpenOffice.org.
14 * OpenOffice.org is free software: you can redistribute it and/or modify
15 * it under the terms of the GNU Lesser General Public License version 3
16 * only, as published by the Free Software Foundation.
18 * OpenOffice.org is distributed in the hope that it will be useful,
19 * but WITHOUT ANY WARRANTY; without even the implied warranty of
20 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21 * GNU Lesser General Public License version 3 for more details
22 * (a copy is included in the LICENSE file that accompanied this code).
24 * You should have received a copy of the GNU Lesser General Public License
25 * version 3 along with OpenOffice.org. If not, see
26 * <http://www.openoffice.org/license.html>
27 * for a copy of the LGPLv3 License.
29 ************************************************************************/
31 // MARKER(update_precomp.py): autogen include statement, do not remove
32 #include "precompiled_connectivity.hxx"
33 #include "hsqldb/HStorageMap.hxx"
34 #include <comphelper/types.hxx>
35 #include <com/sun/star/embed/XTransactionBroadcaster.hpp>
36 #include <com/sun/star/embed/XTransactedObject.hpp>
37 #include <com/sun/star/embed/ElementModes.hpp>
38 #include <com/sun/star/lang/DisposedException.hpp>
39 #include "diagnose_ex.h"
40 #include <osl/thread.h>
42 //........................................................................
43 namespace connectivity
45 //........................................................................
48 //........................................................................
49 using namespace ::com::sun::star::uno
;
50 using namespace ::com::sun::star::lang
;
51 using namespace ::com::sun::star::embed
;
52 using namespace ::com::sun::star::io
;
54 #define ThrowException(env, type, msg) { \
55 env->ThrowNew(env->FindClass(type), msg); }
58 StreamHelper::StreamHelper(const Reference
< XStream
>& _xStream
)
62 // -----------------------------------------------------------------------------
63 StreamHelper::~StreamHelper()
69 if ( m_xInputStream
.is() )
71 m_xInputStream
->closeInput();
72 m_xInputStream
.clear();
74 if ( m_xOutputStream
.is() )
76 m_xOutputStream
->closeOutput();
79 ::comphelper::disposeComponent(m_xOutputStream
);
81 catch(DisposedException
&)
84 catch(const Exception
& e
)
87 OSL_ENSURE(0,"Could not dispose OutputStream");
89 m_xOutputStream
.clear();
94 OSL_ENSURE(0,"Exception catched!");
97 // -----------------------------------------------------------------------------
98 Reference
< XInputStream
> StreamHelper::getInputStream()
100 if ( !m_xInputStream
.is() )
101 m_xInputStream
= m_xStream
->getInputStream();
102 return m_xInputStream
;
104 // -----------------------------------------------------------------------------
105 Reference
< XOutputStream
> StreamHelper::getOutputStream()
107 if ( !m_xOutputStream
.is() )
108 m_xOutputStream
= m_xStream
->getOutputStream();
109 return m_xOutputStream
;
111 // -----------------------------------------------------------------------------
112 Reference
< XSeekable
> StreamHelper::getSeek()
115 m_xSeek
.set(m_xStream
,UNO_QUERY
);
118 // -----------------------------------------------------------------------------
119 TStorages
& lcl_getStorageMap()
121 static TStorages s_aMap
;
124 // -----------------------------------------------------------------------------
125 ::rtl::OUString
lcl_getNextCount()
127 static sal_Int32 s_nCount
= 0;
128 return ::rtl::OUString::valueOf(s_nCount
++);
130 // -----------------------------------------------------------------------------
131 ::rtl::OUString
StorageContainer::removeURLPrefix(const ::rtl::OUString
& _sURL
,const ::rtl::OUString
& _sFileURL
)
133 return _sURL
.copy(_sFileURL
.getLength()+1);
135 // -----------------------------------------------------------------------------
136 ::rtl::OUString
StorageContainer::removeOldURLPrefix(const ::rtl::OUString
& _sURL
)
138 ::rtl::OUString sRet
= _sURL
;
139 #if defined(WIN) || defined(WNT)
140 sal_Int32 nIndex
= sRet
.lastIndexOf('\\');
142 sal_Int32 nIndex
= sRet
.lastIndexOf('/');
146 sRet
= _sURL
.copy(nIndex
+1);
151 /*****************************************************************************/
152 /* convert jstring to rtl_uString */
154 ::rtl::OUString
StorageContainer::jstring2ustring(JNIEnv
* env
, jstring jstr
)
156 if (JNI_FALSE
!= env
->ExceptionCheck())
158 env
->ExceptionClear();
159 OSL_ENSURE(0,"ExceptionClear");
161 ::rtl::OUString aStr
;
164 jboolean
bCopy(sal_True
);
165 const jchar
* pChar
= env
->GetStringChars(jstr
,&bCopy
);
166 jsize len
= env
->GetStringLength(jstr
);
167 aStr
= ::rtl::OUString(pChar
,len
);
170 env
->ReleaseStringChars(jstr
,pChar
);
173 if (JNI_FALSE
!= env
->ExceptionCheck())
175 env
->ExceptionClear();
176 OSL_ENSURE(0,"ExceptionClear");
181 // -----------------------------------------------------------------------------
182 ::rtl::OUString
StorageContainer::registerStorage(const Reference
< XStorage
>& _xStorage
,const ::rtl::OUString
& _sURL
)
184 OSL_ENSURE(_xStorage
.is(),"Storage is NULL!");
185 TStorages
& rMap
= lcl_getStorageMap();
186 // check if the storage is already in our map
187 TStorages::iterator aFind
= ::std::find_if(rMap
.begin(),rMap
.end(),
189 ::std::bind2nd(::std::equal_to
<Reference
<XStorage
> >(),_xStorage
)
190 ,::std::compose1(::std::select1st
<TStorageURLPair
>(),::std::compose1(::std::select1st
<TStorages::mapped_type
>(),::std::select2nd
<TStorages::value_type
>())))
192 if ( aFind
== rMap
.end() )
194 aFind
= rMap
.insert(TStorages::value_type(lcl_getNextCount(),TStorages::mapped_type(TStorageURLPair(_xStorage
,_sURL
),TStreamMap()))).first
;
199 // -----------------------------------------------------------------------------
200 TStorages::mapped_type
StorageContainer::getRegisteredStorage(const ::rtl::OUString
& _sKey
)
202 TStorages::mapped_type aRet
;
203 TStorages
& rMap
= lcl_getStorageMap();
204 TStorages::iterator aFind
= rMap
.find(_sKey
);
205 OSL_ENSURE(aFind
!= rMap
.end(),"Storage could not be found in list!");
206 if ( aFind
!= rMap
.end() )
207 aRet
= aFind
->second
;
211 // -----------------------------------------------------------------------------
212 ::rtl::OUString
StorageContainer::getRegisteredKey(const Reference
< XStorage
>& _xStorage
)
214 ::rtl::OUString sKey
;
215 OSL_ENSURE(_xStorage
.is(),"Storage is NULL!");
216 TStorages
& rMap
= lcl_getStorageMap();
217 // check if the storage is already in our map
218 TStorages::iterator aFind
= ::std::find_if(rMap
.begin(),rMap
.end(),
220 ::std::bind2nd(::std::equal_to
<Reference
<XStorage
> >(),_xStorage
)
221 ,::std::compose1(::std::select1st
<TStorageURLPair
>(),::std::compose1(::std::select1st
<TStorages::mapped_type
>(),::std::select2nd
<TStorages::value_type
>())))
223 if ( aFind
!= rMap
.end() )
227 // -----------------------------------------------------------------------------
228 void StorageContainer::revokeStorage(const ::rtl::OUString
& _sKey
,const Reference
<XTransactionListener
>& _xListener
)
230 TStorages
& rMap
= lcl_getStorageMap();
231 TStorages::iterator aFind
= rMap
.find(_sKey
);
232 if ( aFind
!= rMap
.end() )
236 if ( _xListener
.is() )
238 Reference
<XTransactionBroadcaster
> xBroad(aFind
->second
.first
.first
,UNO_QUERY
);
240 xBroad
->removeTransactionListener(_xListener
);
241 Reference
<XTransactedObject
> xTrans(aFind
->second
.first
.first
,UNO_QUERY
);
252 // -----------------------------------------------------------------------------
253 TStreamMap::mapped_type
StorageContainer::registerStream(JNIEnv
* env
,jstring name
, jstring key
,sal_Int32 _nMode
)
255 TStreamMap::mapped_type pHelper
;
256 TStorages
& rMap
= lcl_getStorageMap();
257 ::rtl::OUString sKey
= jstring2ustring(env
,key
);
258 TStorages::iterator aFind
= rMap
.find(sKey
);
259 OSL_ENSURE(aFind
!= rMap
.end(),"Storage could not be found in list!");
260 if ( aFind
!= rMap
.end() )
262 TStorages::mapped_type aStoragePair
= StorageContainer::getRegisteredStorage(sKey
);
263 OSL_ENSURE(aStoragePair
.first
.first
.is(),"No Storage available!");
264 if ( aStoragePair
.first
.first
.is() )
266 ::rtl::OUString sOrgName
= StorageContainer::jstring2ustring(env
,name
);
267 ::rtl::OUString sName
= removeURLPrefix(sOrgName
,aStoragePair
.first
.second
);
268 TStreamMap::iterator aStreamFind
= aFind
->second
.second
.find(sName
);
269 OSL_ENSURE( aStreamFind
== aFind
->second
.second
.end(),"A Stream was already registered for this object!");
270 if ( aStreamFind
!= aFind
->second
.second
.end() )
272 pHelper
= aStreamFind
->second
;
280 pHelper
.reset(new StreamHelper(aStoragePair
.first
.first
->openStreamElement(sName
,_nMode
)));
284 ::rtl::OUString sStrippedName
= removeOldURLPrefix(sOrgName
);
286 if ( ((_nMode
& ElementModes::WRITE
) != ElementModes::WRITE
) )
288 sal_Bool bIsStream
= sal_True
;
291 bIsStream
= aStoragePair
.first
.first
->isStreamElement(sStrippedName
);
295 bIsStream
= sal_False
;
298 return pHelper
; // readonly file without data stream
300 pHelper
.reset( new StreamHelper(aStoragePair
.first
.first
->openStreamElement( sStrippedName
, _nMode
) ) );
302 aFind
->second
.second
.insert(TStreamMap::value_type(sName
,pHelper
));
306 #if OSL_DEBUG_LEVEL > 0
307 ::rtl::OString
sMessage( "[HSQLDB-SDBC] caught an exception while opening a stream\n" );
308 sMessage
+= "Name: ";
309 sMessage
+= ::rtl::OString( sName
.getStr(), sName
.getLength(), osl_getThreadTextEncoding() );
310 sMessage
+= "\nMode: 0x";
313 sMessage
+= ::rtl::OString::valueOf( _nMode
, 16 ).toAsciiUpperCase();
314 OSL_ENSURE( false, sMessage
.getStr() );
316 StorageContainer::throwJavaException(e
,env
);
323 // -----------------------------------------------------------------------------
324 void StorageContainer::revokeStream( JNIEnv
* env
,jstring name
, jstring key
)
326 TStorages
& rMap
= lcl_getStorageMap();
327 TStorages::iterator aFind
= rMap
.find(jstring2ustring(env
,key
));
328 OSL_ENSURE(aFind
!= rMap
.end(),"Storage could not be found in list!");
329 if ( aFind
!= rMap
.end() )
330 aFind
->second
.second
.erase(removeURLPrefix(jstring2ustring(env
,name
),aFind
->second
.first
.second
));
332 // -----------------------------------------------------------------------------
333 TStreamMap::mapped_type
StorageContainer::getRegisteredStream( JNIEnv
* env
,jstring name
, jstring key
)
335 TStreamMap::mapped_type pRet
;
336 TStorages
& rMap
= lcl_getStorageMap();
337 TStorages::iterator aFind
= rMap
.find(jstring2ustring(env
,key
));
338 OSL_ENSURE(aFind
!= rMap
.end(),"Storage could not be found in list!");
339 if ( aFind
!= rMap
.end() )
341 TStreamMap::iterator aStreamFind
= aFind
->second
.second
.find(removeURLPrefix(jstring2ustring(env
,name
),aFind
->second
.first
.second
));
342 if ( aStreamFind
!= aFind
->second
.second
.end() )
343 pRet
= aStreamFind
->second
;
348 // -----------------------------------------------------------------------------
349 void StorageContainer::throwJavaException(const Exception
& _aException
,JNIEnv
* env
)
351 if (JNI_FALSE
!= env
->ExceptionCheck())
352 env
->ExceptionClear();
353 ::rtl::OString
cstr( ::rtl::OUStringToOString(_aException
.Message
, RTL_TEXTENCODING_JAVA_UTF8
) );
354 OSL_TRACE( __FILE__
": forwarding Exception: %s", cstr
.getStr() );
355 env
->ThrowNew(env
->FindClass("java/io/IOException"), cstr
.getStr());
357 //........................................................................
358 } // namespace hsqldb
359 //........................................................................
360 //........................................................................
362 // namespace connectivity
363 //........................................................................