2015-12-12 13:23:44 -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/.
|
|
|
|
*/
|
|
|
|
|
2016-05-17 06:32:33 -05:00
|
|
|
#ifndef INCLUDED_CHILDSESSION_HPP
|
|
|
|
#define INCLUDED_CHILDSESSION_HPP
|
2015-12-13 11:47:14 -06:00
|
|
|
|
2016-01-10 21:52:00 -06:00
|
|
|
#include <mutex>
|
2017-01-02 07:21:49 -06:00
|
|
|
#include <unordered_map>
|
2017-06-27 06:44:50 -05:00
|
|
|
#include <queue>
|
2016-01-10 21:52:00 -06:00
|
|
|
|
2016-11-15 04:37:47 -06:00
|
|
|
#define LOK_USE_UNSTABLE_API
|
|
|
|
#include <LibreOfficeKit/LibreOfficeKit.hxx>
|
|
|
|
|
2017-11-24 14:29:48 -06:00
|
|
|
#include <Poco/Net/WebSocket.h>
|
2016-10-29 20:15:00 -05:00
|
|
|
#include <Poco/Thread.h>
|
2016-02-26 00:25:13 -06:00
|
|
|
|
|
|
|
#include "Common.hpp"
|
2016-11-24 08:56:06 -06:00
|
|
|
#include "Kit.hpp"
|
|
|
|
#include "Session.hpp"
|
2015-12-12 13:23:44 -06:00
|
|
|
|
2016-08-21 10:12:12 -05:00
|
|
|
class ChildSession;
|
|
|
|
|
2017-08-24 06:47:40 -05:00
|
|
|
enum class LokEventTargetEnum
|
|
|
|
{
|
|
|
|
Document,
|
2017-12-04 10:43:50 -06:00
|
|
|
Window
|
2017-08-24 06:47:40 -05:00
|
|
|
};
|
|
|
|
|
2018-09-12 11:05:01 -05:00
|
|
|
// An abstract interface.
|
|
|
|
class DocumentManagerInterface
|
2016-08-14 17:02:51 -05:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
/// Reqest loading a document, or a new view, if one exists.
|
2016-11-15 04:37:47 -06:00
|
|
|
virtual bool onLoad(const std::string& sessionId,
|
|
|
|
const std::string& jailedFilePath,
|
2018-06-10 13:02:02 -05:00
|
|
|
const std::string& jailedFilePathAnonym,
|
2016-11-15 04:37:47 -06:00
|
|
|
const std::string& userName,
|
2018-06-10 13:02:02 -05:00
|
|
|
const std::string& userNameAnonym,
|
2016-11-15 04:37:47 -06:00
|
|
|
const std::string& docPassword,
|
|
|
|
const std::string& renderOpts,
|
2017-03-24 06:34:32 -05:00
|
|
|
const bool haveDocPassword,
|
2017-09-04 08:40:04 -05:00
|
|
|
const std::string& lang,
|
|
|
|
const std::string& watermarkText) = 0;
|
2016-08-14 17:02:51 -05:00
|
|
|
|
|
|
|
/// Unload a client session, which unloads the document
|
|
|
|
/// if it is the last and only.
|
2016-10-29 20:15:00 -05:00
|
|
|
virtual void onUnload(const ChildSession& session) = 0;
|
2016-08-14 21:20:35 -05:00
|
|
|
|
2016-11-15 04:37:47 -06:00
|
|
|
/// Access to the document instance.
|
|
|
|
virtual std::shared_ptr<lok::Document> getLOKitDocument() = 0;
|
|
|
|
|
2017-05-28 22:59:12 -05:00
|
|
|
/// Send updated view info to all active sessions.
|
|
|
|
virtual void notifyViewInfo() = 0;
|
2017-06-27 06:44:50 -05:00
|
|
|
virtual void updateEditorSpeeds(int id, int speed) = 0;
|
|
|
|
|
|
|
|
virtual int getEditorId() = 0;
|
2017-05-28 22:59:12 -05:00
|
|
|
|
2016-10-26 10:08:25 -05:00
|
|
|
/// Get a view ID <-> UserInfo map.
|
2016-10-29 20:15:00 -05:00
|
|
|
virtual std::map<int, UserInfo> getViewInfo() = 0;
|
|
|
|
virtual std::mutex& getMutex() = 0;
|
2016-11-15 04:37:47 -06:00
|
|
|
|
|
|
|
/// Mutex guarding the document - so that we can lock operations like
|
|
|
|
/// setting a view followed by a tile render, etc.
|
|
|
|
virtual std::mutex& getDocumentMutex() = 0;
|
|
|
|
|
2018-07-17 01:01:05 -05:00
|
|
|
virtual std::string getObfuscatedFileId() = 0;
|
|
|
|
|
2016-10-29 20:15:00 -05:00
|
|
|
virtual std::shared_ptr<TileQueue>& getTileQueue() = 0;
|
|
|
|
|
2018-05-04 11:47:33 -05:00
|
|
|
virtual bool sendFrame(const char* buffer, int length, WSOpCode opCode = WSOpCode::Text) = 0;
|
2016-08-14 17:02:51 -05:00
|
|
|
};
|
2016-01-24 17:25:02 -06:00
|
|
|
|
2017-01-02 07:21:49 -06:00
|
|
|
struct RecordedEvent
|
|
|
|
{
|
|
|
|
int _type;
|
|
|
|
std::string _payload;
|
|
|
|
};
|
|
|
|
|
2017-04-24 12:19:18 -05:00
|
|
|
/// When the session is inactive, we need to record its state for a replay.
|
2017-01-02 07:21:49 -06:00
|
|
|
class StateRecorder
|
|
|
|
{
|
|
|
|
public:
|
2017-04-24 12:19:18 -05:00
|
|
|
bool _invalidate;
|
2017-01-02 07:21:49 -06:00
|
|
|
std::unordered_map<std::string, std::string> _recordedStates;
|
|
|
|
std::unordered_map<int, std::unordered_map<int, RecordedEvent>> _recordedViewEvents;
|
|
|
|
std::unordered_map<int, RecordedEvent> _recordedEvents;
|
2017-04-25 11:17:50 -05:00
|
|
|
std::vector<RecordedEvent> _recordedEventsVector;
|
2017-01-02 07:21:49 -06:00
|
|
|
|
2017-04-24 12:19:18 -05:00
|
|
|
StateRecorder() : _invalidate(false) {}
|
|
|
|
|
|
|
|
// TODO Remember the maximal area we need to invalidate - grow it step by step.
|
|
|
|
void recordInvalidate()
|
|
|
|
{
|
|
|
|
_invalidate = true;
|
|
|
|
}
|
|
|
|
|
2017-01-02 07:21:49 -06:00
|
|
|
void recordEvent(const int type, const std::string& payload)
|
|
|
|
{
|
|
|
|
_recordedEvents[type] = {type, payload};
|
|
|
|
}
|
|
|
|
|
|
|
|
void recordViewEvent(const int viewId, const int type, const std::string& payload)
|
|
|
|
{
|
|
|
|
_recordedViewEvents[viewId][type] = {type, payload};
|
|
|
|
}
|
|
|
|
|
|
|
|
void recordState(const std::string& name, const std::string& value)
|
|
|
|
{
|
|
|
|
_recordedStates[name] = value;
|
|
|
|
}
|
|
|
|
|
2017-04-25 11:17:50 -05:00
|
|
|
/// In the case we need to rememeber all the events that come, not just
|
|
|
|
/// the final state.
|
|
|
|
void recordEventSequence(const int type, const std::string& payload)
|
|
|
|
{
|
|
|
|
_recordedEventsVector.push_back({type, payload});
|
|
|
|
}
|
|
|
|
|
2017-01-02 07:21:49 -06:00
|
|
|
void clear()
|
|
|
|
{
|
2017-04-24 12:19:18 -05:00
|
|
|
_invalidate = false;
|
2017-01-02 07:21:49 -06:00
|
|
|
_recordedEvents.clear();
|
|
|
|
_recordedViewEvents.clear();
|
2017-04-24 12:19:18 -05:00
|
|
|
_recordedStates.clear();
|
2017-04-25 11:17:50 -05:00
|
|
|
_recordedEventsVector.clear();
|
2017-01-02 07:21:49 -06:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-09-28 12:15:51 -05:00
|
|
|
/// Represents a session to the WSD process, in a Kit process. Note that this is not a singleton.
|
2016-12-12 18:53:58 -06:00
|
|
|
class ChildSession final : public Session
|
2015-12-12 13:23:44 -06:00
|
|
|
{
|
|
|
|
public:
|
2016-05-17 06:35:07 -05:00
|
|
|
/// Create a new ChildSession
|
2015-12-23 20:20:49 -06:00
|
|
|
/// ws The socket between master and kit (jailed).
|
2015-12-13 13:25:57 -06:00
|
|
|
/// loKit The LOKit instance.
|
|
|
|
/// loKitDocument The instance to an existing document (when opening
|
|
|
|
/// a new view) or nullptr (when first view).
|
2016-01-06 21:47:01 -06:00
|
|
|
/// jailId The JailID of the jail root directory,
|
|
|
|
// used by downloadas to construct jailed path.
|
2016-05-17 06:35:07 -05:00
|
|
|
ChildSession(const std::string& id,
|
2016-05-17 17:28:59 -05:00
|
|
|
const std::string& jailId,
|
2018-09-12 11:05:01 -05:00
|
|
|
DocumentManagerInterface& docManager);
|
2016-05-17 06:35:07 -05:00
|
|
|
virtual ~ChildSession();
|
2015-12-12 13:23:44 -06:00
|
|
|
|
2016-10-29 20:15:00 -05:00
|
|
|
bool getStatus(const char* buffer, int length);
|
2016-08-29 06:52:06 -05:00
|
|
|
int getViewId() const { return _viewId; }
|
2016-09-17 16:52:54 -05:00
|
|
|
void setViewId(const int viewId) { _viewId = viewId; }
|
2016-10-26 09:35:40 -05:00
|
|
|
const std::string& getViewUserId() const { return _userId; }
|
2016-10-23 15:46:05 -05:00
|
|
|
const std::string& getViewUserName() const { return _userName; }
|
2017-05-28 11:20:49 -05:00
|
|
|
const std::string& getViewUserExtraInfo() const { return _userExtraInfo; }
|
2017-09-04 08:40:04 -05:00
|
|
|
const std::string& getWatermarkText() const { return _watermarkText; }
|
2017-06-27 06:44:50 -05:00
|
|
|
void updateSpeed();
|
|
|
|
int getSpeed();
|
2016-01-03 09:56:41 -06:00
|
|
|
|
2017-01-13 06:52:08 -06:00
|
|
|
void loKitCallback(const int type, const std::string& payload);
|
2016-01-24 17:25:02 -06:00
|
|
|
|
2017-11-24 14:29:48 -06:00
|
|
|
bool sendTextFrame(const char* buffer, int length) override
|
2016-10-09 10:28:22 -05:00
|
|
|
{
|
|
|
|
const auto msg = "client-" + getId() + ' ' + std::string(buffer, length);
|
2018-02-07 03:17:28 -06:00
|
|
|
const std::unique_lock<std::mutex> lock = getLock();
|
2018-05-04 11:47:33 -05:00
|
|
|
return _docManager.sendFrame(msg.data(), msg.size(), WSOpCode::Text);
|
2017-11-24 14:29:48 -06:00
|
|
|
}
|
2016-10-09 10:28:22 -05:00
|
|
|
|
2017-11-24 14:29:48 -06:00
|
|
|
bool sendBinaryFrame(const char* buffer, int length) override
|
|
|
|
{
|
|
|
|
const auto msg = "client-" + getId() + ' ' + std::string(buffer, length);
|
2018-02-07 03:17:28 -06:00
|
|
|
const std::unique_lock<std::mutex> lock = getLock();
|
2018-05-04 11:47:33 -05:00
|
|
|
return _docManager.sendFrame(msg.data(), msg.size(), WSOpCode::Binary);
|
2016-10-09 10:28:22 -05:00
|
|
|
}
|
|
|
|
|
2016-12-12 19:18:25 -06:00
|
|
|
using Session::sendTextFrame;
|
2016-10-09 10:28:22 -05:00
|
|
|
|
2016-08-13 11:14:44 -05:00
|
|
|
private:
|
2017-01-19 19:44:39 -06:00
|
|
|
bool loadDocument(const char* buffer, int length, const std::vector<std::string>& tokens);
|
2015-12-12 13:23:44 -06:00
|
|
|
|
2017-01-19 19:44:39 -06:00
|
|
|
bool sendFontRendering(const char* buffer, int length, const std::vector<std::string>& tokens);
|
|
|
|
bool getCommandValues(const char* buffer, int length, const std::vector<std::string>& tokens);
|
2015-12-12 13:23:44 -06:00
|
|
|
|
2017-01-19 19:44:39 -06:00
|
|
|
bool clientZoom(const char* buffer, int length, const std::vector<std::string>& tokens);
|
|
|
|
bool clientVisibleArea(const char* buffer, int length, const std::vector<std::string>& tokens);
|
2017-11-28 16:38:35 -06:00
|
|
|
bool outlineState(const char* buffer, int length, const std::vector<std::string>& tokens);
|
2017-01-19 19:44:39 -06:00
|
|
|
bool downloadAs(const char* buffer, int length, const std::vector<std::string>& tokens);
|
2015-12-12 13:23:44 -06:00
|
|
|
bool getChildId();
|
2017-01-19 19:44:39 -06:00
|
|
|
bool getTextSelection(const char* buffer, int length, const std::vector<std::string>& tokens);
|
2018-02-25 13:18:04 -06:00
|
|
|
std::string getTextSelectionInternal(const std::string& mimeType);
|
2017-01-19 19:44:39 -06:00
|
|
|
bool paste(const char* buffer, int length, const std::vector<std::string>& tokens);
|
|
|
|
bool insertFile(const char* buffer, int length, const std::vector<std::string>& tokens);
|
2017-08-24 06:47:40 -05:00
|
|
|
bool keyEvent(const char* buffer, int length, const std::vector<std::string>& tokens, const LokEventTargetEnum target);
|
2018-02-07 12:30:45 -06:00
|
|
|
bool extTextInputEvent(const char* /*buffer*/, int /*length*/, const std::vector<std::string>& tokens);
|
2017-08-24 06:47:40 -05:00
|
|
|
bool dialogKeyEvent(const char* buffer, int length, const std::vector<std::string>& tokens);
|
|
|
|
bool mouseEvent(const char* buffer, int length, const std::vector<std::string>& tokens, const LokEventTargetEnum target);
|
2017-01-19 19:44:39 -06:00
|
|
|
bool unoCommand(const char* buffer, int length, const std::vector<std::string>& tokens);
|
|
|
|
bool selectText(const char* buffer, int length, const std::vector<std::string>& tokens);
|
|
|
|
bool selectGraphic(const char* buffer, int length, const std::vector<std::string>& tokens);
|
2017-12-04 10:43:50 -06:00
|
|
|
bool renderWindow(const char* buffer, int length, const std::vector<std::string>& tokens);
|
2017-01-19 19:44:39 -06:00
|
|
|
bool resetSelection(const char* buffer, int length, const std::vector<std::string>& tokens);
|
|
|
|
bool saveAs(const char* buffer, int length, const std::vector<std::string>& tokens);
|
|
|
|
bool setClientPart(const char* buffer, int length, const std::vector<std::string>& tokens);
|
|
|
|
bool setPage(const char* buffer, int length, const std::vector<std::string>& tokens);
|
2017-12-04 13:11:02 -06:00
|
|
|
bool sendWindowCommand(const char* buffer, int length, const std::vector<std::string>& tokens);
|
2015-12-12 13:23:44 -06:00
|
|
|
|
2017-01-13 06:52:08 -06:00
|
|
|
void rememberEventsForInactiveUser(const int type, const std::string& payload);
|
2017-01-02 07:21:49 -06:00
|
|
|
|
2016-08-13 11:14:44 -05:00
|
|
|
virtual void disconnect() override;
|
2016-10-29 20:15:00 -05:00
|
|
|
virtual bool _handleInput(const char* buffer, int length) override;
|
2015-12-25 12:51:32 -06:00
|
|
|
|
2016-11-15 04:37:47 -06:00
|
|
|
std::shared_ptr<lok::Document> getLOKitDocument()
|
|
|
|
{
|
|
|
|
return _docManager.getLOKitDocument();
|
|
|
|
}
|
|
|
|
|
2015-12-25 12:51:32 -06:00
|
|
|
private:
|
2016-01-08 18:58:50 -06:00
|
|
|
const std::string _jailId;
|
2018-09-12 11:05:01 -05:00
|
|
|
DocumentManagerInterface& _docManager;
|
2016-08-21 11:56:58 -05:00
|
|
|
|
2017-06-27 06:44:50 -05:00
|
|
|
std::queue<std::chrono::steady_clock::time_point> _cursorInvalidatedEvent;
|
|
|
|
const unsigned _eventStorageIntervalMs = 15*1000;
|
|
|
|
|
2016-01-03 09:56:41 -06:00
|
|
|
/// View ID, returned by createView() or 0 by default.
|
|
|
|
int _viewId;
|
2016-08-21 11:56:58 -05:00
|
|
|
|
|
|
|
/// Whether document has been opened succesfuly
|
|
|
|
bool _isDocLoaded;
|
|
|
|
|
|
|
|
std::string _docType;
|
2017-01-02 07:21:49 -06:00
|
|
|
|
|
|
|
StateRecorder _stateRecorder;
|
2016-01-10 21:52:00 -06:00
|
|
|
|
2018-02-25 13:18:04 -06:00
|
|
|
/// If we are copying to clipboard.
|
|
|
|
bool _copyToClipboard;
|
|
|
|
|
2016-08-21 11:56:58 -05:00
|
|
|
/// Synchronize _loKitDocument access.
|
2016-01-24 17:25:02 -06:00
|
|
|
/// This should be owned by Document.
|
2016-01-18 02:41:04 -06:00
|
|
|
static std::recursive_mutex Mutex;
|
2015-12-12 13:23:44 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2015-12-13 11:47:14 -06:00
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|