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/*
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#endif /* DOXYGEN_SHOULD_SKIP_THIS */
50
51namespace web::websocket {
52
53 template <typename SubProtocol, typename Request, typename Response>
54 SocketContextUpgrade<SubProtocol, Request, Response>::SocketContextUpgrade(
55 core::socket::stream::SocketConnection* socketConnection,
56 web::http::SocketContextUpgradeFactory<Request, Response>* socketContextUpgradeFactory,
57 Role role)
58 : Super(socketConnection, socketContextUpgradeFactory)
59 , SubProtocolContext(socketConnection, role == Role::CLIENT) {
60 }
61
62 template <typename SubProtocol, typename Request, typename Response>
63 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessage(uint8_t opCode, const char* message, std::size_t messageLength) {
64 Transmitter::sendMessage(opCode, message, messageLength);
65 }
66
67 template <typename SubProtocol, typename Request, typename Response>
68 void
69 SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageStart(uint8_t opCode, const char* message, std::size_t messageLength) {
70 Transmitter::sendMessageStart(opCode, message, messageLength);
71 }
72
73 template <typename SubProtocol, typename Request, typename Response>
74 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageFrame(const char* message, std::size_t messageLength) {
75 Transmitter::sendMessageFrame(message, messageLength);
76 }
77
78 template <typename SubProtocol, typename Request, typename Response>
79 void SocketContextUpgrade<SubProtocol, Request, Response>::sendMessageEnd(const char* message, std::size_t messageLength) {
80 Transmitter::sendMessageEnd(message, messageLength);
81 }
82
83 template <typename SubProtocol, typename Request, typename Response>
84 void SocketContextUpgrade<SubProtocol, Request, Response>::sendPing(const char* reason, std::size_t reasonLength) {
85 sendMessage(9, reason, reasonLength);
86 }
87
88 template <typename SubProtocol, typename Request, typename Response>
89 void SocketContextUpgrade<SubProtocol, Request, Response>::sendPong(const char* reason, std::size_t reasonLength) {
90 sendMessage(10, reason, reasonLength);
91 }
92
93 template <typename SubProtocol, typename Request, typename Response>
94 void
95 SocketContextUpgrade<SubProtocol, Request, Response>::sendClose(uint16_t statusCode, const char* reason, std::size_t reasonLength) {
96 std::size_t closePayloadLength = reasonLength + 2;
97 char* closePayload = new char[closePayloadLength];
98
99 *reinterpret_cast<uint16_t*>(closePayload) = htobe16(statusCode); // cppcheck-suppress uninitdata
100
101 if (reasonLength > 0) {
102 memcpy(closePayload + 2, reason, reasonLength);
104
105 sendClose(closePayload, closePayloadLength);
106
107 delete[] closePayload;
108 }
109
110 template <typename SubProtocol, typename Request, typename Response>
111 void SocketContextUpgrade<SubProtocol, Request, Response>::sendClose(const char* message, std::size_t messageLength) {
112 if (!closeSent) {
113 LOG(DEBUG) << this->getSocketConnection()->getConnectionName() << " WebSocket: Sending close to peer";
115 sendMessage(8, message, messageLength);
119 closeSent = true;
121 }
122
123 template <typename SubProtocol, typename Request, typename Response>
124 core::socket::stream::SocketConnection* SocketContextUpgrade<SubProtocol, Request, Response>::getSocketConnection() {
125 return web::http::SocketContextUpgrade<Request, Response>::getSocketConnection();
127
128 template <typename SubProtocol, typename Request, typename Response>
129 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageStart(int opCode) {
131
132 switch (opCode) {
134 [[fallthrough]];
136 [[fallthrough]];
138 break;
139 default:
140 subProtocol->onMessageStart(opCode);
141 break;
142 }
143 }
144
145 template <typename SubProtocol, typename Request, typename Response>
146 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageData(const char* chunk, uint64_t chunkLen) {
147 switch (receivedOpCode) {
149 [[fallthrough]];
151 [[fallthrough]];
153 pongCloseData += std::string(chunk, static_cast<std::size_t>(chunkLen));
154 break;
155 default:
156 std::size_t chunkOffset = 0;
157
158 do {
159 std::size_t sendChunkLen =
160 (chunkLen - chunkOffset <= SIZE_MAX) ? static_cast<std::size_t>(chunkLen - chunkOffset) : SIZE_MAX;
161 subProtocol->onMessageData(chunk + chunkOffset, sendChunkLen);
162 chunkOffset += sendChunkLen;
163 } while (chunkLen - chunkOffset > 0);
164 break;
165 }
166 }
167
168 template <typename SubProtocol, typename Request, typename Response>
169 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageEnd() {
170 switch (receivedOpCode) {
172 if (closeSent) { // active close
173 closeSent = false;
174 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " WebSocket: Close confirmed from peer";
175 } else { // passive close
176 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " WebSocket: Close request received - replying with close";
177 sendClose(pongCloseData.data(), pongCloseData.length());
178 pongCloseData.clear();
179 shutdownWrite();
180 }
181 break;
184 pongCloseData.clear();
185 break;
187 subProtocol->onPongReceived();
188 break;
189 default:
190 subProtocol->onMessageEnd();
191 break;
192 }
193 }
194
195 template <typename SubProtocol, typename Request, typename Response>
196 void SocketContextUpgrade<SubProtocol, Request, Response>::onMessageError(uint16_t errnum) {
197 subProtocol->onMessageError(errnum);
198 sendClose(errnum);
199 }
200
201 template <typename SubProtocol, typename Request, typename Response>
202 void SocketContextUpgrade<SubProtocol, Request, Response>::onConnected() {
203 LOG(INFO) << getSocketConnection()->getConnectionName() << " WebSocket: connected";
204 subProtocol->onConnected();
205 }
206
207 template <typename SubProtocol, typename Request, typename Response>
208 void SocketContextUpgrade<SubProtocol, Request, Response>::onDisconnected() {
209 subProtocol->onDisconnected();
210 LOG(INFO) << getSocketConnection()->getConnectionName() << " WebSocket: disconnected";
211 }
212
213 template <typename SubProtocol, typename Request, typename Response>
214 bool SocketContextUpgrade<SubProtocol, Request, Response>::onSignal(int sig) {
215 return subProtocol->onSignal(sig);
216 }
217
218 template <typename SubProtocol, typename Request, typename Response>
219 std::size_t SocketContextUpgrade<SubProtocol, Request, Response>::onReceivedFromPeer() {
220 return Receiver::receive();
221 }
222
223} // namespace web::websocket
std::size_t receive()
Definition Receiver.cpp:65
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
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)
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)
friend class web::websocket::SocketContextUpgrade
Definition SubProtocol.h:79
#define CLOSE_SOCKET_TIMEOUT