SNode.C
Loading...
Searching...
No Matches
SocketContextUpgrade.hpp
Go to the documentation of this file.
1/*
2 * SNode.C - a slim toolkit for network communication
3 * Copyright (C) Volker Christian <me@vchrist.at>
4 * 2020, 2021, 2022, 2023, 2024, 2025
5 *
6 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU Lesser General Public License as published
8 * by the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 */
19
20#include "core/socket/stream/SocketConnection.h"
21#include "web/websocket/SocketContextUpgrade.h"
22
23#ifndef DOXYGEN_SHOULD_SKIP_THIS
24
25#include "utils/hexdump.h"
26
27#endif /* DOXYGEN_SHOULD_SKIP_THIS */
28
29namespace web::websocket {
30
31 template <typename SubProtocol, typename Request, typename Response>
32 SocketContextUpgrade<SubProtocol, Request, Response>::SocketContextUpgrade(
33 core::socket::stream::SocketConnection* socketConnection,
34 web::http::SocketContextUpgradeFactory<Request, Response>* socketContextUpgradeFactory,
35 Role role)
36 : Super(socketConnection, socketContextUpgradeFactory)
37 , SubProtocolContext(socketConnection, role == Role::CLIENT) {
38 }
39
40 template <typename SubProtocol, typename Request, typename Response>
41 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessage(uint8_t opCode, const char* message, std::size_t messageLength) {
42 Transmitter::sendMessage(opCode, message, messageLength);
43 }
44
45 template <typename SubProtocol, typename Request, typename Response>
46 void
47 SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageStart(uint8_t opCode, const char* message, std::size_t messageLength) {
48 Transmitter::sendMessageStart(opCode, message, messageLength);
49 }
50
51 template <typename SubProtocol, typename Request, typename Response>
52 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageFrame(const char* message, std::size_t messageLength) {
53 Transmitter::sendMessageFrame(message, messageLength);
54 }
55
56 template <typename SubProtocol, typename Request, typename Response>
57 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageEnd(const char* message, std::size_t messageLength) {
58 Transmitter::sendMessageEnd(message, messageLength);
59 }
60
61 template <typename SubProtocol, typename Request, typename Response>
62 void SocketContextUpgrade<SubProtocol, Request, Response>::sendPing(const char* reason, std::size_t reasonLength) {
63 sendMessage(9, reason, reasonLength);
64 }
65
66 template <typename SubProtocol, typename Request, typename Response>
67 void SocketContextUpgrade<SubProtocol, Request, Response>::sendPong(const char* reason, std::size_t reasonLength) {
68 sendMessage(10, reason, reasonLength);
69 }
70
71 template <typename SubProtocol, typename Request, typename Response>
72 void
73 SocketContextUpgrade<SubProtocol, Request, Response>::sendClose(uint16_t statusCode, const char* reason, std::size_t reasonLength) {
74 std::size_t closePayloadLength = reasonLength + 2;
75 char* closePayload = new char[closePayloadLength];
76
77 *reinterpret_cast<uint16_t*>(closePayload) = htobe16(statusCode); // cppcheck-suppress uninitdata
78
79 if (reasonLength > 0) {
80 memcpy(closePayload + 2, reason, reasonLength);
81 }
82
83 sendClose(closePayload, closePayloadLength);
84
85 delete[] closePayload;
86 }
87
88 template <typename SubProtocol, typename Request, typename Response>
89 void SocketContextUpgrade<SubProtocol, Request, Response>::sendClose(const char* message, std::size_t messageLength) {
90 if (!closeSent) {
91 LOG(DEBUG) << this->getSocketConnection()->getConnectionName() << " WebSocket: Sending close to peer";
93 sendMessage(8, message, messageLength);
97 closeSent = true;
98 }
99 }
100
101 template <typename SubProtocol, typename Request, typename Response>
102 core::socket::stream::SocketConnection* SocketContextUpgrade<SubProtocol, Request, Response>::getSocketConnection() {
103 return web::http::SocketContextUpgrade<Request, Response>::getSocketConnection();
105
106 template <typename SubProtocol, typename Request, typename Response>
107 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageStart(int opCode) {
109
110 switch (opCode) {
111 case SubProtocolContext::OpCode::CLOSE:
112 [[fallthrough]];
113 case SubProtocolContext::OpCode::PING:
114 [[fallthrough]];
115 case SubProtocolContext::OpCode::PONG:
116 break;
117 default:
118 subProtocol->onMessageStart(opCode);
119 break;
120 }
121 }
122
123 template <typename SubProtocol, typename Request, typename Response>
124 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageData(const char* chunk, uint64_t chunkLen) {
125 switch (receivedOpCode) {
126 case SubProtocolContext::OpCode::CLOSE:
127 [[fallthrough]];
128 case SubProtocolContext::OpCode::PING:
129 [[fallthrough]];
130 case SubProtocolContext::OpCode::PONG:
131 pongCloseData += std::string(chunk, static_cast<std::size_t>(chunkLen));
132 break;
133 default:
134 std::size_t chunkOffset = 0;
135
136 do {
137 std::size_t sendChunkLen =
138 (chunkLen - chunkOffset <= SIZE_MAX) ? static_cast<std::size_t>(chunkLen - chunkOffset) : SIZE_MAX;
139 subProtocol->onMessageData(chunk + chunkOffset, sendChunkLen);
140 chunkOffset += sendChunkLen;
141 } while (chunkLen - chunkOffset > 0);
142 break;
143 }
144 }
145
146 template <typename SubProtocol, typename Request, typename Response>
147 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageEnd() {
148 switch (receivedOpCode) {
149 case SubProtocolContext::OpCode::CLOSE:
150 if (closeSent) { // active close
151 closeSent = false;
152 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " WebSocket: Close confirmed from peer";
153 } else { // passive close
154 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " WebSocket: Close request received - replying with close";
155 sendClose(pongCloseData.data(), pongCloseData.length());
156 pongCloseData.clear();
157 shutdownWrite();
158 }
159 break;
160 case SubProtocolContext::OpCode::PING:
161 sendPong(pongCloseData.data(), pongCloseData.length());
162 pongCloseData.clear();
163 break;
164 case SubProtocolContext::OpCode::PONG:
165 subProtocol->onPongReceived();
166 break;
167 default:
168 subProtocol->onMessageEnd();
169 break;
170 }
171 }
172
173 template <typename SubProtocol, typename Request, typename Response>
174 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageError(uint16_t errnum) {
175 subProtocol->onMessageError(errnum);
176 sendClose(errnum);
177 }
178
179 template <typename SubProtocol, typename Request, typename Response>
180 void SocketContextUpgrade<SubProtocol, Request, Response>::onConnected() {
181 LOG(INFO) << getSocketConnection()->getConnectionName() << " WebSocket: connected";
182 subProtocol->onConnected();
183 }
184
185 template <typename SubProtocol, typename Request, typename Response>
186 void SocketContextUpgrade<SubProtocol, Request, Response>::onDisconnected() {
187 subProtocol->onDisconnected();
188 LOG(INFO) << getSocketConnection()->getConnectionName() << " WebSocket: disconnected";
189 }
190
191 template <typename SubProtocol, typename Request, typename Response>
192 bool SocketContextUpgrade<SubProtocol, Request, Response>::onSignal(int sig) {
193 return subProtocol->onSignal(sig);
194 }
195
196 template <typename SubProtocol, typename Request, typename Response>
197 std::size_t SocketContextUpgrade<SubProtocol, Request, Response>::onReceivedFromPeer() {
198 return Receiver::receive();
199 }
200
201} // namespace web::websocket
void sendClose(uint16_t statusCode=1000, const char *reason=nullptr, std::size_t reasonLength=0) override
void sendPing(const char *reason=nullptr, std::size_t reasonLength=0) override
~SocketContextUpgrade() override=default
void onMessageData(const char *chunk, uint64_t chunkLen) override
SocketContextUpgrade(const SocketContextUpgrade &)=delete
SocketContextUpgrade & operator=(const SocketContextUpgrade &)=delete
void sendMessage(uint8_t opCode, const char *message, std::size_t messageLength) override
void sendClose(const char *message, std::size_t messageLength) override
void sendMessageFrame(const char *message, std::size_t messageLength) override
void sendMessageEnd(const char *message, std::size_t messageLength) override
SocketContextUpgrade(core::socket::stream::SocketConnection *socketConnection, web::http::SocketContextUpgradeFactory< Request, Response > *socketContextUpgradeFactory, Role role)
void sendPong(const char *reason=nullptr, std::size_t reasonLength=0) override
void sendMessageStart(uint8_t opCode, const char *message, std::size_t messageLength) override
core::socket::stream::SocketConnection * getSocketConnection() override
void onMessageError(uint16_t errnum) override
virtual void sendMessage(uint8_t opCode, const char *message, std::size_t messageLength)=0
virtual void sendPing(const char *reason=nullptr, std::size_t reasonLength=0)=0
virtual void sendMessageFrame(const char *message, std::size_t messageLength)=0
SubProtocolContext & operator=(const SubProtocolContext &)=delete
virtual void sendPong(const char *reason=nullptr, std::size_t reasonLength=0)=0
virtual void sendClose(const char *message, std::size_t messageLength)=0
SubProtocolContext(core::socket::stream::SocketConnection *socketConnection, bool role)
virtual void sendMessageEnd(const char *message, std::size_t messageLength)=0
virtual void sendClose(uint16_t statusCode=1000, const char *reason=nullptr, std::size_t reasonLength=0)=0
virtual void sendMessageStart(uint8_t opCode, const char *message, std::size_t messageLength)=0
virtual core::socket::stream::SocketConnection * getSocketConnection()=0
SubProtocolContext(const SubProtocolContext &)=delete
void sendMessageEnd(const char *message, std::size_t messageLength)
void sendFrameData(const char *frame, uint64_t frameLength) const
Transmitter & operator=(const Transmitter &)=delete
std::random_device randomDevice
Definition Transmitter.h:68
void sendFrameData(uint8_t data) const
void sendFrameData(uint16_t data) const
void sendMessage(uint8_t opCode, const char *message, std::size_t messageLength)
void sendFrameData(uint32_t data) const
Transmitter(core::socket::stream::SocketConnection *socketConnection, bool masking)
core::socket::stream::SocketConnection * socketConnection
Definition Transmitter.h:71
void send(bool end, uint8_t opCode, const char *message, std::size_t messageLength)
void sendFrame(bool fin, uint8_t opCode, const char *payload, uint64_t payloadLength)
void sendFrameData(uint64_t data) const
std::uniform_int_distribution< uint32_t > distribution
Definition Transmitter.h:69
Transmitter(const Transmitter &)=delete
void sendMessageStart(uint8_t opCode, const char *message, std::size_t messageLength)
void sendMessageFrame(const char *message, std::size_t messageLength)
web::websocket::SubProtocolFactory< SubProtocol > * subProtocolFactory
std::string loadSubProtocol(const std::string &subProtocolName)
SocketContextUpgrade(core::socket::stream::SocketConnection *socketConnection, web::http::SocketContextUpgradeFactory< web::http::client::Request, web::http::client::Response > *socketContextUpgradeFactory)
#define CLOSE_SOCKET_TIMEOUT