SNode.C
Loading...
Searching...
No Matches
echoclient.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 "core/SNodeC.h"
43#include "web/http/legacy/in/Client.h"
44#include "web/http/tls/in/Client.h"
45
46#ifndef DOXYGEN_SHOULD_SKIP_THIS
47
48#include "log/Logger.h"
49
50#endif /* DOXYGEN_SHOULD_SKIP_THIS */
51
52int main(int argc, char* argv[]) {
53 core::SNodeC::init(argc, argv);
54
55 {
56 using LegacyClient = web::http::legacy::in::Client;
57 using MasterRequest = LegacyClient::MasterRequest;
58 using Request = LegacyClient::Request;
59 using Response = LegacyClient::Response;
60 using LegacySocketAddress = LegacyClient::SocketAddress;
61
62 const LegacyClient legacyClient(
63 "legacy",
64 [](const std::shared_ptr<MasterRequest>& req) {
65 const std::string connectionName = req->getSocketContext()->getSocketConnection()->getConnectionName();
66
67 VLOG(1) << connectionName << ": OnRequestBegin";
68
69 req->set("Sec-WebSocket-Protocol", "subprotocol, echo");
70
71 req->upgrade(
72 "/ws",
73 "websocket",
74 [connectionName](bool success) {
75 VLOG(1) << connectionName << ": HTTP Upgrade (http -> websocket) start " << (success ? "success" : "failed");
76 },
77 [connectionName]([[maybe_unused]] const std::shared_ptr<Request>& req,
78 const std::shared_ptr<Response>& res,
79 [[maybe_unused]] bool success) {
80 VLOG(1) << connectionName << ": Upgrade success:";
81
82 VLOG(1) << connectionName << ": Requested: " << req->header("upgrade");
83 VLOG(1) << connectionName << ": Selected: " << res->get("upgrade");
84 },
85 [connectionName](const std::shared_ptr<Request>&, const std::string& message) {
86 VLOG(1) << connectionName << ": Request parse error: " << message;
87 });
88 },
89 []([[maybe_unused]] const std::shared_ptr<MasterRequest>& req) {
90 const std::string connectionName = req->getSocketContext()->getSocketConnection()->getConnectionName();
91
92 VLOG(1) << connectionName << ": OnRequestEnd";
93 });
94
95 legacyClient
96 .setOnInitState([]([[maybe_unused]] core::eventreceiver::ConnectEventReceiver* connectEventReceiver) {
97 VLOG(0) << "------------------- Legacy Client Init: " << connectEventReceiver;
98 })
99 .connect([instanceName = legacyClient.getConfig().getInstanceName()](const LegacySocketAddress& socketAddress,
100 const core::socket::State& state) {
101 switch (state) {
102 case core::socket::State::OK:
103 VLOG(1) << instanceName << " connected to '" << socketAddress.toString() << "'";
104 break;
105 case core::socket::State::DISABLED:
106 VLOG(1) << instanceName << " disabled";
107 break;
108 case core::socket::State::ERROR:
109 VLOG(1) << instanceName << " " << socketAddress.toString() << ": " << state.what();
110 break;
111 case core::socket::State::FATAL:
112 VLOG(1) << instanceName << " " << socketAddress.toString() << ": " << state.what();
113 break;
114 }
115 }); // Connection:keep-alive\r\n\r\n"
116
117 using TlsClient = web::http::tls::in::Client;
118 using MasterRequest = TlsClient::MasterRequest;
119 using Request = TlsClient::Request;
120 using Response = TlsClient::Response;
121 using TLSSocketAddress = TlsClient::SocketAddress;
122
123 const TlsClient tlsClient(
124 "tls",
125 [](const std::shared_ptr<MasterRequest>& req) {
126 const std::string connectionName = req->getSocketContext()->getSocketConnection()->getConnectionName();
127
128 VLOG(1) << connectionName << ": OnRequestBegin";
129
130 req->set("Sec-WebSocket-Protocol", "subprotocol, echo");
131
132 req->upgrade(
133 "/ws",
134 "websocket",
135 [connectionName](bool success) {
136 VLOG(1) << connectionName << ": HTTP Upgrade (http -> websocket) start " << (success ? "success" : "failed");
137 },
138 [connectionName]([[maybe_unused]] const std::shared_ptr<Request>& req,
139 [[maybe_unused]] const std::shared_ptr<Response>& res,
140 [[maybe_unused]] bool success) {
141 },
142 [connectionName](const std::shared_ptr<Request>&, const std::string& message) {
143 VLOG(1) << connectionName << ": Request parse error: " << message;
144 });
145 },
146 []([[maybe_unused]] const std::shared_ptr<MasterRequest>& req) {
147 const std::string connectionName = req->getSocketContext()->getSocketConnection()->getConnectionName();
148
149 VLOG(1) << connectionName << ": OnRequestEnd";
150 });
151
152 tlsClient
153 .setOnInitState([]([[maybe_unused]] core::eventreceiver::ConnectEventReceiver* connectEventReceiver) {
154 VLOG(0) << "------------------- TLS Client Init: " << connectEventReceiver;
155 })
156 .connect([instanceName = tlsClient.getConfig().getInstanceName()](const TLSSocketAddress& socketAddress,
157 const core::socket::State& state) {
158 switch (state) {
159 case core::socket::State::OK:
160 VLOG(1) << instanceName << " connected to '" << socketAddress.toString() << "'";
161 break;
162 case core::socket::State::DISABLED:
163 VLOG(1) << instanceName << " disabled";
164 break;
165 case core::socket::State::ERROR:
166 VLOG(1) << instanceName << " " << socketAddress.toString() << ": " << state.what();
167 break;
168 case core::socket::State::FATAL:
169 VLOG(1) << instanceName << " " << socketAddress.toString() << ": " << state.what();
170 break;
171 }
172 }); // Connection:keep-alive\r\n\r\n"
173 }
174
175 return core::SNodeC::start();
176}
static void init(int argc, char *argv[])
Definition SNodeC.cpp:54
static int start(const utils::Timeval &timeOut={LONG_MAX, 0})
Definition SNodeC.cpp:60
Config & getConfig() const
Definition Socket.hpp:65
static constexpr int DISABLED
Definition State.h:56
static constexpr int ERROR
Definition State.h:57
std::string what() const
Definition State.cpp:114
static constexpr int FATAL
Definition State.h:58
static constexpr int OK
Definition State.h:55
const SocketClient & setOnInitState(const std::function< void(core::eventreceiver::ConnectEventReceiver *)> &onInitState, bool initialize=false) const
const SocketClient & connect(const std::function< void(const SocketAddress &, core::socket::State)> &onStatus) const
const std::string & getConnectionName() const
SocketConnection * getSocketConnection() const
const std::string & getInstanceName() const
std::string toString(bool expanded=true) const override
Client(const std::string &name, std::function< void(const std::shared_ptr< MasterRequest > &)> &&onHttpConnected, std::function< void(const std::shared_ptr< MasterRequest > &)> &&onHttpDisconnected)
Definition Client.h:114
bool upgrade(const std::string &url, const std::string &protocols, const std::function< void(bool)> &onUpgradeInitiate, const std::function< void(const std::shared_ptr< Request > &, const std::shared_ptr< Response > &, bool)> &onResponseReceived, const std::function< void(const std::shared_ptr< Request > &, const std::string &)> &onResponseParseError)
Definition Request.cpp:386
Request & set(const std::string &field, const std::string &value, bool overwrite=true)
Definition Request.cpp:136
SocketContext * getSocketContext() const
Definition Request.cpp:114
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
int main(int argc, char *argv[])
web::http::client::Client< net::in::stream::legacy::SocketClient > Client
Definition Client.h:54
web::http::client::Client< net::in::stream::tls::SocketClient > Client
Definition Client.h:54