/****************************************************************************** * Icinga 2 * * Copyright (C) 2012-2018 Icinga Development Team (https://icinga.com/) * * * * This program is free software; you can redistribute it and/or * * modify it under the terms of the GNU General Public License * * as published by the Free Software Foundation; either version 2 * * of the License, or (at your option) any later version. * * * * This program 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 General Public License for more details. * * * * You should have received a copy of the GNU General Public License * * along with this program; if not, write to the Free Software Foundation * * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. * ******************************************************************************/ #include "remote/httpchunkedencoding.hpp" #include using namespace icinga; StreamReadStatus HttpChunkedEncoding::ReadChunkFromStream(const Stream::Ptr& stream, char **data, size_t *size, ChunkReadContext& context, bool may_wait) { if (context.LengthIndicator == -1) { String line; StreamReadStatus status = stream->ReadLine(&line, context.StreamContext, may_wait); may_wait = false; if (status != StatusNewItem) return status; std::stringstream msgbuf; msgbuf << std::hex << line; msgbuf >> context.LengthIndicator; if (context.LengthIndicator < 0) BOOST_THROW_EXCEPTION(std::invalid_argument("HTTP chunk length must not be negative.")); } StreamReadContext& scontext = context.StreamContext; if (scontext.Eof) return StatusEof; if (scontext.MustRead) { if (!scontext.FillFromStream(stream, may_wait)) { scontext.Eof = true; return StatusEof; } scontext.MustRead = false; } size_t NewlineLength = context.LengthIndicator ? 2 : 0; if (scontext.Size < (size_t)context.LengthIndicator + NewlineLength) { scontext.MustRead = true; return StatusNeedData; } *data = new char[context.LengthIndicator]; *size = context.LengthIndicator; memcpy(*data, scontext.Buffer, context.LengthIndicator); scontext.DropData(context.LengthIndicator + NewlineLength); context.LengthIndicator = -1; return StatusNewItem; } void HttpChunkedEncoding::WriteChunkToStream(const Stream::Ptr& stream, const char *data, size_t count) { std::ostringstream msgbuf; msgbuf << std::hex << count << "\r\n"; String lengthIndicator = msgbuf.str(); stream->Write(lengthIndicator.CStr(), lengthIndicator.GetLength()); stream->Write(data, count); if (count > 0) stream->Write("\r\n", 2); }