2013-07-16 19:02:30 +00:00
|
|
|
/*
|
|
|
|
* synergy -- mouse and keyboard sharing utility
|
2014-11-02 12:12:05 +00:00
|
|
|
* Copyright (C) 2013 Synergy Si Ltd.
|
2013-07-16 19:02:30 +00:00
|
|
|
*
|
|
|
|
* This package is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
2015-05-03 02:33:52 +00:00
|
|
|
* found in the file LICENSE that should have accompanied this file.
|
2013-07-16 19:02:30 +00:00
|
|
|
*
|
|
|
|
* This package is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2014-02-28 12:36:45 +00:00
|
|
|
#include "server/ClientProxy1_5.h"
|
|
|
|
|
|
|
|
#include "server/Server.h"
|
2015-05-18 17:17:22 +00:00
|
|
|
#include "synergy/StreamChunker.h"
|
2014-02-28 12:36:45 +00:00
|
|
|
#include "synergy/ProtocolUtil.h"
|
|
|
|
#include "io/IStream.h"
|
|
|
|
#include "base/Log.h"
|
2013-07-16 19:02:30 +00:00
|
|
|
|
2015-05-15 21:26:57 +00:00
|
|
|
#include <sstream>
|
|
|
|
|
2013-07-16 19:02:30 +00:00
|
|
|
//
|
2014-11-11 13:51:47 +00:00
|
|
|
// ClientProxy1_5
|
2013-07-16 19:02:30 +00:00
|
|
|
//
|
|
|
|
|
2014-11-11 13:51:47 +00:00
|
|
|
const UInt16 ClientProxy1_5::m_intervalThreshold = 1;
|
2013-07-26 12:44:14 +00:00
|
|
|
|
2014-11-11 13:51:47 +00:00
|
|
|
ClientProxy1_5::ClientProxy1_5(const String& name, synergy::IStream* stream, Server* server, IEventQueue* events) :
|
|
|
|
ClientProxy1_4(name, stream, server, events),
|
2013-07-26 12:44:14 +00:00
|
|
|
m_events(events),
|
|
|
|
m_stopwatch(true),
|
|
|
|
m_elapsedTime(0),
|
|
|
|
m_receivedDataSize(0)
|
2013-07-16 19:02:30 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-11-11 13:51:47 +00:00
|
|
|
ClientProxy1_5::~ClientProxy1_5()
|
2013-07-16 19:02:30 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2013-08-30 14:38:43 +00:00
|
|
|
void
|
2014-11-11 13:51:47 +00:00
|
|
|
ClientProxy1_5::sendDragInfo(UInt32 fileCount, const char* info, size_t size)
|
2013-08-30 14:38:43 +00:00
|
|
|
{
|
2014-11-11 13:51:47 +00:00
|
|
|
String data(info, size);
|
2013-08-30 14:38:43 +00:00
|
|
|
|
2014-11-11 13:51:47 +00:00
|
|
|
ProtocolUtil::writef(getStream(), kMsgDDragInfo, fileCount, &data);
|
2013-08-30 14:38:43 +00:00
|
|
|
}
|
|
|
|
|
2013-07-16 19:02:30 +00:00
|
|
|
void
|
2014-11-11 13:51:47 +00:00
|
|
|
ClientProxy1_5::fileChunkSending(UInt8 mark, char* data, size_t dataSize)
|
2013-07-16 19:02:30 +00:00
|
|
|
{
|
2014-11-11 13:51:47 +00:00
|
|
|
String chunk(data, dataSize);
|
2013-07-16 19:02:30 +00:00
|
|
|
|
|
|
|
switch (mark) {
|
2015-05-15 21:26:57 +00:00
|
|
|
case kDataStart:
|
2013-07-24 16:41:12 +00:00
|
|
|
LOG((CLOG_DEBUG2 "file sending start: size=%s", data));
|
2013-07-16 19:02:30 +00:00
|
|
|
break;
|
|
|
|
|
2015-05-15 21:26:57 +00:00
|
|
|
case kDataChunk:
|
2013-07-24 16:41:12 +00:00
|
|
|
LOG((CLOG_DEBUG2 "file chunk sending: size=%i", chunk.size()));
|
2013-07-16 19:02:30 +00:00
|
|
|
break;
|
|
|
|
|
2015-05-15 21:26:57 +00:00
|
|
|
case kDataEnd:
|
2013-07-16 19:02:30 +00:00
|
|
|
LOG((CLOG_DEBUG2 "file sending finished"));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-11-11 13:51:47 +00:00
|
|
|
ProtocolUtil::writef(getStream(), kMsgDFileTransfer, mark, &chunk);
|
2013-07-16 19:02:30 +00:00
|
|
|
}
|
2013-07-24 16:41:12 +00:00
|
|
|
|
2015-05-14 20:51:21 +00:00
|
|
|
void
|
|
|
|
ClientProxy1_5::setClipboard(ClipboardID id, const IClipboard* clipboard)
|
|
|
|
{
|
|
|
|
// ignore if this clipboard is already clean
|
|
|
|
if (m_clipboard[id].m_dirty) {
|
|
|
|
// this clipboard is now clean
|
|
|
|
m_clipboard[id].m_dirty = false;
|
|
|
|
Clipboard::copy(&m_clipboard[id].m_clipboard, clipboard);
|
|
|
|
|
2015-05-15 21:26:57 +00:00
|
|
|
String data = m_clipboard[id].m_clipboard.marshall();
|
|
|
|
|
|
|
|
size_t size = data.size();
|
|
|
|
LOG((CLOG_DEBUG "sending clipboard %d to \"%s\" size=%d", id, getName().c_str(), size));
|
|
|
|
|
|
|
|
//TODO: refactor FileChunker and use thread
|
|
|
|
// send first message (file size)
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << size;
|
|
|
|
String dataSize = ss.str();
|
|
|
|
ProtocolUtil::writef(getStream(), kMsgDClipboard, id, 0, kDataStart, &dataSize);
|
|
|
|
|
|
|
|
// send chunk messages with a fixed chunk size
|
|
|
|
size_t sentLength = 0;
|
|
|
|
size_t chunkSize = 2048;
|
|
|
|
Stopwatch stopwatch;
|
|
|
|
stopwatch.start();
|
|
|
|
while (true) {
|
|
|
|
if (stopwatch.getTime() > 0.1f) {
|
|
|
|
// make sure we don't read too much from the mock data.
|
|
|
|
if (sentLength + chunkSize > size) {
|
|
|
|
chunkSize = size - sentLength;
|
|
|
|
}
|
|
|
|
|
|
|
|
String chunk(data.substr(sentLength, chunkSize).c_str(), chunkSize);
|
|
|
|
ProtocolUtil::writef(getStream(), kMsgDClipboard, id, 0, kDataChunk, &chunk);
|
|
|
|
|
|
|
|
sentLength += chunkSize;
|
|
|
|
|
|
|
|
if (sentLength == size) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
stopwatch.reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// send last message
|
|
|
|
ProtocolUtil::writef(getStream(), kMsgDClipboard, id, 0, kDataEnd, "\0");
|
2015-05-14 20:51:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-24 16:41:12 +00:00
|
|
|
bool
|
2014-11-11 13:51:47 +00:00
|
|
|
ClientProxy1_5::parseMessage(const UInt8* code)
|
2013-07-24 16:41:12 +00:00
|
|
|
{
|
|
|
|
if (memcmp(code, kMsgDFileTransfer, 4) == 0) {
|
|
|
|
fileChunkReceived();
|
|
|
|
}
|
2013-08-30 19:49:38 +00:00
|
|
|
else if (memcmp(code, kMsgDDragInfo, 4) == 0) {
|
|
|
|
dragInfoReceived();
|
|
|
|
}
|
2013-07-24 16:41:12 +00:00
|
|
|
else {
|
2014-11-11 13:51:47 +00:00
|
|
|
return ClientProxy1_4::parseMessage(code);
|
2013-07-24 16:41:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2014-11-11 13:51:47 +00:00
|
|
|
ClientProxy1_5::fileChunkReceived()
|
2013-07-24 16:41:12 +00:00
|
|
|
{
|
|
|
|
// parse
|
2013-08-30 14:38:43 +00:00
|
|
|
UInt8 mark = 0;
|
2014-11-11 13:51:47 +00:00
|
|
|
String content;
|
|
|
|
ProtocolUtil::readf(getStream(), kMsgDFileTransfer + 4, &mark, &content);
|
2013-07-24 16:41:12 +00:00
|
|
|
|
2014-11-11 13:51:47 +00:00
|
|
|
Server* server = getServer();
|
2013-07-24 16:41:12 +00:00
|
|
|
switch (mark) {
|
2015-05-15 21:26:57 +00:00
|
|
|
case kDataStart:
|
2013-07-24 16:41:12 +00:00
|
|
|
server->clearReceivedFileData();
|
|
|
|
server->setExpectedFileSize(content);
|
2013-07-26 12:44:14 +00:00
|
|
|
if (CLOG->getFilter() >= kDEBUG2) {
|
|
|
|
LOG((CLOG_DEBUG2 "recv file data from client: file size=%s", content.c_str()));
|
|
|
|
m_stopwatch.start();
|
|
|
|
}
|
2013-07-24 16:41:12 +00:00
|
|
|
break;
|
|
|
|
|
2015-05-15 21:26:57 +00:00
|
|
|
case kDataChunk:
|
2013-07-24 16:41:12 +00:00
|
|
|
server->fileChunkReceived(content);
|
2013-07-26 12:44:14 +00:00
|
|
|
if (CLOG->getFilter() >= kDEBUG2) {
|
|
|
|
LOG((CLOG_DEBUG2 "recv file data from client: chunck size=%i", content.size()));
|
|
|
|
double interval = m_stopwatch.getTime();
|
|
|
|
m_receivedDataSize += content.size();
|
|
|
|
LOG((CLOG_DEBUG2 "recv file data from client: interval=%f s", interval));
|
|
|
|
if (interval >= m_intervalThreshold) {
|
|
|
|
double averageSpeed = m_receivedDataSize / interval / 1000;
|
|
|
|
LOG((CLOG_DEBUG2 "recv file data from client: average speed=%f kb/s", averageSpeed));
|
|
|
|
|
|
|
|
m_receivedDataSize = 0;
|
|
|
|
m_elapsedTime += interval;
|
|
|
|
m_stopwatch.reset();
|
|
|
|
}
|
|
|
|
}
|
2013-07-24 16:41:12 +00:00
|
|
|
break;
|
|
|
|
|
2015-05-15 21:26:57 +00:00
|
|
|
case kDataEnd:
|
2014-11-11 13:51:47 +00:00
|
|
|
m_events->addEvent(Event(m_events->forIScreen().fileRecieveCompleted(), server));
|
2013-07-26 12:44:14 +00:00
|
|
|
if (CLOG->getFilter() >= kDEBUG2) {
|
|
|
|
LOG((CLOG_DEBUG2 "file data transfer finished"));
|
|
|
|
m_elapsedTime += m_stopwatch.getTime();
|
|
|
|
double averageSpeed = getServer()->getExpectedFileSize() / m_elapsedTime / 1000;
|
|
|
|
LOG((CLOG_DEBUG2 "file data transfer finished: total time consumed=%f s", m_elapsedTime));
|
|
|
|
LOG((CLOG_DEBUG2 "file data transfer finished: total data received=%i kb", getServer()->getExpectedFileSize() / 1000));
|
|
|
|
LOG((CLOG_DEBUG2 "file data transfer finished: total average speed=%f kb/s", averageSpeed));
|
|
|
|
}
|
2013-07-24 16:41:12 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2013-08-30 19:49:38 +00:00
|
|
|
|
|
|
|
void
|
2014-11-11 13:51:47 +00:00
|
|
|
ClientProxy1_5::dragInfoReceived()
|
2013-08-30 19:49:38 +00:00
|
|
|
{
|
|
|
|
// parse
|
|
|
|
UInt32 fileNum = 0;
|
2014-11-11 13:51:47 +00:00
|
|
|
String content;
|
|
|
|
ProtocolUtil::readf(getStream(), kMsgDDragInfo + 4, &fileNum, &content);
|
2013-08-30 19:49:38 +00:00
|
|
|
|
|
|
|
m_server->dragInfoReceived(fileNum, content);
|
|
|
|
}
|