SNode.C
Loading...
Searching...
No Matches
SocketContextUpgradeFactory.cpp
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 "web/websocket/client/SocketContextUpgradeFactory.h"
43
44#include "web/http/client/Request.h"
45#include "web/http/client/Response.h"
46#include "web/websocket/client/SocketContextUpgrade.h"
47
48#ifndef DOXYGEN_SHOULD_SKIP_THIS
49
50#include "utils/base64.h"
51
52#include <stdexcept>
53#include <unistd.h>
54
55#if !defined(HAVE_GETENTROPY)
56#include <cstddef>
57#include <sys/types.h>
58#if defined(SYS_GETRANDOM)
59#include <sys/syscall.h>
60#else
61#include <fcntl.h>
62#include <stdexcept>
63#endif
64#endif
65
66#endif /* DOXYGEN_SHOULD_SKIP_THIS */
67
68namespace web::websocket::client {
69
70#if !defined(HAVE_GETENTROPY)
71 int getentropy(void* buf, size_t buflen);
72 int getentropy(void* buf, size_t buflen) {
73#ifdef SYS_GETRANDOM
74 const ssize_t ret = syscall(SYS_getrandom, buf, buflen, 0);
75#else
76 // Fallback to /dev/urandom if necessary
77 const int fd = open("/dev/urandom", O_RDONLY);
78 if (fd == -1) {
79 return -1;
80 }
81 const ssize_t ret = read(fd, buf, buflen);
82 close(fd);
83#endif
84 return (ret == static_cast<ssize_t>(buflen)) ? 0 : -1;
85 }
86#endif
87
89 unsigned char ebytes[16];
90 if (getentropy(ebytes, 16) != 0) {
91 throw std::runtime_error("getentropy() failed");
92 }
93
94 request.set("Sec-WebSocket-Key", base64::base64_encode(ebytes, 16));
95 request.set("Sec-WebSocket-Version", "13");
96 }
97
99 return "websocket";
100 }
101
102 http::SocketContextUpgrade<web::http::client::Request, web::http::client::Response>*
104 web::http::client::Request* request,
105 web::http::client::Response* response) {
106 SocketContextUpgrade* socketContext = nullptr;
107
108 if (response->get("sec-websocket-accept") == base64::serverWebSocketKey(request->header("Sec-WebSocket-Key"))) {
109 const std::string subProtocolName = response->get("sec-websocket-protocol");
110
111 socketContext = new SocketContextUpgrade(socketConnection, this);
112 const std::string selectedSubProtocolName = socketContext->loadSubProtocol(subProtocolName);
113
114 if (selectedSubProtocolName.empty()) {
115 delete socketContext;
116 socketContext = nullptr;
117 }
118 } else {
120 }
121
122 return socketContext;
123 }
124
126 static bool linked = false;
127
128 if (!linked) {
130 linked = true;
131 }
132 }
133
137
138} // namespace web::websocket::client
Request & set(const std::string &field, const std::string &value, bool overwrite=true)
Definition Request.cpp:136
std::string header(const std::string &field) const
Definition Request.cpp:235
const std::string & get(const std::string &key, int i=0) const
Definition Response.cpp:53
static void link(const std::string &upgradeContextName, SocketContextUpgradeFactory *(*linkedPlugin)())
http::SocketContextUpgrade< web::http::client::Request, web::http::client::Response > * create(core::socket::stream::SocketConnection *socketConnection, web::http::client::Request *request, web::http::client::Response *response) override
void prepare(web::http::client::Request &request) override
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)
std::string serverWebSocketKey(const std::string &clientWebSocketKey)
Definition base64.cpp:56
std::string base64_encode(const unsigned char *bytes_to_encode, std::size_t length)
Definition base64.cpp:69
int getentropy(void *buf, size_t buflen)
web::http::client::SocketContextUpgradeFactory * websocketClientSocketContextUpgradeFactory()