2015-03-04 17:14:04 -06:00
|
|
|
/* -*- 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/.
|
|
|
|
*/
|
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
#include <cassert>
|
2015-03-12 09:18:35 -05:00
|
|
|
#include <cstring>
|
2015-03-04 17:14:04 -06:00
|
|
|
#include <fstream>
|
|
|
|
#include <iostream>
|
2015-03-12 09:18:35 -05:00
|
|
|
#include <memory>
|
2015-03-04 17:14:04 -06:00
|
|
|
|
|
|
|
#include <png.h>
|
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
#include <Poco/Buffer.h>
|
|
|
|
#include <Poco/Process.h>
|
|
|
|
#include <Poco/Random.h>
|
2015-03-04 17:14:04 -06:00
|
|
|
#include <Poco/String.h>
|
|
|
|
#include <Poco/StringTokenizer.h>
|
2015-03-17 18:56:15 -05:00
|
|
|
#include <Poco/Util/Application.h>
|
2015-03-04 17:14:04 -06:00
|
|
|
|
2015-03-09 03:01:30 -05:00
|
|
|
#include "LOOLSession.hpp"
|
2015-03-17 18:56:15 -05:00
|
|
|
#include "LOOLWSD.hpp"
|
2015-03-12 09:18:35 -05:00
|
|
|
#include "TileCache.hpp"
|
2015-03-17 18:56:15 -05:00
|
|
|
#include "Util.hpp"
|
2015-03-04 17:14:04 -06:00
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
using Poco::Buffer;
|
2015-03-04 17:14:04 -06:00
|
|
|
using Poco::Net::WebSocket;
|
2015-03-17 18:56:15 -05:00
|
|
|
using Poco::Process;
|
|
|
|
using Poco::ProcessHandle;
|
|
|
|
using Poco::Random;
|
2015-03-04 17:14:04 -06:00
|
|
|
using Poco::StringTokenizer;
|
2015-03-17 18:56:15 -05:00
|
|
|
using Poco::Thread;
|
|
|
|
using Poco::UInt64;
|
|
|
|
using Poco::Util::Application;
|
|
|
|
|
|
|
|
std::map<UInt64, LOOLSession*> LOOLSession::_childToClient;
|
2015-03-04 17:14:04 -06:00
|
|
|
|
2015-03-09 03:01:30 -05:00
|
|
|
LOOLSession::LOOLSession(WebSocket& ws, LibreOfficeKit *loKit) :
|
2015-03-17 18:56:15 -05:00
|
|
|
_docURL(""),
|
2015-03-04 17:14:04 -06:00
|
|
|
_ws(ws),
|
2015-03-17 18:56:15 -05:00
|
|
|
_toChildProcess(false),
|
|
|
|
_peerWs(nullptr),
|
|
|
|
_childId(0),
|
2015-03-04 17:14:04 -06:00
|
|
|
_loKit(loKit),
|
|
|
|
_loKitDocument(NULL)
|
|
|
|
{
|
2015-03-17 18:56:15 -05:00
|
|
|
std::cout << Util::logPrefix() << "LOOLSesstion ctor this=" << this << " ws=" << &_ws << std::endl;
|
2015-03-04 17:14:04 -06:00
|
|
|
}
|
|
|
|
|
2015-03-09 03:01:30 -05:00
|
|
|
LOOLSession::~LOOLSession()
|
2015-03-07 05:23:46 -06:00
|
|
|
{
|
2015-03-17 18:56:15 -05:00
|
|
|
std::cout << Util::logPrefix() << "LOOLSesstion dtor this=" << this << std::endl;
|
2015-03-07 05:23:46 -06:00
|
|
|
_ws.shutdown();
|
|
|
|
if (_loKitDocument)
|
|
|
|
_loKitDocument->pClass->destroy(_loKitDocument);
|
|
|
|
}
|
|
|
|
|
2015-03-09 03:01:30 -05:00
|
|
|
bool LOOLSession::handleInput(char *buffer, int length)
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
|
|
|
Application& app = Application::instance();
|
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
if (haveSeparateProcess())
|
|
|
|
{
|
|
|
|
forwardRequest(buffer, length);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-03-04 17:14:04 -06:00
|
|
|
char *endl = (char *) memchr(buffer, '\n', length);
|
|
|
|
std::string commandline;
|
2015-03-17 18:56:15 -05:00
|
|
|
if (endl == nullptr)
|
2015-03-04 17:14:04 -06:00
|
|
|
commandline = std::string(buffer, length);
|
|
|
|
else
|
|
|
|
commandline = std::string(buffer, endl-buffer);
|
2015-03-07 05:23:46 -06:00
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
std::cout << Util::logPrefix() << "this=" << this << std::endl;
|
|
|
|
|
|
|
|
app.logger().information(Util::logPrefix() + "Input: '" + commandline + "'" + (endl == nullptr ? "" : " ..."));
|
2015-03-04 17:14:04 -06:00
|
|
|
|
|
|
|
StringTokenizer tokens(commandline, " ", StringTokenizer::TOK_IGNORE_EMPTY | StringTokenizer::TOK_TRIM);
|
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
if (toChildProcess())
|
|
|
|
{
|
|
|
|
assert(_peerWs != nullptr);
|
|
|
|
_peerWs->sendFrame(buffer, length, WebSocket::FRAME_BINARY);
|
|
|
|
}
|
|
|
|
else if (tokens[0] == "child")
|
|
|
|
{
|
|
|
|
if (_peerWs != nullptr ||
|
|
|
|
isChildProcess())
|
|
|
|
{
|
|
|
|
sendTextFrame("error: cmd=child kind=invalid");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (tokens.count() != 2)
|
|
|
|
{
|
|
|
|
sendTextFrame("error: cmd=child kind=syntax");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
UInt64 childId = std::stoull(tokens[1]);
|
|
|
|
app.logger().information(Util::logPrefix() + "childId=" + std::to_string(childId));
|
|
|
|
if (_childToClient.find(childId) == _childToClient.end())
|
|
|
|
{
|
|
|
|
sendTextFrame("error: cmd=child kind=notfound");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
assert(_childToClient[childId]->haveSeparateProcess());
|
|
|
|
assert(_childToClient[childId]->_peerWs == nullptr);
|
|
|
|
_childToClient[childId]->_peerWs = &_ws;
|
|
|
|
_peerWs = &_childToClient[childId]->_ws;
|
|
|
|
for (auto i : _childToClient[childId]->_backLog)
|
|
|
|
{
|
|
|
|
std::cout << Util::logPrefix() << "Sending backlog: '" << std::string(i->begin(), i->size()) << "'" << std::endl;
|
|
|
|
_ws.sendFrame(i->begin(), i->size(), WebSocket::FRAME_BINARY);
|
|
|
|
delete i;
|
|
|
|
}
|
|
|
|
_toChildProcess = true;
|
|
|
|
}
|
|
|
|
else if (tokens[0] == "load")
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
2015-03-17 18:56:15 -05:00
|
|
|
if (_docURL != "")
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
2015-03-13 06:59:51 -05:00
|
|
|
sendTextFrame("error: cms=load kind=docalreadyloaded");
|
2015-03-07 05:23:46 -06:00
|
|
|
return false;
|
2015-03-04 17:14:04 -06:00
|
|
|
}
|
2015-03-17 18:56:15 -05:00
|
|
|
return loadDocument(buffer, length, tokens);
|
2015-03-04 17:14:04 -06:00
|
|
|
}
|
2015-03-17 18:56:15 -05:00
|
|
|
else if (_docURL == "")
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
2015-03-13 06:59:51 -05:00
|
|
|
sendTextFrame("error: cmd=" + tokens[0] + " kind=nodocloaded");
|
2015-03-07 05:23:46 -06:00
|
|
|
return false;
|
2015-03-04 17:14:04 -06:00
|
|
|
}
|
|
|
|
else if (tokens[0] == "status")
|
|
|
|
{
|
2015-03-17 18:56:15 -05:00
|
|
|
return getStatus(buffer, length);
|
2015-03-04 17:14:04 -06:00
|
|
|
}
|
|
|
|
else if (tokens[0] == "tile")
|
|
|
|
{
|
2015-03-17 18:56:15 -05:00
|
|
|
sendTile(buffer, length, tokens);
|
|
|
|
}
|
2015-03-19 07:38:04 -05:00
|
|
|
else if (tokens[0] == "key")
|
|
|
|
{
|
|
|
|
return keyEvent(buffer, length, tokens);
|
|
|
|
}
|
|
|
|
else if (tokens[0] == "mouse")
|
|
|
|
{
|
|
|
|
return mouseEvent(buffer, length, tokens);
|
|
|
|
}
|
|
|
|
else if (tokens[0] == "uno")
|
|
|
|
{
|
|
|
|
return unoCommand(buffer, length, tokens);
|
|
|
|
}
|
|
|
|
else if (tokens[0] == "selecttext")
|
|
|
|
{
|
|
|
|
return selectText(buffer, length, tokens);
|
|
|
|
}
|
|
|
|
else if (tokens[0] == "selectgraphic")
|
|
|
|
{
|
|
|
|
return selectGraphic(buffer, length, tokens);
|
|
|
|
}
|
|
|
|
else if (tokens[0] == "resetselection")
|
|
|
|
{
|
|
|
|
resetSelection();
|
|
|
|
}
|
2015-03-17 18:56:15 -05:00
|
|
|
else
|
|
|
|
{
|
|
|
|
sendTextFrame("error: cmd=" + tokens[0] + " kind=unknown");
|
|
|
|
return false;
|
2015-03-04 17:14:04 -06:00
|
|
|
}
|
2015-03-07 05:23:46 -06:00
|
|
|
return true;
|
2015-03-04 17:14:04 -06:00
|
|
|
}
|
|
|
|
|
2015-03-12 18:34:42 -05:00
|
|
|
bool LOOLSession::haveSeparateProcess() const
|
|
|
|
{
|
2015-03-17 18:56:15 -05:00
|
|
|
return _childId != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LOOLSession::toChildProcess() const
|
|
|
|
{
|
|
|
|
return _toChildProcess;
|
2015-03-12 18:34:42 -05:00
|
|
|
}
|
|
|
|
|
2015-03-09 03:01:30 -05:00
|
|
|
void LOOLSession::sendTextFrame(std::string text)
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
|
|
|
_ws.sendFrame(text.data(), text.size());
|
|
|
|
}
|
|
|
|
|
2015-03-09 03:01:30 -05:00
|
|
|
void LOOLSession::sendBinaryFrame(const char *buffer, int length)
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
|
|
|
_ws.sendFrame(buffer, length, WebSocket::FRAME_BINARY);
|
|
|
|
}
|
|
|
|
|
2015-03-07 05:23:46 -06:00
|
|
|
extern "C"
|
|
|
|
{
|
|
|
|
static void myCallback(int nType, const char* pPayload, void* pData)
|
|
|
|
{
|
2015-03-17 18:56:15 -05:00
|
|
|
LOOLSession *srv = reinterpret_cast<LOOLSession *>(pData);
|
2015-03-07 05:23:46 -06:00
|
|
|
|
|
|
|
switch ((LibreOfficeKitCallbackType) nType)
|
|
|
|
{
|
|
|
|
case LOK_CALLBACK_INVALIDATE_TILES:
|
|
|
|
srv->sendTextFrame("invalidatetiles: " + std::string(pPayload));
|
|
|
|
break;
|
|
|
|
case LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR:
|
|
|
|
srv->sendTextFrame("invalidatecursor:");
|
2015-03-13 04:25:00 -05:00
|
|
|
break;
|
2015-03-07 05:23:46 -06:00
|
|
|
case LOK_CALLBACK_TEXT_SELECTION:
|
|
|
|
srv->sendTextFrame("textselection: " + std::string(pPayload));
|
|
|
|
break;
|
|
|
|
case LOK_CALLBACK_TEXT_SELECTION_START:
|
|
|
|
srv->sendTextFrame("textselectionstart: " + std::string(pPayload));
|
|
|
|
break;
|
|
|
|
case LOK_CALLBACK_TEXT_SELECTION_END:
|
|
|
|
srv->sendTextFrame("textselectionend: " + std::string(pPayload));
|
|
|
|
break;
|
2015-03-13 04:30:23 -05:00
|
|
|
case LOK_CALLBACK_CURSOR_VISIBLE:
|
|
|
|
srv->sendTextFrame("cursorvisible: " + std::string(pPayload));
|
|
|
|
break;
|
|
|
|
case LOK_CALLBACK_GRAPHIC_SELECTION:
|
|
|
|
srv->sendTextFrame("graphicselection: " + std::string(pPayload));
|
|
|
|
break;
|
|
|
|
case LOK_CALLBACK_HYPERLINK_CLICKED:
|
|
|
|
srv->sendTextFrame("hyperlinkclicked: " + std::string(pPayload));
|
|
|
|
break;
|
2015-03-07 05:23:46 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
bool LOOLSession::loadDocument(const char *buffer, int length, StringTokenizer& tokens)
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
|
|
|
if (tokens.count() != 2)
|
|
|
|
{
|
|
|
|
sendTextFrame("error: cmd=load kind=syntax");
|
2015-03-17 18:56:15 -05:00
|
|
|
return false;
|
2015-03-04 17:14:04 -06:00
|
|
|
}
|
2015-03-12 18:34:42 -05:00
|
|
|
|
2015-03-19 07:38:04 -05:00
|
|
|
if (tokens[1].find("url=") == 0)
|
|
|
|
_docURL = tokens[1].substr(strlen("url="));
|
|
|
|
else
|
|
|
|
_docURL = tokens[1];
|
|
|
|
|
2015-03-12 18:34:42 -05:00
|
|
|
_tileCache.reset(new TileCache(_docURL));
|
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
if (isChildProcess())
|
2015-03-07 05:23:46 -06:00
|
|
|
{
|
2015-03-17 18:56:15 -05:00
|
|
|
if ((_loKitDocument = _loKit->pClass->documentLoad(_loKit, _docURL.c_str())) != NULL)
|
|
|
|
{
|
|
|
|
if (!getStatus(buffer, length))
|
|
|
|
return false;
|
|
|
|
_loKitDocument->pClass->registerCallback(_loKitDocument, myCallback, this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2015-03-04 17:14:04 -06:00
|
|
|
}
|
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
bool LOOLSession::getStatus(const char *buffer, int length)
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
2015-03-13 07:17:51 -05:00
|
|
|
std::string status = _tileCache->getStatus();
|
|
|
|
if (status.size() > 0)
|
2015-03-17 18:56:15 -05:00
|
|
|
{
|
|
|
|
sendTextFrame(status);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isChildProcess())
|
|
|
|
{
|
|
|
|
forkOff();
|
|
|
|
forwardRequest(buffer, length);
|
|
|
|
return true;
|
|
|
|
}
|
2015-03-13 07:17:51 -05:00
|
|
|
|
2015-03-04 17:14:04 -06:00
|
|
|
LibreOfficeKitDocumentType type = (LibreOfficeKitDocumentType) _loKitDocument->pClass->getDocumentType(_loKitDocument);
|
|
|
|
std::string typeString;
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case LOK_DOCTYPE_TEXT:
|
|
|
|
typeString = "text";
|
|
|
|
break;
|
|
|
|
case LOK_DOCTYPE_SPREADSHEET:
|
|
|
|
typeString = "spreadsheet";
|
|
|
|
break;
|
|
|
|
case LOK_DOCTYPE_PRESENTATION:
|
|
|
|
typeString = "presentation";
|
|
|
|
break;
|
|
|
|
case LOK_DOCTYPE_DRAWING:
|
|
|
|
typeString = "drawing";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
typeString = "other";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
long width, height;
|
|
|
|
_loKitDocument->pClass->getDocumentSize(_loKitDocument, &width, &height);
|
2015-03-17 18:56:15 -05:00
|
|
|
status = ("status: type=" + typeString + " "
|
|
|
|
"parts=" + std::to_string(_loKitDocument->pClass->getParts(_loKitDocument)) + " "
|
|
|
|
"current=" + std::to_string(_loKitDocument->pClass->getPart(_loKitDocument)) + " "
|
|
|
|
"width=" + std::to_string(width) + " "
|
|
|
|
"height=" + std::to_string(height));
|
|
|
|
_tileCache->saveStatus(status);
|
|
|
|
|
|
|
|
sendTextFrame(status);
|
|
|
|
|
|
|
|
return true;
|
2015-03-04 17:14:04 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
bool getTokenInteger(const std::string& token, const std::string& name, int *value)
|
|
|
|
{
|
|
|
|
size_t nextIdx;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if (token.size() < name.size() + 2 ||
|
|
|
|
token.substr(0, name.size()) != name ||
|
|
|
|
token[name.size()] != '=' ||
|
|
|
|
(*value = std::stoi(token.substr(name.size() + 1), &nextIdx), false) ||
|
|
|
|
nextIdx != token.size() - name.size() - 1)
|
|
|
|
{
|
|
|
|
throw std::invalid_argument("bah");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (std::invalid_argument&)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-07 05:23:46 -06:00
|
|
|
// Callback functions for libpng
|
|
|
|
|
|
|
|
extern "C"
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
2015-03-07 05:23:46 -06:00
|
|
|
static void user_write_status_fn(png_structp, png_uint_32, int)
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-03-07 05:23:46 -06:00
|
|
|
static void user_write_fn(png_structp png_ptr, png_bytep data, png_size_t length)
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
|
|
|
std::vector<char> *outputp = (std::vector<char> *) png_get_io_ptr(png_ptr);
|
|
|
|
size_t oldsize = outputp->size();
|
|
|
|
outputp->resize(oldsize + length);
|
|
|
|
memcpy(outputp->data() + oldsize, data, length);
|
|
|
|
}
|
|
|
|
|
2015-03-07 05:23:46 -06:00
|
|
|
static void user_flush_fn(png_structp)
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
void LOOLSession::sendTile(const char *buffer, int length, StringTokenizer& tokens)
|
2015-03-04 17:14:04 -06:00
|
|
|
{
|
|
|
|
int width, height, tilePosX, tilePosY, tileWidth, tileHeight;
|
|
|
|
|
|
|
|
if (tokens.count() != 7 ||
|
|
|
|
!getTokenInteger(tokens[1], "width", &width) ||
|
2015-03-07 05:23:46 -06:00
|
|
|
!getTokenInteger(tokens[2], "height", &height) ||
|
2015-03-04 17:14:04 -06:00
|
|
|
!getTokenInteger(tokens[3], "tileposx", &tilePosX) ||
|
|
|
|
!getTokenInteger(tokens[4], "tileposy", &tilePosY) ||
|
|
|
|
!getTokenInteger(tokens[5], "tilewidth", &tileWidth) ||
|
|
|
|
!getTokenInteger(tokens[6], "tileheight", &tileHeight))
|
|
|
|
{
|
|
|
|
sendTextFrame("error: cmd=tile kind=syntax");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-03-12 09:18:35 -05:00
|
|
|
std::string response = "tile: " + Poco::cat(std::string(" "), tokens.begin() + 1, tokens.end()) + "\n";
|
|
|
|
|
|
|
|
std::vector<char> output;
|
|
|
|
output.reserve(4 * width * height);
|
|
|
|
output.resize(response.size());
|
|
|
|
memcpy(output.data(), response.data(), response.size());
|
|
|
|
|
2015-03-13 07:17:51 -05:00
|
|
|
std::unique_ptr<std::fstream> cachedTile = _tileCache->lookupTile(width, height, tilePosX, tilePosY, tileWidth, tileHeight);
|
2015-03-12 18:34:42 -05:00
|
|
|
if (cachedTile && cachedTile->is_open())
|
2015-03-12 09:18:35 -05:00
|
|
|
{
|
|
|
|
cachedTile->seekg(0, std::ios_base::end);
|
|
|
|
size_t pos = output.size();
|
|
|
|
std::streamsize size = cachedTile->tellg();
|
|
|
|
output.resize(pos + size);
|
|
|
|
cachedTile->seekg(0, std::ios_base::beg);
|
|
|
|
cachedTile->read(output.data() + pos, size);
|
|
|
|
cachedTile->close();
|
|
|
|
|
|
|
|
sendBinaryFrame(output.data(), output.size());
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
if (!isChildProcess())
|
|
|
|
{
|
|
|
|
forkOff();
|
|
|
|
forwardRequest(buffer, length);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned char *pixmap = new unsigned char[4 * width * height];
|
2015-03-20 06:16:41 -05:00
|
|
|
_loKitDocument->pClass->paintTile(_loKitDocument, pixmap, width, height, tilePosX, tilePosY, tileWidth, tileHeight);
|
2015-03-04 17:14:04 -06:00
|
|
|
|
|
|
|
png_structp png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
|
|
|
|
|
|
|
|
png_infop info_ptr = png_create_info_struct(png_ptr);
|
|
|
|
|
|
|
|
if (setjmp(png_jmpbuf(png_ptr)))
|
|
|
|
{
|
|
|
|
png_destroy_write_struct(&png_ptr, NULL);
|
|
|
|
sendTextFrame("error: cmd=tile kind=failure");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
png_set_IHDR(png_ptr, info_ptr, width, height, 8, PNG_COLOR_TYPE_RGB_ALPHA, PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_DEFAULT, PNG_FILTER_TYPE_DEFAULT);
|
|
|
|
|
|
|
|
png_set_write_fn(png_ptr, &output, user_write_fn, user_flush_fn);
|
|
|
|
png_set_write_status_fn(png_ptr, user_write_status_fn);
|
|
|
|
|
|
|
|
png_write_info(png_ptr, info_ptr);
|
|
|
|
|
|
|
|
for (int y = 0; y < height; ++y)
|
2015-03-17 18:56:15 -05:00
|
|
|
png_write_row(png_ptr, pixmap + y * width * 4);
|
2015-03-04 17:14:04 -06:00
|
|
|
|
|
|
|
png_write_end(png_ptr, info_ptr);
|
|
|
|
|
|
|
|
png_destroy_write_struct(&png_ptr, NULL);
|
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
delete[] pixmap;
|
2015-03-04 17:14:04 -06:00
|
|
|
|
2015-03-13 07:17:51 -05:00
|
|
|
_tileCache->saveTile(width, height, tilePosX, tilePosY, tileWidth, tileHeight, output.data() + response.size(), output.size() - response.size());
|
2015-03-12 09:18:35 -05:00
|
|
|
|
2015-03-04 17:14:04 -06:00
|
|
|
sendBinaryFrame(output.data(), output.size());
|
|
|
|
}
|
|
|
|
|
2015-03-19 07:38:04 -05:00
|
|
|
bool LOOLSession::keyEvent(const char *buffer, int length, Poco::StringTokenizer& tokens)
|
|
|
|
{
|
|
|
|
int type, charcode, keycode;
|
|
|
|
|
|
|
|
if (tokens.count() != 4 ||
|
|
|
|
!getTokenInteger(tokens[1], "type", &type) ||
|
|
|
|
!getTokenInteger(tokens[2], "char", &charcode) ||
|
|
|
|
!getTokenInteger(tokens[3], "key", &keycode))
|
|
|
|
{
|
|
|
|
sendTextFrame("error: cmd=key kind=syntax");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_loKitDocument->pClass->postKeyEvent(_loKitDocument, type, charcode, keycode);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LOOLSession::mouseEvent(const char *buffer, int length, Poco::StringTokenizer& tokens)
|
|
|
|
{
|
|
|
|
int type, x, y, count;
|
|
|
|
|
|
|
|
if (tokens.count() != 5 ||
|
|
|
|
!getTokenInteger(tokens[1], "type", &type) ||
|
|
|
|
!getTokenInteger(tokens[2], "x", &x) ||
|
|
|
|
!getTokenInteger(tokens[3], "y", &y) ||
|
|
|
|
!getTokenInteger(tokens[4], "count", &count))
|
|
|
|
{
|
|
|
|
sendTextFrame("error: cmd=mouse kind=syntax");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_loKitDocument->pClass->postMouseEvent(_loKitDocument, type, x, y, count);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LOOLSession::unoCommand(const char *buffer, int length, Poco::StringTokenizer& tokens)
|
|
|
|
{
|
|
|
|
if (tokens.count() == 1)
|
|
|
|
{
|
|
|
|
sendTextFrame("error: cmd=uno kind=syntax");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_loKitDocument->pClass->postUnoCommand(_loKitDocument, buffer + strlen("uno "));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LOOLSession::selectText(const char *buffer, int length, Poco::StringTokenizer& tokens)
|
|
|
|
{
|
|
|
|
int type, x, y;
|
|
|
|
|
|
|
|
if (tokens.count() != 4 || !getTokenInteger(tokens[1], "type", &type) ||
|
|
|
|
!getTokenInteger(tokens[2], "x", &x) ||
|
|
|
|
!getTokenInteger(tokens[3], "y", &y))
|
|
|
|
{
|
|
|
|
sendTextFrame("error: cmd=selecttext kind=syntax");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_loKitDocument->pClass->setTextSelection(_loKitDocument, type, x, y);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LOOLSession::selectGraphic(const char *buffer, int length, Poco::StringTokenizer& tokens)
|
|
|
|
{
|
|
|
|
int type, x, y;
|
|
|
|
|
|
|
|
if (tokens.count() != 4 ||
|
|
|
|
!getTokenInteger(tokens[1], "type", &type) ||
|
|
|
|
!getTokenInteger(tokens[2], "x", &x) ||
|
|
|
|
!getTokenInteger(tokens[3], "y", &y))
|
|
|
|
{
|
|
|
|
sendTextFrame("error: cmd=selectghraphic kind=syntax");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_loKitDocument->pClass->setGraphicSelection(_loKitDocument, type, x, y);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void LOOLSession::resetSelection()
|
|
|
|
{
|
|
|
|
_loKitDocument->pClass->resetSelection(_loKitDocument);
|
|
|
|
}
|
|
|
|
|
2015-03-12 18:34:42 -05:00
|
|
|
void LOOLSession::forkOff()
|
|
|
|
{
|
2015-03-17 18:56:15 -05:00
|
|
|
assert(!isChildProcess());
|
|
|
|
|
|
|
|
Process::Args args;
|
|
|
|
args.push_back("--port=" + std::to_string(LOOLWSD::portNumber));
|
|
|
|
args.push_back("--lopath=" + LOOLWSD::loPath);
|
|
|
|
|
|
|
|
Random rng;
|
|
|
|
|
|
|
|
_childId = (((UInt64)rng.next()) << 32) | rng.next() | 1;
|
|
|
|
|
|
|
|
_childToClient[_childId] = this;
|
|
|
|
|
|
|
|
args.push_back("--child=" + std::to_string(_childId));
|
|
|
|
|
|
|
|
Application::instance().logger().information(Util::logPrefix() + "Launching child: " + Poco::cat(std::string(" "), args.begin(), args.end()));
|
|
|
|
|
|
|
|
ProcessHandle child = Process::launch(Application::instance().commandPath(), args);
|
|
|
|
|
|
|
|
std::string loadRequest = "load " + _docURL;
|
|
|
|
forwardRequest(loadRequest.c_str(), loadRequest.size());
|
2015-03-12 18:34:42 -05:00
|
|
|
}
|
|
|
|
|
2015-03-17 18:56:15 -05:00
|
|
|
void LOOLSession::forwardRequest(const char *buffer, int length)
|
|
|
|
{
|
|
|
|
if (_peerWs == nullptr)
|
|
|
|
{
|
|
|
|
_backLog.push_back(new Poco::Buffer<char>(buffer, length));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_peerWs->sendFrame(buffer, length, WebSocket::FRAME_BINARY);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LOOLSession::isChildProcess() const
|
|
|
|
{
|
|
|
|
return _loKit != nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-03-04 17:14:04 -06:00
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|