mirror of
https://github.com/pocoproject/poco.git
synced 2025-01-18 00:15:27 +01:00
trunk: add websocket class
This commit is contained in:
parent
7d7c02c579
commit
d066518edd
265
Net/include/Poco/Net/WebSocket.h
Normal file
265
Net/include/Poco/Net/WebSocket.h
Normal file
@ -0,0 +1,265 @@
|
||||
//
|
||||
// WebSocket.h
|
||||
//
|
||||
// $Id: //poco/1.4/Net/include/Poco/Net/WebSocket.h#1 $
|
||||
//
|
||||
// Library: Net
|
||||
// Package: WebSocket
|
||||
// Module: WebSocket
|
||||
//
|
||||
// Definition of the WebSocket class.
|
||||
//
|
||||
// Copyright (c) 2012, Applied Informatics Software Engineering GmbH.
|
||||
// and Contributors.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person or organization
|
||||
// obtaining a copy of the software and accompanying documentation covered by
|
||||
// this license (the "Software") to use, reproduce, display, distribute,
|
||||
// execute, and transmit the Software, and to prepare derivative works of the
|
||||
// Software, and to permit third-parties to whom the Software is furnished to
|
||||
// do so, all subject to the following:
|
||||
//
|
||||
// The copyright notices in the Software and this entire statement, including
|
||||
// the above license grant, this restriction and the following disclaimer,
|
||||
// must be included in all copies of the Software, in whole or in part, and
|
||||
// all derivative works of the Software, unless such copies or derivative
|
||||
// works are solely in the form of machine-executable object code generated by
|
||||
// a source language processor.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
// FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
||||
// SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
||||
// FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
||||
// ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
// DEALINGS IN THE SOFTWARE.
|
||||
//
|
||||
|
||||
|
||||
#ifndef Net_WebSocket_INCLUDED
|
||||
#define Net_WebSocket_INCLUDED
|
||||
|
||||
|
||||
#include "Poco/Net/Net.h"
|
||||
#include "Poco/Net/StreamSocket.h"
|
||||
#include "Poco/Net/HTTPCredentials.h"
|
||||
|
||||
|
||||
namespace Poco {
|
||||
namespace Net {
|
||||
|
||||
|
||||
class WebSocketImpl;
|
||||
class HTTPServerRequest;
|
||||
class HTTPServerResponse;
|
||||
class HTTPClientSession;
|
||||
|
||||
|
||||
class Net_API WebSocket: public StreamSocket
|
||||
/// This class implements a WebSocket according
|
||||
/// to the WebSocket protocol specification in RFC 6455.
|
||||
///
|
||||
/// Both client-side and server-side WebSockets
|
||||
/// are supported.
|
||||
///
|
||||
/// Server-side WebSockets are usually created from within
|
||||
/// a HTTPRequestHandler.
|
||||
///
|
||||
/// Client-side WebSockets are created using a HTTPClientSession.
|
||||
///
|
||||
/// Note that special frames like PING must be handled at
|
||||
/// application level. In the case of a PING, a PONG message
|
||||
/// must be returned.
|
||||
{
|
||||
public:
|
||||
enum Mode
|
||||
{
|
||||
WS_SERVER, /// Server-side WebSocket.
|
||||
WS_CLIENT /// Client-side WebSocket.
|
||||
};
|
||||
|
||||
enum FrameFlags
|
||||
/// Frame header flags.
|
||||
{
|
||||
FRAME_FLAG_FIN = 0x80, /// FIN bit: final fragment of a multi-fragment message.
|
||||
FRAME_FLAG_RSV1 = 0x40, /// Reserved for future use. Must be zero.
|
||||
FRAME_FLAG_RSV2 = 0x20, /// Reserved for future use. Must be zero.
|
||||
FRAME_FLAG_RSV3 = 0x10, /// Reserved for future use. Must be zero.
|
||||
};
|
||||
|
||||
enum FrameOpcodes
|
||||
/// Frame header opcodes.
|
||||
{
|
||||
FRAME_OP_CONT = 0x00, /// Continuation frame.
|
||||
FRAME_OP_TEXT = 0x01, /// Text frame.
|
||||
FRAME_OP_BINARY = 0x02, /// Binary frame.
|
||||
FRAME_OP_CLOSE = 0x08, /// Close connection.
|
||||
FRAME_OP_PING = 0x09, /// Ping frame.
|
||||
FRAME_OP_PONG = 0x0a, /// Pong frame.
|
||||
FRAME_OP_BITMASK = 0x0f /// Bit mask for opcodes.
|
||||
};
|
||||
|
||||
enum SendFlags
|
||||
/// Combined header flags and opcodes for use with sendFrame().
|
||||
{
|
||||
FRAME_TEXT = FRAME_FLAG_FIN | FRAME_OP_TEXT,
|
||||
/// Use this for sending a single text (UTF-8) payload frame.
|
||||
FRAME_BINARY = FRAME_FLAG_FIN | FRAME_OP_BINARY
|
||||
/// Use this for sending a single binary payload frame.
|
||||
};
|
||||
|
||||
enum StatusCodes
|
||||
/// StatusCodes for CLOSE frames sent with shutdown().
|
||||
{
|
||||
WS_NORMAL_CLOSE = 1000,
|
||||
WS_ENDPOINT_GOING_AWAY = 1001,
|
||||
WS_PROTOCOL_ERROR = 1002,
|
||||
WS_PAYLOAD_NOT_ACCEPTABLE = 1003,
|
||||
WS_RESERVED = 1004,
|
||||
WS_RESERVED_NO_STATUS_CODE = 1005,
|
||||
WS_RESERVED_ABNORMAL_CLOSE = 1006,
|
||||
WS_MALFORMED_PAYLOAD = 1007,
|
||||
WS_POLICY_VIOLATION = 1008,
|
||||
WS_PAYLOAD_TOO_BIG = 1009,
|
||||
WS_EXTENSION_REQUIRED = 1010,
|
||||
WS_UNEXPECTED_CONDITION = 1011,
|
||||
WS_RESERVED_TLS_FAILURE = 1015
|
||||
};
|
||||
|
||||
enum ErrorCodes
|
||||
/// These error codes can be obtained from a WebSocketException
|
||||
/// to determine the exact cause of the error.
|
||||
{
|
||||
WS_ERR_NO_HANDSHAKE = 1,
|
||||
/// No Connection: Upgrade or Upgrade: websocket header in handshake request.
|
||||
WS_ERR_HANDSHAKE_NO_VERSION = 2,
|
||||
/// No Sec-WebSocket-Version header in handshake request.
|
||||
WS_ERR_HANDSHAKE_UNSUPPORTED_VERSION = 3,
|
||||
/// Unsupported WebSocket version requested by client.
|
||||
WS_ERR_HANDSHAKE_NO_KEY = 4,
|
||||
/// No Sec-WebSocket-Key header in handshake request.
|
||||
WS_ERR_HANDSHAKE_ACCEPT = 5,
|
||||
/// No Sec-WebSocket-Accept header or wrong value.
|
||||
WS_ERR_UNAUTHORIZED = 6,
|
||||
/// The server rejected the username or password for authentication.
|
||||
WS_ERR_PAYLOAD_TOO_BIG = 10,
|
||||
/// Payload too big for supplied buffer.
|
||||
WS_ERR_INCOMPLETE_FRAME = 11
|
||||
/// Incomplete frame received.
|
||||
};
|
||||
|
||||
WebSocket(HTTPServerRequest& request, HTTPServerResponse& response);
|
||||
/// Creates a server-side WebSocket from within a
|
||||
/// HTTPRequestHandler.
|
||||
///
|
||||
/// First verifies that the request is a valid WebSocket upgrade
|
||||
/// request. If so, completes the handshake by sending
|
||||
/// a proper 101 response.
|
||||
///
|
||||
/// Throws an exception if the request is not a proper WebSocket
|
||||
/// upgrade request.
|
||||
|
||||
WebSocket(HTTPClientSession& cs, HTTPRequest& request, HTTPResponse& response);
|
||||
/// Creates a client-side WebSocket, using the given
|
||||
/// HTTPClientSession and HTTPRequest for the initial handshake
|
||||
/// (HTTP Upgrade request).
|
||||
///
|
||||
/// Additional HTTP headers for the initial handshake request
|
||||
/// (such as Origin or Sec-WebSocket-Protocol) can be given
|
||||
/// in the request object.
|
||||
///
|
||||
/// The result of the handshake can be obtained from the response
|
||||
/// object.
|
||||
|
||||
WebSocket(HTTPClientSession& cs, HTTPRequest& request, HTTPResponse& response, HTTPCredentials& credentials);
|
||||
/// Creates a client-side WebSocket, using the given
|
||||
/// HTTPClientSession and HTTPRequest for the initial handshake
|
||||
/// (HTTP Upgrade request).
|
||||
///
|
||||
/// The given credentials are used for authentication
|
||||
/// if requested by the server.
|
||||
///
|
||||
/// Additional HTTP headers for the initial handshake request
|
||||
/// (such as Origin or Sec-WebSocket-Protocol) can be given
|
||||
/// in the request object.
|
||||
///
|
||||
/// The result of the handshake can be obtained from the response
|
||||
/// object.
|
||||
|
||||
virtual ~WebSocket();
|
||||
/// Destroys the StreamSocket.
|
||||
|
||||
WebSocket& operator = (const Socket& socket);
|
||||
/// Assignment operator.
|
||||
///
|
||||
/// Releases the socket's SocketImpl and
|
||||
/// attaches the SocketImpl from the other socket and
|
||||
/// increments the reference count of the SocketImpl.
|
||||
|
||||
void shutdown();
|
||||
/// Sends a Close control frame to the server end of
|
||||
/// the connection to initiate an orderly shutdown
|
||||
/// of the connection.
|
||||
|
||||
void shutdown(Poco::UInt16 statusCode, const std::string& statusMessage = "");
|
||||
/// Sends a Close control frame to the server end of
|
||||
/// the connection to initiate an orderly shutdown
|
||||
/// of the connection.
|
||||
|
||||
int sendFrame(const void* buffer, int length, int flags = FRAME_TEXT);
|
||||
/// Sends the contents of the given buffer through
|
||||
/// the socket as a single frame.
|
||||
///
|
||||
/// Values from the FrameFlags, FrameOpcodes and SendFlags enumerations
|
||||
/// can be specified in flags.
|
||||
///
|
||||
/// Returns the number of bytes sent, which may be
|
||||
/// less than the number of bytes specified.
|
||||
///
|
||||
/// Certain socket implementations may also return a negative
|
||||
/// value denoting a certain condition.
|
||||
|
||||
int receiveFrame(void* buffer, int length, int& flags);
|
||||
/// Receives a frame from the socket and stores it
|
||||
/// in buffer. Up to length bytes are received. If
|
||||
/// the frame's payload is larger, a WebSocketException
|
||||
/// is thrown and the WebSocket connection must be
|
||||
/// terminated.
|
||||
///
|
||||
/// Returns the number of bytes received.
|
||||
/// A return value of 0 means a graceful shutdown
|
||||
/// of the connection from the peer.
|
||||
///
|
||||
/// Throws a TimeoutException if a receive timeout has
|
||||
/// been set and nothing is received within that interval.
|
||||
/// Throws a NetException (or a subclass) in case of other errors.
|
||||
///
|
||||
/// The frame flags and opcode (FrameFlags and FrameOpcodes)
|
||||
/// is stored in flags.
|
||||
|
||||
Mode mode() const;
|
||||
/// Returns WS_SERVER if the WebSocket is a server-side
|
||||
/// WebSocket, or WS_CLIENT otherwise.
|
||||
|
||||
static const std::string WEBSOCKET_VERSION;
|
||||
/// The WebSocket protocol version supported (13).
|
||||
|
||||
protected:
|
||||
static WebSocketImpl* accept(HTTPServerRequest& request, HTTPServerResponse& response);
|
||||
static WebSocketImpl* connect(HTTPClientSession& cs, HTTPRequest& request, HTTPResponse& response, HTTPCredentials& credentials);
|
||||
static WebSocketImpl* completeHandshake(HTTPClientSession& cs, HTTPResponse& response, const std::string& key);
|
||||
static std::string computeAccept(const std::string& key);
|
||||
static std::string createKey();
|
||||
|
||||
private:
|
||||
WebSocket();
|
||||
|
||||
static const std::string WEBSOCKET_GUID;
|
||||
static HTTPCredentials _defaultCreds;
|
||||
};
|
||||
|
||||
|
||||
} } // namespace Poco::Net
|
||||
|
||||
|
||||
#endif // Net_WebSocket_INCLUDED
|
126
Net/include/Poco/Net/WebSocketImpl.h
Normal file
126
Net/include/Poco/Net/WebSocketImpl.h
Normal file
@ -0,0 +1,126 @@
|
||||
//
|
||||
// WebSocketImpl.h
|
||||
//
|
||||
// $Id: //poco/1.4/Net/include/Poco/Net/WebSocketImpl.h#1 $
|
||||
//
|
||||
// Library: Net
|
||||
// Package: WebSocket
|
||||
// Module: WebSocketImpl
|
||||
//
|
||||
// Definition of the StreamSocketImpl class.
|
||||
//
|
||||
// Copyright (c) 2012, Applied Informatics Software Engineering GmbH.
|
||||
// and Contributors.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person or organization
|
||||
// obtaining a copy of the software and accompanying documentation covered by
|
||||
// this license (the "Software") to use, reproduce, display, distribute,
|
||||
// execute, and transmit the Software, and to prepare derivative works of the
|
||||
// Software, and to permit third-parties to whom the Software is furnished to
|
||||
// do so, all subject to the following:
|
||||
//
|
||||
// The copyright notices in the Software and this entire statement, including
|
||||
// the above license grant, this restriction and the following disclaimer,
|
||||
// must be included in all copies of the Software, in whole or in part, and
|
||||
// all derivative works of the Software, unless such copies or derivative
|
||||
// works are solely in the form of machine-executable object code generated by
|
||||
// a source language processor.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
// FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
||||
// SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
||||
// FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
||||
// ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
// DEALINGS IN THE SOFTWARE.
|
||||
//
|
||||
|
||||
|
||||
#ifndef Net_WebSocketImpl_INCLUDED
|
||||
#define Net_WebSocketImpl_INCLUDED
|
||||
|
||||
|
||||
#include "Poco/Net/StreamSocketImpl.h"
|
||||
#include "Poco/Random.h"
|
||||
|
||||
|
||||
namespace Poco {
|
||||
namespace Net {
|
||||
|
||||
|
||||
class Net_API WebSocketImpl: public StreamSocketImpl
|
||||
/// This class implements a WebSocket, according
|
||||
/// to the WebSocket protocol described in RFC 6455.
|
||||
{
|
||||
public:
|
||||
WebSocketImpl(StreamSocketImpl* pStreamSocketImpl, bool mustMaskPayload);
|
||||
/// Creates a StreamSocketImpl using the given native socket.
|
||||
|
||||
// StreamSocketImpl
|
||||
virtual int sendBytes(const void* buffer, int length, int flags);
|
||||
/// Sends a WebSocket protocol frame.
|
||||
|
||||
virtual int receiveBytes(void* buffer, int length, int flags);
|
||||
/// Receives a WebSocket protocol frame.
|
||||
|
||||
virtual SocketImpl* acceptConnection(SocketAddress& clientAddr);
|
||||
virtual void connect(const SocketAddress& address);
|
||||
virtual void connect(const SocketAddress& address, const Poco::Timespan& timeout);
|
||||
virtual void connectNB(const SocketAddress& address, const Poco::Timespan& timeout);
|
||||
virtual void bind(const SocketAddress& address, bool reuseAddress = false);
|
||||
virtual void bind6(const SocketAddress& address, bool reuseAddress = false, bool ipV6Only = false);
|
||||
virtual void listen(int backlog = 64);
|
||||
virtual void close();
|
||||
virtual void shutdownReceive();
|
||||
virtual void shutdownSend();
|
||||
virtual void shutdown();
|
||||
virtual int sendTo(const void* buffer, int length, const SocketAddress& address, int flags = 0);
|
||||
virtual int receiveFrom(void* buffer, int length, SocketAddress& address, int flags = 0);
|
||||
virtual void sendUrgent(unsigned char data);
|
||||
virtual bool secure() const;
|
||||
|
||||
// Internal
|
||||
int frameFlags() const;
|
||||
/// Returns the frame flags of the most recently received frame.
|
||||
|
||||
bool mustMaskPayload() const;
|
||||
/// Returns true if the payload must be masked.
|
||||
|
||||
protected:
|
||||
enum
|
||||
{
|
||||
FRAME_FLAG_MASK = 0x80,
|
||||
MAX_HEADER_LENGTH = 14
|
||||
};
|
||||
|
||||
virtual ~WebSocketImpl();
|
||||
|
||||
private:
|
||||
WebSocketImpl();
|
||||
|
||||
StreamSocketImpl* _pStreamSocketImpl;
|
||||
int _frameFlags;
|
||||
bool _mustMaskPayload;
|
||||
Poco::Random _rnd;
|
||||
};
|
||||
|
||||
|
||||
//
|
||||
// inlines
|
||||
//
|
||||
inline int WebSocketImpl::frameFlags() const
|
||||
{
|
||||
return _frameFlags;
|
||||
}
|
||||
|
||||
|
||||
inline bool WebSocketImpl::mustMaskPayload() const
|
||||
{
|
||||
return _mustMaskPayload;
|
||||
}
|
||||
|
||||
|
||||
} } // namespace Poco::Net
|
||||
|
||||
|
||||
#endif // Net_WebSocketImpl_INCLUDED
|
244
Net/src/WebSocket.cpp
Normal file
244
Net/src/WebSocket.cpp
Normal file
@ -0,0 +1,244 @@
|
||||
//
|
||||
// WebSocket.cpp
|
||||
//
|
||||
// $Id: //poco/1.4/Net/src/WebSocket.cpp#4 $
|
||||
//
|
||||
// Library: Net
|
||||
// Package: WebSocket
|
||||
// Module: WebSocket
|
||||
//
|
||||
// Copyright (c) 2012, Applied Informatics Software Engineering GmbH.
|
||||
// and Contributors.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person or organization
|
||||
// obtaining a copy of the software and accompanying documentation covered by
|
||||
// this license (the "Software") to use, reproduce, display, distribute,
|
||||
// execute, and transmit the Software, and to prepare derivative works of the
|
||||
// Software, and to permit third-parties to whom the Software is furnished to
|
||||
// do so, all subject to the following:
|
||||
//
|
||||
// The copyright notices in the Software and this entire statement, including
|
||||
// the above license grant, this restriction and the following disclaimer,
|
||||
// must be included in all copies of the Software, in whole or in part, and
|
||||
// all derivative works of the Software, unless such copies or derivative
|
||||
// works are solely in the form of machine-executable object code generated by
|
||||
// a source language processor.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
// FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
||||
// SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
||||
// FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
||||
// ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
// DEALINGS IN THE SOFTWARE.
|
||||
//
|
||||
|
||||
|
||||
#include "Poco/Net/WebSocket.h"
|
||||
#include "Poco/Net/WebSocketImpl.h"
|
||||
#include "Poco/Net/HTTPServerRequestImpl.h"
|
||||
#include "Poco/Net/HTTPServerResponse.h"
|
||||
#include "Poco/Net/HTTPClientSession.h"
|
||||
#include "Poco/Net/NetException.h"
|
||||
#include "Poco/Buffer.h"
|
||||
#include "Poco/MemoryStream.h"
|
||||
#include "Poco/NullStream.h"
|
||||
#include "Poco/BinaryWriter.h"
|
||||
#include "Poco/SHA1Engine.h"
|
||||
#include "Poco/Base64Encoder.h"
|
||||
#include "Poco/String.h"
|
||||
#include "Poco/Random.h"
|
||||
#include "Poco/StreamCopier.h"
|
||||
#include <sstream>
|
||||
|
||||
|
||||
namespace Poco {
|
||||
namespace Net {
|
||||
|
||||
|
||||
const std::string WebSocket::WEBSOCKET_GUID("258EAFA5-E914-47DA-95CA-C5AB0DC85B11");
|
||||
const std::string WebSocket::WEBSOCKET_VERSION("13");
|
||||
HTTPCredentials WebSocket::_defaultCreds;
|
||||
|
||||
|
||||
WebSocket::WebSocket(HTTPServerRequest& request, HTTPServerResponse& response):
|
||||
StreamSocket(accept(request, response))
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
WebSocket::WebSocket(HTTPClientSession& cs, HTTPRequest& request, HTTPResponse& response):
|
||||
StreamSocket(connect(cs, request, response, _defaultCreds))
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
WebSocket::WebSocket(HTTPClientSession& cs, HTTPRequest& request, HTTPResponse& response, HTTPCredentials& credentials):
|
||||
StreamSocket(connect(cs, request, response, credentials))
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
WebSocket::~WebSocket()
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
WebSocket& WebSocket::operator = (const Socket& socket)
|
||||
{
|
||||
if (dynamic_cast<WebSocketImpl*>(socket.impl()))
|
||||
Socket::operator = (socket);
|
||||
else
|
||||
throw InvalidArgumentException("Cannot assign incompatible socket");
|
||||
return *this;
|
||||
}
|
||||
|
||||
|
||||
void WebSocket::shutdown()
|
||||
{
|
||||
shutdown(WS_NORMAL_CLOSE);
|
||||
}
|
||||
|
||||
|
||||
void WebSocket::shutdown(Poco::UInt16 statusCode, const std::string& statusMessage)
|
||||
{
|
||||
Poco::Buffer<char> buffer(statusMessage.size() + 2);
|
||||
Poco::MemoryOutputStream ostr(buffer.begin(), buffer.size());
|
||||
Poco::BinaryWriter writer(ostr, Poco::BinaryWriter::NETWORK_BYTE_ORDER);
|
||||
writer << statusCode;
|
||||
writer.writeRaw(statusMessage);
|
||||
sendFrame(buffer.begin(), static_cast<int>(ostr.charsWritten()), FRAME_FLAG_FIN | FRAME_OP_CLOSE);
|
||||
}
|
||||
|
||||
|
||||
int WebSocket::sendFrame(const void* buffer, int length, int flags)
|
||||
{
|
||||
return static_cast<WebSocketImpl*>(impl())->sendBytes(buffer, length, flags);
|
||||
}
|
||||
|
||||
|
||||
int WebSocket::receiveFrame(void* buffer, int length, int& flags)
|
||||
{
|
||||
int n = static_cast<WebSocketImpl*>(impl())->receiveBytes(buffer, length, 0);
|
||||
flags = static_cast<WebSocketImpl*>(impl())->frameFlags();
|
||||
return n;
|
||||
}
|
||||
|
||||
|
||||
WebSocket::Mode WebSocket::mode() const
|
||||
{
|
||||
return static_cast<WebSocketImpl*>(impl())->mustMaskPayload() ? WS_CLIENT : WS_SERVER;
|
||||
}
|
||||
|
||||
|
||||
WebSocketImpl* WebSocket::accept(HTTPServerRequest& request, HTTPServerResponse& response)
|
||||
{
|
||||
if (icompare(request.get("Connection", ""), "Upgrade") == 0 &&
|
||||
icompare(request.get("Upgrade", ""), "websocket") == 0)
|
||||
{
|
||||
std::string version = request.get("Sec-WebSocket-Version", "");
|
||||
if (version.empty()) throw WebSocketException("Missing Sec-WebSocket-Version in handshake request", WS_ERR_HANDSHAKE_NO_VERSION);
|
||||
if (version != WEBSOCKET_VERSION) throw WebSocketException("Unsupported WebSocket version requested", version, WS_ERR_HANDSHAKE_UNSUPPORTED_VERSION);
|
||||
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.send().flush();
|
||||
return new WebSocketImpl(static_cast<StreamSocketImpl*>(static_cast<HTTPServerRequestImpl&>(request).detachSocket().impl()), false);
|
||||
}
|
||||
else throw WebSocketException("No WebSocket handshake", WS_ERR_NO_HANDSHAKE);
|
||||
}
|
||||
|
||||
|
||||
WebSocketImpl* WebSocket::connect(HTTPClientSession& cs, HTTPRequest& request, HTTPResponse& response, HTTPCredentials& credentials)
|
||||
{
|
||||
if (!cs.getProxyHost().empty() && !cs.secure())
|
||||
{
|
||||
cs.proxyTunnel();
|
||||
}
|
||||
std::string key = createKey();
|
||||
request.set("Connection", "Upgrade");
|
||||
request.set("Upgrade", "websocket");
|
||||
request.set("Sec-WebSocket-Version", WEBSOCKET_VERSION);
|
||||
request.set("Sec-WebSocket-Key", key);
|
||||
request.setChunkedTransferEncoding(false);
|
||||
cs.setKeepAlive(true);
|
||||
cs.sendRequest(request);
|
||||
std::istream& istr = cs.receiveResponse(response);
|
||||
if (response.getStatus() == HTTPResponse::HTTP_SWITCHING_PROTOCOLS)
|
||||
{
|
||||
return completeHandshake(cs, response, key);
|
||||
}
|
||||
else if (response.getStatus() == HTTPResponse::HTTP_UNAUTHORIZED)
|
||||
{
|
||||
Poco::NullOutputStream null;
|
||||
Poco::StreamCopier::copyStream(istr, null);
|
||||
credentials.authenticate(request, response);
|
||||
if (!cs.getProxyHost().empty() && !cs.secure())
|
||||
{
|
||||
cs.reset();
|
||||
cs.proxyTunnel();
|
||||
}
|
||||
cs.sendRequest(request);
|
||||
cs.receiveResponse(response);
|
||||
if (response.getStatus() == HTTPResponse::HTTP_SWITCHING_PROTOCOLS)
|
||||
{
|
||||
return completeHandshake(cs, response, key);
|
||||
}
|
||||
else if (response.getStatus() == HTTPResponse::HTTP_UNAUTHORIZED)
|
||||
{
|
||||
throw WebSocketException("Not authorized", WS_ERR_UNAUTHORIZED);
|
||||
}
|
||||
}
|
||||
throw WebSocketException("Cannot upgrade to WebSocket connection", response.getReason(), WS_ERR_NO_HANDSHAKE);
|
||||
}
|
||||
|
||||
|
||||
WebSocketImpl* WebSocket::completeHandshake(HTTPClientSession& cs, HTTPResponse& response, const std::string& key)
|
||||
{
|
||||
std::string connection = response.get("Connection", "");
|
||||
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)
|
||||
throw WebSocketException("No Upgrade: websocket header in handshake response", WS_ERR_NO_HANDSHAKE);
|
||||
std::string accept = response.get("Sec-WebSocket-Accept", "");
|
||||
if (accept != computeAccept(key))
|
||||
throw WebSocketException("Invalid or missing Sec-WebSocket-Accept header in handshake response", WS_ERR_NO_HANDSHAKE);
|
||||
return new WebSocketImpl(static_cast<StreamSocketImpl*>(cs.socket().impl()), true);
|
||||
}
|
||||
|
||||
|
||||
std::string WebSocket::createKey()
|
||||
{
|
||||
Poco::Random rnd;
|
||||
std::ostringstream ostr;
|
||||
Poco::Base64Encoder base64(ostr);
|
||||
Poco::BinaryWriter writer(base64);
|
||||
writer << rnd.next() << rnd.next() << rnd.next() << rnd.next();
|
||||
base64.close();
|
||||
return ostr.str();
|
||||
}
|
||||
|
||||
|
||||
std::string WebSocket::computeAccept(const std::string& key)
|
||||
{
|
||||
std::string accept(key);
|
||||
accept += WEBSOCKET_GUID;
|
||||
Poco::SHA1Engine sha1;
|
||||
sha1.update(accept);
|
||||
Poco::DigestEngine::Digest d = sha1.digest();
|
||||
std::ostringstream ostr;
|
||||
Poco::Base64Encoder base64(ostr);
|
||||
base64.write(reinterpret_cast<const char*>(&d[0]), d.size());
|
||||
base64.close();
|
||||
return ostr.str();
|
||||
}
|
||||
|
||||
|
||||
} } // namespace Poco::Net
|
279
Net/src/WebSocketImpl.cpp
Normal file
279
Net/src/WebSocketImpl.cpp
Normal file
@ -0,0 +1,279 @@
|
||||
//
|
||||
// WebSocketImpl.cpp
|
||||
//
|
||||
// $Id: //poco/1.4/Net/src/WebSocketImpl.cpp#2 $
|
||||
//
|
||||
// Library: Net
|
||||
// Package: WebSocket
|
||||
// Module: WebSocketImpl
|
||||
//
|
||||
// Copyright (c) 2012, Applied Informatics Software Engineering GmbH.
|
||||
// and Contributors.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person or organization
|
||||
// obtaining a copy of the software and accompanying documentation covered by
|
||||
// this license (the "Software") to use, reproduce, display, distribute,
|
||||
// execute, and transmit the Software, and to prepare derivative works of the
|
||||
// Software, and to permit third-parties to whom the Software is furnished to
|
||||
// do so, all subject to the following:
|
||||
//
|
||||
// The copyright notices in the Software and this entire statement, including
|
||||
// the above license grant, this restriction and the following disclaimer,
|
||||
// must be included in all copies of the Software, in whole or in part, and
|
||||
// all derivative works of the Software, unless such copies or derivative
|
||||
// works are solely in the form of machine-executable object code generated by
|
||||
// a source language processor.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
// FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
||||
// SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
||||
// FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
||||
// ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
// DEALINGS IN THE SOFTWARE.
|
||||
//
|
||||
|
||||
|
||||
#include "Poco/Net/WebSocketImpl.h"
|
||||
#include "Poco/Net/NetException.h"
|
||||
#include "Poco/Net/WebSocket.h"
|
||||
#include "Poco/Buffer.h"
|
||||
#include "Poco/BinaryWriter.h"
|
||||
#include "Poco/BinaryReader.h"
|
||||
#include "Poco/MemoryStream.h"
|
||||
#include "Poco/Format.h"
|
||||
#include <cstring>
|
||||
|
||||
|
||||
namespace Poco {
|
||||
namespace Net {
|
||||
|
||||
|
||||
WebSocketImpl::WebSocketImpl(StreamSocketImpl* pStreamSocketImpl, bool mustMaskPayload):
|
||||
StreamSocketImpl(pStreamSocketImpl->sockfd()),
|
||||
_pStreamSocketImpl(pStreamSocketImpl),
|
||||
_frameFlags(0),
|
||||
_mustMaskPayload(mustMaskPayload)
|
||||
{
|
||||
poco_check_ptr(pStreamSocketImpl);
|
||||
_pStreamSocketImpl->duplicate();
|
||||
}
|
||||
|
||||
|
||||
WebSocketImpl::~WebSocketImpl()
|
||||
{
|
||||
_pStreamSocketImpl->release();
|
||||
reset();
|
||||
}
|
||||
|
||||
|
||||
int WebSocketImpl::sendBytes(const void* buffer, int length, int flags)
|
||||
{
|
||||
Poco::Buffer<char> frame(length + MAX_HEADER_LENGTH);
|
||||
Poco::MemoryOutputStream ostr(frame.begin(), frame.size());
|
||||
Poco::BinaryWriter writer(ostr, Poco::BinaryWriter::NETWORK_BYTE_ORDER);
|
||||
|
||||
writer << static_cast<Poco::UInt8>(flags);
|
||||
Poco::UInt8 lengthByte(0);
|
||||
if (_mustMaskPayload)
|
||||
{
|
||||
lengthByte |= FRAME_FLAG_MASK;
|
||||
}
|
||||
if (length < 126)
|
||||
{
|
||||
lengthByte |= static_cast<Poco::UInt8>(length);
|
||||
writer << lengthByte;
|
||||
}
|
||||
else if (length < 65536)
|
||||
{
|
||||
lengthByte |= 126;
|
||||
writer << lengthByte << static_cast<Poco::UInt16>(length);
|
||||
}
|
||||
else
|
||||
{
|
||||
lengthByte |= 127;
|
||||
writer << lengthByte << static_cast<Poco::UInt64>(length);
|
||||
}
|
||||
if (_mustMaskPayload)
|
||||
{
|
||||
const Poco::UInt32 mask = _rnd.next();
|
||||
const char* m = reinterpret_cast<const char*>(&mask);
|
||||
const char* b = reinterpret_cast<const char*>(buffer);
|
||||
writer.writeRaw(m, 4);
|
||||
char* p = frame.begin() + ostr.charsWritten();
|
||||
for (int i = 0; i < length; i++)
|
||||
{
|
||||
p[i] = b[i] ^ m[i % 4];
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
std::memcpy(frame.begin() + ostr.charsWritten(), buffer, length);
|
||||
}
|
||||
_pStreamSocketImpl->sendBytes(frame.begin(), length + static_cast<int>(ostr.charsWritten()));
|
||||
return length;
|
||||
}
|
||||
|
||||
|
||||
int WebSocketImpl::receiveBytes(void* buffer, int length, int)
|
||||
{
|
||||
char header[MAX_HEADER_LENGTH];
|
||||
int n = _pStreamSocketImpl->receiveBytes(header, MAX_HEADER_LENGTH);
|
||||
if (n > 0)
|
||||
{
|
||||
Poco::MemoryInputStream istr(header, n);
|
||||
Poco::BinaryReader reader(istr, Poco::BinaryReader::NETWORK_BYTE_ORDER);
|
||||
Poco::UInt8 flags;
|
||||
Poco::UInt8 lengthByte;
|
||||
char mask[4];
|
||||
reader >> flags >> lengthByte;
|
||||
_frameFlags = flags;
|
||||
int payloadLength = 0;
|
||||
int payloadOffset = 2;
|
||||
if ((lengthByte & 0x7f) == 127)
|
||||
{
|
||||
Poco::UInt64 l;
|
||||
reader >> l;
|
||||
if (l > length) throw WebSocketException(Poco::format("Insufficient buffer for payload size %Lu", l), WebSocket::WS_ERR_PAYLOAD_TOO_BIG);
|
||||
payloadLength = static_cast<int>(l);
|
||||
payloadOffset += 8;
|
||||
}
|
||||
else if ((lengthByte & 0x7f) == 126)
|
||||
{
|
||||
Poco::UInt16 l;
|
||||
reader >> l;
|
||||
if (l > length) throw WebSocketException(Poco::format("Insufficient buffer for payload size %hu", l), WebSocket::WS_ERR_PAYLOAD_TOO_BIG);
|
||||
payloadLength = static_cast<int>(l);
|
||||
payloadOffset += 2;
|
||||
}
|
||||
else
|
||||
{
|
||||
Poco::UInt8 l = lengthByte & 0x7f;
|
||||
if (l > length) throw WebSocketException(Poco::format("Insufficient buffer for payload size %u", unsigned(l)), WebSocket::WS_ERR_PAYLOAD_TOO_BIG);
|
||||
payloadLength = static_cast<int>(l);
|
||||
}
|
||||
if (lengthByte & FRAME_FLAG_MASK)
|
||||
{
|
||||
reader.readRaw(mask, 4);
|
||||
payloadOffset += 4;
|
||||
}
|
||||
int received = 0;
|
||||
if (payloadOffset < n)
|
||||
{
|
||||
std::memcpy(buffer, header + payloadOffset, n - payloadOffset);
|
||||
received = n - payloadOffset;
|
||||
}
|
||||
while (received < payloadLength)
|
||||
{
|
||||
n = _pStreamSocketImpl->receiveBytes(reinterpret_cast<char*>(buffer) + received, payloadLength - received);
|
||||
if (n > 0)
|
||||
received += n;
|
||||
else
|
||||
throw WebSocketException("Incomplete frame received", WebSocket::WS_ERR_INCOMPLETE_FRAME);
|
||||
}
|
||||
if (lengthByte & FRAME_FLAG_MASK)
|
||||
{
|
||||
char* p = reinterpret_cast<char*>(buffer);
|
||||
for (int i = 0; i < received; i++)
|
||||
{
|
||||
p[i] ^= mask[i % 4];
|
||||
}
|
||||
}
|
||||
return received;
|
||||
}
|
||||
return n;
|
||||
}
|
||||
|
||||
|
||||
SocketImpl* WebSocketImpl::acceptConnection(SocketAddress& clientAddr)
|
||||
{
|
||||
throw Poco::InvalidAccessException("Cannot acceptConnection() on a WebSocketImpl");
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::connect(const SocketAddress& address)
|
||||
{
|
||||
throw Poco::InvalidAccessException("Cannot connect() a WebSocketImpl");
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::connect(const SocketAddress& address, const Poco::Timespan& timeout)
|
||||
{
|
||||
throw Poco::InvalidAccessException("Cannot connect() a WebSocketImpl");
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::connectNB(const SocketAddress& address, const Poco::Timespan& timeout)
|
||||
{
|
||||
throw Poco::InvalidAccessException("Cannot connectNB() a WebSocketImpl");
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::bind(const SocketAddress& address, bool reuseAddress)
|
||||
{
|
||||
throw Poco::InvalidAccessException("Cannot bind() a WebSocketImpl");
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::bind6(const SocketAddress& address, bool reuseAddress, bool ipV6Only)
|
||||
{
|
||||
throw Poco::InvalidAccessException("Cannot bind6() a WebSocketImpl");
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::listen(int backlog)
|
||||
{
|
||||
throw Poco::InvalidAccessException("Cannot listen() on a WebSocketImpl");
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::close()
|
||||
{
|
||||
_pStreamSocketImpl->close();
|
||||
reset();
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::shutdownReceive()
|
||||
{
|
||||
_pStreamSocketImpl->shutdownReceive();
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::shutdownSend()
|
||||
{
|
||||
_pStreamSocketImpl->shutdownSend();
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::shutdown()
|
||||
{
|
||||
_pStreamSocketImpl->shutdown();
|
||||
}
|
||||
|
||||
|
||||
int WebSocketImpl::sendTo(const void* buffer, int length, const SocketAddress& address, int flags)
|
||||
{
|
||||
throw Poco::InvalidAccessException("Cannot sendTo() on a WebSocketImpl");
|
||||
}
|
||||
|
||||
|
||||
int WebSocketImpl::receiveFrom(void* buffer, int length, SocketAddress& address, int flags)
|
||||
{
|
||||
throw Poco::InvalidAccessException("Cannot receiveFrom() on a WebSocketImpl");
|
||||
}
|
||||
|
||||
|
||||
void WebSocketImpl::sendUrgent(unsigned char data)
|
||||
{
|
||||
throw Poco::InvalidAccessException("Cannot sendUrgent() on a WebSocketImpl");
|
||||
}
|
||||
|
||||
|
||||
bool WebSocketImpl::secure() const
|
||||
{
|
||||
return _pStreamSocketImpl->secure();
|
||||
}
|
||||
|
||||
|
||||
} } // namespace Poco::Net
|
Loading…
x
Reference in New Issue
Block a user