3a349b7ccf
Change-Id: I5ca4c1599a1abc4281bac9a4d0164d4b0cb3e24a
596 lines
24 KiB
C++
596 lines
24 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*
|
|
* This file is part of the LibreOffice project.
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*
|
|
* This file incorporates work covered by the following license notice:
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed
|
|
* with this work for additional information regarding copyright
|
|
* ownership. The ASF licenses this file to you under the Apache
|
|
* License, Version 2.0 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy of
|
|
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
|
|
*/
|
|
|
|
#include "SerfRequestProcessor.hxx"
|
|
|
|
#include "AprEnv.hxx"
|
|
#include "SerfCallbacks.hxx"
|
|
#include "SerfSession.hxx"
|
|
#include "SerfPropFindReqProcImpl.hxx"
|
|
#include "SerfPropPatchReqProcImpl.hxx"
|
|
#include "SerfGetReqProcImpl.hxx"
|
|
#include "SerfHeadReqProcImpl.hxx"
|
|
#include "SerfPutReqProcImpl.hxx"
|
|
#include "SerfPostReqProcImpl.hxx"
|
|
#include "SerfDeleteReqProcImpl.hxx"
|
|
#include "SerfMkColReqProcImpl.hxx"
|
|
#include "SerfCopyReqProcImpl.hxx"
|
|
#include "SerfMoveReqProcImpl.hxx"
|
|
#include "SerfLockReqProcImpl.hxx"
|
|
#include "SerfUnlockReqProcImpl.hxx"
|
|
|
|
#include <apr_strings.h>
|
|
|
|
namespace http_dav_ucp
|
|
{
|
|
|
|
SerfRequestProcessor::SerfRequestProcessor( SerfSession& rSerfSession,
|
|
const OUString & inPath,
|
|
const bool bUseChunkedEncoding )
|
|
: mrSerfSession( rSerfSession )
|
|
, mPathStr( nullptr )
|
|
, mbUseChunkedEncoding( bUseChunkedEncoding )
|
|
, mDestPathStr( nullptr )
|
|
, mContentType( nullptr )
|
|
, mReferer( nullptr )
|
|
, mpProcImpl( nullptr )
|
|
, mbProcessingDone( false )
|
|
, mpDAVException()
|
|
, mnHTTPStatusCode( SC_NONE )
|
|
, mHTTPStatusCodeText()
|
|
, mRedirectLocation()
|
|
, mnSuccessfulCredentialAttempts( 0 )
|
|
, mbInputOfCredentialsAborted( false )
|
|
, mbSetupSerfRequestCalled( false )
|
|
, mbAcceptSerfResponseCalled( false )
|
|
, mbHandleSerfResponseCalled( false )
|
|
{
|
|
mPathStr = apr_pstrdup( SerfSession::getAprPool(),
|
|
OUStringToOString( inPath, RTL_TEXTENCODING_UTF8 ).getStr() );
|
|
}
|
|
|
|
SerfRequestProcessor::~SerfRequestProcessor()
|
|
{
|
|
delete mpProcImpl;
|
|
delete mpDAVException;
|
|
}
|
|
|
|
void SerfRequestProcessor::prepareProcessor()
|
|
{
|
|
delete mpDAVException;
|
|
mpDAVException = nullptr;
|
|
mnHTTPStatusCode = SC_NONE;
|
|
mHTTPStatusCodeText.clear();
|
|
mRedirectLocation.clear();
|
|
|
|
mnSuccessfulCredentialAttempts = 0;
|
|
mbInputOfCredentialsAborted = false;
|
|
mbSetupSerfRequestCalled = false;
|
|
mbAcceptSerfResponseCalled = false;
|
|
mbHandleSerfResponseCalled = false;
|
|
}
|
|
|
|
// PROPFIND - allprop & named
|
|
bool SerfRequestProcessor::processPropFind( const Depth inDepth,
|
|
const std::vector< OUString > & inPropNames,
|
|
std::vector< DAVResource > & ioResources,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mpProcImpl = new SerfPropFindReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
inDepth,
|
|
inPropNames,
|
|
ioResources );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// PROPFIND - property names
|
|
bool SerfRequestProcessor::processPropFind( const Depth inDepth,
|
|
std::vector< DAVResourceInfo > & ioResInfo,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mpProcImpl = new SerfPropFindReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
inDepth,
|
|
ioResInfo );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// PROPPATCH
|
|
bool SerfRequestProcessor::processPropPatch( const std::vector< ProppatchValue > & inProperties,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mpProcImpl = new SerfPropPatchReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
inProperties );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// GET
|
|
bool SerfRequestProcessor::processGet( const rtl::Reference< SerfInputStream >& xioInStrm,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mpProcImpl = new SerfGetReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
xioInStrm );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// GET inclusive header fields
|
|
bool SerfRequestProcessor::processGet( const rtl::Reference< SerfInputStream >& xioInStrm,
|
|
const std::vector< OUString > & inHeaderNames,
|
|
DAVResource & ioResource,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mpProcImpl = new SerfGetReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
xioInStrm,
|
|
inHeaderNames,
|
|
ioResource );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// GET
|
|
bool SerfRequestProcessor::processGet( const css::uno::Reference< css::io::XOutputStream >& xioOutStrm,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mpProcImpl = new SerfGetReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
xioOutStrm );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// GET inclusive header fields
|
|
bool SerfRequestProcessor::processGet( const css::uno::Reference< css::io::XOutputStream >& xioOutStrm,
|
|
const std::vector< OUString > & inHeaderNames,
|
|
DAVResource & ioResource,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mpProcImpl = new SerfGetReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
xioOutStrm,
|
|
inHeaderNames,
|
|
ioResource );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// HEAD
|
|
bool SerfRequestProcessor::processHead( const std::vector< OUString > & inHeaderNames,
|
|
DAVResource & ioResource,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mpProcImpl = new SerfHeadReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
inHeaderNames,
|
|
ioResource );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// PUT
|
|
bool SerfRequestProcessor::processPut( const char* inData,
|
|
apr_size_t inDataLen,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
// get the lock from lock store
|
|
const OUString sToken(
|
|
apr_environment::AprEnv::getAprEnv()->getSerfLockStore()->getLockToken(
|
|
OUString::createFromAscii(mPathStr)) );
|
|
|
|
mpProcImpl = new SerfPutReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
inData,
|
|
inDataLen,
|
|
sToken );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// POST
|
|
bool SerfRequestProcessor::processPost( const char* inData,
|
|
apr_size_t inDataLen,
|
|
const OUString & inContentType,
|
|
const OUString & inReferer,
|
|
const rtl::Reference< SerfInputStream >& xioInStrm,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mContentType = apr_pstrdup( SerfSession::getAprPool(),
|
|
OUStringToOString( inContentType, RTL_TEXTENCODING_UTF8 ).getStr() );
|
|
mReferer = apr_pstrdup( SerfSession::getAprPool(),
|
|
OUStringToOString( inReferer, RTL_TEXTENCODING_UTF8 ).getStr() );
|
|
mpProcImpl = new SerfPostReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
inData,
|
|
inDataLen,
|
|
mContentType,
|
|
mReferer,
|
|
xioInStrm );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// POST
|
|
bool SerfRequestProcessor::processPost( const char* inData,
|
|
apr_size_t inDataLen,
|
|
const OUString & inContentType,
|
|
const OUString & inReferer,
|
|
const css::uno::Reference< css::io::XOutputStream >& xioOutStrm,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mContentType = apr_pstrdup( SerfSession::getAprPool(),
|
|
OUStringToOString( inContentType, RTL_TEXTENCODING_UTF8 ).getStr() );
|
|
mReferer = apr_pstrdup( SerfSession::getAprPool(),
|
|
OUStringToOString( inReferer, RTL_TEXTENCODING_UTF8 ).getStr() );
|
|
mpProcImpl = new SerfPostReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
inData,
|
|
inDataLen,
|
|
mContentType,
|
|
mReferer,
|
|
xioOutStrm );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// DELETE
|
|
bool SerfRequestProcessor::processDelete( apr_status_t& outSerfStatus )
|
|
{
|
|
mpProcImpl = new SerfDeleteReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// MKCOL
|
|
bool SerfRequestProcessor::processMkCol( apr_status_t& outSerfStatus )
|
|
{
|
|
mpProcImpl = new SerfMkColReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// COPY
|
|
bool SerfRequestProcessor::processCopy( const OUString & inDestinationPath,
|
|
const bool inOverwrite,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mDestPathStr = apr_pstrdup( SerfSession::getAprPool(),
|
|
OUStringToOString( inDestinationPath, RTL_TEXTENCODING_UTF8 ).getStr() );
|
|
mpProcImpl = new SerfCopyReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
mDestPathStr,
|
|
inOverwrite );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
// MOVE
|
|
bool SerfRequestProcessor::processMove( const OUString & inDestinationPath,
|
|
const bool inOverwrite,
|
|
apr_status_t& outSerfStatus )
|
|
{
|
|
mDestPathStr = apr_pstrdup( SerfSession::getAprPool(),
|
|
OUStringToOString( inDestinationPath, RTL_TEXTENCODING_UTF8 ).getStr() );
|
|
mpProcImpl = new SerfMoveReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
mDestPathStr,
|
|
inOverwrite );
|
|
outSerfStatus = runProcessor();
|
|
|
|
return outSerfStatus == APR_SUCCESS;
|
|
}
|
|
|
|
|
|
bool SerfRequestProcessor::processLock( const css::ucb::Lock & rLock, sal_Int32 *plastChanceToSendRefreshRequest )
|
|
{
|
|
mpProcImpl = new SerfLockReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
mrSerfSession,
|
|
rLock,
|
|
plastChanceToSendRefreshRequest );
|
|
|
|
return runProcessor() == APR_SUCCESS;
|
|
}
|
|
|
|
bool SerfRequestProcessor::processUnlock()
|
|
{
|
|
// get the lock from lock store
|
|
const OUString sToken(
|
|
apr_environment::AprEnv::getAprEnv()->getSerfLockStore()->getLockToken(
|
|
OUString::createFromAscii(mPathStr)) );
|
|
if ( sToken.isEmpty() )
|
|
throw DAVException( DAVException::DAV_NOT_LOCKED );
|
|
|
|
mpProcImpl = new SerfUnlockReqProcImpl( mPathStr,
|
|
mrSerfSession.getRequestEnvironment().m_aRequestHeaders,
|
|
sToken );
|
|
|
|
return runProcessor() == APR_SUCCESS;
|
|
}
|
|
|
|
apr_status_t SerfRequestProcessor::runProcessor()
|
|
{
|
|
prepareProcessor();
|
|
|
|
// activate chunked encoding, if requested
|
|
if ( mbUseChunkedEncoding )
|
|
{
|
|
mpProcImpl->activateChunkedEncoding();
|
|
}
|
|
|
|
// create serf request
|
|
serf_connection_request_create( mrSerfSession.getSerfConnection(),
|
|
Serf_SetupRequest,
|
|
this );
|
|
|
|
// perform serf request
|
|
mbProcessingDone = false;
|
|
apr_status_t status = APR_SUCCESS;
|
|
serf_context_t* pSerfContext = mrSerfSession.getSerfContext();
|
|
apr_pool_t* pAprPool = SerfSession::getAprPool();
|
|
while ( true )
|
|
{
|
|
status = serf_context_run( pSerfContext,
|
|
SERF_DURATION_FOREVER,
|
|
pAprPool );
|
|
if ( APR_STATUS_IS_TIMEUP( status ) )
|
|
{
|
|
continue;
|
|
}
|
|
if ( status != APR_SUCCESS )
|
|
{
|
|
break;
|
|
}
|
|
if ( mbProcessingDone )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
postprocessProcessor( status );
|
|
|
|
return status;
|
|
}
|
|
|
|
void SerfRequestProcessor::postprocessProcessor( const apr_status_t inStatus )
|
|
{
|
|
if ( inStatus == APR_SUCCESS )
|
|
{
|
|
return;
|
|
}
|
|
|
|
switch ( inStatus )
|
|
{
|
|
case APR_EGENERAL:
|
|
case SERF_ERROR_AUTHN_FAILED:
|
|
// general error; <mnHTTPStatusCode> provides more information
|
|
{
|
|
switch ( mnHTTPStatusCode )
|
|
{
|
|
case SC_NONE:
|
|
if ( !mbSetupSerfRequestCalled )
|
|
{
|
|
mpDAVException = new DAVException( DAVException::DAV_HTTP_LOOKUP,
|
|
SerfUri::makeConnectionEndPointString( mrSerfSession.getHostName(),
|
|
mrSerfSession.getPort() ) );
|
|
}
|
|
else if ( mbInputOfCredentialsAborted )
|
|
{
|
|
mpDAVException = new DAVException( DAVException::DAV_HTTP_NOAUTH,
|
|
SerfUri::makeConnectionEndPointString( mrSerfSession.getHostName(),
|
|
mrSerfSession.getPort() ) );
|
|
}
|
|
else
|
|
{
|
|
mpDAVException = new DAVException( DAVException::DAV_HTTP_ERROR,
|
|
mHTTPStatusCodeText,
|
|
mnHTTPStatusCode );
|
|
}
|
|
break;
|
|
case SC_MOVED_PERMANENTLY:
|
|
case SC_MOVED_TEMPORARILY:
|
|
case SC_SEE_OTHER:
|
|
case SC_TEMPORARY_REDIRECT:
|
|
mpDAVException = new DAVException( DAVException::DAV_HTTP_REDIRECT,
|
|
mRedirectLocation );
|
|
break;
|
|
default:
|
|
mpDAVException = new DAVException( DAVException::DAV_HTTP_ERROR,
|
|
mHTTPStatusCodeText,
|
|
mnHTTPStatusCode );
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
mpDAVException = new DAVException( DAVException::DAV_HTTP_ERROR );
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
apr_status_t SerfRequestProcessor::provideSerfCredentials( char ** outUsername,
|
|
char ** outPassword,
|
|
serf_request_t * inRequest,
|
|
int inCode,
|
|
const char *inAuthProtocol,
|
|
const char *inRealm,
|
|
apr_pool_t *inAprPool )
|
|
{
|
|
// as each successful provided credentials are tried twice - see below - the
|
|
// number of real attempts is half of the value of <mnSuccessfulCredentialAttempts>
|
|
if ( (mnSuccessfulCredentialAttempts / 2) >= 5 ||
|
|
mbInputOfCredentialsAborted )
|
|
{
|
|
mbInputOfCredentialsAborted = true;
|
|
return SERF_ERROR_AUTHN_FAILED;
|
|
}
|
|
|
|
// because serf keeps credentials only for a connection in case of digest authentication
|
|
// we give each successful provided credentials a second try in order to workaround the
|
|
// situation that the connection for which the credentials have been provided has been closed
|
|
// before the provided credentials could be applied for the request.
|
|
apr_status_t status = mrSerfSession.provideSerfCredentials( (mnSuccessfulCredentialAttempts % 2) == 1,
|
|
outUsername,
|
|
outPassword,
|
|
inRequest,
|
|
inCode,
|
|
inAuthProtocol,
|
|
inRealm,
|
|
inAprPool );
|
|
if ( status != APR_SUCCESS )
|
|
{
|
|
mbInputOfCredentialsAborted = true;
|
|
}
|
|
else
|
|
{
|
|
++mnSuccessfulCredentialAttempts;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
apr_status_t SerfRequestProcessor::setupSerfRequest( serf_request_t * inSerfRequest,
|
|
serf_bucket_t ** outSerfRequestBucket,
|
|
serf_response_acceptor_t * outSerfResponseAcceptor,
|
|
void ** outSerfResponseAcceptorBaton,
|
|
serf_response_handler_t * outSerfResponseHandler,
|
|
void ** outSerfResponseHandlerBaton,
|
|
apr_pool_t * /*inAprPool*/ )
|
|
{
|
|
mbSetupSerfRequestCalled = true;
|
|
*outSerfRequestBucket = mpProcImpl->createSerfRequestBucket( inSerfRequest );
|
|
|
|
// apply callbacks for accepting response and handling response
|
|
*outSerfResponseAcceptor = Serf_AcceptResponse;
|
|
*outSerfResponseAcceptorBaton = this;
|
|
*outSerfResponseHandler = Serf_HandleResponse;
|
|
*outSerfResponseHandlerBaton = this;
|
|
|
|
return APR_SUCCESS;
|
|
}
|
|
|
|
serf_bucket_t* SerfRequestProcessor::acceptSerfResponse( serf_request_t * inSerfRequest,
|
|
serf_bucket_t * inSerfStreamBucket,
|
|
apr_pool_t * inAprPool )
|
|
{
|
|
mbAcceptSerfResponseCalled = true;
|
|
return mrSerfSession.acceptSerfResponse( inSerfRequest,
|
|
inSerfStreamBucket,
|
|
inAprPool );
|
|
}
|
|
|
|
apr_status_t SerfRequestProcessor::handleSerfResponse( serf_request_t * inSerfRequest,
|
|
serf_bucket_t * inSerfResponseBucket,
|
|
apr_pool_t * inAprPool )
|
|
{
|
|
mbHandleSerfResponseCalled = true;
|
|
|
|
// some general response handling and error handling
|
|
{
|
|
if ( !inSerfResponseBucket )
|
|
{
|
|
/* A NULL response can come back if the request failed completely */
|
|
mbProcessingDone = true;
|
|
return APR_EGENERAL;
|
|
}
|
|
|
|
serf_status_line sl;
|
|
apr_status_t status = serf_bucket_response_status( inSerfResponseBucket, &sl );
|
|
if ( status )
|
|
{
|
|
mbProcessingDone = false; // allow another try in order to get a response
|
|
return status;
|
|
}
|
|
// TODO - check, if response status code handling is correct
|
|
mnHTTPStatusCode = ( sl.version != 0 && sl.code >= 0 )
|
|
? static_cast< sal_uInt16 >( sl.code )
|
|
: SC_NONE;
|
|
if ( sl.reason )
|
|
{
|
|
mHTTPStatusCodeText = OUString::createFromAscii( sl.reason );
|
|
}
|
|
if ( ( sl.version == 0 || sl.code < 0 ) ||
|
|
mnHTTPStatusCode >= 300 )
|
|
{
|
|
if ( mnHTTPStatusCode == 301 ||
|
|
mnHTTPStatusCode == 302 ||
|
|
mnHTTPStatusCode == 303 ||
|
|
mnHTTPStatusCode == 307 )
|
|
{
|
|
// new location for certain redirections
|
|
serf_bucket_t *headers = serf_bucket_response_get_headers( inSerfResponseBucket );
|
|
const char* location = serf_bucket_headers_get( headers, "Location" );
|
|
if ( location )
|
|
{
|
|
mRedirectLocation = OUString::createFromAscii( location );
|
|
}
|
|
mbProcessingDone = true;
|
|
return APR_EGENERAL;
|
|
}
|
|
else if ( mrSerfSession.isHeadRequestInProgress() &&
|
|
( mnHTTPStatusCode == 401 || mnHTTPStatusCode == 407 ) )
|
|
{
|
|
// keep going as authentication is not required on HEAD request.
|
|
// the response already contains header fields.
|
|
}
|
|
else
|
|
{
|
|
mbProcessingDone = true;
|
|
return APR_EGENERAL;
|
|
}
|
|
}
|
|
}
|
|
|
|
// request specific processing of the response bucket
|
|
apr_status_t status = APR_SUCCESS;
|
|
mbProcessingDone = mpProcImpl->processSerfResponseBucket( inSerfRequest,
|
|
inSerfResponseBucket,
|
|
inAprPool,
|
|
status );
|
|
|
|
return status;
|
|
}
|
|
|
|
} // namespace http_dav_ucp
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|