2018-12-23 14:09:23 -05:00
|
|
|
/*******************************************************************************
|
|
|
|
* gui/statistics/TurtleRouterStatistics.cpp *
|
|
|
|
* *
|
|
|
|
* Copyright (c) 2011 Retroshare Team <retroshare.project@gmail.com> *
|
|
|
|
* *
|
|
|
|
* This program is free software: you can redistribute it and/or modify *
|
|
|
|
* it under the terms of the GNU Affero General Public License as *
|
|
|
|
* published by the Free Software Foundation, either version 3 of the *
|
|
|
|
* License, or (at your option) any later version. *
|
|
|
|
* *
|
|
|
|
* This program 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 Affero General Public License for more details. *
|
|
|
|
* *
|
|
|
|
* You should have received a copy of the GNU Affero General Public License *
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>. *
|
|
|
|
* *
|
|
|
|
*******************************************************************************/
|
2011-05-31 17:39:44 -04:00
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
#include <QTimer>
|
|
|
|
#include <QObject>
|
|
|
|
|
|
|
|
#include <QPainter>
|
|
|
|
#include <QStylePainter>
|
|
|
|
|
|
|
|
#include <retroshare/rsturtle.h>
|
|
|
|
#include <retroshare/rspeers.h>
|
|
|
|
#include "TurtleRouterStatistics.h"
|
2014-09-10 10:26:37 -04:00
|
|
|
#include "TurtleRouterDialog.h"
|
2011-05-31 17:39:44 -04:00
|
|
|
|
|
|
|
#include "gui/settings/rsharesettings.h"
|
|
|
|
|
2017-07-15 12:40:03 -04:00
|
|
|
//static const int MAX_TUNNEL_REQUESTS_DISPLAY = 10 ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
2018-02-17 17:37:25 -05:00
|
|
|
template<class TURTLE_REQ_DISPLAY_INFO> class TRHistogram
|
2011-05-31 17:39:44 -04:00
|
|
|
{
|
|
|
|
public:
|
2018-02-17 17:37:25 -05:00
|
|
|
TRHistogram(const std::vector<TURTLE_REQ_DISPLAY_INFO>& info) :_infos(info) {}
|
2011-05-31 17:39:44 -04:00
|
|
|
|
|
|
|
QColor colorScale(float f)
|
|
|
|
{
|
|
|
|
if(f == 0)
|
|
|
|
return QColor::fromHsv(0,0,192) ;
|
|
|
|
else
|
|
|
|
return QColor::fromHsv((int)((1.0-f)*280),200,255) ;
|
|
|
|
}
|
|
|
|
|
2015-06-24 16:55:09 -04:00
|
|
|
virtual void draw(QPainter *painter,int& ox,int& oy,const QString& title,float fontHeight)
|
|
|
|
{
|
|
|
|
static const int MaxTime = 61 ;
|
|
|
|
static const int MaxDepth = 8 ;
|
|
|
|
|
|
|
|
float fact = fontHeight/14.0 ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
2015-06-24 16:55:09 -04:00
|
|
|
const int cellx = fact*7 ;
|
|
|
|
const int celly = fact*12 ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
2015-06-24 16:55:09 -04:00
|
|
|
int save_ox = ox ;
|
|
|
|
painter->setPen(QColor::fromRgb(0,0,0)) ;
|
|
|
|
painter->drawText(2*fact+ox,celly+oy,title) ;
|
|
|
|
oy+=2*fact+2*celly ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
2015-06-24 16:55:09 -04:00
|
|
|
if(_infos.empty())
|
|
|
|
return ;
|
|
|
|
|
|
|
|
ox += 10 *fact;
|
2014-03-17 16:56:06 -04:00
|
|
|
std::map<RsPeerId,std::vector<int> > hits ;
|
|
|
|
std::map<RsPeerId,std::vector<int> > depths ;
|
|
|
|
std::map<RsPeerId,std::vector<int> >::iterator it ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
2015-06-24 16:55:09 -04:00
|
|
|
int max_hits = 1;
|
|
|
|
int max_depth = 1;
|
|
|
|
|
|
|
|
for(uint32_t i=0;i<_infos.size();++i)
|
|
|
|
{
|
|
|
|
std::vector<int>& h(hits[_infos[i].source_peer_id]) ;
|
|
|
|
std::vector<int>& g(depths[_infos[i].source_peer_id]) ;
|
|
|
|
|
|
|
|
if(h.size() <= _infos[i].age)
|
|
|
|
h.resize(MaxTime,0) ;
|
|
|
|
|
|
|
|
if(g.empty())
|
|
|
|
g.resize(MaxDepth,0) ;
|
|
|
|
|
|
|
|
if(_infos[i].age < h.size())
|
|
|
|
{
|
|
|
|
++h[_infos[i].age] ;
|
|
|
|
if(h[_infos[i].age] > max_hits)
|
|
|
|
max_hits = h[_infos[i].age] ;
|
|
|
|
}
|
|
|
|
if(_infos[i].depth < g.size())
|
|
|
|
{
|
|
|
|
++g[_infos[i].depth] ;
|
|
|
|
|
|
|
|
if(g[_infos[i].depth] > max_depth)
|
|
|
|
max_depth = g[_infos[i].depth] ;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int max_bi = std::max(max_hits,max_depth) ;
|
|
|
|
int p=0 ;
|
|
|
|
|
|
|
|
for(it=depths.begin();it!=depths.end();++it,++p)
|
|
|
|
for(int i=0;i<MaxDepth;++i)
|
|
|
|
painter->fillRect(ox+MaxTime*cellx+20*fact+i*cellx,oy+p*celly,cellx,celly,colorScale(it->second[i]/(float)max_bi)) ;
|
|
|
|
|
|
|
|
painter->setPen(QColor::fromRgb(0,0,0)) ;
|
|
|
|
painter->drawRect(ox+MaxTime*cellx+20*fact,oy,MaxDepth*cellx,p*celly) ;
|
|
|
|
|
|
|
|
for(int i=0;i<MaxTime;i+=5)
|
|
|
|
painter->drawText(ox+i*cellx,oy+(p+1)*celly+4*fact,QString::number(i)) ;
|
|
|
|
|
|
|
|
p=0 ;
|
|
|
|
int great_total = 0 ;
|
|
|
|
|
|
|
|
for(it=hits.begin();it!=hits.end();++it,++p)
|
|
|
|
{
|
|
|
|
int total = 0 ;
|
|
|
|
|
|
|
|
for(int i=0;i<MaxTime;++i)
|
|
|
|
{
|
|
|
|
painter->fillRect(ox+i*cellx,oy+p*celly,cellx,celly,colorScale(it->second[i]/(float)max_bi)) ;
|
|
|
|
total += it->second[i] ;
|
|
|
|
}
|
|
|
|
|
|
|
|
painter->setPen(QColor::fromRgb(0,0,0)) ;
|
|
|
|
painter->drawText(ox+MaxDepth*cellx+30*fact+(MaxTime+1)*cellx,oy+(p+1)*celly,TurtleRouterStatistics::getPeerName(it->first)) ;
|
|
|
|
painter->drawText(ox+MaxDepth*cellx+30*fact+(MaxTime+1)*cellx+120*fact,oy+(p+1)*celly,"("+QString::number(total)+")") ;
|
|
|
|
great_total += total ;
|
|
|
|
}
|
|
|
|
|
|
|
|
painter->drawRect(ox,oy,MaxTime*cellx,p*celly) ;
|
|
|
|
|
|
|
|
for(int i=0;i<MaxTime;i+=5)
|
|
|
|
painter->drawText(ox+i*cellx,oy+(p+1)*celly+4*fact,QString::number(i)) ;
|
|
|
|
for(int i=0;i<MaxDepth;++i)
|
|
|
|
painter->drawText(ox+MaxTime*cellx+20*fact+i*cellx,oy+(p+1)*celly+4*fact,QString::number(i)) ;
|
|
|
|
painter->setPen(QColor::fromRgb(255,130,80)) ;
|
|
|
|
painter->drawText(ox+MaxDepth*cellx+30*fact+(MaxTime+1)*cellx+120*fact,oy+(p+1)*celly+4*fact,"("+QString::number(great_total)+")");
|
|
|
|
|
|
|
|
oy += (p+1)*celly+6 ;
|
|
|
|
|
|
|
|
painter->setPen(QColor::fromRgb(0,0,0)) ;
|
|
|
|
painter->drawText(ox,oy+celly,"("+QApplication::translate("TurtleRouterStatistics", "Age in seconds")+")");
|
|
|
|
painter->drawText(ox+MaxTime*cellx+20*fact,oy+celly,"("+QApplication::translate("TurtleRouterStatistics", "Depth")+")");
|
|
|
|
|
|
|
|
painter->drawText(ox+MaxDepth*cellx+30*fact+(MaxTime+1)*cellx+120,oy+celly,"("+QApplication::translate("TurtleRouterStatistics", "total")+")");
|
|
|
|
|
|
|
|
oy += 3*celly ;
|
|
|
|
|
|
|
|
// now, draw a scale
|
|
|
|
|
|
|
|
int last_hts = -1 ;
|
|
|
|
int cellid = 0 ;
|
|
|
|
|
|
|
|
for(int i=0;i<=10;++i)
|
|
|
|
{
|
|
|
|
int hts = (int)(max_bi*i/10.0) ;
|
|
|
|
|
|
|
|
if(hts > last_hts)
|
|
|
|
{
|
|
|
|
painter->fillRect(ox+cellid*(cellx+22*fact),oy,cellx,celly,colorScale(i/10.0f)) ;
|
|
|
|
painter->setPen(QColor::fromRgb(0,0,0)) ;
|
|
|
|
painter->drawRect(ox+cellid*(cellx+22*fact),oy,cellx,celly) ;
|
|
|
|
painter->drawText(ox+cellid*(cellx+22*fact)+cellx+4*fact,oy+celly,QString::number(hts)) ;
|
|
|
|
last_hts = hts ;
|
|
|
|
++cellid ;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
oy += celly*2 ;
|
|
|
|
|
|
|
|
ox = save_ox ;
|
2011-05-31 17:39:44 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2018-02-17 17:37:25 -05:00
|
|
|
const std::vector<TURTLE_REQ_DISPLAY_INFO>& _infos ;
|
2011-05-31 17:39:44 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
TurtleRouterStatistics::TurtleRouterStatistics(QWidget *parent)
|
|
|
|
: RsAutoUpdatePage(2000,parent)
|
|
|
|
{
|
|
|
|
setupUi(this) ;
|
|
|
|
|
|
|
|
m_bProcessSettings = false;
|
|
|
|
|
|
|
|
_tunnel_statistics_F->setWidget( _tst_CW = new TurtleRouterStatisticsWidget() ) ;
|
|
|
|
_tunnel_statistics_F->setWidgetResizable(true);
|
|
|
|
_tunnel_statistics_F->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
|
|
_tunnel_statistics_F->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOn);
|
|
|
|
_tunnel_statistics_F->viewport()->setBackgroundRole(QPalette::NoRole);
|
|
|
|
_tunnel_statistics_F->setFrameStyle(QFrame::NoFrame);
|
|
|
|
_tunnel_statistics_F->setFocusPolicy(Qt::NoFocus);
|
|
|
|
|
2018-07-15 08:40:50 -04:00
|
|
|
routertabWidget->addTab(new TurtleRouterDialog(), QString(tr("File transfer tunnels")));
|
|
|
|
routertabWidget->addTab(new GxsAuthenticatedTunnelsDialog(),QString(tr("Authenticated tunnels")));
|
|
|
|
routertabWidget->addTab(new GxsNetTunnelsDialog(), QString(tr("GXS sync tunnels") ));
|
|
|
|
|
|
|
|
float fontHeight = QFontMetricsF(font()).height();
|
|
|
|
float fact = fontHeight/14.0;
|
2015-06-25 16:11:39 -04:00
|
|
|
|
|
|
|
frmGraph->setMinimumHeight(200*fact);
|
2014-09-10 10:26:37 -04:00
|
|
|
|
2011-05-31 17:39:44 -04:00
|
|
|
// load settings
|
|
|
|
processSettings(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
TurtleRouterStatistics::~TurtleRouterStatistics()
|
|
|
|
{
|
|
|
|
|
|
|
|
// save settings
|
|
|
|
processSettings(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TurtleRouterStatistics::processSettings(bool bLoad)
|
|
|
|
{
|
|
|
|
m_bProcessSettings = true;
|
|
|
|
|
|
|
|
Settings->beginGroup(QString("TurtleRouterStatistics"));
|
|
|
|
|
|
|
|
if (bLoad) {
|
|
|
|
// load settings
|
|
|
|
|
|
|
|
// state of splitter
|
|
|
|
//splitter->restoreState(Settings->value("Splitter").toByteArray());
|
|
|
|
} else {
|
|
|
|
// save settings
|
|
|
|
|
|
|
|
// state of splitter
|
|
|
|
//Settings->setValue("Splitter", splitter->saveState());
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Settings->endGroup();
|
|
|
|
|
|
|
|
m_bProcessSettings = false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void TurtleRouterStatistics::updateDisplay()
|
|
|
|
{
|
|
|
|
std::vector<std::vector<std::string> > hashes_info ;
|
|
|
|
std::vector<std::vector<std::string> > tunnels_info ;
|
2018-02-17 17:37:25 -05:00
|
|
|
std::vector<TurtleSearchRequestDisplayInfo > search_reqs_info ;
|
|
|
|
std::vector<TurtleTunnelRequestDisplayInfo > tunnel_reqs_info ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
|
|
|
rsTurtle->getInfo(hashes_info,tunnels_info,search_reqs_info,tunnel_reqs_info) ;
|
|
|
|
|
|
|
|
//updateTunnelRequests(hashes_info,tunnels_info,search_reqs_info,tunnel_reqs_info) ;
|
|
|
|
_tst_CW->updateTunnelStatistics(hashes_info,tunnels_info,search_reqs_info,tunnel_reqs_info) ;
|
|
|
|
_tst_CW->update();
|
|
|
|
}
|
|
|
|
|
2014-03-17 16:56:06 -04:00
|
|
|
QString TurtleRouterStatistics::getPeerName(const RsPeerId &peer_id)
|
2011-05-31 17:39:44 -04:00
|
|
|
{
|
2014-03-17 16:56:06 -04:00
|
|
|
static std::map<RsPeerId, QString> names ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
2014-03-17 16:56:06 -04:00
|
|
|
std::map<RsPeerId,QString>::const_iterator it = names.find(peer_id) ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
|
|
|
if( it != names.end())
|
|
|
|
return it->second ;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
RsPeerDetails detail ;
|
|
|
|
if(!rsPeers->getPeerDetails(peer_id,detail))
|
2012-02-05 16:05:11 -05:00
|
|
|
return tr("Unknown Peer");
|
2011-05-31 17:39:44 -04:00
|
|
|
|
2011-07-17 19:07:29 -04:00
|
|
|
return (names[peer_id] = QString::fromUtf8(detail.name.c_str())) ;
|
2011-05-31 17:39:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TurtleRouterStatisticsWidget::TurtleRouterStatisticsWidget(QWidget *parent)
|
|
|
|
: QWidget(parent)
|
|
|
|
{
|
|
|
|
maxWidth = 200 ;
|
|
|
|
maxHeight = 0 ;
|
|
|
|
}
|
|
|
|
|
2011-08-12 10:06:29 -04:00
|
|
|
void TurtleRouterStatisticsWidget::updateTunnelStatistics(const std::vector<std::vector<std::string> >& /*hashes_info*/,
|
|
|
|
const std::vector<std::vector<std::string> >& /*tunnels_info*/,
|
2018-02-17 17:37:25 -05:00
|
|
|
const std::vector<TurtleSearchRequestDisplayInfo >& search_reqs_info,
|
|
|
|
const std::vector<TurtleTunnelRequestDisplayInfo >& tunnel_reqs_info)
|
2011-05-31 17:39:44 -04:00
|
|
|
|
|
|
|
{
|
|
|
|
QPixmap tmppixmap(maxWidth, maxHeight);
|
2015-05-20 15:50:10 -04:00
|
|
|
tmppixmap.fill(Qt::transparent);
|
2011-05-31 17:39:44 -04:00
|
|
|
setFixedHeight(maxHeight);
|
|
|
|
|
|
|
|
QPainter painter(&tmppixmap);
|
|
|
|
painter.initFrom(this);
|
|
|
|
|
2015-06-24 16:55:09 -04:00
|
|
|
|
|
|
|
// extracts the height of the fonts in pixels. This is used to callibrate the size of the objects to draw.
|
|
|
|
|
|
|
|
float fontHeight = QFontMetricsF(font()).height();
|
|
|
|
float fact = fontHeight/14.0;
|
|
|
|
maxHeight = 500*fact ;
|
|
|
|
|
|
|
|
int cellx = 6*fact ;
|
|
|
|
int celly = (10+4)*fact ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
|
|
|
// std::cerr << "Drawing into pixmap of size " << maxWidth << "x" << maxHeight << std::endl;
|
|
|
|
// draw...
|
2015-06-24 16:55:09 -04:00
|
|
|
int ox=5*fact,oy=5*fact ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
2018-02-17 17:37:25 -05:00
|
|
|
TRHistogram<TurtleSearchRequestDisplayInfo>(search_reqs_info).draw(&painter,ox,oy,tr("Search requests repartition") + ":",fontHeight) ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
|
|
|
painter.setPen(QColor::fromRgb(70,70,70)) ;
|
|
|
|
painter.drawLine(0,oy,maxWidth,oy) ;
|
|
|
|
oy += celly ;
|
|
|
|
|
2018-02-17 17:37:25 -05:00
|
|
|
TRHistogram<TurtleTunnelRequestDisplayInfo>(tunnel_reqs_info).draw(&painter,ox,oy,tr("Tunnel requests repartition") + ":",fontHeight) ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
|
|
|
// now give information about turtle traffic.
|
|
|
|
//
|
|
|
|
TurtleTrafficStatisticsInfo info ;
|
|
|
|
rsTurtle->getTrafficStatistics(info) ;
|
|
|
|
|
|
|
|
painter.setPen(QColor::fromRgb(70,70,70)) ;
|
|
|
|
painter.drawLine(0,oy,maxWidth,oy) ;
|
|
|
|
oy += celly ;
|
|
|
|
|
2012-02-05 16:05:11 -05:00
|
|
|
painter.drawText(ox,oy+celly,tr("Turtle router traffic")+":") ; oy += celly*2 ;
|
2011-05-31 17:39:44 -04:00
|
|
|
painter.drawText(ox+2*cellx,oy+celly,tr("Tunnel requests Dn")+"\t: " + speedString(info.tr_dn_Bps) ) ; oy += celly ;
|
2014-08-29 10:18:50 -04:00
|
|
|
painter.drawText(ox+2*cellx,oy+celly,tr("Tunnel requests Up")+"\t: " + speedString(info.tr_up_Bps) ) ; oy += celly ;
|
2011-05-31 17:39:44 -04:00
|
|
|
painter.drawText(ox+2*cellx,oy+celly,tr("Incoming file data")+"\t: " + speedString(info.data_dn_Bps) ) ; oy += celly ;
|
|
|
|
painter.drawText(ox+2*cellx,oy+celly,tr("Outgoing file data")+"\t: " + speedString(info.data_up_Bps) ) ; oy += celly ;
|
2015-06-12 04:27:26 -04:00
|
|
|
painter.drawText(ox+2*cellx,oy+celly,tr("Forwarded data")+" \t: " + speedString(info.unknown_updn_Bps) ) ; oy += celly ;
|
2011-05-31 17:39:44 -04:00
|
|
|
|
2011-10-01 07:46:00 -04:00
|
|
|
QString prob_string ;
|
|
|
|
|
|
|
|
for(uint i=0;i<info.forward_probabilities.size();++i)
|
|
|
|
prob_string += QString::number(info.forward_probabilities[i],'g',2) + " (" + QString::number(i) + ") " ;
|
|
|
|
|
|
|
|
painter.drawText(ox+2*cellx,oy+celly,tr("TR Forward probabilities")+"\t: " + prob_string ) ;
|
|
|
|
oy += celly ;
|
|
|
|
oy += celly ;
|
|
|
|
|
2011-05-31 17:39:44 -04:00
|
|
|
// update the pixmap
|
|
|
|
//
|
|
|
|
pixmap = tmppixmap;
|
|
|
|
maxHeight = oy ;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString TurtleRouterStatisticsWidget::speedString(float f)
|
|
|
|
{
|
|
|
|
if(f < 1.0f)
|
|
|
|
return QString("0 B/s") ;
|
|
|
|
if(f < 1024.0f)
|
|
|
|
return QString::number((int)f)+" B/s" ;
|
|
|
|
|
|
|
|
return QString::number(f/1024.0,'f',2) + " KB/s";
|
|
|
|
}
|
|
|
|
|
2011-08-12 10:06:29 -04:00
|
|
|
void TurtleRouterStatisticsWidget::paintEvent(QPaintEvent */*event*/)
|
2011-05-31 17:39:44 -04:00
|
|
|
{
|
|
|
|
QStylePainter(this).drawPixmap(0, 0, pixmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TurtleRouterStatisticsWidget::resizeEvent(QResizeEvent *event)
|
|
|
|
{
|
|
|
|
QRect TaskGraphRect = geometry();
|
|
|
|
maxWidth = TaskGraphRect.width();
|
|
|
|
maxHeight = TaskGraphRect.height() ;
|
|
|
|
|
|
|
|
QWidget::resizeEvent(event);
|
|
|
|
update();
|
|
|
|
}
|