2022-01-20 18:27:56 +00:00
|
|
|
// This file Copyright © 2007-2022 Mnemosyne LLC.
|
2022-02-07 16:25:02 +00:00
|
|
|
// It may be used under GPLv2 (SPDX: GPL-2.0-only), GPLv3 (SPDX: GPL-3.0-only),
|
2022-01-20 18:27:56 +00:00
|
|
|
// or any future license endorsed by Mnemosyne LLC.
|
|
|
|
// License text can be found in the licenses/ folder.
|
2007-11-21 20:03:53 +00:00
|
|
|
|
2021-12-15 21:25:42 +00:00
|
|
|
#include <ctime>
|
2022-01-13 02:13:58 +00:00
|
|
|
#include <string>
|
2021-12-15 21:25:42 +00:00
|
|
|
|
2007-11-21 20:03:53 +00:00
|
|
|
#include "transmission.h"
|
2021-12-15 21:25:42 +00:00
|
|
|
|
2013-01-25 23:34:20 +00:00
|
|
|
#include "log.h"
|
2021-12-15 21:25:42 +00:00
|
|
|
#include "session.h"
|
2008-08-11 19:05:02 +00:00
|
|
|
#include "stats.h"
|
2021-11-13 00:10:04 +00:00
|
|
|
#include "utils.h"
|
2012-12-14 04:34:42 +00:00
|
|
|
#include "variant.h"
|
2007-11-21 20:03:53 +00:00
|
|
|
|
|
|
|
/***
|
|
|
|
****
|
|
|
|
***/
|
|
|
|
|
2010-01-19 19:37:00 +00:00
|
|
|
/** @brief Opaque, per-session data structure for bandwidth use statistics */
|
2007-11-26 18:55:36 +00:00
|
|
|
struct tr_stats_handle
|
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_session_stats single;
|
|
|
|
tr_session_stats old;
|
|
|
|
time_t startTime;
|
|
|
|
bool isDirty;
|
2007-11-26 18:55:36 +00:00
|
|
|
};
|
|
|
|
|
2021-11-13 00:10:04 +00:00
|
|
|
static std::string getOldFilename(tr_session const* session)
|
2008-04-18 16:23:59 +00:00
|
|
|
{
|
2021-11-13 00:10:04 +00:00
|
|
|
return tr_strvPath(tr_sessionGetConfigDir(session), "stats.benc");
|
2008-04-18 16:23:59 +00:00
|
|
|
}
|
|
|
|
|
2021-11-13 00:10:04 +00:00
|
|
|
static std::string getFilename(tr_session const* session)
|
2008-06-12 16:25:36 +00:00
|
|
|
{
|
2021-11-13 00:10:04 +00:00
|
|
|
return tr_strvPath(tr_sessionGetConfigDir(session), "stats.json");
|
2008-06-12 16:25:36 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static void loadCumulativeStats(tr_session const* session, tr_session_stats* setme)
|
2007-11-21 20:03:53 +00:00
|
|
|
{
|
2021-10-23 15:43:15 +00:00
|
|
|
auto top = tr_variant{};
|
2007-11-21 20:03:53 +00:00
|
|
|
|
2021-11-13 00:10:04 +00:00
|
|
|
auto filename = getFilename(session);
|
2022-01-15 19:33:57 +00:00
|
|
|
bool loaded = tr_variantFromFile(&top, TR_VARIANT_PARSE_JSON, filename, nullptr);
|
2008-10-14 03:03:29 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
if (!loaded)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
filename = getOldFilename(session);
|
2022-01-15 19:33:57 +00:00
|
|
|
loaded = tr_variantFromFile(&top, TR_VARIANT_PARSE_BENC, filename, nullptr);
|
2008-06-12 16:25:36 +00:00
|
|
|
}
|
2008-10-14 03:03:29 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
if (loaded)
|
2007-11-21 20:03:53 +00:00
|
|
|
{
|
2021-10-23 15:43:15 +00:00
|
|
|
auto i = int64_t{};
|
2017-04-19 12:04:45 +00:00
|
|
|
|
|
|
|
if (tr_variantDictFindInt(&top, TR_KEY_downloaded_bytes, &i))
|
|
|
|
{
|
|
|
|
setme->downloadedBytes = (uint64_t)i;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tr_variantDictFindInt(&top, TR_KEY_files_added, &i))
|
|
|
|
{
|
|
|
|
setme->filesAdded = (uint64_t)i;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tr_variantDictFindInt(&top, TR_KEY_seconds_active, &i))
|
|
|
|
{
|
|
|
|
setme->secondsActive = (uint64_t)i;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tr_variantDictFindInt(&top, TR_KEY_session_count, &i))
|
|
|
|
{
|
|
|
|
setme->sessionCount = (uint64_t)i;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tr_variantDictFindInt(&top, TR_KEY_uploaded_bytes, &i))
|
|
|
|
{
|
|
|
|
setme->uploadedBytes = (uint64_t)i;
|
|
|
|
}
|
|
|
|
|
|
|
|
tr_variantFree(&top);
|
2007-11-21 20:03:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static void saveCumulativeStats(tr_session const* session, tr_session_stats const* s)
|
2007-11-26 18:55:36 +00:00
|
|
|
{
|
2021-10-23 15:43:15 +00:00
|
|
|
auto top = tr_variant{};
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_variantInitDict(&top, 5);
|
|
|
|
tr_variantDictAddInt(&top, TR_KEY_downloaded_bytes, s->downloadedBytes);
|
|
|
|
tr_variantDictAddInt(&top, TR_KEY_files_added, s->filesAdded);
|
|
|
|
tr_variantDictAddInt(&top, TR_KEY_seconds_active, s->secondsActive);
|
|
|
|
tr_variantDictAddInt(&top, TR_KEY_session_count, s->sessionCount);
|
|
|
|
tr_variantDictAddInt(&top, TR_KEY_uploaded_bytes, s->uploadedBytes);
|
2012-12-05 17:29:46 +00:00
|
|
|
|
2022-03-10 05:51:14 +00:00
|
|
|
tr_variantToFile(&top, TR_VARIANT_FMT_JSON, getFilename(session));
|
2012-12-05 17:29:46 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_variantFree(&top);
|
2007-11-26 18:55:36 +00:00
|
|
|
}
|
|
|
|
|
2007-11-21 20:03:53 +00:00
|
|
|
/***
|
|
|
|
****
|
|
|
|
***/
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_statsInit(tr_session* session)
|
2007-11-21 20:03:53 +00:00
|
|
|
{
|
2022-01-17 18:39:50 +00:00
|
|
|
auto* const stats = tr_new0(struct tr_stats_handle, 1);
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
loadCumulativeStats(session, &stats->old);
|
|
|
|
stats->single.sessionCount = 1;
|
|
|
|
stats->startTime = tr_time();
|
|
|
|
session->sessionStats = stats;
|
2007-11-21 20:03:53 +00:00
|
|
|
}
|
|
|
|
|
2021-10-23 15:43:15 +00:00
|
|
|
static tr_stats_handle* getStats(tr_session const* session)
|
2009-12-08 20:51:45 +00:00
|
|
|
{
|
2021-09-15 00:18:09 +00:00
|
|
|
return session != nullptr ? session->sessionStats : nullptr;
|
2009-12-08 20:51:45 +00:00
|
|
|
}
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_statsSaveDirty(tr_session* session)
|
2007-11-21 20:03:53 +00:00
|
|
|
{
|
2021-10-23 15:43:15 +00:00
|
|
|
auto* const h = getStats(session);
|
2012-12-05 17:29:46 +00:00
|
|
|
|
2021-09-15 00:18:09 +00:00
|
|
|
if (h != nullptr && h->isDirty)
|
2009-12-08 20:51:45 +00:00
|
|
|
{
|
2021-09-12 17:41:49 +00:00
|
|
|
auto cumulative = tr_session_stats{};
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_sessionGetCumulativeStats(session, &cumulative);
|
|
|
|
saveCumulativeStats(session, &cumulative);
|
|
|
|
h->isDirty = false;
|
2009-12-08 20:51:45 +00:00
|
|
|
}
|
|
|
|
}
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_statsClose(tr_session* session)
|
2009-12-08 20:51:45 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_statsSaveDirty(session);
|
2007-11-21 20:03:53 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_free(session->sessionStats);
|
2021-09-15 00:18:09 +00:00
|
|
|
session->sessionStats = nullptr;
|
2007-11-21 20:03:53 +00:00
|
|
|
}
|
|
|
|
|
2008-02-04 19:54:47 +00:00
|
|
|
/***
|
|
|
|
****
|
|
|
|
***/
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
static void updateRatio(tr_session_stats* setme)
|
2007-12-22 03:01:14 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
setme->ratio = tr_getRatio(setme->uploadedBytes, setme->downloadedBytes);
|
2008-02-04 19:54:47 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static void addStats(tr_session_stats* setme, tr_session_stats const* a, tr_session_stats const* b)
|
2008-02-04 19:54:47 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
setme->uploadedBytes = a->uploadedBytes + b->uploadedBytes;
|
|
|
|
setme->downloadedBytes = a->downloadedBytes + b->downloadedBytes;
|
|
|
|
setme->filesAdded = a->filesAdded + b->filesAdded;
|
|
|
|
setme->sessionCount = a->sessionCount + b->sessionCount;
|
|
|
|
setme->secondsActive = a->secondsActive + b->secondsActive;
|
|
|
|
updateRatio(setme);
|
2007-12-22 03:01:14 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
void tr_sessionGetStats(tr_session const* session, tr_session_stats* setme)
|
2007-11-21 20:03:53 +00:00
|
|
|
{
|
2017-04-20 16:02:19 +00:00
|
|
|
struct tr_stats_handle const* stats = getStats(session);
|
2012-12-05 17:29:46 +00:00
|
|
|
|
2021-09-15 00:18:09 +00:00
|
|
|
if (stats != nullptr)
|
2008-11-17 05:05:11 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
*setme = stats->single;
|
|
|
|
setme->secondsActive = tr_time() - stats->startTime;
|
|
|
|
updateRatio(setme);
|
2008-11-17 05:05:11 +00:00
|
|
|
}
|
2007-11-21 20:03:53 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
void tr_sessionGetCumulativeStats(tr_session const* session, tr_session_stats* setme)
|
2007-11-21 20:03:53 +00:00
|
|
|
{
|
2017-04-20 16:02:19 +00:00
|
|
|
struct tr_stats_handle const* stats = getStats(session);
|
2021-09-12 17:41:49 +00:00
|
|
|
auto current = tr_session_stats{};
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2021-09-15 00:18:09 +00:00
|
|
|
if (stats != nullptr)
|
2008-11-17 05:05:11 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_sessionGetStats(session, ¤t);
|
|
|
|
addStats(setme, &stats->old, ¤t);
|
2008-11-17 05:05:11 +00:00
|
|
|
}
|
2007-11-21 20:03:53 +00:00
|
|
|
}
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_sessionClearStats(tr_session* session)
|
2008-04-22 14:07:42 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_session_stats zero;
|
2012-12-05 17:29:46 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
zero.uploadedBytes = 0;
|
|
|
|
zero.downloadedBytes = 0;
|
|
|
|
zero.ratio = TR_RATIO_NA;
|
|
|
|
zero.filesAdded = 0;
|
|
|
|
zero.sessionCount = 0;
|
|
|
|
zero.secondsActive = 0;
|
2012-12-05 17:29:46 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
session->sessionStats->isDirty = true;
|
|
|
|
session->sessionStats->single = session->sessionStats->old = zero;
|
|
|
|
session->sessionStats->startTime = tr_time();
|
2008-04-22 14:07:42 +00:00
|
|
|
}
|
|
|
|
|
2007-11-21 20:03:53 +00:00
|
|
|
/**
|
|
|
|
***
|
|
|
|
**/
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_statsAddUploaded(tr_session* session, uint32_t bytes)
|
2007-11-21 20:03:53 +00:00
|
|
|
{
|
2021-10-23 15:43:15 +00:00
|
|
|
auto* const s = getStats(session);
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2021-10-23 15:43:15 +00:00
|
|
|
if (s != nullptr)
|
2009-12-08 20:51:45 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
s->single.uploadedBytes += bytes;
|
|
|
|
s->isDirty = true;
|
2009-12-08 20:51:45 +00:00
|
|
|
}
|
2007-11-21 20:03:53 +00:00
|
|
|
}
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_statsAddDownloaded(tr_session* session, uint32_t bytes)
|
2007-11-21 20:03:53 +00:00
|
|
|
{
|
2021-10-23 15:43:15 +00:00
|
|
|
auto* const s = getStats(session);
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2021-10-23 15:43:15 +00:00
|
|
|
if (s != nullptr)
|
2009-12-08 20:51:45 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
s->single.downloadedBytes += bytes;
|
|
|
|
s->isDirty = true;
|
2009-12-08 20:51:45 +00:00
|
|
|
}
|
2007-11-26 18:55:36 +00:00
|
|
|
}
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_statsFileCreated(tr_session* session)
|
2007-11-26 18:55:36 +00:00
|
|
|
{
|
2021-10-23 15:43:15 +00:00
|
|
|
auto* const s = getStats(session);
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2021-10-23 15:43:15 +00:00
|
|
|
if (s != nullptr)
|
2017-04-19 12:04:45 +00:00
|
|
|
{
|
|
|
|
s->single.filesAdded++;
|
|
|
|
}
|
2007-11-21 20:03:53 +00:00
|
|
|
}
|