[Libreoffice-commits] online.git: 2 commits - common/UnitHTTP.cpp common/UnitHTTP.hpp Makefile.am tools/Connect.cpp tools/WebSocketDump.cpp
Michael Meeks
michael.meeks at collabora.com
Tue Apr 17 16:12:53 UTC 2018
Makefile.am | 8 +
common/UnitHTTP.cpp | 41 --------
common/UnitHTTP.hpp | 29 +++++
tools/Connect.cpp | 11 --
tools/WebSocketDump.cpp | 240 ++++++++++++++++++++++++++++++++++++++++++++++++
5 files changed, 276 insertions(+), 53 deletions(-)
New commits:
commit f68d54e02a43738754ba82a40ab42644cc6565aa
Author: Michael Meeks <michael.meeks at collabora.com>
Date: Tue Apr 17 14:01:49 2018 +0100
Initial websocket test tool for remote admin connections.
Change-Id: I8be2068bf7d77c70720a044556d11f5fc80b2f0c
diff --git a/Makefile.am b/Makefile.am
index bbff1083e..6aedc561c 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -2,7 +2,9 @@ SUBDIRS = . test loleaflet
export ENABLE_DEBUG
-bin_PROGRAMS = loolwsd loolforkit loolmap loolmount looltool loolstress loolconfig
+bin_PROGRAMS = \
+ loolwsd loolforkit loolmap loolmount \
+ looltool loolstress loolconfig loolsocketdump
dist_bin_SCRIPTS = loolwsd-systemplate-setup
@@ -131,6 +133,9 @@ loolconfig_SOURCES = tools/Config.cpp \
common/Log.cpp \
common/Util.cpp
+loolsocketdump_SOURCES = tools/WebSocketDump.cpp \
+ $(shared_sources)
+
wsd_headers = wsd/Admin.hpp \
wsd/AdminModel.hpp \
wsd/Auth.hpp \
diff --git a/tools/Connect.cpp b/tools/Connect.cpp
index 04b55ea0c..26db06674 100644
--- a/tools/Connect.cpp
+++ b/tools/Connect.cpp
@@ -236,13 +236,10 @@ private:
namespace Util
{
-
-void alertAllUsers(const std::string& cmd, const std::string& kind)
-{
- std::cout << "error: cmd=" << cmd << " kind=" << kind << std::endl;
- (void) kind;
-}
-
+ void alertAllUsers(const std::string& cmd, const std::string& kind)
+ {
+ std::cout << "error: cmd=" << cmd << " kind=" << kind << std::endl;
+ }
}
POCO_APP_MAIN(Connect)
diff --git a/tools/WebSocketDump.cpp b/tools/WebSocketDump.cpp
new file mode 100644
index 000000000..a72e646c4
--- /dev/null
+++ b/tools/WebSocketDump.cpp
@@ -0,0 +1,240 @@
+/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */
+/*
+ * This file is part of the LibreOffice project.
+ *
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/.
+ */
+
+/* A simple tool that accepts web-socket connections and dumps the contents */
+
+#include <config.h>
+
+#include <unistd.h>
+
+#include <Poco/URI.h>
+#include <Poco/MemoryStream.h>
+#include <Poco/Net/HTTPRequest.h>
+#include <Poco/Net/HTTPResponse.h>
+#include <Poco/StringTokenizer.h>
+
+#include <Log.hpp>
+#include <Util.hpp>
+#include <Protocol.hpp>
+#include <ServerSocket.hpp>
+#include <WebSocketHandler.hpp>
+#if ENABLE_SSL
+# include <SslSocket.hpp>
+#endif
+
+// Dumps incoming websocket messages and doesn't respond.
+class DumpSocketHandler : public WebSocketHandler
+{
+public:
+ DumpSocketHandler()
+ {
+ }
+
+private:
+ /// Process incoming websocket messages
+ void handleMessage(bool fin, WSOpCode code, std::vector<char> &data)
+ {
+ std::cout << "WebSocket message code " << (int)code << " fin " << fin << " data:\n";
+ Util::dumpHex(std::cout, "", " ", data, false);
+ }
+};
+
+/// Handles incoming connections and dispatches to the appropriate handler.
+class ClientRequestDispatcher : public SocketHandlerInterface
+{
+public:
+ ClientRequestDispatcher()
+ {
+ }
+
+private:
+
+ /// Set the socket associated with this ResponseClient.
+ void onConnect(const std::shared_ptr<StreamSocket>& socket) override
+ {
+ _socket = socket;
+ LOG_TRC("#" << socket->getFD() << " Connected to ClientRequestDispatcher.");
+ }
+
+ /// Called after successful socket reads.
+ void handleIncomingMessage(SocketDisposition & /* disposition */) override
+ {
+ std::shared_ptr<StreamSocket> socket = _socket.lock();
+ std::vector<char>& in = socket->_inBuffer;
+ LOG_TRC("#" << socket->getFD() << " handling incoming " << in.size() << " bytes.");
+
+ // Find the end of the header, if any.
+ static const std::string marker("\r\n\r\n");
+ auto itBody = std::search(in.begin(), in.end(),
+ marker.begin(), marker.end());
+ if (itBody == in.end())
+ {
+ LOG_DBG("#" << socket->getFD() << " doesn't have enough data yet.");
+ return;
+ }
+
+ // Skip the marker.
+ itBody += marker.size();
+
+ Poco::MemoryInputStream message(&in[0], in.size());
+ Poco::Net::HTTPRequest request;
+ try
+ {
+ request.read(message);
+
+ Log::StreamLogger logger = Log::info();
+ if (logger.enabled())
+ {
+ logger << "#" << socket->getFD() << ": Client HTTP Request: "
+ << request.getMethod() << ' '
+ << request.getURI() << ' '
+ << request.getVersion();
+
+ for (const auto& it : request)
+ {
+ logger << " / " << it.first << ": " << it.second;
+ }
+
+ LOG_END(logger);
+ }
+
+ const std::streamsize contentLength = request.getContentLength();
+ const auto offset = itBody - in.begin();
+ const std::streamsize available = in.size() - offset;
+
+ if (contentLength != Poco::Net::HTTPMessage::UNKNOWN_CONTENT_LENGTH && available < contentLength)
+ {
+ LOG_DBG("Not enough content yet: ContentLength: " << contentLength << ", available: " << available);
+ return;
+ }
+ }
+ catch (const std::exception& exc)
+ {
+ // Probably don't have enough data just yet.
+ // TODO: timeout if we never get enough.
+ return;
+ }
+
+ try
+ {
+ // Routing
+ Poco::URI requestUri(request.getURI());
+ std::vector<std::string> reqPathSegs;
+ requestUri.getPathSegments(reqPathSegs);
+
+ LOG_INF("Incoming websocket request: " << request.getURI());
+
+ const std::string& requestURI = request.getURI();
+ Poco::StringTokenizer pathTokens(requestURI, "/", Poco::StringTokenizer::TOK_IGNORE_EMPTY |
+ Poco::StringTokenizer::TOK_TRIM);
+
+ if (request.find("Upgrade") != request.end() && Poco::icompare(request["Upgrade"], "websocket") == 0)
+ socket->setHandler(std::make_shared<DumpSocketHandler>());
+ else
+ {
+ Poco::Net::HTTPResponse response;
+ response.setStatusAndReason(Poco::Net::HTTPResponse::HTTP_BAD_REQUEST);
+ response.setContentLength(0);
+ LOG_INF("DumpWebSockets bad request");
+ socket->send(response);
+ socket->shutdown();
+ }
+ }
+ catch (const std::exception& exc)
+ {
+ // Bad request.
+ std::ostringstream oss;
+ oss << "HTTP/1.1 400\r\n"
+ << "Date: " << Poco::DateTimeFormatter::format(Poco::Timestamp(), Poco::DateTimeFormat::HTTP_FORMAT) << "\r\n"
+ << "User-Agent: LOOLWSD WOPI Agent\r\n"
+ << "Content-Length: 0\r\n"
+ << "\r\n";
+ socket->send(oss.str());
+ socket->shutdown();
+
+ // NOTE: Check _wsState to choose between HTTP response or WebSocket (app-level) error.
+ LOG_INF("#" << socket->getFD() << " Exception while processing incoming request: [" <<
+ LOOLProtocol::getAbbreviatedMessage(in) << "]: " << exc.what());
+ }
+
+ // if we succeeded - remove the request from our input buffer
+ // we expect one request per socket
+ in.erase(in.begin(), itBody);
+ }
+
+ int getPollEvents(std::chrono::steady_clock::time_point /* now */,
+ int & /* timeoutMaxMs */) override
+ {
+ return POLLIN;
+ }
+
+ void performWrites() override
+ {
+ }
+
+private:
+ // The socket that owns us (we can't own it).
+ std::weak_ptr<StreamSocket> _socket;
+};
+
+class DumpSocketFactory final : public SocketFactory
+{
+ std::shared_ptr<Socket> create(const int physicalFd) override
+ {
+#if ENABLE_SSL
+ return StreamSocket::create<SslStreamSocket>(physicalFd, std::unique_ptr<SocketHandlerInterface>{ new ClientRequestDispatcher });
+#else
+ return StreamSocket::create<StreamSocket>(physicalFd, std::unique_ptr<SocketHandlerInterface>{ new ClientRequestDispatcher });
+#endif
+ }
+};
+
+namespace Util
+{
+ void alertAllUsers(const std::string& cmd, const std::string& kind)
+ {
+ std::cout << "error: cmd=" << cmd << " kind=" << kind << std::endl;
+ }
+}
+
+int main (int argc, char **argv)
+{
+ int port = 9042;
+ (void) argc; (void) argv;
+
+ SocketPoll acceptPoll("accept");
+ SocketPoll DumpSocketPoll("websocket");
+
+ // Setup listening socket with a factory for connected sockets.
+ auto serverSocket = std::make_shared<ServerSocket>(
+ Socket::Type::IPv4, DumpSocketPoll,
+ std::make_shared<DumpSocketFactory>());
+
+ if (!serverSocket->bind(ServerSocket::Type::Public, port))
+ {
+ fprintf(stderr, "Failed to bind websocket to port %d\n", port);
+ return -1;
+ }
+
+ if (!serverSocket->listen())
+ {
+ fprintf(stderr, "Failed to listen on websocket, port %d\n", port);
+ return -1;
+ }
+
+ acceptPoll.startThread();
+ acceptPoll.insertNewSocket(serverSocket);
+
+ while (true)
+ {
+ DumpSocketPoll.poll(1000);
+ }
+}
+
+/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
commit 81fd84cf4eaefaac7a1412c6bfbfdd271f75e0ec
Author: Michael Meeks <michael.meeks at collabora.com>
Date: Tue Apr 17 16:40:55 2018 +0100
Move UnitHTTP code to header to avoid linking trouble.
Change-Id: I430110e840fa8b3862c21c1d4e02288ed704e0a3
diff --git a/Makefile.am b/Makefile.am
index ec1faf917..bbff1083e 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -61,7 +61,6 @@ shared_sources = common/FileUtil.cpp \
common/SigUtil.cpp \
common/SpookyV2.cpp \
common/Unit.cpp \
- common/UnitHTTP.cpp \
common/Util.cpp \
net/DelaySocket.cpp \
net/Socket.cpp
diff --git a/common/UnitHTTP.cpp b/common/UnitHTTP.cpp
deleted file mode 100644
index 0d4b23c07..000000000
--- a/common/UnitHTTP.cpp
+++ /dev/null
@@ -1,41 +0,0 @@
-/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-/*
- * This file is part of the LibreOffice project.
- *
- * This Source Code Form is subject to the terms of the Mozilla Public
- * License, v. 2.0. If a copy of the MPL was not distributed with this
- * file, You can obtain one at http://mozilla.org/MPL/2.0/.
- */
-
-#include <config.h>
-
-#include <iostream>
-#include "UnitHTTP.hpp"
-
-Poco::Net::HTTPClientSession *UnitHTTP::createSession()
-{
- // HTTP forced in configure hook.
- return new Poco::Net::HTTPClientSession ("127.0.0.1",
- ClientPortNumber);
-}
-
-UnitWebSocket::UnitWebSocket(const std::string &docURL)
-{
- try {
- UnitHTTPServerResponse response;
- UnitHTTPServerRequest request(response, docURL);
-
- _session = UnitHTTP::createSession();
-
- // FIXME: leaking the session - hey ho ... do we need a UnitSocket ?
- _socket = new LOOLWebSocket(*_session, request, response);
- } catch (const Poco::Exception &ex) {
- std::cerr << "Exception creating websocket " << ex.displayText() << std::endl;
- throw;
- }
-}
-
-LOOLWebSocket* UnitWebSocket::getLOOLWebSocket() const
-{
- return _socket;
-}
diff --git a/common/UnitHTTP.hpp b/common/UnitHTTP.hpp
index a6a107d9f..32f2b7416 100644
--- a/common/UnitHTTP.hpp
+++ b/common/UnitHTTP.hpp
@@ -114,7 +114,12 @@ public:
namespace UnitHTTP
{
- Poco::Net::HTTPClientSession* createSession();
+ Poco::Net::HTTPClientSession* createSession()
+ {
+ // HTTP forced in configure hook.
+ return new Poco::Net::HTTPClientSession ("127.0.0.1",
+ ClientPortNumber);
+ }
}
class UnitWebSocket
@@ -124,14 +129,32 @@ class UnitWebSocket
public:
/// Get a websocket connected for a given URL
- UnitWebSocket(const std::string& docURL);
+ UnitWebSocket(const std::string& docURL)
+ {
+ try {
+ UnitHTTPServerResponse response;
+ UnitHTTPServerRequest request(response, docURL);
+
+ _session = UnitHTTP::createSession();
+
+ // FIXME: leaking the session - hey ho ... do we need a UnitSocket ?
+ _socket = new LOOLWebSocket(*_session, request, response);
+ } catch (const Poco::Exception &ex) {
+ std::cerr << "Exception creating websocket " << ex.displayText() << std::endl;
+ throw;
+ }
+ }
+
~UnitWebSocket()
{
delete _socket;
delete _session;
}
- LOOLWebSocket* getLOOLWebSocket() const;
+ LOOLWebSocket* getLOOLWebSocket() const
+ {
+ return _socket;
+ }
};
#endif
More information about the Libreoffice-commits
mailing list