gecko-dev/netwerk/protocol/http/src/nsHttpChunkedDecoder.cpp

134 lines
3.9 KiB
C++
Raw Normal View History

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
* The contents of this file are subject to the Mozilla Public
* License Version 1.1 (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.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
* implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code is Mozilla.
*
* The Initial Developer of the Original Code is Netscape
* Communications. Portions created by Netscape Communications are
* Copyright (C) 2001 by Netscape Communications. All
* Rights Reserved.
*
* Contributor(s):
* Darin Fisher <darin@netscape.com> (original author)
*/
#include "nsHttpChunkedDecoder.h"
#include "nsHttp.h"
//-----------------------------------------------------------------------------
// nsHttpChunkedDecoder <public>
//-----------------------------------------------------------------------------
nsresult
nsHttpChunkedDecoder::HandleChunkedContent(char *buf,
PRUint32 count,
PRUint32 *countRead)
{
LOG(("nsHttpChunkedDecoder::HandleChunkedContent [count=%u]\n", count));
*countRead = 0;
while (count) {
if (mChunkRemaining) {
PRUint32 amt = PR_MIN(mChunkRemaining, count);
count -= amt;
mChunkRemaining -= amt;
*countRead += amt;
buf += amt;
}
if (count) {
PRUint32 bytesConsumed = 0;
nsresult rv = ParseChunkRemaining(buf, count, &bytesConsumed);
if (NS_FAILED(rv)) return rv;
count -= bytesConsumed;
if (count) {
// shift buf by bytesConsumed
memmove(buf, buf + bytesConsumed, count);
}
}
}
return NS_OK;
}
//-----------------------------------------------------------------------------
// nsHttpChunkedDecoder <private>
//-----------------------------------------------------------------------------
nsresult
nsHttpChunkedDecoder::ParseChunkRemaining(char *buf,
PRUint32 count,
PRUint32 *countRead)
{
NS_PRECONDITION(mChunkRemaining == 0, "chunk remaining should be zero");
char *p;
// we may have a partial line from the last time we were called
if (mLineBuf.IsEmpty()) {
p = PL_strstr(buf, "\r\n");
if (p) {
*p = 0;
p = buf;
}
}
else {
p = PL_strstr(buf, "\r\n");
// it would be really sad if this ever returned null, b/c it
// would mean we were reading really small chunks from the net.
if (p) {
*p = 0;
// append to the line buf and use that as the buf to parse.
mLineBuf.Append(buf);
p = (char *) mLineBuf.get();
}
}
if (!p) {
mLineBuf.Append(buf, count);
*countRead = count;
return NS_OK;
}
// indicate that the entire line has been consumed... the \r has been
// nulled out, so we have to add 2 to include the \r\n in the count read.
*countRead = PL_strlen(buf) + 2;
buf = p;
if ((p = PL_strchr(buf, ';')) != nsnull) {
// ignore any chunk-extensions
*p = 0;
}
if (!sscanf(buf, "%x", &mChunkRemaining)) {
LOG(("sscanf failed parsing hex on string [%s]\n", buf));
return NS_ERROR_UNEXPECTED;
}
if ((mChunkRemaining == 0) && (*buf != 0)) {
// XXX need to add code to consume "trailer" headers
mReachedEOF = PR_TRUE;
}
// clear the line buffer if it is not already empty
if (!mLineBuf.IsEmpty())
mLineBuf.SetLength(0);
return NS_OK;
}