Skip to content

Commit

Permalink
FEAT(server, client): Add rolling connection quality information
Browse files Browse the repository at this point in the history
  • Loading branch information
Hartmnt committed Jan 9, 2025
1 parent 8b8c4ea commit 29b376a
Show file tree
Hide file tree
Showing 9 changed files with 434 additions and 111 deletions.
1 change: 1 addition & 0 deletions src/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,7 @@ set(SHARED_SOURCES

"crypto/CryptographicHash.cpp"
"crypto/CryptographicRandom.cpp"
"crypto/CryptState.cpp"
"crypto/CryptStateOCB2.cpp"

"${3RDPARTY_DIR}/arc4random/arc4random_uniform.cpp"
Expand Down
6 changes: 6 additions & 0 deletions src/Mumble.proto
Original file line number Diff line number Diff line change
Expand Up @@ -553,6 +553,12 @@ message UserStats {
// True if the user has a strong certificate.
optional bool strong_certificate = 18 [default = false];
optional bool opus = 19 [default = false];

optional uint32 rolling_time = 20;
// Rolling packet statistics for packets received from the client.
optional Stats rolling_from_client = 21;
// Rolling packet statistics for packets sent by the server.
optional Stats rolling_from_server = 22;
}

// Used by the client to request binary data from the server. By default large
Expand Down
49 changes: 49 additions & 0 deletions src/crypto/CryptState.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
// Copyright The Mumble Developers. All rights reserved.
// Use of this source code is governed by a BSD-style license
// that can be found in the LICENSE file at the root of the
// Mumble source tree or at <https://www.mumble.info/LICENSE>.

#include "CryptState.h"

void CryptState::handleRollingStats() {
bool noData = m_statsLocalReference.empty() && m_statsRemoteReference.empty();

std::chrono now = std::chrono::steady_clock::now();

// Update every 5 seconds
if ((now - m_rollingTimer) > m_rollingScanInterval) {
return;
}

PacketStatsSnapshot snapshotLocal;
snapshotLocal.stats = m_statsLocal;
snapshotLocal.timestamp = now;
m_statsLocalReference.push(snapshotLocal);

PacketStatsSnapshot snapshotRemote;
snapshotRemote.stats = m_statsRemote;
snapshotRemote.timestamp = now;
m_statsRemoteReference.push(snapshotRemote);

while (!m_statsLocalReference.empty() && (m_statsLocalReference.front().timestamp + m_rollingWindow) > now) {
m_statsLocalReference.pop();
}

while (!m_statsRemoteReference.empty() && (m_statsRemoteReference.front().timestamp + m_rollingWindow) > now) {
m_statsRemoteReference.pop();
}

if (!m_statsLocalReference.empty()) {
m_statsLocalRolling.good = m_statsLocal.good - m_statsLocalReference.front().stats.good;
m_statsLocalRolling.late = m_statsLocal.late - m_statsLocalReference.front().stats.late;
m_statsLocalRolling.lost = m_statsLocal.lost - m_statsLocalReference.front().stats.lost;
m_statsLocalRolling.resync = m_statsLocal.resync - m_statsLocalReference.front().stats.resync;
}

if (!m_statsRemoteReference.empty()) {
m_statsRemoteRolling.good = m_statsRemote.good - m_statsRemoteReference.front().stats.good;
m_statsRemoteRolling.late = m_statsRemote.late - m_statsRemoteReference.front().stats.late;
m_statsRemoteRolling.lost = m_statsRemote.lost - m_statsRemoteReference.front().stats.lost;
m_statsRemoteRolling.resync = m_statsRemote.resync - m_statsRemoteReference.front().stats.resync;
}
}
34 changes: 28 additions & 6 deletions src/crypto/CryptState.h
Original file line number Diff line number Diff line change
Expand Up @@ -7,21 +7,43 @@
#define MUMBLE_CRYPTSTATE_H_

#include "Timer.h"
#include <chrono>
#include <queue>
#include <string>

struct PacketStats {
unsigned int good = 0;
unsigned int late = 0;
unsigned int lost = 0;
unsigned int rsync = 0;
unsigned int good = 0;
unsigned int late = 0;
unsigned int lost = 0;
unsigned int resync = 0;
};

struct PacketStatsSnapshot {
PacketStats stats;
std::chrono timestamp;
};

class CryptState {
private:
Q_DISABLE_COPY(CryptState)

const std::chrono::duration m_rollingScanInterval = std::chrono::seconds(5);
const std::chrono m_rollingTimer = std::chrono::steady_clock::now();

std::queue< PacketStatsSnapshot > m_statsLocalReference;
std::queue< PacketStatsSnapshot > m_statsRemoteReference;

protected:
void handleRollingStats();

public:
PacketStats m_statsLocal = {};
PacketStats m_statsRemote = {};
PacketStats m_statsLocal = {};
PacketStats m_statsRemote = {};
PacketStats m_statsLocalRolling = {};
PacketStats m_statsRemoteRolling = {};

/// This is the packet statistics sliding time window size in seconds
std::chrono::duration m_rollingWindow = std::chrono::seconds(60 * 5);

Timer tLastGood;
Timer tLastRequest;
Expand Down
1 change: 1 addition & 0 deletions src/crypto/CryptStateOCB2.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -220,6 +220,7 @@ bool CryptStateOCB2::decrypt(const unsigned char *source, unsigned char *dst, un
m_statsLocal.lost -= static_cast< unsigned int >(std::abs(lost));
}

handleRollingStats();
tLastGood.restart();
return true;
}
Expand Down
68 changes: 60 additions & 8 deletions src/mumble/UserInformation.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -163,8 +163,12 @@ void UserInformation::update(const MumbleProto::UserStats &msg) {
qlTCPVar->setText(QString::number(msg.tcp_ping_var() > 0.0f ? sqrtf(msg.tcp_ping_var()) : 0.0f, 'f', 2));
qlUDPVar->setText(QString::number(msg.udp_ping_var() > 0.0f ? sqrtf(msg.udp_ping_var()) : 0.0f, 'f', 2));

if (msg.has_from_client() && msg.has_from_server()) {
qgbUDP->setVisible(true);
bool hasTotalStats = msg.has_from_client() && msg.has_from_server();
bool hasRollingStats = msg.has_rolling_time() && msg.has_rolling_from_client() && msg.has_rolling_from_server();

qgbUDP->setVisible(hasTotalStats || hasRollingStats);

if (hasTotalStats) {
const MumbleProto::UserStats_Stats &from = msg.from_client();
qlFromGood->setText(QString::number(from.good()));
qlFromLate->setText(QString::number(from.late()));
Expand All @@ -179,17 +183,65 @@ void UserInformation::update(const MumbleProto::UserStats &msg) {

quint32 allFromPackets = from.good() + from.late() + from.lost();
qlFromLatePercent->setText(
QString::number(allFromPackets > 0 ? from.late() * 100.0 / allFromPackets : 0., 'f', 2));
QString::number(allFromPackets > 0 ? from.late() * 100.0 / allFromPackets : 0., 'f', 1));
qlFromLostPercent->setText(
QString::number(allFromPackets > 0 ? from.lost() * 100.0 / allFromPackets : 0., 'f', 2));
QString::number(allFromPackets > 0 ? from.lost() * 100.0 / allFromPackets : 0., 'f', 1));

quint32 allToPackets = to.good() + to.late() + to.lost();
qlToLatePercent->setText(QString::number(allToPackets > 0 ? to.late() * 100.0 / allToPackets : 0., 'f', 2));
qlToLostPercent->setText(QString::number(allToPackets > 0 ? to.lost() * 100.0 / allToPackets : 0., 'f', 2));
} else {
qgbUDP->setVisible(false);
qlToLatePercent->setText(QString::number(allToPackets > 0 ? to.late() * 100.0 / allToPackets : 0., 'f', 1));
qlToLostPercent->setText(QString::number(allToPackets > 0 ? to.lost() * 100.0 / allToPackets : 0., 'f', 1));
}

if (hasRollingStats) {
const MumbleProto::UserStats_Stats &from = msg.rolling_from_client();
qlFromGoodRolling->setText(QString::number(from.good()));
qlFromLateRolling->setText(QString::number(from.late()));
qlFromLostRolling->setText(QString::number(from.lost()));
qlFromResyncRolling->setText(QString::number(from.resync()));

const MumbleProto::UserStats_Stats &to = msg.rolling_from_server();
qlToGoodRolling->setText(QString::number(to.good()));
qlToLateRolling->setText(QString::number(to.late()));
qlToLostRolling->setText(QString::number(to.lost()));
qlToResyncRolling->setText(QString::number(to.resync()));

quint32 allFromPackets = from.good() + from.late() + from.lost();
qlFromLatePercentRolling->setText(
QString::number(allFromPackets > 0 ? from.late() * 100.0 / allFromPackets : 0., 'f', 1));
qlFromLostPercentRolling->setText(
QString::number(allFromPackets > 0 ? from.lost() * 100.0 / allFromPackets : 0., 'f', 1));

quint32 allToPackets = to.good() + to.late() + to.lost();
qlToLatePercentRolling->setText(
QString::number(allToPackets > 0 ? to.late() * 100.0 / allToPackets : 0., 'f', 1));
qlToLostPercentRolling->setText(
QString::number(allToPackets > 0 ? to.lost() * 100.0 / allToPackets : 0., 'f', 1));

uint32_t rollingSeconds = msg.rolling_time();
QString rollingText = tr("Last %1 %2:");
if (rollingSeconds < 120) {
qliRolling->setText(rollingText.arg(QString::number(rollingSeconds)).arg(tr("seconds")));
} else {
qliRolling->setText(rollingText.arg(QString::number(rollingSeconds / 60)).arg(tr("minutes")));
}
}

qlFromGoodRolling->setVisible(hasRollingStats);
qlFromLateRolling->setVisible(hasRollingStats);
qlFromLostRolling->setVisible(hasRollingStats);
qlFromResyncRolling->setVisible(hasRollingStats);
qlToGoodRolling->setVisible(hasRollingStats);
qlToLateRolling->setVisible(hasRollingStats);
qlToLostRolling->setVisible(hasRollingStats);
qlToResyncRolling->setVisible(hasRollingStats);
qlFromLatePercentRolling->setVisible(hasRollingStats);
qlFromLostPercentRolling->setVisible(hasRollingStats);
qlToLatePercentRolling->setVisible(hasRollingStats);
qlToLostPercentRolling->setVisible(hasRollingStats);
qliRolling->setVisible(hasRollingStats);
qliRollingFrom->setVisible(hasRollingStats);
qliRollingTo->setVisible(hasRollingStats);

if (msg.has_onlinesecs()) {
if (msg.has_idlesecs())
qlTime->setText(
Expand Down
Loading

0 comments on commit 29b376a

Please sign in to comment.