Ed Tanous | 40e9b92 | 2024-09-10 13:50:16 -0700 | [diff] [blame] | 1 | // SPDX-License-Identifier: Apache-2.0 |
| 2 | // SPDX-FileCopyrightText: Copyright OpenBMC Authors |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 3 | #pragma once |
Ed Tanous | 3ccb3ad | 2023-01-13 17:40:03 -0800 | [diff] [blame] | 4 | #include "app.hpp" |
Ed Tanous | d785720 | 2025-01-28 15:32:26 -0800 | [diff] [blame] | 5 | #include "logging.hpp" |
Ed Tanous | faf100f | 2023-05-25 10:03:14 -0700 | [diff] [blame] | 6 | #include "websocket.hpp" |
Ed Tanous | 3ccb3ad | 2023-01-13 17:40:03 -0800 | [diff] [blame] | 7 | |
Ed Tanous | d785720 | 2025-01-28 15:32:26 -0800 | [diff] [blame] | 8 | #include <sys/types.h> |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 9 | |
Ed Tanous | d785720 | 2025-01-28 15:32:26 -0800 | [diff] [blame] | 10 | #include <boost/asio/buffer.hpp> |
| 11 | #include <boost/asio/error.hpp> |
| 12 | #include <boost/asio/ip/address.hpp> |
| 13 | #include <boost/asio/ip/tcp.hpp> |
| 14 | #include <boost/beast/core/flat_static_buffer.hpp> |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 15 | #include <boost/container/flat_map.hpp> |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 16 | |
Ed Tanous | d785720 | 2025-01-28 15:32:26 -0800 | [diff] [blame] | 17 | #include <cstddef> |
| 18 | #include <memory> |
| 19 | #include <string> |
| 20 | |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 21 | namespace crow |
| 22 | { |
| 23 | namespace obmc_kvm |
| 24 | { |
| 25 | |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 26 | static constexpr const uint maxSessions = 4; |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 27 | |
Xinnan Xie | 8e73b90 | 2023-08-23 11:14:48 +0800 | [diff] [blame] | 28 | class KvmSession : public std::enable_shared_from_this<KvmSession> |
Jae Hyun Yoo | c68604b | 2019-02-26 15:46:07 -0800 | [diff] [blame] | 29 | { |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 30 | public: |
Ed Tanous | 23a21a1 | 2020-07-25 04:45:05 +0000 | [diff] [blame] | 31 | explicit KvmSession(crow::websocket::Connection& connIn) : |
Ed Tanous | f5b191a | 2022-02-15 11:30:39 -0800 | [diff] [blame] | 32 | conn(connIn), hostSocket(conn.getIoContext()) |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 33 | { |
| 34 | boost::asio::ip::tcp::endpoint endpoint( |
Johnathan Mantey | b8c7eb2 | 2020-02-19 12:44:19 -0800 | [diff] [blame] | 35 | boost::asio::ip::make_address("127.0.0.1"), 5900); |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 36 | hostSocket.async_connect( |
Ed Tanous | 23a21a1 | 2020-07-25 04:45:05 +0000 | [diff] [blame] | 37 | endpoint, [this, &connIn](const boost::system::error_code& ec) { |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 38 | if (ec) |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 39 | { |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 40 | BMCWEB_LOG_ERROR( |
| 41 | "conn:{}, Couldn't connect to KVM socket port: {}", |
| 42 | logPtr(&conn), ec); |
| 43 | if (ec != boost::asio::error::operation_aborted) |
| 44 | { |
| 45 | connIn.close("Error in connecting to KVM port"); |
| 46 | } |
| 47 | return; |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 48 | } |
Jae Hyun Yoo | c68604b | 2019-02-26 15:46:07 -0800 | [diff] [blame] | 49 | |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 50 | doRead(); |
| 51 | }); |
Jae Hyun Yoo | c68604b | 2019-02-26 15:46:07 -0800 | [diff] [blame] | 52 | } |
| 53 | |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 54 | void onMessage(const std::string& data) |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 55 | { |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 56 | if (data.length() > inputBuffer.capacity()) |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 57 | { |
Ed Tanous | 62598e3 | 2023-07-17 17:06:25 -0700 | [diff] [blame] | 58 | BMCWEB_LOG_ERROR("conn:{}, Buffer overrun when writing {} bytes", |
| 59 | logPtr(&conn), data.length()); |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 60 | conn.close("Buffer overrun"); |
| 61 | return; |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 62 | } |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 63 | |
Ed Tanous | 62598e3 | 2023-07-17 17:06:25 -0700 | [diff] [blame] | 64 | BMCWEB_LOG_DEBUG("conn:{}, Read {} bytes from websocket", logPtr(&conn), |
| 65 | data.size()); |
Ed Tanous | 44106f3 | 2024-04-06 13:48:50 -0700 | [diff] [blame] | 66 | size_t copied = boost::asio::buffer_copy( |
| 67 | inputBuffer.prepare(data.size()), boost::asio::buffer(data)); |
Ed Tanous | 62598e3 | 2023-07-17 17:06:25 -0700 | [diff] [blame] | 68 | BMCWEB_LOG_DEBUG("conn:{}, Committing {} bytes from websocket", |
Ed Tanous | 44106f3 | 2024-04-06 13:48:50 -0700 | [diff] [blame] | 69 | logPtr(&conn), copied); |
| 70 | inputBuffer.commit(copied); |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 71 | |
Ed Tanous | 62598e3 | 2023-07-17 17:06:25 -0700 | [diff] [blame] | 72 | BMCWEB_LOG_DEBUG("conn:{}, inputbuffer size {}", logPtr(&conn), |
| 73 | inputBuffer.size()); |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 74 | doWrite(); |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 75 | } |
| 76 | |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 77 | protected: |
| 78 | void doRead() |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 79 | { |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 80 | std::size_t bytes = outputBuffer.capacity() - outputBuffer.size(); |
Ed Tanous | 62598e3 | 2023-07-17 17:06:25 -0700 | [diff] [blame] | 81 | BMCWEB_LOG_DEBUG("conn:{}, Reading {} from kvm socket", logPtr(&conn), |
| 82 | bytes); |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 83 | hostSocket.async_read_some( |
| 84 | outputBuffer.prepare(outputBuffer.capacity() - outputBuffer.size()), |
Xinnan Xie | 8e73b90 | 2023-08-23 11:14:48 +0800 | [diff] [blame] | 85 | [this, weak(weak_from_this())](const boost::system::error_code& ec, |
| 86 | std::size_t bytesRead) { |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 87 | auto self = weak.lock(); |
| 88 | if (self == nullptr) |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 89 | { |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 90 | return; |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 91 | } |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 92 | BMCWEB_LOG_DEBUG("conn:{}, read done. Read {} bytes", |
| 93 | logPtr(&conn), bytesRead); |
| 94 | if (ec) |
| 95 | { |
| 96 | BMCWEB_LOG_ERROR( |
| 97 | "conn:{}, Couldn't read from KVM socket port: {}", |
| 98 | logPtr(&conn), ec); |
| 99 | if (ec != boost::asio::error::operation_aborted) |
| 100 | { |
| 101 | conn.close("Error in connecting to KVM port"); |
| 102 | } |
| 103 | return; |
| 104 | } |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 105 | |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 106 | outputBuffer.commit(bytesRead); |
| 107 | std::string_view payload( |
| 108 | static_cast<const char*>(outputBuffer.data().data()), |
| 109 | bytesRead); |
| 110 | BMCWEB_LOG_DEBUG("conn:{}, Sending payload size {}", |
| 111 | logPtr(&conn), payload.size()); |
| 112 | conn.sendBinary(payload); |
| 113 | outputBuffer.consume(bytesRead); |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 114 | |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 115 | doRead(); |
| 116 | }); |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 117 | } |
| 118 | |
| 119 | void doWrite() |
| 120 | { |
| 121 | if (doingWrite) |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 122 | { |
Ed Tanous | 62598e3 | 2023-07-17 17:06:25 -0700 | [diff] [blame] | 123 | BMCWEB_LOG_DEBUG("conn:{}, Already writing. Bailing out", |
| 124 | logPtr(&conn)); |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 125 | return; |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 126 | } |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 127 | if (inputBuffer.size() == 0) |
| 128 | { |
Ed Tanous | 62598e3 | 2023-07-17 17:06:25 -0700 | [diff] [blame] | 129 | BMCWEB_LOG_DEBUG("conn:{}, inputBuffer empty. Bailing out", |
| 130 | logPtr(&conn)); |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 131 | return; |
| 132 | } |
| 133 | |
| 134 | doingWrite = true; |
Xinnan Xie | 8e73b90 | 2023-08-23 11:14:48 +0800 | [diff] [blame] | 135 | hostSocket.async_write_some( |
| 136 | inputBuffer.data(), |
| 137 | [this, weak(weak_from_this())](const boost::system::error_code& ec, |
Ed Tanous | 002d39b | 2022-05-31 08:59:27 -0700 | [diff] [blame] | 138 | std::size_t bytesWritten) { |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 139 | auto self = weak.lock(); |
| 140 | if (self == nullptr) |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 141 | { |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 142 | return; |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 143 | } |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 144 | BMCWEB_LOG_DEBUG("conn:{}, Wrote {}bytes", logPtr(&conn), |
| 145 | bytesWritten); |
| 146 | doingWrite = false; |
| 147 | inputBuffer.consume(bytesWritten); |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 148 | |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 149 | if (ec == boost::asio::error::eof) |
| 150 | { |
| 151 | conn.close("KVM socket port closed"); |
| 152 | return; |
| 153 | } |
| 154 | if (ec) |
| 155 | { |
| 156 | BMCWEB_LOG_ERROR("conn:{}, Error in KVM socket write {}", |
| 157 | logPtr(&conn), ec); |
| 158 | if (ec != boost::asio::error::operation_aborted) |
| 159 | { |
| 160 | conn.close("Error in reading to host port"); |
| 161 | } |
| 162 | return; |
| 163 | } |
| 164 | |
| 165 | doWrite(); |
| 166 | }); |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 167 | } |
| 168 | |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 169 | crow::websocket::Connection& conn; |
| 170 | boost::asio::ip::tcp::socket hostSocket; |
Ed Tanous | 6de264c | 2022-01-06 12:47:59 -0800 | [diff] [blame] | 171 | boost::beast::flat_static_buffer<1024UL * 50UL> outputBuffer; |
| 172 | boost::beast::flat_static_buffer<1024UL> inputBuffer; |
Ed Tanous | f5b191a | 2022-02-15 11:30:39 -0800 | [diff] [blame] | 173 | bool doingWrite{false}; |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 174 | }; |
| 175 | |
Ed Tanous | cf9e417 | 2022-12-21 09:30:16 -0800 | [diff] [blame] | 176 | using SessionMap = boost::container::flat_map<crow::websocket::Connection*, |
Xinnan Xie | 8e73b90 | 2023-08-23 11:14:48 +0800 | [diff] [blame] | 177 | std::shared_ptr<KvmSession>>; |
Ed Tanous | cf9e417 | 2022-12-21 09:30:16 -0800 | [diff] [blame] | 178 | // NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables) |
| 179 | static SessionMap sessions; |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 180 | |
Ed Tanous | 52cc112 | 2020-07-18 13:51:21 -0700 | [diff] [blame] | 181 | inline void requestRoutes(App& app) |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 182 | { |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 183 | sessions.reserve(maxSessions); |
| 184 | |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 185 | BMCWEB_ROUTE(app, "/kvm/0") |
Ed Tanous | 432a890 | 2021-06-14 15:28:56 -0700 | [diff] [blame] | 186 | .privileges({{"ConfigureComponents", "ConfigureManager"}}) |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 187 | .websocket() |
zhanghch05 | 7772638 | 2021-10-21 14:07:57 +0800 | [diff] [blame] | 188 | .onopen([](crow::websocket::Connection& conn) { |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 189 | BMCWEB_LOG_DEBUG("Connection {} opened", logPtr(&conn)); |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 190 | |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 191 | if (sessions.size() == maxSessions) |
| 192 | { |
| 193 | conn.close("Max sessions are already connected"); |
| 194 | return; |
| 195 | } |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 196 | |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 197 | sessions[&conn] = std::make_shared<KvmSession>(conn); |
| 198 | }) |
Ed Tanous | cb13a39 | 2020-07-25 19:02:03 +0000 | [diff] [blame] | 199 | .onclose([](crow::websocket::Connection& conn, const std::string&) { |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 200 | sessions.erase(&conn); |
| 201 | }) |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 202 | .onmessage([](crow::websocket::Connection& conn, |
Ed Tanous | cb13a39 | 2020-07-25 19:02:03 +0000 | [diff] [blame] | 203 | const std::string& data, bool) { |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 204 | if (sessions[&conn]) |
| 205 | { |
| 206 | sessions[&conn]->onMessage(data); |
| 207 | } |
| 208 | }); |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 209 | } |
Jae Hyun Yoo | a133b29 | 2019-09-16 10:38:47 -0700 | [diff] [blame] | 210 | |
Ed Tanous | 3eb2f35 | 2018-12-20 12:30:45 -0800 | [diff] [blame] | 211 | } // namespace obmc_kvm |
| 212 | } // namespace crow |