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