1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 // MARKER(update_precomp.py): autogen include statement, do not remove
23 #include "precompiled_ucb.hxx"
24 
25 #include <hash_map>
26 #include <vector>
27 #include <string.h>
28 #include <rtl/string.h>
29 #include "comphelper/sequence.hxx"
30 #include "ucbhelper/simplecertificatevalidationrequest.hxx"
31 
32 #include "AprEnv.hxx"
33 #include <apr_strings.h>
34 
35 #include "DAVAuthListener.hxx"
36 #include "SerfTypes.hxx"
37 #include "SerfSession.hxx"
38 #include "SerfUri.hxx"
39 #include "SerfRequestProcessor.hxx"
40 #include "SerfCallbacks.hxx"
41 #include "SerfInputStream.hxx"
42 #include "UCBDeadPropertyValue.hxx"
43 
44 #include <com/sun/star/xml/crypto/XSecurityEnvironment.hpp>
45 #include <com/sun/star/security/XCertificate.hpp>
46 #include <com/sun/star/security/CertificateValidity.hpp>
47 #include <com/sun/star/security/CertificateContainerStatus.hpp>
48 #include <com/sun/star/security/CertificateContainer.hpp>
49 #include <com/sun/star/security/XCertificateContainer.hpp>
50 #include <com/sun/star/ucb/Lock.hpp>
51 #include <com/sun/star/xml/crypto/XSEInitializer.hpp>
52 
53 using namespace com::sun::star;
54 using namespace http_dav_ucp;
55 
56 
57 // -------------------------------------------------------------------
58 // static members!
59 //SerfLockStore SerfSession::m_aSerfLockStore;
60 
61 // -------------------------------------------------------------------
62 // Constructor
63 // -------------------------------------------------------------------
64 SerfSession::SerfSession(
65         const rtl::Reference< DAVSessionFactory > & rSessionFactory,
66         const rtl::OUString& inUri,
67         const ucbhelper::InternetProxyDecider & rProxyDecider )
68     throw ( DAVException )
69     : DAVSession( rSessionFactory )
70     , m_aMutex()
71     , m_aUri( inUri )
72     , m_aProxyName()
73     , m_nProxyPort( 0 )
74     , m_pSerfConnection( 0 )
75     , m_pSerfContext( 0 )
76     , m_bIsHeadRequestInProgress( false )
77     , m_bUseChunkedEncoding( false )
78     , m_bNoOfTransferEncodingSwitches( 0 )
79     , m_rProxyDecider( rProxyDecider )
80     , m_aEnv()
81 {
82     m_pSerfContext = serf_context_create( getAprPool() );
83 
84     m_pSerfBucket_Alloc = serf_bucket_allocator_create( getAprPool(), NULL, NULL );
85 }
86 
87 // -------------------------------------------------------------------
88 // Destructor
89 // -------------------------------------------------------------------
90 SerfSession::~SerfSession( )
91 {
92     if ( m_pSerfConnection )
93     {
94         serf_connection_close( m_pSerfConnection );
95         m_pSerfConnection = 0;
96     }
97 }
98 
99 // -------------------------------------------------------------------
100 void SerfSession::Init( const DAVRequestEnvironment & rEnv )
101   throw ( DAVException )
102 {
103     osl::Guard< osl::Mutex > theGuard( m_aMutex );
104     m_aEnv = rEnv;
105     Init();
106 }
107 
108 // -------------------------------------------------------------------
109 void SerfSession::Init()
110     throw ( DAVException )
111 {
112     osl::Guard< osl::Mutex > theGuard( m_aMutex );
113 
114     bool bCreateNewSession = false;
115 
116     if ( m_pSerfConnection == 0 )
117     {
118         const ucbhelper::InternetProxyServer & rProxyCfg = getProxySettings();
119 
120         m_aProxyName = rProxyCfg.aName;
121         m_nProxyPort = rProxyCfg.nPort;
122 
123         // Not yet initialized. Create new session.
124         bCreateNewSession = true;
125     }
126     else
127     {
128         const ucbhelper::InternetProxyServer & rProxyCfg = getProxySettings();
129 
130         if ( ( rProxyCfg.aName != m_aProxyName )
131              || ( rProxyCfg.nPort != m_nProxyPort ) )
132         {
133             m_aProxyName = rProxyCfg.aName;
134             m_nProxyPort = rProxyCfg.nPort;
135 
136             // new session needed, destroy old first
137             serf_connection_close( m_pSerfConnection );
138             m_pSerfConnection = 0;
139             bCreateNewSession = true;
140         }
141     }
142 
143     if ( bCreateNewSession )
144     {
145         // TODO - close_connection callback
146         apr_status_t status = serf_connection_create2( &m_pSerfConnection,
147                                                        m_pSerfContext,
148                                                        *(m_aUri.getAprUri()),
149                                                        Serf_ConnectSetup, this,
150                                                        0 /* close connection callback */, 0 /* close connection baton */,
151                                                        getAprPool() );
152 
153         if ( m_pSerfConnection == 0 ||status != APR_SUCCESS )
154         {
155             throw DAVException( DAVException::DAV_SESSION_CREATE,
156                                 SerfUri::makeConnectionEndPointString( m_aUri.GetHost(), m_aUri.GetPort() ) );
157         }
158 
159         // Register the session with the lock store
160 //        m_aSerfLockStore.registerSession( m_pSerfConnection );
161 
162         if ( m_aProxyName.getLength() )
163         {
164             apr_sockaddr_t *proxy_address = NULL;
165             status = apr_sockaddr_info_get( &proxy_address,
166                                                                rtl::OUStringToOString( m_aProxyName, RTL_TEXTENCODING_UTF8 ).getStr(),
167                                                                APR_UNSPEC,
168                                                                static_cast<apr_port_t>(m_nProxyPort),
169                                                                0, getAprPool() );
170 
171             if ( status != APR_SUCCESS )
172             {
173                 throw DAVException( DAVException::DAV_SESSION_CREATE,
174                                     SerfUri::makeConnectionEndPointString( m_aUri.GetHost(), m_aUri.GetPort() ) );
175             }
176 
177             serf_config_proxy( m_pSerfContext, proxy_address );
178         }
179 
180 
181         serf_config_credentials_callback( m_pSerfContext, Serf_Credentials );
182 
183         m_bUseChunkedEncoding = isSSLNeeded();
184     }
185 }
186 
187 apr_pool_t* SerfSession::getAprPool()
188 {
189     return apr_environment::AprEnv::getAprEnv()->getAprPool();
190 }
191 
192 serf_bucket_alloc_t* SerfSession::getSerfBktAlloc()
193 {
194     return m_pSerfBucket_Alloc;
195 }
196 
197 serf_context_t* SerfSession::getSerfContext()
198 {
199     return m_pSerfContext;
200 }
201 
202 SerfConnection* SerfSession::getSerfConnection()
203 {
204     return m_pSerfConnection;
205 }
206 
207 bool SerfSession::isHeadRequestInProgress()
208 {
209     return m_bIsHeadRequestInProgress;
210 }
211 
212 bool SerfSession::isSSLNeeded()
213 {
214     return m_aUri.GetScheme().equalsIgnoreAsciiCase( rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "https" ) ) );
215 }
216 
217 char* SerfSession::getHostinfo()
218 {
219     return m_aUri.getAprUri()->hostinfo;
220 }
221 
222 
223 // -------------------------------------------------------------------
224 // virtual
225 sal_Bool SerfSession::CanUse( const rtl::OUString & inUri )
226 {
227     try
228     {
229         SerfUri theUri( inUri );
230         if ( ( theUri.GetPort() == m_aUri.GetPort() ) &&
231              ( theUri.GetHost() == m_aUri.GetHost() ) &&
232              ( theUri.GetScheme() == m_aUri.GetScheme() ) )
233         {
234             return sal_True;
235         }
236     }
237     catch ( DAVException const & )
238     {
239         return sal_False;
240     }
241     return sal_False;
242 }
243 
244 // -------------------------------------------------------------------
245 // virtual
246 sal_Bool SerfSession::UsesProxy()
247 {
248     Init();
249     return ( m_aProxyName.getLength() > 0 );
250 }
251 
252 apr_status_t SerfSession::setupSerfConnection( apr_socket_t * inAprSocket,
253                                                serf_bucket_t **outSerfInputBucket,
254                                                serf_bucket_t **outSerfOutputBucket,
255                                                apr_pool_t* /*inAprPool*/ )
256 {
257     serf_bucket_t *tmpInputBkt;
258     tmpInputBkt = serf_context_bucket_socket_create( getSerfContext(),
259                                                      inAprSocket,
260                                                      getSerfBktAlloc() );
261 
262     if ( isSSLNeeded() )
263     {
264         tmpInputBkt = serf_bucket_ssl_decrypt_create( tmpInputBkt,
265                                                       0,
266                                                       getSerfBktAlloc() );
267         /** Set the callback that is called to authenticate the
268             certifcate (chain).
269         */
270         serf_ssl_server_cert_chain_callback_set(
271             serf_bucket_ssl_decrypt_context_get(tmpInputBkt),
272             NULL,
273             Serf_CertificateChainValidation,
274             this);
275         serf_ssl_set_hostname( serf_bucket_ssl_decrypt_context_get( tmpInputBkt ),
276                                getHostinfo() );
277 
278         *outSerfOutputBucket = serf_bucket_ssl_encrypt_create( *outSerfOutputBucket,
279                                                                serf_bucket_ssl_decrypt_context_get( tmpInputBkt ),
280                                                                getSerfBktAlloc() );
281     }
282 
283     *outSerfInputBucket = tmpInputBkt;
284 
285     return APR_SUCCESS;
286 }
287 
288 apr_status_t SerfSession::provideSerfCredentials( bool bGiveProvidedCredentialsASecondTry,
289                                                   char ** outUsername,
290                                                   char ** outPassword,
291                                                   serf_request_t * /*inRequest*/,
292                                                   int /*inCode*/,
293                                                   const char *inAuthProtocol,
294                                                   const char *inRealm,
295                                                   apr_pool_t *inAprPool )
296 {
297     DAVAuthListener * pListener = getRequestEnvironment().m_xAuthListener.get();
298     if ( !pListener )
299     {
300         // abort
301         return SERF_ERROR_AUTHN_FAILED;
302     }
303 
304     rtl::OUString theUserName;
305     rtl::OUString thePassWord;
306     try
307     {
308         SerfUri uri( getRequestEnvironment().m_aRequestURI );
309         rtl::OUString aUserInfo( uri.GetUserInfo() );
310         if ( aUserInfo.getLength() )
311         {
312             sal_Int32 nPos = aUserInfo.indexOf( '@' );
313             if ( nPos == -1 )
314             {
315                 theUserName = aUserInfo;
316             }
317             else
318             {
319                 theUserName = aUserInfo.copy( 0, nPos );
320                 thePassWord = aUserInfo.copy( nPos + 1 );
321             }
322         }
323     }
324     catch ( DAVException const & )
325     {
326         // abort
327         return SERF_ERROR_AUTHN_FAILED;
328     }
329 
330     const bool bCanUseSystemCreds = ( ( strcasecmp( inAuthProtocol, "NTLM" ) == 0 ) ||
331                                       ( strcasecmp( inAuthProtocol, "Negotiate" ) == 0 ) );
332 
333     int theRetVal = pListener->authenticate( rtl::OUString::createFromAscii( inRealm ),
334                                              getHostName(),
335                                              theUserName,
336                                              thePassWord,
337                                              bCanUseSystemCreds,
338                                              bGiveProvidedCredentialsASecondTry ? sal_False : sal_True );
339 
340     if ( theRetVal == 0 )
341     {
342         *outUsername = apr_pstrdup( inAprPool, rtl::OUStringToOString( theUserName, RTL_TEXTENCODING_UTF8 ).getStr() );
343         *outPassword = apr_pstrdup( inAprPool, rtl::OUStringToOString( thePassWord, RTL_TEXTENCODING_UTF8 ).getStr() );
344     }
345 
346     return theRetVal != 0 ? SERF_ERROR_AUTHN_FAILED : APR_SUCCESS;
347 }
348 
349 namespace {
350     // -------------------------------------------------------------------
351     // Helper function
352     ::rtl::OUString GetHostnamePart( const ::rtl::OUString& _rRawString )
353     {
354         ::rtl::OUString sPart;
355         ::rtl::OUString sPartId = ::rtl::OUString::createFromAscii( "CN=" );
356         sal_Int32 nContStart = _rRawString.indexOf( sPartId );
357         if ( nContStart != -1 )
358         {
359             nContStart = nContStart + sPartId.getLength();
360             sal_Int32 nContEnd
361                 = _rRawString.indexOf( sal_Unicode( ',' ), nContStart );
362             sPart = _rRawString.copy( nContStart, nContEnd - nContStart );
363         }
364         return sPart;
365     }
366 } // namespace
367 
368 
369 apr_status_t SerfSession::verifySerfCertificateChain (
370     int,
371     const serf_ssl_certificate_t * const * pCertificateChainBase64Encoded,
372     int nCertificateChainLength)
373 {
374     // Check arguments.
375     if (pCertificateChainBase64Encoded == NULL || nCertificateChainLength<=0)
376     {
377         OSL_ASSERT(pCertificateChainBase64Encoded != NULL);
378         OSL_ASSERT(nCertificateChainLength>0);
379         return SERF_SSL_CERT_UNKNOWN_FAILURE;
380     }
381 
382     // Create some crypto objects to decode and handle the base64
383     // encoded certificate chain.
384     uno::Reference< xml::crypto::XSEInitializer > xSEInitializer;
385     uno::Reference< security::XCertificateContainer > xCertificateContainer;
386     uno::Reference< xml::crypto::XXMLSecurityContext > xSecurityContext;
387     uno::Reference< xml::crypto::XSecurityEnvironment > xSecurityEnv;
388     try
389     {
390         // Create a certificate container.
391         xCertificateContainer = uno::Reference< security::XCertificateContainer >(
392             getMSF()->createInstance(
393                 rtl::OUString::createFromAscii(
394                     "com.sun.star.security.CertificateContainer" ) ),
395             uno::UNO_QUERY_THROW);
396 
397         xSEInitializer = uno::Reference< xml::crypto::XSEInitializer >(
398             getMSF()->createInstance(
399                 rtl::OUString::createFromAscii( "com.sun.star.xml.crypto.SEInitializer" ) ),
400             uno::UNO_QUERY_THROW);
401 
402         xSecurityContext = xSEInitializer->createSecurityContext( rtl::OUString() );
403         if (xSecurityContext.is())
404             xSecurityEnv = xSecurityContext->getSecurityEnvironment();
405 
406         if ( ! xSecurityContext.is() || ! xSecurityEnv.is())
407         {
408             // Do we have to dispose xSEInitializer or xCertificateContainer?
409             return SERF_SSL_CERT_UNKNOWN_FAILURE;
410         }
411     }
412     catch ( uno::Exception const &)
413     {
414         return SERF_SSL_CERT_UNKNOWN_FAILURE;
415     }
416 
417     // Decode the server certificate.
418     const char* sBase64EncodedServerCertificate (
419         serf_ssl_cert_export(
420             pCertificateChainBase64Encoded[0],
421             getAprPool()));
422     uno::Reference< security::XCertificate > xServerCertificate(
423         xSecurityEnv->createCertificateFromAscii(
424             rtl::OUString::createFromAscii(sBase64EncodedServerCertificate)));
425     if ( ! xServerCertificate.is())
426         return SERF_SSL_CERT_UNKNOWN_FAILURE;
427 
428     // Get the subject from the server certificate.
429     ::rtl::OUString sServerCertificateSubject (xServerCertificate->getSubjectName());
430     sal_Int32 nIndex = 0;
431     while (nIndex >= 0)
432     {
433         const ::rtl::OUString sToken (sServerCertificateSubject.getToken(0, ',', nIndex));
434         if (sToken.compareToAscii("CN=", 3) == 0)
435         {
436             sServerCertificateSubject = sToken.copy(3);
437             break;
438         }
439         else if (sToken.compareToAscii(" CN=", 4) == 0)
440         {
441             sServerCertificateSubject = sToken.copy(4);
442             break;
443         }
444     }
445 
446     // When the certificate container already contains a (trusted)
447     // entry for the server then we do not have to authenticate any
448     // certificate.
449     const security::CertificateContainerStatus eStatus (
450         xCertificateContainer->hasCertificate(
451             getHostName(), sServerCertificateSubject ) );
452     if (eStatus != security::CertificateContainerStatus_NOCERT)
453     {
454         return eStatus == security::CertificateContainerStatus_TRUSTED
455                ? APR_SUCCESS
456                : SERF_SSL_CERT_UNKNOWN_FAILURE;
457     }
458 
459     // The shortcut failed, so try to verify the whole chain.  This is
460     // done outside the isDomainMatch() block because the result is
461     // used by the interaction handler.
462     std::vector< uno::Reference< security::XCertificate > > aChain;
463     for (nIndex=1; nIndex<nCertificateChainLength; ++nIndex)
464     {
465         const char* sBase64EncodedCertificate (
466             serf_ssl_cert_export(
467                 pCertificateChainBase64Encoded[nIndex],
468                 getAprPool()));
469         uno::Reference< security::XCertificate > xCertificate(
470             xSecurityEnv->createCertificateFromAscii(
471                 rtl::OUString::createFromAscii(sBase64EncodedCertificate)));
472         if ( ! xCertificate.is())
473             return SERF_SSL_CERT_UNKNOWN_FAILURE;
474         aChain.push_back(xCertificate);
475     }
476     const sal_Int64 nVerificationResult (xSecurityEnv->verifyCertificate(
477             xServerCertificate,
478             ::comphelper::containerToSequence(aChain)));
479 
480     // When the certificate matches the host name then we can use the
481     // result of the verification.
482     if (isDomainMatch(sServerCertificateSubject))
483     {
484 
485         if (nVerificationResult == 0)
486         {
487             // Certificate (chain) is valid.
488             xCertificateContainer->addCertificate(getHostName(), sServerCertificateSubject,  sal_True);
489             return APR_SUCCESS;
490         }
491         else if ((nVerificationResult & security::CertificateValidity::CHAIN_INCOMPLETE) != 0)
492         {
493             // We do not have enough information for verification,
494             // neither automatically (as we just discovered) nor
495             // manually (so there is no point in showing any dialog.)
496             return SERF_SSL_CERT_UNKNOWN_FAILURE;
497         }
498         else if ((nVerificationResult &
499                 (security::CertificateValidity::INVALID | security::CertificateValidity::REVOKED)) != 0)
500         {
501             // Certificate (chain) is invalid.
502             xCertificateContainer->addCertificate(getHostName(), sServerCertificateSubject,  sal_False);
503             return SERF_SSL_CERT_UNKNOWN_FAILURE;
504         }
505         else
506         {
507             // For all other we have to ask the user.
508         }
509     }
510 
511     // We have not been able to automatically verify (or falsify) the
512     // certificate chain.  To resolve this we have to ask the user.
513     const uno::Reference< ucb::XCommandEnvironment > xEnv( getRequestEnvironment().m_xEnv );
514     if ( xEnv.is() )
515     {
516         uno::Reference< task::XInteractionHandler > xIH( xEnv->getInteractionHandler() );
517         if ( xIH.is() )
518         {
519             rtl::Reference< ucbhelper::SimpleCertificateValidationRequest >
520                 xRequest( new ucbhelper::SimpleCertificateValidationRequest(
521                         static_cast<sal_Int32>(nVerificationResult), xServerCertificate, getHostName() ) );
522             xIH->handle( xRequest.get() );
523 
524             rtl::Reference< ucbhelper::InteractionContinuation > xSelection
525                 = xRequest->getSelection();
526 
527             if ( xSelection.is() )
528             {
529                 uno::Reference< task::XInteractionApprove > xApprove(
530                     xSelection.get(), uno::UNO_QUERY );
531                 if ( xApprove.is() )
532                 {
533                     xCertificateContainer->addCertificate( getHostName(), sServerCertificateSubject,  sal_True );
534                     return APR_SUCCESS;
535                 }
536                 else
537                 {
538                     // Don't trust cert
539                     xCertificateContainer->addCertificate( getHostName(), sServerCertificateSubject, sal_False );
540                     return SERF_SSL_CERT_UNKNOWN_FAILURE;
541                 }
542             }
543         }
544         else
545         {
546             // Don't trust cert
547             xCertificateContainer->addCertificate( getHostName(), sServerCertificateSubject, sal_False );
548             return SERF_SSL_CERT_UNKNOWN_FAILURE;
549         }
550     }
551 
552     return SERF_SSL_CERT_UNKNOWN_FAILURE;
553 }
554 
555 serf_bucket_t* SerfSession::acceptSerfResponse( serf_request_t * inSerfRequest,
556                                                 serf_bucket_t * inSerfStreamBucket,
557                                                 apr_pool_t* /*inAprPool*/ )
558 {
559     // get the per-request bucket allocator
560     serf_bucket_alloc_t* SerfBktAlloc = serf_request_get_alloc( inSerfRequest );
561 
562     // create a barrier bucket so the response doesn't eat us!
563     serf_bucket_t *responseBkt = serf_bucket_barrier_create( inSerfStreamBucket,
564                                                              SerfBktAlloc );
565 
566     // create response bucket
567     responseBkt = serf_bucket_response_create( responseBkt,
568                                                SerfBktAlloc );
569 
570     if ( isHeadRequestInProgress() )
571     {
572         // advise the response bucket that this was from a HEAD request and that it should not expect to see a response body.
573         serf_bucket_response_set_head( responseBkt );
574     }
575 
576     return responseBkt;
577 }
578 
579 SerfRequestProcessor* SerfSession::createReqProc( const rtl::OUString & inPath )
580 {
581     return new SerfRequestProcessor( *this,
582                                      inPath,
583                                      m_bUseChunkedEncoding );
584 }
585 
586 // -------------------------------------------------------------------
587 // PROPFIND - allprop & named
588 // -------------------------------------------------------------------
589 void SerfSession::PROPFIND( const rtl::OUString & inPath,
590                             const Depth inDepth,
591                             const std::vector< rtl::OUString > & inPropNames,
592                             std::vector< DAVResource > & ioResources,
593                             const DAVRequestEnvironment & rEnv )
594     throw ( DAVException )
595 {
596     osl::Guard< osl::Mutex > theGuard( m_aMutex );
597 
598     Init( rEnv );
599 
600     apr_status_t status = APR_SUCCESS;
601     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
602     aReqProc->processPropFind( inDepth,
603                                inPropNames,
604                                ioResources,
605                                status );
606 
607     if ( status == APR_SUCCESS &&
608          aReqProc->mpDAVException == 0 &&
609          ioResources.empty() )
610     {
611         m_aEnv = DAVRequestEnvironment();
612         throw DAVException( DAVException::DAV_HTTP_ERROR, inPath, (sal_uInt16)APR_EGENERAL );
613     }
614     HandleError( aReqProc );
615 }
616 
617 // -------------------------------------------------------------------
618 // PROPFIND - propnames
619 // -------------------------------------------------------------------
620 void SerfSession::PROPFIND( const rtl::OUString & inPath,
621                             const Depth inDepth,
622                             std::vector< DAVResourceInfo > & ioResInfo,
623                             const DAVRequestEnvironment & rEnv )
624     throw( DAVException )
625 {
626     osl::Guard< osl::Mutex > theGuard( m_aMutex );
627 
628     Init( rEnv );
629 
630     apr_status_t status = APR_SUCCESS;
631     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
632     aReqProc->processPropFind( inDepth,
633                                ioResInfo,
634                                status );
635 
636     if ( status == APR_SUCCESS &&
637          aReqProc->mpDAVException == 0 &&
638          ioResInfo.empty() )
639     {
640         m_aEnv = DAVRequestEnvironment();
641         throw DAVException( DAVException::DAV_HTTP_ERROR, inPath, (sal_uInt16)APR_EGENERAL );
642     }
643     HandleError( aReqProc );
644 }
645 
646 // -------------------------------------------------------------------
647 // PROPPATCH
648 // -------------------------------------------------------------------
649 void SerfSession::PROPPATCH( const rtl::OUString & inPath,
650                              const std::vector< ProppatchValue > & inValues,
651                              const DAVRequestEnvironment & rEnv )
652     throw( DAVException )
653 {
654     osl::Guard< osl::Mutex > theGuard( m_aMutex );
655 
656     Init( rEnv );
657 
658     apr_status_t status = APR_SUCCESS;
659     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
660     aReqProc->processPropPatch( inValues,
661                                 status );
662 
663     HandleError( aReqProc );
664 }
665 
666 // -------------------------------------------------------------------
667 // HEAD
668 // -------------------------------------------------------------------
669 void SerfSession::HEAD( const ::rtl::OUString & inPath,
670                         const std::vector< ::rtl::OUString > & inHeaderNames,
671                         DAVResource & ioResource,
672                         const DAVRequestEnvironment & rEnv )
673     throw( DAVException )
674 {
675     osl::Guard< osl::Mutex > theGuard( m_aMutex );
676 
677     Init( rEnv );
678 
679     m_bIsHeadRequestInProgress = true;
680 
681     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
682     ioResource.uri = inPath;
683     ioResource.properties.clear();
684     apr_status_t status = APR_SUCCESS;
685     aReqProc->processHead( inHeaderNames,
686                            ioResource,
687                            status );
688 
689     m_bIsHeadRequestInProgress = false;
690 
691     HandleError( aReqProc );
692 }
693 
694 // -------------------------------------------------------------------
695 // GET
696 // -------------------------------------------------------------------
697 uno::Reference< io::XInputStream >
698 SerfSession::GET( const rtl::OUString & inPath,
699                   const DAVRequestEnvironment & rEnv )
700     throw ( DAVException )
701 {
702     osl::Guard< osl::Mutex > theGuard( m_aMutex );
703 
704     Init( rEnv );
705 
706     uno::Reference< SerfInputStream > xInputStream( new SerfInputStream );
707     apr_status_t status = APR_SUCCESS;
708     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
709     aReqProc->processGet( xInputStream,
710                           status );
711 
712     HandleError( aReqProc );
713 
714     return uno::Reference< io::XInputStream >( xInputStream.get() );
715 }
716 
717 // -------------------------------------------------------------------
718 // GET
719 // -------------------------------------------------------------------
720 void SerfSession::GET( const rtl::OUString & inPath,
721                        uno::Reference< io::XOutputStream > & ioOutputStream,
722                        const DAVRequestEnvironment & rEnv )
723     throw ( DAVException )
724 {
725     osl::Guard< osl::Mutex > theGuard( m_aMutex );
726 
727     Init( rEnv );
728 
729     apr_status_t status = APR_SUCCESS;
730     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
731     aReqProc->processGet( ioOutputStream,
732                           status );
733 
734     HandleError( aReqProc );
735 }
736 
737 // -------------------------------------------------------------------
738 // GET
739 // -------------------------------------------------------------------
740 uno::Reference< io::XInputStream >
741 SerfSession::GET( const rtl::OUString & inPath,
742                   const std::vector< ::rtl::OUString > & inHeaderNames,
743                   DAVResource & ioResource,
744                   const DAVRequestEnvironment & rEnv )
745     throw ( DAVException )
746 {
747     osl::Guard< osl::Mutex > theGuard( m_aMutex );
748 
749     Init( rEnv );
750 
751     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
752     uno::Reference< SerfInputStream > xInputStream( new SerfInputStream );
753     ioResource.uri = inPath;
754     ioResource.properties.clear();
755     apr_status_t status = APR_SUCCESS;
756     aReqProc->processGet( xInputStream,
757                           inHeaderNames,
758                           ioResource,
759                           status );
760 
761     HandleError( aReqProc );
762 
763     return uno::Reference< io::XInputStream >( xInputStream.get() );
764 }
765 
766 
767 // -------------------------------------------------------------------
768 // GET
769 // -------------------------------------------------------------------
770 void SerfSession::GET( const rtl::OUString & inPath,
771                        uno::Reference< io::XOutputStream > & ioOutputStream,
772                        const std::vector< ::rtl::OUString > & inHeaderNames,
773                        DAVResource & ioResource,
774                        const DAVRequestEnvironment & rEnv )
775     throw ( DAVException )
776 {
777     osl::Guard< osl::Mutex > theGuard( m_aMutex );
778 
779     Init( rEnv );
780 
781     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
782     ioResource.uri = inPath;
783     ioResource.properties.clear();
784     apr_status_t status = APR_SUCCESS;
785     aReqProc->processGet( ioOutputStream,
786                           inHeaderNames,
787                           ioResource,
788                           status );
789 
790     HandleError( aReqProc );
791 }
792 
793 // -------------------------------------------------------------------
794 // PUT
795 // -------------------------------------------------------------------
796 void SerfSession::PUT( const rtl::OUString & inPath,
797                        const uno::Reference< io::XInputStream > & inInputStream,
798                        const DAVRequestEnvironment & rEnv )
799     throw ( DAVException )
800 {
801     osl::Guard< osl::Mutex > theGuard( m_aMutex );
802 
803     Init( rEnv );
804 
805     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
806     uno::Sequence< sal_Int8 > aDataToSend;
807     if ( !getDataFromInputStream( inInputStream, aDataToSend, false ) )
808         throw DAVException( DAVException::DAV_INVALID_ARG );
809     apr_status_t status = APR_SUCCESS;
810     aReqProc->processPut( reinterpret_cast< const char * >( aDataToSend.getConstArray() ),
811                           aDataToSend.getLength(),
812                           status );
813 
814     HandleError( aReqProc );
815 }
816 
817 // -------------------------------------------------------------------
818 // POST
819 // -------------------------------------------------------------------
820 uno::Reference< io::XInputStream >
821 SerfSession::POST( const rtl::OUString & inPath,
822                    const rtl::OUString & rContentType,
823                    const rtl::OUString & rReferer,
824                    const uno::Reference< io::XInputStream > & inInputStream,
825                    const DAVRequestEnvironment & rEnv )
826     throw ( DAVException )
827 {
828     osl::Guard< osl::Mutex > theGuard( m_aMutex );
829 
830     uno::Sequence< sal_Int8 > aDataToSend;
831     if ( !getDataFromInputStream( inInputStream, aDataToSend, true ) )
832     {
833         throw DAVException( DAVException::DAV_INVALID_ARG );
834     }
835 
836     Init( rEnv );
837 
838     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
839     uno::Reference< SerfInputStream > xInputStream( new SerfInputStream );
840     apr_status_t status = APR_SUCCESS;
841     aReqProc->processPost( reinterpret_cast< const char * >( aDataToSend.getConstArray() ),
842                            aDataToSend.getLength(),
843                            rContentType,
844                            rReferer,
845                            xInputStream,
846                            status );
847 
848     HandleError( aReqProc );
849     return uno::Reference< io::XInputStream >( xInputStream.get() );
850 }
851 
852 // -------------------------------------------------------------------
853 // POST
854 // -------------------------------------------------------------------
855 void SerfSession::POST( const rtl::OUString & inPath,
856                         const rtl::OUString & rContentType,
857                         const rtl::OUString & rReferer,
858                         const uno::Reference< io::XInputStream > & inInputStream,
859                         uno::Reference< io::XOutputStream > & oOutputStream,
860                         const DAVRequestEnvironment & rEnv )
861     throw ( DAVException )
862 {
863     osl::Guard< osl::Mutex > theGuard( m_aMutex );
864 
865     uno::Sequence< sal_Int8 > aDataToSend;
866     if ( !getDataFromInputStream( inInputStream, aDataToSend, true ) )
867     {
868         throw DAVException( DAVException::DAV_INVALID_ARG );
869     }
870 
871     Init( rEnv );
872 
873     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
874     apr_status_t status = APR_SUCCESS;
875     aReqProc->processPost( reinterpret_cast< const char * >( aDataToSend.getConstArray() ),
876                            aDataToSend.getLength(),
877                            rContentType,
878                            rReferer,
879                            oOutputStream,
880                            status );
881 
882     HandleError( aReqProc );
883 }
884 
885 // -------------------------------------------------------------------
886 // MKCOL
887 // -------------------------------------------------------------------
888 void SerfSession::MKCOL( const rtl::OUString & inPath,
889                          const DAVRequestEnvironment & rEnv )
890     throw ( DAVException )
891 {
892     osl::Guard< osl::Mutex > theGuard( m_aMutex );
893 
894     Init( rEnv );
895 
896     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
897     apr_status_t status = APR_SUCCESS;
898     aReqProc->processMkCol( status );
899 
900     HandleError( aReqProc );
901 }
902 
903 // -------------------------------------------------------------------
904 // COPY
905 // -------------------------------------------------------------------
906 void SerfSession::COPY( const rtl::OUString & inSourceURL,
907                         const rtl::OUString & inDestinationURL,
908                         const DAVRequestEnvironment & rEnv,
909                         sal_Bool inOverWrite )
910     throw ( DAVException )
911 {
912     osl::Guard< osl::Mutex > theGuard( m_aMutex );
913 
914     Init( rEnv );
915 
916     SerfUri theSourceUri( inSourceURL );
917     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( theSourceUri.GetPath() ) );
918     apr_status_t status = APR_SUCCESS;
919     aReqProc->processCopy( inDestinationURL,
920                            (inOverWrite ? true : false),
921                            status );
922 
923     HandleError( aReqProc );
924 }
925 
926 // -------------------------------------------------------------------
927 // MOVE
928 // -------------------------------------------------------------------
929 void SerfSession::MOVE( const rtl::OUString & inSourceURL,
930                         const rtl::OUString & inDestinationURL,
931                         const DAVRequestEnvironment & rEnv,
932                         sal_Bool inOverWrite )
933     throw ( DAVException )
934 {
935     osl::Guard< osl::Mutex > theGuard( m_aMutex );
936 
937     Init( rEnv );
938 
939     SerfUri theSourceUri( inSourceURL );
940     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( theSourceUri.GetPath() ) );
941     apr_status_t status = APR_SUCCESS;
942     aReqProc->processMove( inDestinationURL,
943                            (inOverWrite ? true : false),
944                            status );
945 
946     HandleError( aReqProc );
947 }
948 
949 // -------------------------------------------------------------------
950 // DESTROY
951 // -------------------------------------------------------------------
952 void SerfSession::DESTROY( const rtl::OUString & inPath,
953                            const DAVRequestEnvironment & rEnv )
954     throw ( DAVException )
955 {
956     osl::Guard< osl::Mutex > theGuard( m_aMutex );
957 
958     Init( rEnv );
959 
960     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
961     apr_status_t status = APR_SUCCESS;
962     aReqProc->processDelete( status );
963 
964     HandleError( aReqProc );
965 }
966 
967 // -------------------------------------------------------------------
968 /*
969 namespace
970 {
971     sal_Int32 lastChanceToSendRefreshRequest( TimeValue const & rStart,
972                                               int timeout )
973     {
974         TimeValue aEnd;
975         osl_getSystemTime( &aEnd );
976 
977         // Try to estimate a safe absolute time for sending the
978         // lock refresh request.
979         sal_Int32 lastChanceToSendRefreshRequest = -1;
980         if ( timeout != NE_TIMEOUT_INFINITE )
981         {
982             sal_Int32 calltime = aEnd.Seconds - rStart.Seconds;
983             if ( calltime <= timeout )
984             {
985                 lastChanceToSendRefreshRequest
986                     = aEnd.Seconds + timeout - calltime;
987             }
988             else
989             {
990                 OSL_TRACE( "No chance to refresh lock before timeout!" );
991             }
992         }
993         return lastChanceToSendRefreshRequest;
994     }
995 
996 } // namespace
997 */
998 // -------------------------------------------------------------------
999 // LOCK (set new lock)
1000 // -------------------------------------------------------------------
1001 void SerfSession::LOCK( const ::rtl::OUString & inPath,
1002                         ucb::Lock & /*rLock*/,
1003                         const DAVRequestEnvironment & rEnv )
1004     throw ( DAVException )
1005 {
1006     osl::Guard< osl::Mutex > theGuard( m_aMutex );
1007 
1008     Init( rEnv );
1009 
1010     boost::shared_ptr<SerfRequestProcessor> aReqProc( createReqProc( inPath ) );
1011     HandleError( aReqProc );
1012     /* Create a depth zero, exclusive write lock, with default timeout
1013      * (allowing a server to pick a default).  token, owner and uri are
1014      * unset. */
1015     /*
1016     SerfLock * theLock = ne_lock_create();
1017 
1018     // Set the lock uri
1019     ne_uri aUri;
1020     ne_uri_parse( rtl::OUStringToOString( makeAbsoluteURL( inPath ),
1021                                           RTL_TEXTENCODING_UTF8 ).getStr(),
1022                   &aUri );
1023     theLock->uri = aUri;
1024 
1025     // Set the lock depth
1026     switch( rLock.Depth )
1027     {
1028     case ucb::LockDepth_ZERO:
1029         theLock->depth = NE_DEPTH_ZERO;
1030         break;
1031     case ucb::LockDepth_ONE:
1032         theLock->depth = NE_DEPTH_ONE;
1033         break;
1034     case ucb::LockDepth_INFINITY:
1035         theLock->depth = NE_DEPTH_INFINITE;
1036         break;
1037     default:
1038         throw DAVException( DAVException::DAV_INVALID_ARG );
1039     }
1040 
1041     // Set the lock scope
1042     switch ( rLock.Scope )
1043     {
1044     case ucb::LockScope_EXCLUSIVE:
1045         theLock->scope = ne_lockscope_exclusive;
1046         break;
1047     case ucb::LockScope_SHARED:
1048         theLock->scope = ne_lockscope_shared;
1049         break;
1050     default:
1051         throw DAVException( DAVException::DAV_INVALID_ARG );
1052     }
1053 
1054     // Set the lock timeout
1055     theLock->timeout = (long)rLock.Timeout;
1056 
1057     // Set the lock owner
1058     rtl::OUString aValue;
1059     rLock.Owner >>= aValue;
1060     theLock->owner =
1061         ne_strdup( rtl::OUStringToOString( aValue,
1062                                            RTL_TEXTENCODING_UTF8 ).getStr() );
1063     TimeValue startCall;
1064     osl_getSystemTime( &startCall );
1065 
1066     int theRetVal = ne_lock( m_pHttpSession, theLock );
1067 
1068     if ( theRetVal == NE_OK )
1069     {
1070         m_aSerfLockStore.addLock( theLock,
1071                                   this,
1072                                   lastChanceToSendRefreshRequest(
1073                                       startCall, theLock->timeout ) );
1074 
1075         uno::Sequence< rtl::OUString > aTokens( 1 );
1076         aTokens[ 0 ] = rtl::OUString::createFromAscii( theLock->token );
1077         rLock.LockTokens = aTokens;
1078 
1079         OSL_TRACE( "SerfSession::LOCK: created lock for %s. token: %s",
1080                    rtl::OUStringToOString( makeAbsoluteURL( inPath ),
1081                                            RTL_TEXTENCODING_UTF8 ).getStr(),
1082                    theLock->token );
1083     }
1084     else
1085     {
1086         ne_lock_destroy( theLock );
1087 
1088         OSL_TRACE( "SerfSession::LOCK: obtaining lock for %s failed!",
1089                    rtl::OUStringToOString( makeAbsoluteURL( inPath ),
1090                                            RTL_TEXTENCODING_UTF8 ).getStr() );
1091     }
1092 
1093     HandleError( theRetVal, inPath, rEnv );
1094     */
1095 }
1096 
1097 // -------------------------------------------------------------------
1098 // LOCK (refresh existing lock)
1099 // -------------------------------------------------------------------
1100 sal_Int64 SerfSession::LOCK( const ::rtl::OUString & /*inPath*/,
1101                              sal_Int64 nTimeout,
1102                              const DAVRequestEnvironment & /*rEnv*/ )
1103     throw ( DAVException )
1104 {
1105     osl::Guard< osl::Mutex > theGuard( m_aMutex );
1106 
1107     return nTimeout;
1108     /*
1109     // Try to get the neon lock from lock store
1110     SerfLock * theLock
1111         = m_aSerfLockStore.findByUri( makeAbsoluteURL( inPath ) );
1112     if ( !theLock )
1113          throw DAVException( DAVException::DAV_NOT_LOCKED );
1114 
1115     Init( rEnv );
1116 
1117     // refresh existing lock.
1118     theLock->timeout = static_cast< long >( nTimeout );
1119 
1120     TimeValue startCall;
1121     osl_getSystemTime( &startCall );
1122 
1123     int theRetVal = ne_lock_refresh( m_pHttpSession, theLock );
1124 
1125     if ( theRetVal == NE_OK )
1126     {
1127         m_aSerfLockStore.updateLock( theLock,
1128                                      lastChanceToSendRefreshRequest(
1129                                          startCall, theLock->timeout ) );
1130     }
1131 
1132     HandleError( theRetVal, inPath, rEnv );
1133 
1134     return theLock->timeout;
1135     */
1136 }
1137 
1138 // -------------------------------------------------------------------
1139 // LOCK (refresh existing lock)
1140 // -------------------------------------------------------------------
1141 bool SerfSession::LOCK( SerfLock * /*pLock*/,
1142                         sal_Int32 & /*rlastChanceToSendRefreshRequest*/ )
1143 {
1144     osl::Guard< osl::Mutex > theGuard( m_aMutex );
1145 
1146     return true;
1147     /*
1148     // refresh existing lock.
1149 
1150     TimeValue startCall;
1151     osl_getSystemTime( &startCall );
1152 
1153     if ( ne_lock_refresh( m_pHttpSession, pLock ) == NE_OK )
1154     {
1155         rlastChanceToSendRefreshRequest
1156             = lastChanceToSendRefreshRequest( startCall, pLock->timeout );
1157 
1158         OSL_TRACE( "Lock successfully refreshed." );
1159         return true;
1160     }
1161     else
1162     {
1163         OSL_TRACE( "Lock not refreshed!" );
1164         return false;
1165     }
1166     */
1167 }
1168 
1169 // -------------------------------------------------------------------
1170 // UNLOCK
1171 // -------------------------------------------------------------------
1172 void SerfSession::UNLOCK( const ::rtl::OUString & /*inPath*/,
1173                           const DAVRequestEnvironment & /*rEnv*/ )
1174     throw ( DAVException )
1175 {
1176     osl::Guard< osl::Mutex > theGuard( m_aMutex );
1177 
1178     /*
1179     // get the neon lock from lock store
1180     SerfLock * theLock
1181         = m_aSerfLockStore.findByUri( makeAbsoluteURL( inPath ) );
1182     if ( !theLock )
1183         throw DAVException( DAVException::DAV_NOT_LOCKED );
1184 
1185     Init( rEnv );
1186 
1187     int theRetVal = ne_unlock( m_pHttpSession, theLock );
1188 
1189     if ( theRetVal == NE_OK )
1190     {
1191         m_aSerfLockStore.removeLock( theLock );
1192         ne_lock_destroy( theLock );
1193     }
1194     else
1195     {
1196         OSL_TRACE( "SerfSession::UNLOCK: unlocking of %s failed.",
1197                    rtl::OUStringToOString( makeAbsoluteURL( inPath ),
1198                                            RTL_TEXTENCODING_UTF8 ).getStr() );
1199     }
1200 
1201     HandleError( theRetVal, inPath, rEnv );
1202     */
1203 }
1204 
1205 // -------------------------------------------------------------------
1206 // UNLOCK
1207 // -------------------------------------------------------------------
1208 bool SerfSession::UNLOCK( SerfLock * /*pLock*/ )
1209 {
1210     osl::Guard< osl::Mutex > theGuard( m_aMutex );
1211 
1212     return true;
1213     /*
1214     if ( ne_unlock( m_pHttpSession, pLock ) == NE_OK )
1215     {
1216         OSL_TRACE( "UNLOCK succeeded." );
1217         return true;
1218     }
1219     else
1220     {
1221         OSL_TRACE( "UNLOCK failed!" );
1222         return false;
1223     }
1224     */
1225 }
1226 
1227 // -------------------------------------------------------------------
1228 void SerfSession::abort()
1229     throw ( DAVException )
1230 {
1231     // 11.11.09 (tkr): The following code lines causing crashes if
1232     // closing a ongoing connection. It turned out that this existing
1233     // solution doesn't work in multi-threading environments.
1234     // So I disabled them in 3.2. . Issue #73893# should fix it in OOo 3.3.
1235     //if ( m_pHttpSession )
1236     //    ne_close_connection( m_pHttpSession );
1237 }
1238 
1239 // -------------------------------------------------------------------
1240 const ucbhelper::InternetProxyServer & SerfSession::getProxySettings() const
1241 {
1242     if ( m_aUri.GetScheme().equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "http" ) ) ||
1243          m_aUri.GetScheme().equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "https" ) ) )
1244     {
1245         return m_rProxyDecider.getProxy( m_aUri.GetScheme(),
1246                                          m_aUri.GetHost(),
1247                                          m_aUri.GetPort() );
1248     }
1249     else
1250     {
1251         // TODO: figure out, if this case can occur
1252         return m_rProxyDecider.getProxy( m_aUri.GetScheme(),
1253                                          rtl::OUString() /* not used */,
1254                                          -1 /* not used */ );
1255     }
1256 }
1257 
1258 /*
1259 // -------------------------------------------------------------------
1260 namespace {
1261 
1262 bool containsLocktoken( const uno::Sequence< ucb::Lock > & rLocks,
1263                         const char * token )
1264 {
1265     for ( sal_Int32 n = 0; n < rLocks.getLength(); ++n )
1266     {
1267         const uno::Sequence< rtl::OUString > & rTokens
1268             = rLocks[ n ].LockTokens;
1269         for ( sal_Int32 m = 0; m < rTokens.getLength(); ++m )
1270         {
1271             if ( rTokens[ m ].equalsAscii( token ) )
1272                 return true;
1273         }
1274     }
1275     return false;
1276 }
1277 
1278 } // namespace
1279 */
1280 
1281 // -------------------------------------------------------------------
1282 bool SerfSession::removeExpiredLocktoken( const rtl::OUString & /*inURL*/,
1283                                           const DAVRequestEnvironment & /*rEnv*/ )
1284 {
1285     return true;
1286     /*
1287     SerfLock * theLock = m_aSerfLockStore.findByUri( inURL );
1288     if ( !theLock )
1289         return false;
1290 
1291     // do a lockdiscovery to check whether this lock is still valid.
1292     try
1293     {
1294         // @@@ Alternative: use ne_lock_discover() => less overhead
1295 
1296         std::vector< DAVResource > aResources;
1297         std::vector< rtl::OUString > aPropNames;
1298         aPropNames.push_back( DAVProperties::LOCKDISCOVERY );
1299 
1300         PROPFIND( rEnv.m_aRequestURI, DAVZERO, aPropNames, aResources, rEnv );
1301 
1302         if ( aResources.size() == 0 )
1303             return false;
1304 
1305         std::vector< DAVPropertyValue >::const_iterator it
1306             = aResources[ 0 ].properties.begin();
1307         std::vector< DAVPropertyValue >::const_iterator end
1308             = aResources[ 0 ].properties.end();
1309 
1310         while ( it != end )
1311         {
1312             if ( (*it).Name.equals( DAVProperties::LOCKDISCOVERY ) )
1313             {
1314                 uno::Sequence< ucb::Lock > aLocks;
1315                 if ( !( (*it).Value >>= aLocks ) )
1316                     return false;
1317 
1318                 if ( !containsLocktoken( aLocks, theLock->token ) )
1319                 {
1320                     // expired!
1321                     break;
1322                 }
1323 
1324                 // still valid.
1325                 return false;
1326             }
1327             ++it;
1328         }
1329 
1330         // No lockdiscovery prop in propfind result / locktoken not found
1331         // in propfind result -> not locked
1332         OSL_TRACE( "SerfSession::removeExpiredLocktoken: Removing "
1333                    " expired lock token for %s. token: %s",
1334                    rtl::OUStringToOString( inURL,
1335                                            RTL_TEXTENCODING_UTF8 ).getStr(),
1336                    theLock->token );
1337 
1338         m_aSerfLockStore.removeLock( theLock );
1339         ne_lock_destroy( theLock );
1340         return true;
1341     }
1342     catch ( DAVException const & )
1343     {
1344     }
1345     return false;
1346     */
1347 }
1348 
1349 // -------------------------------------------------------------------
1350 // HandleError
1351 // Common Error Handler
1352 // -------------------------------------------------------------------
1353 void SerfSession::HandleError( boost::shared_ptr<SerfRequestProcessor> rReqProc )
1354     throw ( DAVException )
1355 {
1356     m_aEnv = DAVRequestEnvironment();
1357 
1358     if ( rReqProc->mpDAVException )
1359     {
1360         DAVException* mpDAVExp( rReqProc->mpDAVException );
1361 
1362         serf_connection_reset( getSerfConnection() );
1363 
1364         if ( mpDAVExp->getStatus() == 413 &&
1365              m_bNoOfTransferEncodingSwitches < 2 )
1366         {
1367             m_bUseChunkedEncoding = !m_bUseChunkedEncoding;
1368             ++m_bNoOfTransferEncodingSwitches;
1369         }
1370 
1371         throw DAVException( mpDAVExp->getError(),
1372                             mpDAVExp->getData(),
1373                             mpDAVExp->getStatus() );
1374     }
1375 
1376     /*
1377     // Map error code to DAVException.
1378     switch ( nError )
1379     {
1380         case NE_OK:
1381             return;
1382 
1383         case NE_ERROR:        // Generic error
1384         {
1385             rtl::OUString aText = rtl::OUString::createFromAscii(
1386                 ne_get_error( m_pHttpSession ) );
1387 
1388             sal_uInt16 code = makeStatusCode( aText );
1389 
1390             if ( code == SC_LOCKED )
1391             {
1392                 if ( m_aSerfLockStore.findByUri(
1393                          makeAbsoluteURL( inPath ) ) == 0 )
1394                 {
1395                     // locked by 3rd party
1396                     throw DAVException( DAVException::DAV_LOCKED );
1397                 }
1398                 else
1399                 {
1400                     // locked by ourself
1401                     throw DAVException( DAVException::DAV_LOCKED_SELF );
1402                 }
1403             }
1404 
1405             // Special handling for 400 and 412 status codes, which may indicate
1406             // that a lock previously obtained by us has been released meanwhile
1407             // by the server. Unfortunately, RFC is not clear at this point,
1408             // thus server implementations behave different...
1409             else if ( code == SC_BAD_REQUEST || code == SC_PRECONDITION_FAILED )
1410             {
1411                 if ( removeExpiredLocktoken( makeAbsoluteURL( inPath ), rEnv ) )
1412                     throw DAVException( DAVException::DAV_LOCK_EXPIRED );
1413             }
1414 
1415             throw DAVException( DAVException::DAV_HTTP_ERROR, aText, code );
1416         }
1417         case NE_LOOKUP:       // Name lookup failed.
1418             throw DAVException( DAVException::DAV_HTTP_LOOKUP,
1419                                 SerfUri::makeConnectionEndPointString(
1420                                     m_aHostName, m_nPort ) );
1421 
1422         case NE_AUTH:         // User authentication failed on server
1423             throw DAVException( DAVException::DAV_HTTP_AUTH,
1424                                 SerfUri::makeConnectionEndPointString(
1425                                     m_aHostName, m_nPort ) );
1426 
1427         case NE_PROXYAUTH:    // User authentication failed on proxy
1428             throw DAVException( DAVException::DAV_HTTP_AUTHPROXY,
1429                                 SerfUri::makeConnectionEndPointString(
1430                                     m_aProxyName, m_nProxyPort ) );
1431 
1432         case NE_CONNECT:      // Could not connect to server
1433             throw DAVException( DAVException::DAV_HTTP_CONNECT,
1434                                 SerfUri::makeConnectionEndPointString(
1435                                     m_aHostName, m_nPort ) );
1436 
1437         case NE_TIMEOUT:      // Connection timed out
1438             throw DAVException( DAVException::DAV_HTTP_TIMEOUT,
1439                                 SerfUri::makeConnectionEndPointString(
1440                                     m_aHostName, m_nPort ) );
1441 
1442         case NE_FAILED:       // The precondition failed
1443             throw DAVException( DAVException::DAV_HTTP_FAILED,
1444                                 SerfUri::makeConnectionEndPointString(
1445                                     m_aHostName, m_nPort ) );
1446 
1447         case NE_RETRY:        // Retry request (ne_end_request ONLY)
1448             throw DAVException( DAVException::DAV_HTTP_RETRY,
1449                                 SerfUri::makeConnectionEndPointString(
1450                                     m_aHostName, m_nPort ) );
1451 
1452         case NE_REDIRECT:
1453         {
1454             SerfUri aUri( ne_redirect_location( m_pHttpSession ) );
1455             throw DAVException(
1456                 DAVException::DAV_HTTP_REDIRECT, aUri.GetURI() );
1457         }
1458         default:
1459         {
1460             OSL_TRACE( "SerfSession::HandleError : Unknown Serf error code!" );
1461             throw DAVException( DAVException::DAV_HTTP_ERROR,
1462                                 rtl::OUString::createFromAscii(
1463                                     ne_get_error( m_pHttpSession ) ) );
1464         }
1465     }
1466     */
1467 }
1468 
1469 // -------------------------------------------------------------------
1470 // static
1471 bool
1472 SerfSession::getDataFromInputStream(
1473     const uno::Reference< io::XInputStream > & xStream,
1474     uno::Sequence< sal_Int8 > & rData,
1475     bool bAppendTrailingZeroByte )
1476 {
1477     if ( xStream.is() )
1478     {
1479         uno::Reference< io::XSeekable > xSeekable( xStream, uno::UNO_QUERY );
1480         if ( xSeekable.is() )
1481         {
1482             try
1483             {
1484                 sal_Int32 nSize
1485                     = sal::static_int_cast<sal_Int32>(xSeekable->getLength());
1486                 sal_Int32 nRead
1487                     = xStream->readBytes( rData, nSize );
1488 
1489                 if ( nRead == nSize )
1490                 {
1491                     if ( bAppendTrailingZeroByte )
1492                     {
1493                         rData.realloc( nSize + 1 );
1494                         rData[ nSize ] = sal_Int8( 0 );
1495                     }
1496                     return true;
1497                 }
1498             }
1499             catch ( io::NotConnectedException const & )
1500             {
1501                 // readBytes
1502             }
1503             catch ( io::BufferSizeExceededException const & )
1504             {
1505                 // readBytes
1506             }
1507             catch ( io::IOException const & )
1508             {
1509                 // getLength, readBytes
1510             }
1511         }
1512         else
1513         {
1514             try
1515             {
1516                 uno::Sequence< sal_Int8 > aBuffer;
1517                 sal_Int32 nPos = 0;
1518 
1519                 sal_Int32 nRead = xStream->readSomeBytes( aBuffer, 65536 );
1520                 while ( nRead > 0 )
1521                 {
1522                     if ( rData.getLength() < ( nPos + nRead ) )
1523                         rData.realloc( nPos + nRead );
1524 
1525                     aBuffer.realloc( nRead );
1526                     rtl_copyMemory( (void*)( rData.getArray() + nPos ),
1527                                     (const void*)aBuffer.getConstArray(),
1528                                     nRead );
1529                     nPos += nRead;
1530 
1531                     aBuffer.realloc( 0 );
1532                     nRead = xStream->readSomeBytes( aBuffer, 65536 );
1533                 }
1534 
1535                 if ( bAppendTrailingZeroByte )
1536                 {
1537                     rData.realloc( nPos + 1 );
1538                     rData[ nPos ] = sal_Int8( 0 );
1539                 }
1540                 return true;
1541             }
1542             catch ( io::NotConnectedException const & )
1543             {
1544                 // readBytes
1545             }
1546             catch ( io::BufferSizeExceededException const & )
1547             {
1548                 // readBytes
1549             }
1550             catch ( io::IOException const & )
1551             {
1552                 // readBytes
1553             }
1554         }
1555     }
1556     return false;
1557 }
1558 
1559 // ---------------------------------------------------------------------
1560 sal_Bool
1561 SerfSession::isDomainMatch( rtl::OUString certHostName )
1562 {
1563     rtl::OUString hostName = getHostName();
1564 
1565     if (hostName.equalsIgnoreAsciiCase( certHostName ) )
1566         return sal_True;
1567 
1568     if ( 0 == certHostName.indexOf( rtl::OUString::createFromAscii( "*" ) ) &&
1569          hostName.getLength() >= certHostName.getLength()  )
1570     {
1571         rtl::OUString cmpStr = certHostName.copy( 1 );
1572 
1573         if ( hostName.matchIgnoreAsciiCase(
1574                 cmpStr, hostName.getLength() -  cmpStr.getLength() ) )
1575             return sal_True;
1576     }
1577     return sal_False;
1578 }
1579 
1580 /*
1581 // ---------------------------------------------------------------------
1582 rtl::OUString SerfSession::makeAbsoluteURL( rtl::OUString const & rURL ) const
1583 {
1584     try
1585     {
1586         // Is URL relative or already absolute?
1587         if ( rURL[ 0 ] != sal_Unicode( '/' ) )
1588         {
1589             // absolute.
1590             return rtl::OUString( rURL );
1591         }
1592         else
1593         {
1594             ne_uri aUri;
1595             memset( &aUri, 0, sizeof( aUri ) );
1596 
1597             ne_fill_server_uri( m_pHttpSession, &aUri );
1598             aUri.path
1599                 = ne_strdup( rtl::OUStringToOString(
1600                     rURL, RTL_TEXTENCODING_UTF8 ).getStr() );
1601             SerfUri aSerfUri( &aUri );
1602             ne_uri_free( &aUri );
1603             return aSerfUri.GetURI();
1604         }
1605     }
1606     catch ( DAVException const & )
1607     {
1608     }
1609     // error.
1610     return rtl::OUString();
1611 }
1612 */
1613