Introduces AIHTTPTimeoutPolicy objects which do not just specify a single "timeout" in seconds, but a plethora of timings related to the life cycle of the average HTTP transaction. This knowledge is that moved to the Responder being used instead of floating constants hardcoded in the callers of http requests. This assumes that the same timeout policy is wanted for each transaction that uses the same Responder, which can be enforced is needed. I added a AIHTTPTimeoutPolicy for EVERY responder, only to make it easier later to tune timeout values and/or to get feedback about which responder runs into HTTP errors in debug output (especially time outs), so that they can be tuned later. If we already understood exactly what we were doing then most responders could have been left alone and just return the default timeout policy: by far most timeout policies are just a copy of the default policy, currently. This commit is not finished... It's a work in progress (viewer runs fine with it though).
749 lines
20 KiB
C++
749 lines
20 KiB
C++
/**
|
|
* @file llurlrequest.cpp
|
|
* @author Phoenix
|
|
* @date 2005-04-28
|
|
* @brief Implementation of the URLRequest class and related classes.
|
|
*
|
|
* $LicenseInfo:firstyear=2005&license=viewerlgpl$
|
|
* Second Life Viewer Source Code
|
|
* Copyright (C) 2010, Linden Research, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation;
|
|
* version 2.1 of the License only.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
* Linden Research, Inc., 945 Battery Street, San Francisco, CA 94111 USA
|
|
* $/LicenseInfo$
|
|
*/
|
|
|
|
#include "linden_common.h"
|
|
#include "llurlrequest.h"
|
|
|
|
#ifdef CWDEBUG
|
|
#include <libcwd/buf2str.h>
|
|
#endif
|
|
|
|
#include <algorithm>
|
|
#include <openssl/x509_vfy.h>
|
|
#include <openssl/ssl.h>
|
|
#include "aicurleasyrequeststatemachine.h"
|
|
#include "llioutil.h"
|
|
#include "llmemtype.h"
|
|
#include "llpumpio.h"
|
|
#include "llsd.h"
|
|
#include "llstring.h"
|
|
#include "apr_env.h"
|
|
#include "llapr.h"
|
|
#include "llscopedvolatileaprpool.h"
|
|
#include "llfasttimer.h"
|
|
#include "message.h"
|
|
static const U32 HTTP_STATUS_PIPE_ERROR = 499;
|
|
|
|
/**
|
|
* String constants
|
|
*/
|
|
const std::string CONTEXT_TRANSFERED_BYTES("transfered_bytes");
|
|
|
|
|
|
//static size_t headerCallback(char* data, size_t size, size_t nmemb, void* user);
|
|
|
|
#if 0
|
|
/**
|
|
* class LLURLRequestDetail
|
|
*/
|
|
class LLURLRequestDetail
|
|
{
|
|
public:
|
|
LLURLRequestDetail();
|
|
~LLURLRequestDetail();
|
|
std::string mURL;
|
|
AICurlEasyRequestStateMachine* mStateMachine;
|
|
LLIOPipe::buffer_ptr_t mResponseBuffer;
|
|
LLChannelDescriptors mChannels;
|
|
U8* mLastRead;
|
|
U32 mBodyLimit;
|
|
S32 mByteAccumulator;
|
|
bool mIsBodyLimitSet;
|
|
};
|
|
|
|
LLURLRequestDetail::LLURLRequestDetail() :
|
|
mStateMachine(new AICurlEasyRequestStateMachine(false)),
|
|
mLastRead(NULL),
|
|
mBodyLimit(0),
|
|
mByteAccumulator(0),
|
|
mIsBodyLimitSet(false)
|
|
{
|
|
}
|
|
|
|
LLURLRequestDetail::~LLURLRequestDetail()
|
|
{
|
|
mLastRead = NULL;
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* class LLURLRequest
|
|
*/
|
|
|
|
// static
|
|
std::string LLURLRequest::actionAsVerb(LLURLRequest::ERequestAction action)
|
|
{
|
|
static int const array_size = HTTP_MOVE + 1; // INVALID == 0
|
|
static char const* const VERBS[array_size] =
|
|
{
|
|
"(invalid)",
|
|
"HEAD",
|
|
"GET",
|
|
"PUT",
|
|
"POST",
|
|
"DELETE",
|
|
"MOVE"
|
|
};
|
|
return VERBS[action >= array_size ? INVALID : action];
|
|
}
|
|
|
|
// This might throw AICurlNoEasyHandle.
|
|
LLURLRequest::LLURLRequest(LLURLRequest::ERequestAction action, std::string const& url, Injector* body, AICurlInterface::ResponderPtr responder, AIHTTPHeaders& headers) :
|
|
AICurlEasyRequestStateMachine(true), mAction(action), mURL(url), mBody(body), mResponder(responder), mHeaders(headers)
|
|
{
|
|
}
|
|
|
|
void LLURLRequest::initialize_impl(void)
|
|
{
|
|
if (mHeaders.hasHeader("Cookie"))
|
|
{
|
|
allowCookies();
|
|
}
|
|
|
|
// If the header is "Pragma" with no value, the caller intends to
|
|
// force libcurl to drop the Pragma header it so gratuitously inserts.
|
|
// Before inserting the header, force libcurl to not use the proxy.
|
|
std::string pragma_value;
|
|
if (mHeaders.getValue("Pragma", pragma_value) && pragma_value.empty())
|
|
{
|
|
useProxy(false);
|
|
}
|
|
|
|
if (mAction == HTTP_PUT || mAction == HTTP_POST)
|
|
{
|
|
// If the Content-Type header was passed in we defer to the caller's wisdom,
|
|
// but if they did not specify a Content-Type, then ask the injector.
|
|
mHeaders.addHeader("Content-Type", mBody->contentType(), AIHTTPHeaders::keep_existing_header);
|
|
}
|
|
else
|
|
{
|
|
// Check to see if we have already set Accept or not. If no one
|
|
// set it, set it to application/llsd+xml since that's what we
|
|
// almost always want.
|
|
mHeaders.addHeader("Accept", "application/llsd+xml", AIHTTPHeaders::keep_existing_header);
|
|
}
|
|
|
|
if (mAction == HTTP_POST && gMessageSystem)
|
|
{
|
|
mHeaders.addHeader("X-SecondLife-UDP-Listen-Port", llformat("%d", gMessageSystem->mPort));
|
|
}
|
|
|
|
bool success = false;
|
|
try
|
|
{
|
|
AICurlEasyRequest_wat buffered_easy_request_w(*mCurlEasyRequest);
|
|
AICurlResponderBuffer_wat buffer_w(*mCurlEasyRequest);
|
|
buffer_w->prepRequest(buffered_easy_request_w, mHeaders, mResponder);
|
|
|
|
if (mBody)
|
|
{
|
|
// This might throw AICurlNoBody.
|
|
mBodySize = mBody->get_body(buffer_w->sChannels, buffer_w->getInput());
|
|
}
|
|
|
|
success = configure(buffered_easy_request_w);
|
|
}
|
|
catch (AICurlNoBody const& error)
|
|
{
|
|
llwarns << "Injector::get_body() failed: " << error.what() << llendl;
|
|
}
|
|
|
|
if (success)
|
|
{
|
|
// Continue to initialize base class.
|
|
AICurlEasyRequestStateMachine::initialize_impl();
|
|
}
|
|
else
|
|
{
|
|
abort();
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
void LLURLRequest::setURL2(const std::string& url)
|
|
{
|
|
mDetail->mURL = url;
|
|
}
|
|
|
|
std::string LLURLRequest::getURL2() const
|
|
{
|
|
return mDetail->mURL;
|
|
}
|
|
#endif
|
|
|
|
void LLURLRequest::addHeader(const char* header)
|
|
{
|
|
AICurlEasyRequest_wat curlEasyRequest_w(*mCurlEasyRequest);
|
|
curlEasyRequest_w->addHeader(header);
|
|
}
|
|
|
|
#ifdef AI_UNUSED
|
|
void LLURLRequest::setBodyLimit(U32 size)
|
|
{
|
|
mDetail->mBodyLimit = size;
|
|
mDetail->mIsBodyLimitSet = true;
|
|
}
|
|
|
|
void LLURLRequest::setCallback(LLURLRequestComplete* callback)
|
|
{
|
|
mCompletionCallback = callback;
|
|
AICurlEasyRequest_wat curlEasyRequest_w(*mCurlEasyRequest);
|
|
curlEasyRequest_w->setHeaderCallback(&headerCallback, (void*)callback);
|
|
}
|
|
#endif
|
|
|
|
// Added to mitigate the effect of libcurl looking
|
|
// for the ALL_PROXY and http_proxy env variables
|
|
// and deciding to insert a Pragma: no-cache
|
|
// header! The only usage of this method at the
|
|
// time of this writing is in llhttpclient.cpp
|
|
// in the request() method, where this method
|
|
// is called with use_proxy = FALSE
|
|
void LLURLRequest::useProxy(bool use_proxy)
|
|
{
|
|
static std::string env_proxy;
|
|
|
|
if (use_proxy && env_proxy.empty())
|
|
{
|
|
char* env_proxy_str;
|
|
LLScopedVolatileAPRPool scoped_pool;
|
|
apr_status_t status = apr_env_get(&env_proxy_str, "ALL_PROXY", scoped_pool);
|
|
if (status != APR_SUCCESS)
|
|
{
|
|
status = apr_env_get(&env_proxy_str, "http_proxy", scoped_pool);
|
|
}
|
|
if (status != APR_SUCCESS)
|
|
{
|
|
use_proxy = false;
|
|
}
|
|
else
|
|
{
|
|
// env_proxy_str is stored in the scoped_pool, so we have to make a copy.
|
|
env_proxy = env_proxy_str;
|
|
}
|
|
}
|
|
|
|
LL_DEBUGS("Proxy") << "use_proxy = " << (use_proxy?'Y':'N') << ", env_proxy = " << (!env_proxy.empty() ? env_proxy : "(null)") << LL_ENDL;
|
|
|
|
AICurlEasyRequest_wat curlEasyRequest_w(*mCurlEasyRequest);
|
|
curlEasyRequest_w->setoptString(CURLOPT_PROXY, (use_proxy && !env_proxy.empty()) ? env_proxy : std::string(""));
|
|
}
|
|
|
|
#ifdef AI_UNUSED
|
|
void LLURLRequest::useProxy(const std::string &proxy)
|
|
{
|
|
AICurlEasyRequest_wat curlEasyRequest_w(*mCurlEasyRequest);
|
|
curlEasyRequest_w->setoptString(CURLOPT_PROXY, proxy);
|
|
}
|
|
#endif
|
|
|
|
void LLURLRequest::allowCookies()
|
|
{
|
|
AICurlEasyRequest_wat curlEasyRequest_w(*mCurlEasyRequest);
|
|
curlEasyRequest_w->setoptString(CURLOPT_COOKIEFILE, "");
|
|
}
|
|
|
|
#ifdef AI_UNUSED // no longer derived from LLIOPipe
|
|
//virtual
|
|
bool LLURLRequest::hasExpiration(void) const
|
|
{
|
|
// Currently, this ALWAYS returns false -- because only AICurlEasyRequestStateMachine uses buffered
|
|
// AICurlEasyRequest objects, and LLURLRequest uses (unbuffered) AICurlEasyRequest directly, which
|
|
// have no expiration facility.
|
|
return mDetail->mStateMachine->isBuffered();
|
|
}
|
|
|
|
//virtual
|
|
bool LLURLRequest::hasNotExpired(void) const
|
|
{
|
|
if (!mDetail->mStateMachine->isBuffered())
|
|
return true;
|
|
AICurlEasyRequest_wat buffered_easy_request_w(*mCurlEasyRequest);
|
|
AICurlResponderBuffer_wat buffer_w(*mCurlEasyRequest);
|
|
return buffer_w->isValid();
|
|
}
|
|
|
|
// virtual
|
|
LLIOPipe::EStatus LLURLRequest::handleError(
|
|
LLIOPipe::EStatus status,
|
|
LLPumpIO* pump)
|
|
{
|
|
DoutEntering(dc::curl, "LLURLRequest::handleError(" << LLIOPipe::lookupStatusString(status) << ", " << (void*)pump << ") [" << (void*)mCurlEasyRequest.get_ptr().get() << "]");
|
|
|
|
if (LL_LIKELY(!mDetail->mStateMachine->isBuffered())) // Currently always true.
|
|
{
|
|
// The last reference will be deleted when the pump that this chain belongs to
|
|
// is removed from the running chains vector, upon returning from this function.
|
|
// This keeps the CurlEasyRequest object alive until the curl thread cleanly removed it.
|
|
Dout(dc::curl, "Calling mDetail->mStateMachine->removeRequest() [" << (void*)mCurlEasyRequest.get_ptr().get() << "]");
|
|
mDetail->mStateMachine->removeRequest();
|
|
}
|
|
else if (!hasNotExpired())
|
|
{
|
|
// The buffered version has it's own time out handling, and that already expired,
|
|
// so we can ignore the expiration of this timer (currently never happens).
|
|
// I left it here because it's what LL did (in the form if (!isValid() ...),
|
|
// and it would be relevant if this characteristic of mDetail->mStateMachine
|
|
// would change. --Aleric
|
|
return STATUS_EXPIRED ;
|
|
}
|
|
|
|
if(mCompletionCallback && pump)
|
|
{
|
|
LLURLRequestComplete* complete = NULL;
|
|
complete = (LLURLRequestComplete*)mCompletionCallback.get();
|
|
complete->httpStatus(
|
|
HTTP_STATUS_PIPE_ERROR,
|
|
LLIOPipe::lookupStatusString(status));
|
|
complete->responseStatus(status);
|
|
pump->respond(complete);
|
|
mCompletionCallback = NULL;
|
|
}
|
|
return status;
|
|
}
|
|
|
|
static LLFastTimer::DeclareTimer FTM_PROCESS_URL_REQUEST("URL Request");
|
|
|
|
// virtual
|
|
LLIOPipe::EStatus LLURLRequest::process_impl(
|
|
const LLChannelDescriptors& channels,
|
|
buffer_ptr_t& buffer,
|
|
bool& eos,
|
|
LLSD& context,
|
|
LLPumpIO* pump)
|
|
{
|
|
LLFastTimer t(FTM_PROCESS_URL_REQUEST);
|
|
PUMP_DEBUG;
|
|
//llinfos << "LLURLRequest::process_impl()" << llendl;
|
|
if (!buffer) return STATUS_ERROR;
|
|
|
|
if (!mDetail) return STATUS_ERROR; //Seems to happen on occasion. Need to hunt down why.
|
|
|
|
//AIFIXME: implement this again:
|
|
// we're still waiting or processing, check how many
|
|
// bytes we have accumulated.
|
|
const S32 MIN_ACCUMULATION = 100000;
|
|
if(pump && (mDetail->mByteAccumulator > MIN_ACCUMULATION))
|
|
{
|
|
static LLFastTimer::DeclareTimer FTM_URL_ADJUST_TIMEOUT("Adjust Timeout");
|
|
LLFastTimer t(FTM_URL_ADJUST_TIMEOUT);
|
|
// This is a pretty sloppy calculation, but this
|
|
// tries to make the gross assumption that if data
|
|
// is coming in at 56kb/s, then this transfer will
|
|
// probably succeed. So, if we're accumlated
|
|
// 100,000 bytes (MIN_ACCUMULATION) then let's
|
|
// give this client another 2s to complete.
|
|
const F32 TIMEOUT_ADJUSTMENT = 2.0f;
|
|
mDetail->mByteAccumulator = 0;
|
|
pump->adjustTimeoutSeconds(TIMEOUT_ADJUSTMENT);
|
|
lldebugs << "LLURLRequest adjustTimeoutSeconds for request: " << mDetail->mURL << llendl;
|
|
if (mState == STATE_INITIALIZED)
|
|
{
|
|
llinfos << "LLURLRequest adjustTimeoutSeconds called during upload" << llendl;
|
|
}
|
|
}
|
|
|
|
switch(mState)
|
|
{
|
|
case STATE_INITIALIZED:
|
|
{
|
|
PUMP_DEBUG;
|
|
// We only need to wait for input if we are uploading
|
|
// something.
|
|
if(((HTTP_PUT == mAction) || (HTTP_POST == mAction)) && !eos)
|
|
{
|
|
// we're waiting to get all of the information
|
|
return STATUS_BREAK;
|
|
}
|
|
|
|
// *FIX: bit of a hack, but it should work. The configure and
|
|
// callback method expect this information to be ready.
|
|
mDetail->mResponseBuffer = buffer;
|
|
mDetail->mChannels = channels;
|
|
if(!configure())
|
|
{
|
|
return STATUS_ERROR;
|
|
}
|
|
mRemoved = false;
|
|
mState = STATE_WAITING_FOR_RESPONSE;
|
|
mDetail->mStateMachine->addRequest(); // Add easy handle to multi handle.
|
|
|
|
return STATUS_BREAK;
|
|
}
|
|
case STATE_WAITING_FOR_RESPONSE:
|
|
case STATE_PROCESSING_RESPONSE:
|
|
{
|
|
if (!mRemoved) // Not removed from multi handle yet?
|
|
{
|
|
// Easy handle is still being processed.
|
|
return STATUS_BREAK;
|
|
}
|
|
// Curl thread finished with this easy handle.
|
|
mState = STATE_CURL_FINISHED;
|
|
}
|
|
case STATE_CURL_FINISHED:
|
|
{
|
|
PUMP_DEBUG;
|
|
LLIOPipe::EStatus status = STATUS_NO_CONNECTION; // Catch-all failure code.
|
|
|
|
// Left braces in order not to change indentation.
|
|
{
|
|
CURLcode result;
|
|
|
|
static LLFastTimer::DeclareTimer FTM_PROCESS_URL_REQUEST_GET_RESULT("Get Result");
|
|
|
|
AICurlEasyRequest_wat(*mCurlEasyRequest)->getResult(&result);
|
|
|
|
mState = STATE_HAVE_RESPONSE;
|
|
context[CONTEXT_REQUEST][CONTEXT_TRANSFERED_BYTES] = mRequestTransferedBytes;
|
|
context[CONTEXT_RESPONSE][CONTEXT_TRANSFERED_BYTES] = mResponseTransferedBytes;
|
|
lldebugs << this << "Setting context to " << context << llendl;
|
|
switch(result)
|
|
{
|
|
case CURLE_OK:
|
|
case CURLE_WRITE_ERROR:
|
|
// NB: The error indication means that we stopped the
|
|
// writing due the body limit being reached
|
|
if(mCompletionCallback && pump)
|
|
{
|
|
LLURLRequestComplete* complete = NULL;
|
|
complete = (LLURLRequestComplete*)
|
|
mCompletionCallback.get();
|
|
complete->responseStatus(
|
|
result == CURLE_OK
|
|
? STATUS_OK : STATUS_STOP);
|
|
LLPumpIO::links_t chain;
|
|
LLPumpIO::LLLinkInfo link;
|
|
link.mPipe = mCompletionCallback;
|
|
link.mChannels = LLBufferArray::makeChannelConsumer(
|
|
channels);
|
|
chain.push_back(link);
|
|
static LLFastTimer::DeclareTimer FTM_PROCESS_URL_PUMP_RESPOND("Pump Respond");
|
|
{
|
|
LLFastTimer t(FTM_PROCESS_URL_PUMP_RESPOND);
|
|
pump->respond(chain, buffer, context);
|
|
}
|
|
mCompletionCallback = NULL;
|
|
}
|
|
status = STATUS_BREAK; // This is what the old code returned. Does it make sense?
|
|
break;
|
|
case CURLE_FAILED_INIT:
|
|
case CURLE_COULDNT_CONNECT:
|
|
status = STATUS_NO_CONNECTION;
|
|
break;
|
|
default:
|
|
llwarns << "URLRequest Error: " << result
|
|
<< ", "
|
|
<< LLCurl::strerror(result)
|
|
<< ", "
|
|
<< (mDetail->mURL.empty() ? "<EMPTY URL>" : mDetail->mURL)
|
|
<< llendl;
|
|
status = STATUS_ERROR;
|
|
break;
|
|
}
|
|
}
|
|
return status;
|
|
}
|
|
case STATE_HAVE_RESPONSE:
|
|
PUMP_DEBUG;
|
|
// we already stuffed everything into channel in in the curl
|
|
// callback, so we are done.
|
|
eos = true;
|
|
context[CONTEXT_REQUEST][CONTEXT_TRANSFERED_BYTES] = mRequestTransferedBytes;
|
|
context[CONTEXT_RESPONSE][CONTEXT_TRANSFERED_BYTES] = mResponseTransferedBytes;
|
|
lldebugs << this << "Setting context to " << context << llendl;
|
|
return STATUS_DONE;
|
|
|
|
default:
|
|
PUMP_DEBUG;
|
|
context[CONTEXT_REQUEST][CONTEXT_TRANSFERED_BYTES] = mRequestTransferedBytes;
|
|
context[CONTEXT_RESPONSE][CONTEXT_TRANSFERED_BYTES] = mResponseTransferedBytes;
|
|
lldebugs << this << "Setting context to " << context << llendl;
|
|
return STATUS_ERROR;
|
|
}
|
|
}
|
|
#endif // AI_UNUSED
|
|
|
|
bool LLURLRequest::configure(AICurlEasyRequest_wat const& curlEasyRequest_w)
|
|
{
|
|
bool rv = false;
|
|
{
|
|
switch(mAction)
|
|
{
|
|
case HTTP_HEAD:
|
|
curlEasyRequest_w->setopt(CURLOPT_HEADER, 1);
|
|
curlEasyRequest_w->setopt(CURLOPT_NOBODY, 1);
|
|
curlEasyRequest_w->setopt(CURLOPT_FOLLOWLOCATION, 1);
|
|
rv = true;
|
|
break;
|
|
|
|
case HTTP_GET:
|
|
curlEasyRequest_w->setopt(CURLOPT_HTTPGET, 1);
|
|
curlEasyRequest_w->setopt(CURLOPT_FOLLOWLOCATION, 1);
|
|
|
|
// Set Accept-Encoding to allow response compression
|
|
curlEasyRequest_w->setoptString(CURLOPT_ENCODING, "");
|
|
rv = true;
|
|
break;
|
|
|
|
case HTTP_PUT:
|
|
{
|
|
// Disable the expect http 1.1 extension. POST and PUT default
|
|
// to turning this on, and I am not too sure what it means.
|
|
curlEasyRequest_w->addHeader("Expect:");
|
|
curlEasyRequest_w->setopt(CURLOPT_UPLOAD, 1);
|
|
curlEasyRequest_w->setopt(CURLOPT_INFILESIZE, mBodySize);
|
|
rv = true;
|
|
break;
|
|
}
|
|
case HTTP_POST:
|
|
{
|
|
// Set the handle for an http post
|
|
curlEasyRequest_w->setPost(mBodySize);
|
|
rv = true;
|
|
break;
|
|
}
|
|
case HTTP_DELETE:
|
|
// Set the handle for an http post
|
|
curlEasyRequest_w->setoptString(CURLOPT_CUSTOMREQUEST, "DELETE");
|
|
rv = true;
|
|
break;
|
|
|
|
case HTTP_MOVE:
|
|
// Set the handle for an http post
|
|
curlEasyRequest_w->setoptString(CURLOPT_CUSTOMREQUEST, "MOVE");
|
|
rv = true;
|
|
break;
|
|
|
|
default:
|
|
llwarns << "Unhandled URLRequest action: " << mAction << llendl;
|
|
break;
|
|
}
|
|
if(rv)
|
|
{
|
|
curlEasyRequest_w->finalizeRequest(mURL, mResponder->getHTTPTimeoutPolicy(), this);
|
|
}
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
#if 0
|
|
// static
|
|
size_t LLURLRequest::downCallback(
|
|
char* data,
|
|
size_t size,
|
|
size_t nmemb,
|
|
void* user)
|
|
{
|
|
LLURLRequest* req = (LLURLRequest*)user;
|
|
if(STATE_WAITING_FOR_RESPONSE == req->mState)
|
|
{
|
|
req->mState = STATE_PROCESSING_RESPONSE;
|
|
}
|
|
U32 bytes = size * nmemb;
|
|
if (req->mDetail->mIsBodyLimitSet)
|
|
{
|
|
if (bytes > req->mDetail->mBodyLimit)
|
|
{
|
|
bytes = req->mDetail->mBodyLimit;
|
|
req->mDetail->mBodyLimit = 0;
|
|
}
|
|
else
|
|
{
|
|
req->mDetail->mBodyLimit -= bytes;
|
|
}
|
|
}
|
|
|
|
req->mDetail->mResponseBuffer->append(
|
|
req->mDetail->mChannels.out(),
|
|
(U8*)data,
|
|
bytes);
|
|
req->mResponseTransferedBytes += bytes;
|
|
req->mDetail->mByteAccumulator += bytes;
|
|
return bytes;
|
|
}
|
|
|
|
// static
|
|
size_t LLURLRequest::upCallback(
|
|
char* data,
|
|
size_t size,
|
|
size_t nmemb,
|
|
void* user)
|
|
{
|
|
LLURLRequest* req = (LLURLRequest*)user;
|
|
S32 bytes = llmin(
|
|
(S32)(size * nmemb),
|
|
req->mDetail->mResponseBuffer->countAfter(
|
|
req->mDetail->mChannels.in(),
|
|
req->mDetail->mLastRead));
|
|
req->mDetail->mLastRead = req->mDetail->mResponseBuffer->readAfter(
|
|
req->mDetail->mChannels.in(),
|
|
req->mDetail->mLastRead,
|
|
(U8*)data,
|
|
bytes);
|
|
req->mRequestTransferedBytes += bytes;
|
|
return bytes;
|
|
}
|
|
|
|
static size_t headerCallback(char* header_line, size_t size, size_t nmemb, void* user)
|
|
{
|
|
size_t header_len = size * nmemb;
|
|
LLURLRequestComplete* complete = (LLURLRequestComplete*)user;
|
|
|
|
if (!complete || !header_line)
|
|
{
|
|
return header_len;
|
|
}
|
|
|
|
// *TODO: This should be a utility in llstring.h: isascii()
|
|
for (size_t i = 0; i < header_len; ++i)
|
|
{
|
|
if (header_line[i] < 0)
|
|
{
|
|
return header_len;
|
|
}
|
|
}
|
|
|
|
std::string header(header_line, header_len);
|
|
|
|
// Per HTTP spec the first header line must be the status line.
|
|
if (header.substr(0,5) == "HTTP/")
|
|
{
|
|
std::string::iterator end = header.end();
|
|
std::string::iterator pos1 = std::find(header.begin(), end, ' ');
|
|
if (pos1 != end) ++pos1;
|
|
std::string::iterator pos2 = std::find(pos1, end, ' ');
|
|
if (pos2 != end) ++pos2;
|
|
std::string::iterator pos3 = std::find(pos2, end, '\r');
|
|
|
|
std::string version(header.begin(), pos1);
|
|
std::string status(pos1, pos2);
|
|
std::string reason(pos2, pos3);
|
|
|
|
S32 status_code = atoi(status.c_str());
|
|
if (status_code > 0)
|
|
{
|
|
complete->httpStatus((U32)status_code, reason);
|
|
return header_len;
|
|
}
|
|
}
|
|
|
|
std::string::iterator sep = std::find(header.begin(),header.end(),':');
|
|
|
|
if (sep != header.end())
|
|
{
|
|
std::string key(header.begin(), sep);
|
|
std::string value(sep + 1, header.end());
|
|
|
|
key = utf8str_tolower(utf8str_trim(key));
|
|
value = utf8str_trim(value);
|
|
|
|
complete->header(key, value);
|
|
}
|
|
else
|
|
{
|
|
LLStringUtil::trim(header);
|
|
if (!header.empty())
|
|
{
|
|
llwarns << "Unable to parse header: " << header << llendl;
|
|
}
|
|
}
|
|
|
|
return header_len;
|
|
}
|
|
#endif
|
|
|
|
#ifdef AI_UNUSED
|
|
/**
|
|
* LLURLRequestComplete
|
|
*/
|
|
LLURLRequestComplete::LLURLRequestComplete() :
|
|
mRequestStatus(LLIOPipe::STATUS_ERROR)
|
|
{
|
|
}
|
|
|
|
// virtual
|
|
LLURLRequestComplete::~LLURLRequestComplete()
|
|
{
|
|
}
|
|
|
|
//virtual
|
|
void LLURLRequestComplete::header(const std::string& header, const std::string& value)
|
|
{
|
|
}
|
|
|
|
//virtual
|
|
void LLURLRequestComplete::complete(const LLChannelDescriptors& channels,
|
|
const buffer_ptr_t& buffer)
|
|
{
|
|
if(STATUS_OK == mRequestStatus)
|
|
{
|
|
response(channels, buffer);
|
|
}
|
|
else
|
|
{
|
|
noResponse();
|
|
}
|
|
}
|
|
|
|
//virtual
|
|
void LLURLRequestComplete::response(const LLChannelDescriptors& channels,
|
|
const buffer_ptr_t& buffer)
|
|
{
|
|
llwarns << "LLURLRequestComplete::response default implementation called"
|
|
<< llendl;
|
|
}
|
|
|
|
//virtual
|
|
void LLURLRequestComplete::noResponse()
|
|
{
|
|
llwarns << "LLURLRequestComplete::noResponse default implementation called"
|
|
<< llendl;
|
|
}
|
|
|
|
void LLURLRequestComplete::responseStatus(LLIOPipe::EStatus status)
|
|
{
|
|
mRequestStatus = status;
|
|
}
|
|
|
|
static LLFastTimer::DeclareTimer FTM_PROCESS_URL_COMPLETE("URL Complete");
|
|
// virtual
|
|
LLIOPipe::EStatus LLURLRequestComplete::process_impl(
|
|
const LLChannelDescriptors& channels,
|
|
buffer_ptr_t& buffer,
|
|
bool& eos,
|
|
LLSD& context,
|
|
LLPumpIO* pump)
|
|
{
|
|
LLFastTimer t(FTM_PROCESS_URL_COMPLETE);
|
|
PUMP_DEBUG;
|
|
complete(channels, buffer);
|
|
return STATUS_OK;
|
|
}
|
|
#endif
|