Implement /console0 websocket

This commit ipmlements the serial console websocket in a way that is
compatible with phosphor-rest.  This allows the webui serial console to
function.  Latency doesn't appear improved, but I suspect that the
obmc-console server has issues.

Tested By:
Booted phosphor-webui serial console.  Serial console works as
expected.  Also implemented a serial console in python using python
websocket, and it appears to send and receive data correctly.

Change-Id: I0e571beb70a51923d6d7d148779a1154432c45c9
Signed-off-by: Ed Tanous <ed.tanous@intel.com>
diff --git a/CMakeLists.txt b/CMakeLists.txt
index dd70f94..97b3aad 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -10,6 +10,7 @@
 option (BMCWEB_ENABLE_KVM "Enable KVM websocket interfaces" ON)
 option (BMCWEB_ENABLE_DBUS_REST "Enable rest dbus interfaces" ON)
 option (BMCWEB_ENABLE_REDFISH "Enable redfish interfaces" ON)
+option (BMCWEB_ENABLE_HOST_SERIAL_WEBSOCKET "Enable host serial websocket" ON)
 option (BMCWEB_ENABLE_STATIC_HOSTING "Enable hosting of static files.
        For example, redfish schema and webui files" ON)
 
diff --git a/include/obmc_console.hpp b/include/obmc_console.hpp
new file mode 100644
index 0000000..a15004e
--- /dev/null
+++ b/include/obmc_console.hpp
@@ -0,0 +1,157 @@
+#pragma once
+#include <crow/app.h>
+#include <crow/websocket.h>
+#include <sys/socket.h>
+
+#include <boost/container/flat_map.hpp>
+#include <boost/container/flat_set.hpp>
+#include <webserver_common.hpp>
+
+namespace crow
+{
+namespace obmc_console
+{
+
+static std::unique_ptr<boost::asio::local::stream_protocol::socket> host_socket;
+
+static std::array<char, 4096> outputBuffer;
+static std::string inputBuffer;
+
+static boost::container::flat_set<crow::websocket::Connection*> sessions;
+
+static bool doingWrite = false;
+
+void doWrite()
+{
+    if (doingWrite)
+    {
+        BMCWEB_LOG_DEBUG << "Already writing.  Bailing out";
+        return;
+    }
+
+    if (inputBuffer.empty())
+    {
+        BMCWEB_LOG_DEBUG << "Outbuffer empty.  Bailing out";
+        return;
+    }
+
+    doingWrite = true;
+    host_socket->async_write_some(
+        boost::asio::buffer(inputBuffer.data(), inputBuffer.size()),
+        [](boost::beast::error_code ec, std::size_t bytes_written) {
+            doingWrite = false;
+            inputBuffer.erase(0, bytes_written);
+
+            if (ec == boost::asio::error::eof)
+            {
+                for (auto session : sessions)
+                {
+                    session->close("Error in reading to host port");
+                }
+                return;
+            }
+            if (ec)
+            {
+                BMCWEB_LOG_ERROR << "Error in host serial write " << ec;
+                return;
+            }
+            doWrite();
+        });
+}
+
+void doRead()
+{
+    BMCWEB_LOG_DEBUG << "Reading from socket";
+    host_socket->async_read_some(
+        boost::asio::buffer(outputBuffer.data(), outputBuffer.size()),
+        [](const boost::system::error_code& ec, std::size_t bytesRead) {
+            BMCWEB_LOG_DEBUG << "read done.  Read " << bytesRead << " bytes";
+            if (ec)
+            {
+                BMCWEB_LOG_ERROR << "Couldn't read from host serial port: "
+                                 << ec;
+                for (auto session : sessions)
+                {
+                    session->close("Error in connecting to host port");
+                }
+                return;
+            }
+            boost::beast::string_view payload(outputBuffer.data(), bytesRead);
+            for (auto session : sessions)
+            {
+                session->sendText(payload);
+            }
+            doRead();
+        });
+}
+
+void connectHandler(const boost::system::error_code& ec)
+{
+    if (ec)
+    {
+        BMCWEB_LOG_ERROR << "Couldn't connect to host serial port: " << ec;
+        for (auto session : sessions)
+        {
+            session->close("Error in connecting to host port");
+        }
+        return;
+    }
+
+    doWrite();
+    doRead();
+}
+
+void requestRoutes(CrowApp& app)
+{
+    BMCWEB_ROUTE(app, "/console0")
+        .websocket()
+        .onopen([](crow::websocket::Connection& conn) {
+            BMCWEB_LOG_DEBUG << "Connection " << &conn << " opened";
+
+            sessions.insert(&conn);
+            if (host_socket == nullptr)
+            {
+                const std::string consoleName("\0obmc-console", 13);
+                boost::asio::local::stream_protocol::endpoint ep(consoleName);
+
+                // This is a hack.  For whatever reason boost local endpoint has
+                // a check to see if a string is null terminated, and if it is,
+                // it drops the path character count by 1.  For abstract
+                // sockets, we need the count to be the full sizeof(s->sun_path)
+                // (ie 108), even though our path _looks_ like it's null
+                // terminated.  This is likely a bug in asio that needs to be
+                // submitted Todo(ed).  so the cheat here is to break the
+                // abstraction for a minute, write a 1 to the last byte, this
+                // causes the check at the end of resize here:
+                // https://www.boost.org/doc/libs/1_68_0/boost/asio/local/detail/impl/endpoint.ipp
+                // to not decrement us unesssesarily.
+                struct sockaddr_un* s =
+                    reinterpret_cast<sockaddr_un*>(ep.data());
+                s->sun_path[sizeof(s->sun_path) - 1] = 1;
+                ep.resize(sizeof(sockaddr_un));
+                s->sun_path[sizeof(s->sun_path) - 1] = 0;
+
+                host_socket = std::make_unique<
+                    boost::asio::local::stream_protocol::socket>(
+                    conn.getIoService());
+                host_socket->async_connect(ep, connectHandler);
+            }
+        })
+        .onclose(
+            [](crow::websocket::Connection& conn, const std::string& reason) {
+                sessions.erase(&conn);
+                if (sessions.empty())
+                {
+                    host_socket = nullptr;
+                    inputBuffer.clear();
+                    inputBuffer.shrink_to_fit();
+                }
+            })
+        .onmessage([](crow::websocket::Connection& conn,
+                      const std::string& data, bool is_binary) {
+            inputBuffer += data;
+            doWrite();
+        });
+}
+} // namespace obmc_console
+} // namespace crow
diff --git a/settings.hpp.in b/settings.hpp.in
index 44d8f17..897dca5 100644
--- a/settings.hpp.in
+++ b/settings.hpp.in
@@ -4,6 +4,7 @@
 #cmakedefine BMCWEB_ENABLE_DBUS_REST
 #cmakedefine BMCWEB_ENABLE_REDFISH
 #cmakedefine BMCWEB_ENABLE_STATIC_HOSTING
+#cmakedefine BMCWEB_ENABLE_HOST_SERIAL_WEBSOCKET
 #cmakedefine BMCWEB_INSECURE_DISABLE_CSRF_PREVENTION
 #cmakedefine BMCWEB_INSECURE_DISABLE_SSL
 #cmakedefine BMCWEB_INSECURE_DISABLE_XSS_PREVENTION