2017-03-02 12:12:52 -06:00
|
|
|
/* -*- 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/.
|
|
|
|
*/
|
|
|
|
|
2017-03-08 10:38:22 -06:00
|
|
|
#include "config.h"
|
|
|
|
|
2017-03-06 15:26:53 -06:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <ctype.h>
|
2017-03-18 09:59:09 -05:00
|
|
|
#include <iomanip>
|
2017-03-19 10:45:29 -05:00
|
|
|
#include <zlib.h>
|
2017-03-30 06:08:47 -05:00
|
|
|
|
2017-03-15 13:21:59 -05:00
|
|
|
#include <Poco/DateTime.h>
|
|
|
|
#include <Poco/DateTimeFormat.h>
|
|
|
|
#include <Poco/DateTimeFormatter.h>
|
|
|
|
|
2017-03-06 22:09:09 -06:00
|
|
|
#include "SigUtil.hpp"
|
2017-03-02 12:12:52 -06:00
|
|
|
#include "Socket.hpp"
|
2017-03-06 09:45:34 -06:00
|
|
|
#include "ServerSocket.hpp"
|
2017-03-18 09:59:09 -05:00
|
|
|
#include "WebSocketHandler.hpp"
|
2017-03-02 12:12:52 -06:00
|
|
|
|
2017-03-09 14:35:04 -06:00
|
|
|
int SocketPoll::DefaultPollTimeoutMs = 5000;
|
2017-04-05 11:58:52 -05:00
|
|
|
std::atomic<bool> Socket::InhibitThreadChecks(false);
|
2017-03-09 14:35:04 -06:00
|
|
|
|
2017-03-03 15:18:27 -06:00
|
|
|
// help with initialization order
|
|
|
|
namespace {
|
|
|
|
std::vector<int> &getWakeupsArray()
|
|
|
|
{
|
|
|
|
static std::vector<int> pollWakeups;
|
|
|
|
return pollWakeups;
|
|
|
|
}
|
|
|
|
std::mutex &getPollWakeupsMutex()
|
|
|
|
{
|
|
|
|
static std::mutex pollWakeupsMutex;
|
|
|
|
return pollWakeupsMutex;
|
|
|
|
}
|
|
|
|
}
|
2017-03-02 12:12:52 -06:00
|
|
|
|
2017-03-09 13:00:04 -06:00
|
|
|
SocketPoll::SocketPoll(const std::string& threadName)
|
|
|
|
: _name(threadName),
|
2017-03-09 13:23:21 -06:00
|
|
|
_stop(false),
|
2017-03-12 21:11:25 -05:00
|
|
|
_threadStarted(false),
|
2017-04-02 16:49:14 -05:00
|
|
|
_threadFinished(false),
|
|
|
|
_owner(std::this_thread::get_id())
|
2017-03-02 12:12:52 -06:00
|
|
|
{
|
|
|
|
// Create the wakeup fd.
|
|
|
|
if (::pipe2(_wakeup, O_CLOEXEC | O_NONBLOCK) == -1)
|
|
|
|
{
|
2017-03-28 00:07:07 -05:00
|
|
|
throw std::runtime_error("Failed to allocate pipe for SocketPoll [" + threadName + "] waking.");
|
2017-03-02 12:12:52 -06:00
|
|
|
}
|
2017-03-06 22:09:09 -06:00
|
|
|
|
2017-04-02 17:43:21 -05:00
|
|
|
std::lock_guard<std::mutex> lock(getPollWakeupsMutex());
|
|
|
|
getWakeupsArray().push_back(_wakeup[1]);
|
2017-03-02 12:12:52 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
SocketPoll::~SocketPoll()
|
|
|
|
{
|
2017-03-31 11:28:20 -05:00
|
|
|
joinThread();
|
2017-03-06 22:09:09 -06:00
|
|
|
|
2017-03-17 13:56:59 -05:00
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> lock(getPollWakeupsMutex());
|
|
|
|
auto it = std::find(getWakeupsArray().begin(),
|
|
|
|
getWakeupsArray().end(),
|
|
|
|
_wakeup[1]);
|
2017-03-02 12:12:52 -06:00
|
|
|
|
2017-03-17 13:56:59 -05:00
|
|
|
if (it != getWakeupsArray().end())
|
|
|
|
getWakeupsArray().erase(it);
|
|
|
|
}
|
2017-03-03 15:18:27 -06:00
|
|
|
|
2017-03-17 13:56:59 -05:00
|
|
|
::close(_wakeup[0]);
|
|
|
|
::close(_wakeup[1]);
|
|
|
|
_wakeup[0] = -1;
|
|
|
|
_wakeup[1] = -1;
|
2017-03-03 15:18:27 -06:00
|
|
|
}
|
|
|
|
|
2017-03-09 13:00:04 -06:00
|
|
|
void SocketPoll::startThread()
|
|
|
|
{
|
2017-03-09 13:23:21 -06:00
|
|
|
if (!_threadStarted)
|
|
|
|
{
|
2017-03-19 12:22:12 -05:00
|
|
|
_threadStarted = true;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
_thread = std::thread(&SocketPoll::pollingThreadEntry, this);
|
|
|
|
}
|
|
|
|
catch (const std::exception& exc)
|
|
|
|
{
|
|
|
|
LOG_ERR("Failed to start poll thread: " << exc.what());
|
|
|
|
_threadStarted = false;
|
|
|
|
}
|
2017-03-09 13:23:21 -06:00
|
|
|
}
|
2017-03-09 13:00:04 -06:00
|
|
|
}
|
|
|
|
|
2017-03-31 11:28:20 -05:00
|
|
|
void SocketPoll::joinThread()
|
|
|
|
{
|
2017-04-06 01:56:21 -05:00
|
|
|
addCallback([this](){ removeSockets(); });
|
2017-03-31 11:28:20 -05:00
|
|
|
stop();
|
|
|
|
if (_threadStarted && _thread.joinable())
|
|
|
|
{
|
|
|
|
if (_thread.get_id() == std::this_thread::get_id())
|
|
|
|
LOG_ERR("DEADLOCK PREVENTED: joining own thread!");
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_thread.join();
|
|
|
|
_threadStarted = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-03 15:18:27 -06:00
|
|
|
void SocketPoll::wakeupWorld()
|
|
|
|
{
|
|
|
|
for (const auto& fd : getWakeupsArray())
|
|
|
|
wakeup(fd);
|
2017-03-02 12:12:52 -06:00
|
|
|
}
|
|
|
|
|
2017-03-12 18:03:45 -05:00
|
|
|
void ServerSocket::dumpState(std::ostream& os)
|
2017-03-06 09:45:34 -06:00
|
|
|
{
|
2017-03-12 18:03:45 -05:00
|
|
|
os << "\t" << getFD() << "\t<accept>\n";
|
2017-03-06 09:45:34 -06:00
|
|
|
}
|
|
|
|
|
2017-03-06 15:26:53 -06:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
void dump_hex (const char *legend, const char *prefix, std::vector<char> buffer)
|
|
|
|
{
|
|
|
|
unsigned int i, j;
|
|
|
|
fprintf (stderr, "%s", legend);
|
|
|
|
for (j = 0; j < buffer.size() + 15; j += 16)
|
|
|
|
{
|
|
|
|
fprintf (stderr, "%s0x%.4x ", prefix, j);
|
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
{
|
|
|
|
if ((j + i) < buffer.size())
|
|
|
|
fprintf (stderr, "%.2x ", (unsigned char)buffer[j+i]);
|
|
|
|
else
|
|
|
|
fprintf (stderr, " ");
|
|
|
|
if (i == 8)
|
|
|
|
fprintf (stderr, " ");
|
|
|
|
}
|
|
|
|
fprintf (stderr, " | ");
|
|
|
|
|
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
if ((j + i) < buffer.size() && ::isprint(buffer[j+i]))
|
|
|
|
fprintf (stderr, "%c", buffer[j+i]);
|
|
|
|
else
|
|
|
|
fprintf (stderr, ".");
|
|
|
|
fprintf (stderr, "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2017-03-18 09:59:09 -05:00
|
|
|
void WebSocketHandler::dumpState(std::ostream& os)
|
|
|
|
{
|
|
|
|
os << (_shuttingDown ? "shutd " : "alive ")
|
|
|
|
<< std::setw(5) << 1.0*_pingTimeUs/1000 << "ms ";
|
|
|
|
if (_wsPayload.size() > 0)
|
|
|
|
dump_hex("\t\tws queued payload:\n", "\t\t", _wsPayload);
|
|
|
|
}
|
|
|
|
|
2017-03-12 18:03:45 -05:00
|
|
|
void StreamSocket::dumpState(std::ostream& os)
|
2017-03-06 09:45:34 -06:00
|
|
|
{
|
2017-03-17 16:59:09 -05:00
|
|
|
int timeoutMaxMs = SocketPoll::DefaultPollTimeoutMs;
|
|
|
|
int events = getPollEvents(std::chrono::steady_clock::now(), timeoutMaxMs);
|
|
|
|
os << "\t" << getFD() << "\t" << events << "\t"
|
2017-03-18 09:59:09 -05:00
|
|
|
<< _inBuffer.size() << "\t" << _outBuffer.size() << "\t";
|
|
|
|
_socketHandler->dumpState(os);
|
2017-03-06 15:26:53 -06:00
|
|
|
if (_inBuffer.size() > 0)
|
|
|
|
dump_hex("\t\tinBuffer:\n", "\t\t", _inBuffer);
|
|
|
|
if (_outBuffer.size() > 0)
|
|
|
|
dump_hex("\t\toutBuffer:\n", "\t\t", _inBuffer);
|
2017-03-06 09:45:34 -06:00
|
|
|
}
|
|
|
|
|
2017-03-12 18:03:45 -05:00
|
|
|
void SocketPoll::dumpState(std::ostream& os)
|
2017-03-03 15:18:55 -06:00
|
|
|
{
|
2017-03-12 18:03:45 -05:00
|
|
|
// FIXME: NOT thread-safe! _pollSockets is modified from the polling thread!
|
|
|
|
os << " Poll [" << _pollSockets.size() << "] - wakeup r: "
|
|
|
|
<< _wakeup[0] << " w: " << _wakeup[1] << "\n";
|
2017-03-20 12:18:41 -05:00
|
|
|
if (_newCallbacks.size() > 0)
|
|
|
|
os << "\tcallbacks: " << _newCallbacks.size() << "\n";
|
2017-03-12 18:03:45 -05:00
|
|
|
os << "\tfd\tevents\trsize\twsize\n";
|
2017-03-03 15:18:55 -06:00
|
|
|
for (auto &i : _pollSockets)
|
2017-03-12 18:03:45 -05:00
|
|
|
i->dumpState(os);
|
2017-03-03 15:18:55 -06:00
|
|
|
}
|
|
|
|
|
2017-03-15 13:21:59 -05:00
|
|
|
namespace HttpHelper
|
|
|
|
{
|
2017-04-09 10:48:05 -05:00
|
|
|
void sendFile(const std::shared_ptr<StreamSocket>& socket, const std::string& path, const std::string& mediaType,
|
2017-03-19 10:45:29 -05:00
|
|
|
Poco::Net::HTTPResponse& response, bool noCache, bool deflate)
|
2017-03-15 13:21:59 -05:00
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
if (stat(path.c_str(), &st) != 0)
|
|
|
|
{
|
|
|
|
LOG_WRN("#" << socket->getFD() << ": Failed to stat [" << path << "]. File will not be sent.");
|
|
|
|
throw Poco::FileNotFoundException("Failed to stat [" + path + "]. File will not be sent.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
response.set("User-Agent", HTTP_AGENT_STRING);
|
|
|
|
response.set("Date", Poco::DateTimeFormatter::format(Poco::Timestamp(), Poco::DateTimeFormat::HTTP_FORMAT));
|
|
|
|
if (!noCache)
|
|
|
|
{
|
|
|
|
// 60 * 60 * 24 * 128 (days) = 11059200
|
|
|
|
response.set("Cache-Control", "max-age=11059200");
|
|
|
|
response.set("ETag", "\"" LOOLWSD_VERSION_HASH "\"");
|
|
|
|
}
|
|
|
|
|
2017-04-09 10:48:05 -05:00
|
|
|
response.setContentType(mediaType);
|
|
|
|
response.add("X-Content-Type-Options", "nosniff");
|
|
|
|
|
2017-04-03 22:57:59 -05:00
|
|
|
int bufferSize = std::min(st.st_size, (off_t)Socket::MaximumSendBufferSize);
|
|
|
|
if (st.st_size >= socket->getSendBufferSize())
|
2017-03-19 10:45:29 -05:00
|
|
|
{
|
2017-04-03 22:57:59 -05:00
|
|
|
socket->setSocketBufferSize(bufferSize);
|
|
|
|
bufferSize = socket->getSendBufferSize();
|
|
|
|
}
|
2017-03-19 10:45:29 -05:00
|
|
|
|
2017-04-04 07:17:23 -05:00
|
|
|
// Disable deflate for now - until we can cache deflated data.
|
2017-04-06 10:07:40 -05:00
|
|
|
// FIXME: IE/Edge doesn't work well with deflate, so check with
|
|
|
|
// IE/Edge before enabling the deflate again
|
|
|
|
if (!deflate || true)
|
2017-04-03 22:57:59 -05:00
|
|
|
{
|
2017-03-19 10:45:29 -05:00
|
|
|
response.setContentLength(st.st_size);
|
|
|
|
std::ostringstream oss;
|
|
|
|
response.write(oss);
|
|
|
|
const std::string header = oss.str();
|
|
|
|
LOG_TRC("#" << socket->getFD() << ": Sending file [" << path << "]: " << header);
|
|
|
|
socket->send(header);
|
2017-03-15 13:21:59 -05:00
|
|
|
|
2017-03-19 10:45:29 -05:00
|
|
|
std::ifstream file(path, std::ios::binary);
|
|
|
|
bool flush = true;
|
2017-04-04 07:17:23 -05:00
|
|
|
std::unique_ptr<char[]> buf(new char[bufferSize]);
|
2017-03-19 10:45:29 -05:00
|
|
|
do
|
|
|
|
{
|
2017-04-04 07:17:23 -05:00
|
|
|
file.read(&buf[0], bufferSize);
|
2017-03-19 10:45:29 -05:00
|
|
|
const int size = file.gcount();
|
|
|
|
if (size > 0)
|
2017-04-04 07:17:23 -05:00
|
|
|
socket->send(&buf[0], size, flush);
|
2017-03-19 10:45:29 -05:00
|
|
|
else
|
|
|
|
break;
|
|
|
|
flush = false;
|
|
|
|
}
|
|
|
|
while (file);
|
|
|
|
}
|
|
|
|
else
|
2017-03-15 13:21:59 -05:00
|
|
|
{
|
2017-03-19 10:45:29 -05:00
|
|
|
response.set("Content-Encoding", "deflate");
|
|
|
|
std::ostringstream oss;
|
|
|
|
response.write(oss);
|
|
|
|
const std::string header = oss.str();
|
|
|
|
LOG_TRC("#" << socket->getFD() << ": Sending file [" << path << "]: " << header);
|
|
|
|
socket->send(header);
|
|
|
|
|
|
|
|
std::ifstream file(path, std::ios::binary);
|
|
|
|
bool flush = true;
|
2017-04-04 07:17:23 -05:00
|
|
|
|
|
|
|
// FIXME: Should compress once ahead of time
|
|
|
|
// compression of bundle.js takes significant time:
|
|
|
|
// 200's ms for level 9 (468k), 72ms for level 1(587k)
|
|
|
|
// down from 2Mb.
|
|
|
|
std::unique_ptr<char[]> buf(new char[st.st_size]);
|
2017-03-19 10:45:29 -05:00
|
|
|
do
|
|
|
|
{
|
2017-04-04 07:17:23 -05:00
|
|
|
static const unsigned int level = 1;
|
|
|
|
file.read(&buf[0], st.st_size);
|
2017-04-03 22:57:59 -05:00
|
|
|
const long unsigned int size = file.gcount();
|
2017-03-19 10:45:29 -05:00
|
|
|
long unsigned int compSize = compressBound(size);
|
2017-04-04 07:17:23 -05:00
|
|
|
std::unique_ptr<char[]> cbuf(new char[compSize]);
|
|
|
|
compress2((Bytef *)&cbuf[0], &compSize, (Bytef *)&buf[0], size, level);
|
2017-03-19 10:45:29 -05:00
|
|
|
if (size > 0)
|
2017-04-04 07:17:23 -05:00
|
|
|
socket->send(&cbuf[0], compSize, flush);
|
2017-03-19 10:45:29 -05:00
|
|
|
else
|
|
|
|
break;
|
|
|
|
flush = false;
|
|
|
|
}
|
|
|
|
while(file);
|
2017-03-15 13:21:59 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-02 12:12:52 -06:00
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|