xref: /openbmc/bmcweb/include/vm_websocket.hpp (revision 1516c21b27faf8dcf7c41e9b7253da97025a5f28)
1 #pragma once
2 
3 #include "app.hpp"
4 #include "dbus_utility.hpp"
5 #include "privileges.hpp"
6 #include "websocket.hpp"
7 
8 #include <boost/asio/local/stream_protocol.hpp>
9 #include <boost/asio/readable_pipe.hpp>
10 #include <boost/asio/writable_pipe.hpp>
11 #include <boost/asio/write.hpp>
12 #include <boost/beast/core/buffers_to_string.hpp>
13 #include <boost/beast/core/flat_static_buffer.hpp>
14 #include <boost/container/flat_map.hpp>
15 #include <boost/process/v2/process.hpp>
16 #include <boost/process/v2/stdio.hpp>
17 #include <sdbusplus/asio/property.hpp>
18 
19 #include <csignal>
20 #include <string_view>
21 
22 namespace crow
23 {
24 
25 namespace obmc_vm
26 {
27 
28 // NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables)
29 static crow::websocket::Connection* session = nullptr;
30 
31 // The max network block device buffer size is 128kb plus 16bytes
32 // for the message header:
33 // https://github.com/NetworkBlockDevice/nbd/blob/master/doc/proto.md#simple-reply-message
34 static constexpr auto nbdBufferSize = (128 * 1024 + 16) * 4;
35 
36 class Handler : public std::enable_shared_from_this<Handler>
37 {
38   public:
39     Handler(const std::string& media, boost::asio::io_context& ios) :
40         pipeOut(ios), pipeIn(ios),
41         proxy(ios, "/usr/bin/nbd-proxy", {media},
42               boost::process::v2::process_stdio{
43                   .in = pipeIn, .out = pipeOut, .err = nullptr}),
44         outputBuffer(new boost::beast::flat_static_buffer<nbdBufferSize>),
45         inputBuffer(new boost::beast::flat_static_buffer<nbdBufferSize>)
46     {}
47 
48     ~Handler() = default;
49 
50     Handler(const Handler&) = delete;
51     Handler(Handler&&) = delete;
52     Handler& operator=(const Handler&) = delete;
53     Handler& operator=(Handler&&) = delete;
54 
55     void doClose()
56     {
57         // boost::process::child::terminate uses SIGKILL, need to send SIGTERM
58         // to allow the proxy to stop nbd-client and the USB device gadget.
59         int rc = kill(proxy.id(), SIGTERM);
60         if (rc != 0)
61         {
62             BMCWEB_LOG_ERROR("Failed to terminate nbd-proxy: {}", errno);
63             return;
64         }
65 
66         proxy.wait();
67     }
68 
69     void connect()
70     {
71         std::error_code ec;
72         if (ec)
73         {
74             BMCWEB_LOG_ERROR("Couldn't connect to nbd-proxy: {}", ec.message());
75             if (session != nullptr)
76             {
77                 session->close("Error connecting to nbd-proxy");
78             }
79             return;
80         }
81         doWrite();
82         doRead();
83     }
84 
85     void doWrite()
86     {
87         if (doingWrite)
88         {
89             BMCWEB_LOG_DEBUG("Already writing.  Bailing out");
90             return;
91         }
92 
93         if (inputBuffer->size() == 0)
94         {
95             BMCWEB_LOG_DEBUG("inputBuffer empty.  Bailing out");
96             return;
97         }
98 
99         doingWrite = true;
100         pipeIn.async_write_some(
101             inputBuffer->data(),
102             [this, self(shared_from_this())](const boost::beast::error_code& ec,
103                                              std::size_t bytesWritten) {
104                 BMCWEB_LOG_DEBUG("Wrote {}bytes", bytesWritten);
105                 doingWrite = false;
106                 inputBuffer->consume(bytesWritten);
107 
108                 if (session == nullptr)
109                 {
110                     return;
111                 }
112                 if (ec == boost::asio::error::eof)
113                 {
114                     session->close("VM socket port closed");
115                     return;
116                 }
117                 if (ec)
118                 {
119                     session->close("Error in writing to proxy port");
120                     BMCWEB_LOG_ERROR("Error in VM socket write {}", ec);
121                     return;
122                 }
123                 doWrite();
124             });
125     }
126 
127     void doRead()
128     {
129         std::size_t bytes = outputBuffer->capacity() - outputBuffer->size();
130 
131         pipeOut.async_read_some(
132             outputBuffer->prepare(bytes),
133             [this, self(shared_from_this())](
134                 const boost::system::error_code& ec, std::size_t bytesRead) {
135                 BMCWEB_LOG_DEBUG("Read done.  Read {} bytes", bytesRead);
136                 if (ec)
137                 {
138                     BMCWEB_LOG_ERROR("Couldn't read from VM port: {}", ec);
139                     if (session != nullptr)
140                     {
141                         session->close("Error in connecting to VM port");
142                     }
143                     return;
144                 }
145                 if (session == nullptr)
146                 {
147                     return;
148                 }
149 
150                 outputBuffer->commit(bytesRead);
151                 std::string_view payload(
152                     static_cast<const char*>(outputBuffer->data().data()),
153                     bytesRead);
154                 session->sendBinary(payload);
155                 outputBuffer->consume(bytesRead);
156 
157                 doRead();
158             });
159     }
160 
161     boost::asio::readable_pipe pipeOut;
162     boost::asio::writable_pipe pipeIn;
163     boost::process::v2::process proxy;
164     bool doingWrite{false};
165 
166     std::unique_ptr<boost::beast::flat_static_buffer<nbdBufferSize>>
167         outputBuffer;
168     std::unique_ptr<boost::beast::flat_static_buffer<nbdBufferSize>>
169         inputBuffer;
170 };
171 
172 // NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables)
173 static std::shared_ptr<Handler> handler;
174 
175 } // namespace obmc_vm
176 
177 namespace nbd_proxy
178 {
179 using boost::asio::local::stream_protocol;
180 
181 // The max network block device buffer size is 128kb plus 16bytes
182 // for the message header:
183 // https://github.com/NetworkBlockDevice/nbd/blob/master/doc/proto.md#simple-reply-message
184 static constexpr auto nbdBufferSize = (128 * 1024 + 16) * 4;
185 
186 struct NbdProxyServer : std::enable_shared_from_this<NbdProxyServer>
187 {
188     NbdProxyServer(crow::websocket::Connection& connIn,
189                    const std::string& socketIdIn,
190                    const std::string& endpointIdIn, const std::string& pathIn) :
191         socketId(socketIdIn), endpointId(endpointIdIn), path(pathIn),
192 
193         peerSocket(connIn.getIoContext()),
194         acceptor(connIn.getIoContext(), stream_protocol::endpoint(socketId)),
195         connection(connIn)
196     {}
197 
198     NbdProxyServer(const NbdProxyServer&) = delete;
199     NbdProxyServer(NbdProxyServer&&) = delete;
200     NbdProxyServer& operator=(const NbdProxyServer&) = delete;
201     NbdProxyServer& operator=(NbdProxyServer&&) = delete;
202 
203     ~NbdProxyServer()
204     {
205         BMCWEB_LOG_DEBUG("NbdProxyServer destructor");
206 
207         BMCWEB_LOG_DEBUG("peerSocket->close()");
208         boost::system::error_code ec;
209         peerSocket.close(ec);
210 
211         BMCWEB_LOG_DEBUG("std::filesystem::remove({})", socketId);
212         std::error_code ec2;
213         std::filesystem::remove(socketId.c_str(), ec2);
214         if (ec2)
215         {
216             BMCWEB_LOG_DEBUG("Failed to remove file, ignoring");
217         }
218 
219         crow::connections::systemBus->async_method_call(
220             dbus::utility::logError, "xyz.openbmc_project.VirtualMedia", path,
221             "xyz.openbmc_project.VirtualMedia.Proxy", "Unmount");
222     }
223 
224     std::string getEndpointId() const
225     {
226         return endpointId;
227     }
228 
229     static void afterMount(const std::weak_ptr<NbdProxyServer>& weak,
230                            const boost::system::error_code& ec,
231                            bool /*isBinary*/)
232     {
233         std::shared_ptr<NbdProxyServer> self = weak.lock();
234         if (self == nullptr)
235         {
236             return;
237         }
238         if (ec)
239         {
240             BMCWEB_LOG_ERROR("DBus error: cannot call mount method = {}",
241                              ec.message());
242 
243             self->connection.close("Failed to mount media");
244             return;
245         }
246     }
247 
248     static void afterAccept(const std::weak_ptr<NbdProxyServer>& weak,
249                             const boost::system::error_code& ec,
250                             stream_protocol::socket socket)
251     {
252         if (ec)
253         {
254             BMCWEB_LOG_ERROR("UNIX socket: async_accept error = {}",
255                              ec.message());
256             return;
257         }
258 
259         BMCWEB_LOG_DEBUG("Connection opened");
260         std::shared_ptr<NbdProxyServer> self = weak.lock();
261         if (self == nullptr)
262         {
263             return;
264         }
265 
266         self->connection.resumeRead();
267         self->peerSocket = std::move(socket);
268         //  Start reading from socket
269         self->doRead();
270     }
271 
272     void run()
273     {
274         acceptor.async_accept(
275             std::bind_front(&NbdProxyServer::afterAccept, weak_from_this()));
276 
277         crow::connections::systemBus->async_method_call(
278             [weak{weak_from_this()}](const boost::system::error_code& ec,
279                                      bool isBinary) {
280                 afterMount(weak, ec, isBinary);
281             },
282             "xyz.openbmc_project.VirtualMedia", path,
283             "xyz.openbmc_project.VirtualMedia.Proxy", "Mount");
284     }
285 
286     void send(std::string_view buffer, std::function<void()>&& onDone)
287     {
288         size_t copied = boost::asio::buffer_copy(
289             ws2uxBuf.prepare(buffer.size()), boost::asio::buffer(buffer));
290         ws2uxBuf.commit(copied);
291 
292         doWrite(std::move(onDone));
293     }
294 
295   private:
296     static void afterSendEx(const std::weak_ptr<NbdProxyServer>& weak)
297     {
298         std::shared_ptr<NbdProxyServer> self2 = weak.lock();
299         if (self2 != nullptr)
300         {
301             self2->ux2wsBuf.consume(self2->ux2wsBuf.size());
302             self2->doRead();
303         }
304     }
305 
306     void afterRead(const std::weak_ptr<NbdProxyServer>& weak,
307                    const boost::system::error_code& ec, size_t bytesRead)
308     {
309         if (ec)
310         {
311             BMCWEB_LOG_ERROR("UNIX socket: async_read_some error = {}",
312                              ec.message());
313             return;
314         }
315         std::shared_ptr<NbdProxyServer> self = weak.lock();
316         if (self == nullptr)
317         {
318             return;
319         }
320 
321         // Send to websocket
322         self->ux2wsBuf.commit(bytesRead);
323         self->connection.sendEx(
324             crow::websocket::MessageType::Binary,
325             boost::beast::buffers_to_string(self->ux2wsBuf.data()),
326             std::bind_front(&NbdProxyServer::afterSendEx, weak_from_this()));
327     }
328 
329     void doRead()
330     {
331         // Trigger async read
332         peerSocket.async_read_some(ux2wsBuf.prepare(nbdBufferSize),
333                                    std::bind_front(&NbdProxyServer::afterRead,
334                                                    this, weak_from_this()));
335     }
336 
337     static void afterWrite(const std::weak_ptr<NbdProxyServer>& weak,
338                            std::function<void()>&& onDone,
339                            const boost::system::error_code& ec,
340                            size_t bytesWritten)
341     {
342         std::shared_ptr<NbdProxyServer> self = weak.lock();
343         if (self == nullptr)
344         {
345             return;
346         }
347 
348         self->ws2uxBuf.consume(bytesWritten);
349         self->uxWriteInProgress = false;
350 
351         if (ec)
352         {
353             BMCWEB_LOG_ERROR("UNIX: async_write error = {}", ec.message());
354             self->connection.close("Internal error");
355             return;
356         }
357 
358         // Retrigger doWrite if there is something in buffer
359         if (self->ws2uxBuf.size() > 0)
360         {
361             self->doWrite(std::move(onDone));
362             return;
363         }
364         onDone();
365     }
366 
367     void doWrite(std::function<void()>&& onDone)
368     {
369         if (uxWriteInProgress)
370         {
371             BMCWEB_LOG_ERROR("Write in progress");
372             return;
373         }
374 
375         if (ws2uxBuf.size() == 0)
376         {
377             BMCWEB_LOG_ERROR("No data to write to UNIX socket");
378             return;
379         }
380 
381         uxWriteInProgress = true;
382         peerSocket.async_write_some(
383             ws2uxBuf.data(),
384             std::bind_front(&NbdProxyServer::afterWrite, weak_from_this(),
385                             std::move(onDone)));
386     }
387 
388     // Keeps UNIX socket endpoint file path
389     const std::string socketId;
390     const std::string endpointId;
391     const std::string path;
392 
393     bool uxWriteInProgress = false;
394 
395     // UNIX => WebSocket buffer
396     boost::beast::flat_static_buffer<nbdBufferSize> ux2wsBuf;
397 
398     // WebSocket => UNIX buffer
399     boost::beast::flat_static_buffer<nbdBufferSize> ws2uxBuf;
400 
401     // The socket used to communicate with the client.
402     stream_protocol::socket peerSocket;
403 
404     // Default acceptor for UNIX socket
405     stream_protocol::acceptor acceptor;
406 
407     crow::websocket::Connection& connection;
408 };
409 
410 using SessionMap = boost::container::flat_map<crow::websocket::Connection*,
411                                               std::shared_ptr<NbdProxyServer>>;
412 // NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables)
413 static SessionMap sessions;
414 
415 inline void
416     afterGetSocket(crow::websocket::Connection& conn,
417                    const sdbusplus::message::object_path& path,
418                    const boost::system::error_code& ec,
419                    const dbus::utility::DBusPropertiesMap& propertiesList)
420 {
421     if (ec)
422     {
423         BMCWEB_LOG_ERROR("DBus getAllProperties error: {}", ec.message());
424         conn.close("Internal Error");
425         return;
426     }
427     std::string endpointId;
428     std::string socket;
429 
430     bool success = sdbusplus::unpackPropertiesNoThrow(
431         redfish::dbus_utils::UnpackErrorPrinter(), propertiesList, "EndpointId",
432         endpointId, "Socket", socket);
433 
434     if (!success)
435     {
436         BMCWEB_LOG_ERROR("Failed to unpack properties");
437         conn.close("Internal Error");
438         return;
439     }
440 
441     for (const auto& session : sessions)
442     {
443         if (session.second->getEndpointId() == conn.url().path())
444         {
445             BMCWEB_LOG_ERROR("Cannot open new connection - socket is in use");
446             conn.close("Slot is in use");
447             return;
448         }
449     }
450 
451     // If the socket file exists (i.e. after bmcweb crash),
452     // we cannot reuse it.
453     std::error_code ec2;
454     std::filesystem::remove(socket.c_str(), ec2);
455     // Ignore failures.  File might not exist.
456 
457     sessions[&conn] =
458         std::make_shared<NbdProxyServer>(conn, socket, endpointId, path);
459     sessions[&conn]->run();
460 }
461 
462 inline void onOpen(crow::websocket::Connection& conn)
463 {
464     BMCWEB_LOG_DEBUG("nbd-proxy.onopen({})", logPtr(&conn));
465 
466     if (conn.url().segments().size() < 2)
467     {
468         BMCWEB_LOG_ERROR("Invalid path - \"{}\"", conn.url().path());
469         conn.close("Internal error");
470         return;
471     }
472 
473     std::string index = conn.url().segments().back();
474     std::string path =
475         std::format("/xyz/openbmc_project/VirtualMedia/Proxy/Slot_{}", index);
476 
477     sdbusplus::asio::getAllProperties(
478         *crow::connections::systemBus, "xyz.openbmc_project.VirtualMedia", path,
479         "xyz.openbmc_project.VirtualMedia.MountPoint",
480         [&conn, path](const boost::system::error_code& ec,
481                       const dbus::utility::DBusPropertiesMap& propertiesList) {
482             afterGetSocket(conn, path, ec, propertiesList);
483         });
484 
485     // We need to wait for dbus and the websockets to hook up before data is
486     // sent/received.  Tell the core to hold off messages until the sockets are
487     // up
488     conn.deferRead();
489 }
490 
491 inline void onClose(crow::websocket::Connection& conn,
492                     const std::string& reason)
493 {
494     BMCWEB_LOG_DEBUG("nbd-proxy.onclose(reason = '{}')", reason);
495     auto session = sessions.find(&conn);
496     if (session == sessions.end())
497     {
498         BMCWEB_LOG_DEBUG("No session to close");
499         return;
500     }
501     // Remove reference to session in global map
502     sessions.erase(session);
503 }
504 
505 inline void onMessage(crow::websocket::Connection& conn, std::string_view data,
506                       crow::websocket::MessageType /*type*/,
507                       std::function<void()>&& whenComplete)
508 {
509     BMCWEB_LOG_DEBUG("nbd-proxy.onMessage(len = {})", data.size());
510 
511     // Acquire proxy from sessions
512     auto session = sessions.find(&conn);
513     if (session == sessions.end() || session->second == nullptr)
514     {
515         whenComplete();
516         return;
517     }
518 
519     session->second->send(data, std::move(whenComplete));
520 }
521 } // namespace nbd_proxy
522 
523 namespace obmc_vm
524 {
525 
526 inline void requestRoutes(App& app)
527 {
528     static_assert(
529         !(BMCWEB_VM_WEBSOCKET && BMCWEB_VM_NBDPROXY),
530         "nbd proxy cannot be turned on at the same time as vm websocket.");
531 
532     if constexpr (BMCWEB_VM_NBDPROXY)
533     {
534         BMCWEB_ROUTE(app, "/nbd/<str>")
535             .privileges({{"ConfigureComponents", "ConfigureManager"}})
536             .websocket()
537             .onopen(nbd_proxy::onOpen)
538             .onclose(nbd_proxy::onClose)
539             .onmessageex(nbd_proxy::onMessage);
540 
541         BMCWEB_ROUTE(app, "/vm/0/0")
542             .privileges({{"ConfigureComponents", "ConfigureManager"}})
543             .websocket()
544             .onopen(nbd_proxy::onOpen)
545             .onclose(nbd_proxy::onClose)
546             .onmessageex(nbd_proxy::onMessage);
547     }
548     if constexpr (BMCWEB_VM_WEBSOCKET)
549     {
550         BMCWEB_ROUTE(app, "/vm/0/0")
551             .privileges({{"ConfigureComponents", "ConfigureManager"}})
552             .websocket()
553             .onopen([](crow::websocket::Connection& conn) {
554                 BMCWEB_LOG_DEBUG("Connection {} opened", logPtr(&conn));
555 
556                 if (session != nullptr)
557                 {
558                     conn.close("Session already connected");
559                     return;
560                 }
561 
562                 if (handler != nullptr)
563                 {
564                     conn.close("Handler already running");
565                     return;
566                 }
567 
568                 session = &conn;
569 
570                 // media is the last digit of the endpoint /vm/0/0. A future
571                 // enhancement can include supporting different endpoint values.
572                 const char* media = "0";
573                 handler = std::make_shared<Handler>(media, conn.getIoContext());
574                 handler->connect();
575             })
576             .onclose([](crow::websocket::Connection& conn,
577                         const std::string& /*reason*/) {
578                 if (&conn != session)
579                 {
580                     return;
581                 }
582 
583                 session = nullptr;
584                 handler->doClose();
585                 handler->inputBuffer->clear();
586                 handler->outputBuffer->clear();
587                 handler.reset();
588             })
589             .onmessage([](crow::websocket::Connection& conn,
590                           const std::string& data, bool) {
591                 if (data.length() > handler->inputBuffer->capacity() -
592                                         handler->inputBuffer->size())
593                 {
594                     BMCWEB_LOG_ERROR("Buffer overrun when writing {} bytes",
595                                      data.length());
596                     conn.close("Buffer overrun");
597                     return;
598                 }
599 
600                 size_t copied = boost::asio::buffer_copy(
601                     handler->inputBuffer->prepare(data.size()),
602                     boost::asio::buffer(data));
603                 handler->inputBuffer->commit(copied);
604                 handler->doWrite();
605             });
606     }
607 }
608 
609 } // namespace obmc_vm
610 
611 } // namespace crow
612