2012-10-29 07:19:47 +00:00
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
|
|
#include <Poco/Net/HTTPServerResponse.h>
|
|
|
|
|
|
|
|
|
|
#include <DB/Core/Exception.h>
|
|
|
|
|
#include <DB/Core/ErrorCodes.h>
|
|
|
|
|
|
|
|
|
|
#include <DB/IO/WriteBuffer.h>
|
|
|
|
|
#include <DB/IO/BufferWithOwnMemory.h>
|
2015-03-26 08:47:02 +00:00
|
|
|
|
#include <statdaemons/NetException.h>
|
2012-10-29 07:19:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
namespace DB
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** Отличается от WriteBufferFromOStream тем, что инициализируется не std::ostream, а Poco::Net::HTTPServerResponse.
|
|
|
|
|
* При первом сбросе данных, получает из него std::ostream (с помощью метода send).
|
|
|
|
|
* Это нужно в HTTP серверах, чтобы после передачи в какой-нибудь метод WriteBuffer-а,
|
|
|
|
|
* но до вывода первых данных клиенту, можно было изменить какие-нибудь HTTP заголовки (например, код ответа).
|
|
|
|
|
* (После вызова Poco::Net::HTTPServerResponse::send() изменить заголовки уже нельзя.)
|
|
|
|
|
* То есть, суть в том, чтобы вызывать метод Poco::Net::HTTPServerResponse::send() не сразу.
|
|
|
|
|
*/
|
|
|
|
|
class WriteBufferFromHTTPServerResponse : public BufferWithOwnMemory<WriteBuffer>
|
|
|
|
|
{
|
|
|
|
|
private:
|
|
|
|
|
Poco::Net::HTTPServerResponse & response;
|
2014-04-08 07:31:51 +00:00
|
|
|
|
std::ostream * ostr = nullptr;
|
2012-10-29 07:19:47 +00:00
|
|
|
|
|
|
|
|
|
void nextImpl()
|
|
|
|
|
{
|
|
|
|
|
if (!ostr)
|
|
|
|
|
ostr = &response.send();
|
|
|
|
|
|
|
|
|
|
if (!offset())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
ostr->write(working_buffer.begin(), offset());
|
|
|
|
|
ostr->flush();
|
|
|
|
|
|
|
|
|
|
if (!ostr->good())
|
2015-03-26 08:47:02 +00:00
|
|
|
|
throw NetException("Cannot write to ostream", ErrorCodes::CANNOT_WRITE_TO_OSTREAM);
|
2012-10-29 07:19:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
WriteBufferFromHTTPServerResponse(Poco::Net::HTTPServerResponse & response_, size_t size = DBMS_DEFAULT_BUFFER_SIZE)
|
2014-04-08 07:31:51 +00:00
|
|
|
|
: BufferWithOwnMemory<WriteBuffer>(size), response(response_) {}
|
2012-10-29 07:19:47 +00:00
|
|
|
|
|
2013-12-17 19:45:18 +00:00
|
|
|
|
/** Если данные ещё не были отправлены - отправить хотя бы HTTP заголовки.
|
|
|
|
|
* Используйте эту функцию после того, как данные, возможно, были отправлены,
|
|
|
|
|
* и не было ошибок (вы не планируете поменять код ответа).
|
|
|
|
|
*/
|
|
|
|
|
void finalize()
|
|
|
|
|
{
|
|
|
|
|
if (!ostr)
|
|
|
|
|
ostr = &response.send();
|
|
|
|
|
}
|
|
|
|
|
|
2012-10-29 07:19:47 +00:00
|
|
|
|
~WriteBufferFromHTTPServerResponse()
|
|
|
|
|
{
|
2013-11-28 20:05:54 +00:00
|
|
|
|
if (!offset())
|
|
|
|
|
return;
|
|
|
|
|
|
2013-11-18 17:17:45 +00:00
|
|
|
|
try
|
2012-11-19 04:16:00 +00:00
|
|
|
|
{
|
2012-10-29 07:19:47 +00:00
|
|
|
|
next();
|
2012-11-19 04:16:00 +00:00
|
|
|
|
}
|
2013-11-18 17:17:45 +00:00
|
|
|
|
catch (...)
|
|
|
|
|
{
|
2013-11-18 19:18:03 +00:00
|
|
|
|
tryLogCurrentException(__PRETTY_FUNCTION__);
|
2013-11-18 17:17:45 +00:00
|
|
|
|
}
|
2012-10-29 07:19:47 +00:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
}
|