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#ifndef DOXYGEN_SHOULD_SKIP_THIS
46
47#include "utils/hexdump.h"
48
49#include <vector>
50
51#endif /* DOXYGEN_SHOULD_SKIP_THIS */
52
53namespace web::websocket {
54
55 template <typename SubProtocol, typename Request, typename Response>
56 SocketContextUpgrade<SubProtocol, Request, Response>::SocketContextUpgrade(
57 core::socket::stream::SocketConnection* socketConnection,
58 web::http::SocketContextUpgradeFactory<Request, Response>* socketContextUpgradeFactory,
59 Role role)
60 : Super(socketConnection, socketContextUpgradeFactory)
61 , SubProtocolContext(role == Role::CLIENT) {
62 }
63
64 template <typename SubProtocol, typename Request, typename Response>
65 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessage(uint8_t opCode, const char* message, std::size_t messageLength) {
66 Transmitter::sendMessage(opCode, message, messageLength);
67 }
68
69 template <typename SubProtocol, typename Request, typename Response>
70 void
71 SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageStart(uint8_t opCode, const char* message, std::size_t messageLength) {
72 Transmitter::sendMessageStart(opCode, message, messageLength);
73 }
74
75 template <typename SubProtocol, typename Request, typename Response>
76 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageFrame(const char* message, std::size_t messageLength) {
77 Transmitter::sendMessageFrame(message, messageLength);
78 }
79
80 template <typename SubProtocol, typename Request, typename Response>
81 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageEnd(const char* message, std::size_t messageLength) {
82 Transmitter::sendMessageEnd(message, messageLength);
83 }
84
85 template <typename SubProtocol, typename Request, typename Response>
86 void SocketContextUpgrade<SubProtocol, Request, Response>::sendPing(const char* reason, std::size_t reasonLength) {
87 sendMessage(9, reason, reasonLength);
88 }
89
90 template <typename SubProtocol, typename Request, typename Response>
91 void SocketContextUpgrade<SubProtocol, Request, Response>::sendPong(const char* reason, std::size_t reasonLength) {
92 sendMessage(10, reason, reasonLength);
93 }
94
95 template <typename SubProtocol, typename Request, typename Response>
96 void
97 SocketContextUpgrade<SubProtocol, Request, Response>::sendClose(uint16_t statusCode, const char* reason, std::size_t reasonLength) {
98 const std::size_t closePayloadLength = reasonLength + 2;
99 std::vector<char> closePayload(closePayloadLength);
100
101 // Avoid unaligned stores.
102 const uint16_t beStatus = htobe16(statusCode);
103 std::memcpy(closePayload.data(), &beStatus, sizeof(beStatus));
104
105 if (reasonLength > 0) {
106 std::memcpy(closePayload.data() + 2, reason, reasonLength);
107 }
108
109 sendClose(closePayload.data(), closePayloadLength);
110 }
112 template <typename SubProtocol, typename Request, typename Response>
113 void SocketContextUpgrade<SubProtocol, Request, Response>::sendClose(const char* message, std::size_t messageLength) {
114 if (!closeSent) {
115 LOG(DEBUG) << this->getSocketConnection()->getConnectionName() << " WebSocketContext: Subprotocol '" << subProtocol->name
116 << "' sending close to peer";
118 sendMessage(8, message, messageLength);
119
122 closeSent = true;
124 }
125
126 template <typename SubProtocol, typename Request, typename Response>
127 void SocketContextUpgrade<SubProtocol, Request, Response>::sendFrameChunk(const char* message, std::size_t messageLength) const {
128 Super::sendToPeer(message, messageLength);
130
131 template <typename SubProtocol, typename Request, typename Response>
132 std::size_t SocketContextUpgrade<SubProtocol, Request, Response>::readFrameChunk(char* chunk, std::size_t chunkLen) const {
133 return Super::readFromPeer(chunk, chunkLen);
134 }
136 template <typename SubProtocol, typename Request, typename Response>
137 core::socket::stream::SocketConnection* SocketContextUpgrade<SubProtocol, Request, Response>::getSocketConnection() const {
138 return web::http::SocketContextUpgrade<Request, Response>::getSocketConnection();
139 }
141 template <typename SubProtocol, typename Request, typename Response>
142 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageStart(int opCode) {
145 switch (opCode) {
147 [[fallthrough]];
149 [[fallthrough]];
151 break;
152 default:
153 subProtocol->onMessageStart(opCode);
154 break;
155 }
156 }
157
158 template <typename SubProtocol, typename Request, typename Response>
159 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageData(const char* chunk, uint64_t chunkLen) {
160 switch (receivedOpCode) {
162 [[fallthrough]];
164 [[fallthrough]];
166 pongCloseData += std::string(chunk, static_cast<std::size_t>(chunkLen));
167 break;
168 default:
169 std::size_t chunkOffset = 0;
170 do {
171 std::size_t sendChunkLen =
172 (chunkLen - chunkOffset <= SIZE_MAX) ? static_cast<std::size_t>(chunkLen - chunkOffset) : SIZE_MAX;
173 subProtocol->onMessageData(chunk + chunkOffset, sendChunkLen);
174 chunkOffset += sendChunkLen;
175 } while (chunkLen - chunkOffset > 0);
176 break;
177 }
178 }
179
180 template <typename SubProtocol, typename Request, typename Response>
181 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageEnd() {
182 switch (receivedOpCode) {
184 if (closeSent) { // active close
185 closeSent = false;
186 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " WebSocketContext: Subprotocol '" << subProtocol->name
187 << "' close confirmed from peer";
188
189 shutdownWrite();
190 } else { // passive close
191 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " WebSocketContext: Subprotocol '" << subProtocol->name
192 << "' close request received - replying with close";
193
194 sendClose(pongCloseData.data(), pongCloseData.length());
195 pongCloseData.clear();
196 }
197 break;
200 pongCloseData.clear();
201 break;
203 subProtocol->onPongReceived();
204 break;
205 default:
206 subProtocol->onMessageEnd();
207 break;
208 }
209 }
210
211 template <typename SubProtocol, typename Request, typename Response>
212 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageError(uint16_t errnum) {
213 subProtocol->onMessageError(errnum);
214 sendClose(errnum);
215 }
216
217 template <typename SubProtocol, typename Request, typename Response>
218 void SocketContextUpgrade<SubProtocol, Request, Response>::onConnected() {
219 LOG(INFO) << getSocketConnection()->getConnectionName() << " WebSocketContext: Subprotocol '" << subProtocol->name << "' connect";
220 subProtocol->attach();
221 }
222
223 template <typename SubProtocol, typename Request, typename Response>
224 void SocketContextUpgrade<SubProtocol, Request, Response>::onDisconnected() {
225 subProtocol->detach();
226 LOG(INFO) << getSocketConnection()->getConnectionName() << " WebSocketContext: Subprotocol '" << subProtocol->name
227 << "' disconnected";
228 }
229
230 template <typename SubProtocol, typename Request, typename Response>
231 bool SocketContextUpgrade<SubProtocol, Request, Response>::onSignal(int sig) {
232 return subProtocol->onSignal(sig);
233 }
234
235 template <typename SubProtocol, typename Request, typename Response>
236 std::size_t SocketContextUpgrade<SubProtocol, Request, Response>::onReceivedFromPeer() {
237 return Receiver::receive();
238 }
239
240 template <typename SubProtocol, typename Request, typename Response>
241 std::size_t SocketContextUpgrade<SubProtocol, Request, Response>::getPayloadTotalSent() const {
243 }
244
245 template <typename SubProtocol, typename Request, typename Response>
246 std::size_t SocketContextUpgrade<SubProtocol, Request, Response>::getPayloadTotalRead() const {
248 }
249
250 template <typename SubProtocol, typename Request, typename Response>
251 std::string SocketContextUpgrade<SubProtocol, Request, Response>::getOnlineSince() const {
252 return Super::getOnlineSince();
253 }
254
255 template <typename SubProtocol, typename Request, typename Response>
256 std::string SocketContextUpgrade<SubProtocol, Request, Response>::getOnlineDuration() const {
257 return Super::getOnlineDuration();
258 }
259
260} // namespace web::websocket
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