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, 2026
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/*
21 * MIT License
22 *
23 * Permission is hereby granted, free of charge, to any person obtaining a copy
24 * of this software and associated documentation files (the "Software"), to deal
25 * in the Software without restriction, including without limitation the rights
26 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
27 * copies of the Software, and to permit persons to whom the Software is
28 * furnished to do so, subject to the following conditions:
29 *
30 * The above copyright notice and this permission notice shall be included in
31 * all copies or substantial portions of the Software.
32 *
33 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
34 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
35 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
36 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
37 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
38 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
39 * THE SOFTWARE.
40 */
41
42#include "core/socket/stream/SocketConnection.h"
43#include "web/websocket/SocketContextUpgrade.h"
44
45#include <cstring>
46
47#ifndef DOXYGEN_SHOULD_SKIP_THIS
48
49#include "log/Logger.h"
50#include "utils/hexdump.h"
51
52#include <vector>
53
54#endif /* DOXYGEN_SHOULD_SKIP_THIS */
55
56namespace web::websocket {
57
58 template <typename SubProtocol, typename Request, typename Response>
59 SocketContextUpgrade<SubProtocol, Request, Response>::SocketContextUpgrade(
60 core::socket::stream::SocketConnection* socketConnection,
61 web::http::SocketContextUpgradeFactory<Request, Response>* socketContextUpgradeFactory,
62 Role role)
63 : Super(socketConnection, socketContextUpgradeFactory)
64 , SubProtocolContext(role == Role::CLIENT) {
65 }
66
67 template <typename SubProtocol, typename Request, typename Response>
68 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessage(uint8_t opCode, const char* message, std::size_t messageLength) {
69 Transmitter::sendMessage(opCode, message, messageLength);
70 }
71
72 template <typename SubProtocol, typename Request, typename Response>
73 void
74 SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageStart(uint8_t opCode, const char* message, std::size_t messageLength) {
75 Transmitter::sendMessageStart(opCode, message, messageLength);
76 }
77
78 template <typename SubProtocol, typename Request, typename Response>
79 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageFrame(const char* message, std::size_t messageLength) {
80 Transmitter::sendMessageFrame(message, messageLength);
81 }
82
83 template <typename SubProtocol, typename Request, typename Response>
84 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageEnd(const char* message, std::size_t messageLength) {
85 Transmitter::sendMessageEnd(message, messageLength);
86 }
87
88 template <typename SubProtocol, typename Request, typename Response>
89 void SocketContextUpgrade<SubProtocol, Request, Response>::sendPing(const char* reason, std::size_t reasonLength) {
90 sendMessage(9, reason, reasonLength);
91 }
92
93 template <typename SubProtocol, typename Request, typename Response>
94 void SocketContextUpgrade<SubProtocol, Request, Response>::sendPong(const char* reason, std::size_t reasonLength) {
95 sendMessage(10, reason, reasonLength);
96 }
97
98 template <typename SubProtocol, typename Request, typename Response>
99 void
100 SocketContextUpgrade<SubProtocol, Request, Response>::sendClose(uint16_t statusCode, const char* reason, std::size_t reasonLength) {
101 const std::size_t closePayloadLength = reasonLength + 2;
102 std::vector<char> closePayload(closePayloadLength);
103
104 // Avoid unaligned stores.
105 const uint16_t beStatus = htobe16(statusCode);
106 std::memcpy(closePayload.data(), &beStatus, sizeof(beStatus));
107
108 if (reasonLength > 0) {
109 std::memcpy(closePayload.data() + 2, reason, reasonLength);
112 sendClose(closePayload.data(), closePayloadLength);
115 template <typename SubProtocol, typename Request, typename Response>
116 void SocketContextUpgrade<SubProtocol, Request, Response>::sendClose(const char* message, std::size_t messageLength) {
117 if (!closeSent) {
118 LOG(DEBUG) << this->getSocketConnection()->getConnectionName() << " WebSocketContext: Subprotocol '" << subProtocol->name
119 << "' sending close to peer";
120
121 sendMessage(8, message, messageLength);
122
123 setTimeout(CLOSE_SOCKET_TIMEOUT);
125 closeSent = true;
128
129 template <typename SubProtocol, typename Request, typename Response>
130 void SocketContextUpgrade<SubProtocol, Request, Response>::sendFrameChunk(const char* chunk, std::size_t chunkLen) const {
131 Super::sendToPeer(chunk, chunkLen);
132 }
134 template <typename SubProtocol, typename Request, typename Response>
135 std::size_t SocketContextUpgrade<SubProtocol, Request, Response>::readFrameChunk(char* chunk, std::size_t chunkLen) const {
136 return Super::readFromPeer(chunk, chunkLen);
137 }
139 template <typename SubProtocol, typename Request, typename Response>
140 core::socket::stream::SocketConnection* SocketContextUpgrade<SubProtocol, Request, Response>::getSocketConnection() const {
141 return web::http::SocketContextUpgrade<Request, Response>::getSocketConnection();
143
144 template <typename SubProtocol, typename Request, typename Response>
145 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageStart(int opCode) {
146 receivedOpCode = opCode;
147
148 switch (opCode) {
150 [[fallthrough]];
152 [[fallthrough]];
154 break;
155 default:
156 subProtocol->onMessageStart(opCode);
157 break;
158 }
159 }
160
161 template <typename SubProtocol, typename Request, typename Response>
162 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageData(const char* chunk, uint64_t chunkLen) {
163 switch (receivedOpCode) {
165 [[fallthrough]];
167 [[fallthrough]];
169 pongCloseData += std::string(chunk, static_cast<std::size_t>(chunkLen));
170 break;
171 default:
172 std::size_t chunkOffset = 0;
173 do {
174 std::size_t sendChunkLen =
175 (chunkLen - chunkOffset <= SIZE_MAX) ? static_cast<std::size_t>(chunkLen - chunkOffset) : SIZE_MAX;
176 subProtocol->onMessageData(chunk + chunkOffset, sendChunkLen);
177 chunkOffset += sendChunkLen;
178 } while (chunkLen - chunkOffset > 0);
179 break;
180 }
181 }
182
183 template <typename SubProtocol, typename Request, typename Response>
184 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageEnd() {
185 switch (receivedOpCode) {
187 if (closeSent) { // active close
188 closeSent = false;
189 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " WebSocketContext: Subprotocol '" << subProtocol->name
190 << "' close confirmed from peer";
191
192 shutdownWrite();
193 } else { // passive close
194 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " WebSocketContext: Subprotocol '" << subProtocol->name
195 << "' close request received - replying with close";
196
197 sendClose(pongCloseData.data(), pongCloseData.length());
198 pongCloseData.clear();
199 }
200 break;
203 pongCloseData.clear();
204 break;
206 subProtocol->onPongReceived();
207 break;
208 default:
209 subProtocol->onMessageEnd();
210 break;
211 }
212 }
213
214 template <typename SubProtocol, typename Request, typename Response>
215 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageError(uint16_t errnum) {
216 subProtocol->onMessageError(errnum);
217 sendClose(errnum);
218 }
219
220 template <typename SubProtocol, typename Request, typename Response>
221 void SocketContextUpgrade<SubProtocol, Request, Response>::onConnected() {
222 LOG(INFO) << getSocketConnection()->getConnectionName() << " WebSocketContext: Subprotocol '" << subProtocol->name << "' connect";
223 subProtocol->attach();
224 }
225
226 template <typename SubProtocol, typename Request, typename Response>
227 void SocketContextUpgrade<SubProtocol, Request, Response>::onDisconnected() {
228 subProtocol->detach();
229 LOG(INFO) << getSocketConnection()->getConnectionName() << " WebSocketContext: Subprotocol '" << subProtocol->name
230 << "' disconnected";
231 }
232
233 template <typename SubProtocol, typename Request, typename Response>
234 bool SocketContextUpgrade<SubProtocol, Request, Response>::onSignal(int sig) {
235 return subProtocol->onSignal(sig);
236 }
237
238 template <typename SubProtocol, typename Request, typename Response>
239 std::size_t SocketContextUpgrade<SubProtocol, Request, Response>::onReceivedFromPeer() {
240 return Receiver::receive();
241 }
242
243 template <typename SubProtocol, typename Request, typename Response>
244 std::size_t SocketContextUpgrade<SubProtocol, Request, Response>::getPayloadTotalSent() const {
246 }
247
248 template <typename SubProtocol, typename Request, typename Response>
249 std::size_t SocketContextUpgrade<SubProtocol, Request, Response>::getPayloadTotalRead() const {
251 }
252
253 template <typename SubProtocol, typename Request, typename Response>
254 std::string SocketContextUpgrade<SubProtocol, Request, Response>::getOnlineSince() const {
255 return Super::getOnlineSince();
256 }
257
258 template <typename SubProtocol, typename Request, typename Response>
259 std::string SocketContextUpgrade<SubProtocol, Request, Response>::getOnlineDuration() const {
260 return Super::getOnlineDuration();
261 }
262
263} // namespace web::websocket
#define LOG(level)
Definition Logger.h:148
LogMessage(Level level, int verboseLevel=-1, bool withErrno=false)
Definition Logger.cpp:280
std::size_t receive()
Definition Receiver.cpp:67
std::size_t getPayloadTotalRead() const
Definition Receiver.cpp:104
web::http::SocketContextUpgrade< RequestT, ResponseT > Super
void sendFrameChunk(const char *chunk, std::size_t chunkLen) const final
void sendClose(uint16_t statusCode=1000, const char *reason=nullptr, std::size_t reasonLength=0) override
std::size_t getPayloadTotalRead() const 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
std::size_t readFrameChunk(char *chunk, std::size_t chunkLen) const final
std::string getOnlineSince() const override
void sendClose(const char *message, std::size_t messageLength) override
core::socket::stream::SocketConnection * getSocketConnection() const 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)
std::string getOnlineDuration() const override
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
std::size_t getPayloadTotalSent() const override
void onMessageError(uint16_t errnum) override
SubProtocolFactory * select(const std::string &subProtocolName, Role role)
virtual std::size_t deleteSubProtocol(SubProtocol *subProtocol)
SubProtocol * createSubProtocol(SubProtocolContext *subProtocolContext)
void sendMessageEnd(const char *message, std::size_t messageLength)
void sendMessage(uint8_t opCode, const char *message, std::size_t messageLength)
std::size_t getPayloadTotalSent() const
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