mirror of
https://github.com/pocoproject/poco.git
synced 2025-10-29 20:59:45 +01:00
Poco::Net::WebSocket::setMaxPayloadSize() and Poco::Net::WebSocket::getMaxPayloadSize()
to specify a maximum acceptable payload size for Poco::Net::WebSocket::receiveFrame()
This commit is contained in:
@@ -44,7 +44,7 @@ WebSocket::WebSocket(HTTPServerRequest& request, HTTPServerResponse& response):
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
|
||||
WebSocket::WebSocket(HTTPClientSession& cs, HTTPRequest& request, HTTPResponse& response):
|
||||
StreamSocket(connect(cs, request, response, _defaultCreds))
|
||||
{
|
||||
@@ -57,7 +57,7 @@ WebSocket::WebSocket(HTTPClientSession& cs, HTTPRequest& request, HTTPResponse&
|
||||
}
|
||||
|
||||
|
||||
WebSocket::WebSocket(const Socket& socket):
|
||||
WebSocket::WebSocket(const Socket& socket):
|
||||
StreamSocket(socket)
|
||||
{
|
||||
if (!dynamic_cast<WebSocketImpl*>(impl()))
|
||||
@@ -111,7 +111,7 @@ int WebSocket::receiveFrame(void* buffer, int length, int& flags)
|
||||
return n;
|
||||
}
|
||||
|
||||
|
||||
|
||||
int WebSocket::receiveFrame(Poco::Buffer<char>& buffer, int& flags)
|
||||
{
|
||||
int n = static_cast<WebSocketImpl*>(impl())->receiveBytes(buffer, 0);
|
||||
@@ -126,6 +126,18 @@ WebSocket::Mode WebSocket::mode() const
|
||||
}
|
||||
|
||||
|
||||
void WebSocket::setMaxPayloadSize(int maxPayloadSize)
|
||||
{
|
||||
static_cast<WebSocketImpl*>(impl())->setMaxPayloadSize(maxPayloadSize);
|
||||
}
|
||||
|
||||
|
||||
int WebSocket::getMaxPayloadSize() const
|
||||
{
|
||||
return static_cast<WebSocketImpl*>(impl())->getMaxPayloadSize();
|
||||
}
|
||||
|
||||
|
||||
WebSocketImpl* WebSocket::accept(HTTPServerRequest& request, HTTPServerResponse& response)
|
||||
{
|
||||
if (request.hasToken("Connection", "upgrade") && icompare(request.get("Upgrade", ""), "websocket") == 0)
|
||||
@@ -136,14 +148,14 @@ WebSocketImpl* WebSocket::accept(HTTPServerRequest& request, HTTPServerResponse&
|
||||
std::string key = request.get("Sec-WebSocket-Key", "");
|
||||
Poco::trimInPlace(key);
|
||||
if (key.empty()) throw WebSocketException("Missing Sec-WebSocket-Key in handshake request", WS_ERR_HANDSHAKE_NO_KEY);
|
||||
|
||||
|
||||
response.setStatusAndReason(HTTPResponse::HTTP_SWITCHING_PROTOCOLS);
|
||||
response.set("Upgrade", "websocket");
|
||||
response.set("Connection", "Upgrade");
|
||||
response.set("Sec-WebSocket-Accept", computeAccept(key));
|
||||
response.setContentLength(0);
|
||||
response.setContentLength(HTTPResponse::UNKNOWN_CONTENT_LENGTH);
|
||||
response.send().flush();
|
||||
|
||||
|
||||
HTTPServerRequestImpl& requestImpl = static_cast<HTTPServerRequestImpl&>(request);
|
||||
return new WebSocketImpl(static_cast<StreamSocketImpl*>(requestImpl.detachSocket().impl()), requestImpl.session(), false);
|
||||
}
|
||||
@@ -205,7 +217,7 @@ WebSocketImpl* WebSocket::connect(HTTPClientSession& cs, HTTPRequest& request, H
|
||||
WebSocketImpl* WebSocket::completeHandshake(HTTPClientSession& cs, HTTPResponse& response, const std::string& key)
|
||||
{
|
||||
std::string connection = response.get("Connection", "");
|
||||
if (Poco::icompare(connection, "Upgrade") != 0)
|
||||
if (Poco::icompare(connection, "Upgrade") != 0)
|
||||
throw WebSocketException("No Connection: Upgrade header in handshake response", WS_ERR_NO_HANDSHAKE);
|
||||
std::string upgrade = response.get("Upgrade", "");
|
||||
if (Poco::icompare(upgrade, "websocket") != 0)
|
||||
|
||||
@@ -21,9 +21,13 @@
|
||||
#include "Poco/BinaryReader.h"
|
||||
#include "Poco/MemoryStream.h"
|
||||
#include "Poco/Format.h"
|
||||
#include <limits>
|
||||
#include <cstring>
|
||||
|
||||
|
||||
#undef max
|
||||
|
||||
|
||||
namespace Poco {
|
||||
namespace Net {
|
||||
|
||||
@@ -31,6 +35,7 @@ namespace Net {
|
||||
WebSocketImpl::WebSocketImpl(StreamSocketImpl* pStreamSocketImpl, HTTPSession& session, bool mustMaskPayload):
|
||||
StreamSocketImpl(pStreamSocketImpl->sockfd()),
|
||||
_pStreamSocketImpl(pStreamSocketImpl),
|
||||
_maxPayloadSize(std::numeric_limits<int>::max()),
|
||||
_buffer(0),
|
||||
_bufferOffset(0),
|
||||
_frameFlags(0),
|
||||
@@ -134,6 +139,7 @@ int WebSocketImpl::receiveHeader(char mask[4], bool& useMask)
|
||||
Poco::BinaryReader reader(istr, Poco::BinaryReader::NETWORK_BYTE_ORDER);
|
||||
Poco::UInt64 l;
|
||||
reader >> l;
|
||||
if (l > _maxPayloadSize) throw WebSocketException("Payload too big", WebSocket::WS_ERR_PAYLOAD_TOO_BIG);
|
||||
payloadLength = static_cast<int>(l);
|
||||
}
|
||||
else if (lengthByte == 126)
|
||||
@@ -148,10 +154,12 @@ int WebSocketImpl::receiveHeader(char mask[4], bool& useMask)
|
||||
Poco::BinaryReader reader(istr, Poco::BinaryReader::NETWORK_BYTE_ORDER);
|
||||
Poco::UInt16 l;
|
||||
reader >> l;
|
||||
if (l > _maxPayloadSize) throw WebSocketException("Payload too big", WebSocket::WS_ERR_PAYLOAD_TOO_BIG);
|
||||
payloadLength = static_cast<int>(l);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (lengthByte > _maxPayloadSize) throw WebSocketException("Payload too big", WebSocket::WS_ERR_PAYLOAD_TOO_BIG);
|
||||
payloadLength = lengthByte;
|
||||
}
|
||||
|
||||
@@ -169,6 +177,14 @@ int WebSocketImpl::receiveHeader(char mask[4], bool& useMask)
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::setMaxPayloadSize(int maxPayloadSize)
|
||||
{
|
||||
poco_assert (maxPayloadSize > 0);
|
||||
|
||||
_maxPayloadSize = maxPayloadSize;
|
||||
}
|
||||
|
||||
|
||||
int WebSocketImpl::receivePayload(char *buffer, int payloadLength, char mask[4], bool useMask)
|
||||
{
|
||||
int received = receiveNBytes(reinterpret_cast<char*>(buffer), payloadLength);
|
||||
@@ -205,7 +221,7 @@ int WebSocketImpl::receiveBytes(Poco::Buffer<char>& buffer, int)
|
||||
int payloadLength = receiveHeader(mask, useMask);
|
||||
if (payloadLength <= 0)
|
||||
return payloadLength;
|
||||
int oldSize = buffer.size();
|
||||
std::size_t oldSize = buffer.size();
|
||||
buffer.resize(oldSize + payloadLength);
|
||||
return receivePayload(buffer.begin() + oldSize, payloadLength, mask, useMask);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user