|
Packit |
8f70b4 |
/*
|
|
Packit |
8f70b4 |
* lftp - file transfer program
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* Copyright (c) 1996-2012 by Alexander V. Lukyanov (lav@yars.free.net)
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* This program is free software; you can redistribute it and/or modify
|
|
Packit |
8f70b4 |
* it under the terms of the GNU General Public License as published by
|
|
Packit |
8f70b4 |
* the Free Software Foundation; either version 3 of the License, or
|
|
Packit |
8f70b4 |
* (at your option) any later version.
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* This program is distributed in the hope that it will be useful,
|
|
Packit |
8f70b4 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
8f70b4 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
Packit |
8f70b4 |
* GNU General Public License for more details.
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* You should have received a copy of the GNU General Public License
|
|
Packit |
8f70b4 |
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
Packit |
8f70b4 |
*/
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
#include <config.h>
|
|
Packit |
8f70b4 |
#include <math.h>
|
|
Packit |
8f70b4 |
#include <stdlib.h>
|
|
Packit |
8f70b4 |
#include "Speedometer.h"
|
|
Packit |
8f70b4 |
#include "misc.h"
|
|
Packit |
8f70b4 |
#include "xstring.h"
|
|
Packit |
8f70b4 |
#include "ResMgr.h"
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
#define now SMTask::now
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
Speedometer::Speedometer(const char *p)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
period=15;
|
|
Packit |
8f70b4 |
rate=0;
|
|
Packit |
8f70b4 |
last_second=now;
|
|
Packit |
8f70b4 |
start=now;
|
|
Packit |
8f70b4 |
last_bytes=0;
|
|
Packit |
8f70b4 |
terse=true;
|
|
Packit |
8f70b4 |
period_resource=p;
|
|
Packit |
8f70b4 |
Reconfig(0);
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
bool Speedometer::Valid()
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
return now>=start+TimeDiff(1,0) && now
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
float Speedometer::Get()
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
Add(0);
|
|
Packit |
8f70b4 |
SMTask::current->Timeout(500);
|
|
Packit |
8f70b4 |
return rate;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
void Speedometer::Add(int b)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
if(b==0 && (now==last_second || TimeDiff(now,last_second).MilliSeconds()<100))
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
// This makes Speedometer start only when first data come.
|
|
Packit |
8f70b4 |
if(rate==0)
|
|
Packit |
8f70b4 |
Reset();
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
double div=period;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
if(start>now)
|
|
Packit |
8f70b4 |
start=now; // time was adjusted?
|
|
Packit |
8f70b4 |
if(now
|
|
Packit |
8f70b4 |
last_second=now;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
double time_passed_since_start=TimeDiff(now,start);
|
|
Packit |
8f70b4 |
double time_passed=TimeDiff(now,last_second);
|
|
Packit |
8f70b4 |
if(time_passed_since_start
|
|
Packit |
8f70b4 |
div=time_passed_since_start;
|
|
Packit |
8f70b4 |
if(div<1)
|
|
Packit |
8f70b4 |
div=1;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
rate*=1-time_passed/div;
|
|
Packit |
8f70b4 |
rate+=b/div;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
last_second=now;
|
|
Packit |
8f70b4 |
if(b>0)
|
|
Packit |
8f70b4 |
last_bytes=now;
|
|
Packit |
8f70b4 |
if(rate<0)
|
|
Packit |
8f70b4 |
rate=0;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
// These `incorrect' units are used to fit on status line
|
|
Packit |
8f70b4 |
xstring& Speedometer::GetStr(float r)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
if(r<1)
|
|
Packit |
8f70b4 |
return xstring::get_tmp("");
|
|
Packit |
8f70b4 |
if(r<1024)
|
|
Packit |
8f70b4 |
// for translator: those are the units. This is 'byte per second'
|
|
Packit |
8f70b4 |
return xstring::format(_("%.0fb/s"),r);
|
|
Packit |
8f70b4 |
else if(r<1024*1024)
|
|
Packit |
8f70b4 |
// for translator: This is 'Kibibyte per second'
|
|
Packit |
8f70b4 |
return xstring::format(_("%.1fK/s"),r/1024.);
|
|
Packit |
8f70b4 |
else
|
|
Packit |
8f70b4 |
// for translator: This is 'Mebibyte per second'
|
|
Packit |
8f70b4 |
return xstring::format(_("%.2fM/s"),r/(1024.*1024));
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
xstring& Speedometer::GetStrProper(float r)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
if(r<1)
|
|
Packit |
8f70b4 |
return xstring::get_tmp("");
|
|
Packit |
8f70b4 |
if(r<1024)
|
|
Packit |
8f70b4 |
return xstring::format(_("%.0f B/s"),r);
|
|
Packit |
8f70b4 |
else if(r<1024*1024)
|
|
Packit |
8f70b4 |
return xstring::format(_("%.1f KiB/s"),r/1024.);
|
|
Packit |
8f70b4 |
else
|
|
Packit |
8f70b4 |
return xstring::format(_("%.2f MiB/s"),r/(1024.*1024));
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
xstring& Speedometer::GetStr()
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
return Valid() ? GetStr(Get()) : xstring::get_tmp("");
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
xstring& Speedometer::GetETAStrFromSize(off_t size)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
if(!Valid() || Get()<1)
|
|
Packit |
8f70b4 |
return xstring::get_tmp("");
|
|
Packit |
8f70b4 |
return GetETAStrFromTime(long(size/rate+.5));
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
xstring& Speedometer::GetETAStrFromTime(long eta)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
if(eta<0)
|
|
Packit |
8f70b4 |
return xstring::get_tmp("");
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
unsigned flags=TimeInterval::TO_STR_TRANSLATE;
|
|
Packit |
8f70b4 |
if(terse)
|
|
Packit |
8f70b4 |
flags+=TimeInterval::TO_STR_TERSE;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
// for translator: Estimated Time of Arrival.
|
|
Packit |
8f70b4 |
return xstring::cat(_("eta:"),TimeInterval(eta,0).toString(flags),NULL);
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
const char *Speedometer::GetStrS(float r)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
xstring &rate=GetStr(r);
|
|
Packit |
8f70b4 |
if(rate.length())
|
|
Packit |
8f70b4 |
rate.append(' ');
|
|
Packit |
8f70b4 |
return rate;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
const char *Speedometer::GetStrS()
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
return Valid() ? GetStrS(Get()) : "";
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
const char *Speedometer::GetETAStrSFromSize(off_t s)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
xstring &eta=GetETAStrFromSize(s);
|
|
Packit |
8f70b4 |
if(eta.length())
|
|
Packit |
8f70b4 |
eta.append(' ');
|
|
Packit |
8f70b4 |
return eta;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
const char *Speedometer::GetETAStrSFromTime(long s)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
xstring &eta=GetETAStrFromTime(s);
|
|
Packit |
8f70b4 |
if(eta.length())
|
|
Packit |
8f70b4 |
eta.append(' ');
|
|
Packit |
8f70b4 |
return eta;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
void Speedometer::Reset()
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
start=now;
|
|
Packit |
8f70b4 |
last_second=now;
|
|
Packit |
8f70b4 |
rate=0;
|
|
Packit |
8f70b4 |
last_bytes=0;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
ResDecl res_eta_terse("xfer:eta-terse", "yes",ResMgr::BoolValidate,ResMgr::NoClosure);
|
|
Packit |
8f70b4 |
void Speedometer::Reconfig(const char *n)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
terse=res_eta_terse.QueryBool(0);
|
|
Packit |
8f70b4 |
SetPeriod(ResMgr::Query(period_resource,0));
|
|
Packit |
8f70b4 |
}
|