2020-08-14 18:58:22 +02:00
|
|
|
#include "NetworkManager.h"
|
|
|
|
|
|
|
|
#include "Message.h"
|
|
|
|
|
|
|
|
#include "p2p/base/basic_packet_socket_factory.h"
|
2023-02-18 22:24:25 +01:00
|
|
|
#include "v2/ReflectorRelayPortFactory.h"
|
2020-08-14 18:58:22 +02:00
|
|
|
#include "p2p/client/basic_port_allocator.h"
|
|
|
|
#include "p2p/base/p2p_transport_channel.h"
|
|
|
|
#include "p2p/base/basic_async_resolver_factory.h"
|
|
|
|
#include "api/packet_socket_factory.h"
|
|
|
|
#include "p2p/base/ice_credentials_iterator.h"
|
|
|
|
#include "api/jsep_ice_candidate.h"
|
2021-06-25 02:43:10 +02:00
|
|
|
#include "rtc_base/network_monitor_factory.h"
|
|
|
|
|
2021-07-19 17:56:43 +02:00
|
|
|
#include "TurnCustomizerImpl.h"
|
2021-06-25 02:43:10 +02:00
|
|
|
#include "platform/PlatformInterface.h"
|
2020-08-14 18:58:22 +02:00
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
#include <openssl/sha.h>
|
|
|
|
#include <openssl/aes.h>
|
|
|
|
#ifndef OPENSSL_IS_BORINGSSL
|
|
|
|
#include <openssl/modes.h>
|
|
|
|
#endif
|
|
|
|
#include <openssl/rand.h>
|
|
|
|
#include <openssl/crypto.h>
|
|
|
|
} // extern "C"
|
|
|
|
|
|
|
|
namespace tgcalls {
|
|
|
|
|
2020-12-23 08:48:30 +01:00
|
|
|
class TgCallsCryptStringImpl : public rtc::CryptStringImpl {
|
|
|
|
public:
|
|
|
|
TgCallsCryptStringImpl(std::string const &value) :
|
|
|
|
_value(value) {
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual ~TgCallsCryptStringImpl() override {
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual size_t GetLength() const override {
|
|
|
|
return _value.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void CopyTo(char* dest, bool nullterminate) const override {
|
|
|
|
memcpy(dest, _value.data(), _value.size());
|
|
|
|
if (nullterminate) {
|
|
|
|
dest[_value.size()] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
virtual std::string UrlEncode() const override {
|
|
|
|
return _value;
|
|
|
|
}
|
|
|
|
virtual CryptStringImpl* Copy() const override {
|
|
|
|
return new TgCallsCryptStringImpl(_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void CopyRawTo(std::vector<unsigned char>* dest) const override {
|
|
|
|
dest->resize(_value.size());
|
|
|
|
memcpy(dest->data(), _value.data(), _value.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::string _value;
|
|
|
|
};
|
|
|
|
|
2020-08-14 18:58:22 +02:00
|
|
|
NetworkManager::NetworkManager(
|
|
|
|
rtc::Thread *thread,
|
|
|
|
EncryptionKey encryptionKey,
|
|
|
|
bool enableP2P,
|
2020-10-01 03:59:32 +02:00
|
|
|
bool enableTCP,
|
|
|
|
bool enableStunMarking,
|
2020-08-14 18:58:22 +02:00
|
|
|
std::vector<RtcServer> const &rtcServers,
|
2020-12-23 08:48:30 +01:00
|
|
|
std::unique_ptr<Proxy> proxy,
|
2020-08-14 18:58:22 +02:00
|
|
|
std::function<void(const NetworkManager::State &)> stateUpdated,
|
|
|
|
std::function<void(DecryptedMessage &&)> transportMessageReceived,
|
|
|
|
std::function<void(Message &&)> sendSignalingMessage,
|
|
|
|
std::function<void(int delayMs, int cause)> sendTransportServiceAsync) :
|
|
|
|
_thread(thread),
|
2020-08-15 23:06:36 +02:00
|
|
|
_enableP2P(enableP2P),
|
2020-10-01 03:59:32 +02:00
|
|
|
_enableTCP(enableTCP),
|
|
|
|
_enableStunMarking(enableStunMarking),
|
2020-08-15 23:06:36 +02:00
|
|
|
_rtcServers(rtcServers),
|
2020-12-23 08:48:30 +01:00
|
|
|
_proxy(std::move(proxy)),
|
2020-08-14 18:58:22 +02:00
|
|
|
_transport(
|
|
|
|
EncryptedConnection::Type::Transport,
|
|
|
|
encryptionKey,
|
|
|
|
[=](int delayMs, int cause) { sendTransportServiceAsync(delayMs, cause); }),
|
|
|
|
_isOutgoing(encryptionKey.isOutgoing),
|
|
|
|
_stateUpdated(std::move(stateUpdated)),
|
|
|
|
_transportMessageReceived(std::move(transportMessageReceived)),
|
|
|
|
_sendSignalingMessage(std::move(sendSignalingMessage)),
|
2022-04-21 20:03:20 +02:00
|
|
|
_localIceParameters(rtc::CreateRandomString(cricket::ICE_UFRAG_LENGTH), rtc::CreateRandomString(cricket::ICE_PWD_LENGTH), false) {
|
2020-08-14 18:58:22 +02:00
|
|
|
assert(_thread->IsCurrent());
|
2021-06-25 02:43:10 +02:00
|
|
|
|
|
|
|
_networkMonitorFactory = PlatformInterface::SharedInstance()->createNetworkMonitorFactory();
|
2020-08-15 23:06:36 +02:00
|
|
|
}
|
2020-08-14 18:58:22 +02:00
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
NetworkManager::~NetworkManager() {
|
|
|
|
assert(_thread->IsCurrent());
|
|
|
|
|
|
|
|
RTC_LOG(LS_INFO) << "NetworkManager::~NetworkManager()";
|
2020-08-14 18:58:22 +02:00
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
_transportChannel.reset();
|
|
|
|
_asyncResolverFactory.reset();
|
|
|
|
_portAllocator.reset();
|
|
|
|
_networkManager.reset();
|
|
|
|
_socketFactory.reset();
|
2022-04-16 16:43:17 +02:00
|
|
|
_networkMonitorFactory.reset();
|
2020-08-15 23:06:36 +02:00
|
|
|
}
|
2020-08-14 18:58:22 +02:00
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
void NetworkManager::start() {
|
2022-03-11 17:49:54 +01:00
|
|
|
_socketFactory.reset(new rtc::BasicPacketSocketFactory(_thread->socketserver()));
|
2020-08-14 18:58:22 +02:00
|
|
|
|
2023-02-18 22:24:25 +01:00
|
|
|
_networkManager = std::make_unique<rtc::BasicNetworkManager>(_networkMonitorFactory.get(), _thread->socketserver());
|
2020-10-01 03:59:32 +02:00
|
|
|
|
|
|
|
if (_enableStunMarking) {
|
|
|
|
_turnCustomizer.reset(new TurnCustomizerImpl());
|
|
|
|
}
|
|
|
|
|
2023-02-18 22:24:25 +01:00
|
|
|
_relayPortFactory.reset(new ReflectorRelayPortFactory(_rtcServers));
|
|
|
|
|
|
|
|
_portAllocator.reset(new cricket::BasicPortAllocator(_networkManager.get(), _socketFactory.get(), _turnCustomizer.get(), _relayPortFactory.get()));
|
2020-08-14 18:58:22 +02:00
|
|
|
|
2021-03-19 11:25:58 +01:00
|
|
|
uint32_t flags = _portAllocator->flags();
|
|
|
|
|
|
|
|
flags |=
|
|
|
|
//cricket::PORTALLOCATOR_ENABLE_SHARED_SOCKET |
|
|
|
|
cricket::PORTALLOCATOR_ENABLE_IPV6 |
|
|
|
|
cricket::PORTALLOCATOR_ENABLE_IPV6_ON_WIFI;
|
|
|
|
|
2020-10-01 03:59:32 +02:00
|
|
|
if (!_enableTCP) {
|
|
|
|
flags |= cricket::PORTALLOCATOR_DISABLE_TCP;
|
|
|
|
}
|
2020-08-15 23:06:36 +02:00
|
|
|
if (!_enableP2P) {
|
|
|
|
flags |= cricket::PORTALLOCATOR_DISABLE_UDP;
|
|
|
|
flags |= cricket::PORTALLOCATOR_DISABLE_STUN;
|
2021-02-23 12:53:38 +01:00
|
|
|
uint32_t candidateFilter = _portAllocator->candidate_filter();
|
|
|
|
candidateFilter &= ~(cricket::CF_REFLEXIVE);
|
|
|
|
_portAllocator->SetCandidateFilter(candidateFilter);
|
2020-08-15 23:06:36 +02:00
|
|
|
}
|
2020-12-23 08:48:30 +01:00
|
|
|
|
2021-03-19 11:25:58 +01:00
|
|
|
_portAllocator->set_step_delay(cricket::kMinimumStepDelay);
|
|
|
|
|
2020-12-23 08:48:30 +01:00
|
|
|
if (_proxy) {
|
|
|
|
rtc::ProxyInfo proxyInfo;
|
|
|
|
proxyInfo.type = rtc::ProxyType::PROXY_SOCKS5;
|
|
|
|
proxyInfo.address = rtc::SocketAddress(_proxy->host, _proxy->port);
|
|
|
|
proxyInfo.username = _proxy->login;
|
|
|
|
proxyInfo.password = rtc::CryptString(TgCallsCryptStringImpl(_proxy->password));
|
|
|
|
_portAllocator->set_proxy("t/1.0", proxyInfo);
|
|
|
|
}
|
|
|
|
|
2021-03-19 11:25:58 +01:00
|
|
|
_portAllocator->set_flags(flags);
|
2020-08-15 23:06:36 +02:00
|
|
|
_portAllocator->Initialize();
|
|
|
|
|
|
|
|
cricket::ServerAddresses stunServers;
|
|
|
|
std::vector<cricket::RelayServerConfig> turnServers;
|
|
|
|
|
|
|
|
for (auto &server : _rtcServers) {
|
2023-02-18 22:24:25 +01:00
|
|
|
if (server.isTcp) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-08-14 18:58:22 +02:00
|
|
|
if (server.isTurn) {
|
|
|
|
turnServers.push_back(cricket::RelayServerConfig(
|
|
|
|
rtc::SocketAddress(server.host, server.port),
|
|
|
|
server.login,
|
|
|
|
server.password,
|
|
|
|
cricket::PROTO_UDP
|
|
|
|
));
|
|
|
|
} else {
|
|
|
|
rtc::SocketAddress stunAddress = rtc::SocketAddress(server.host, server.port);
|
|
|
|
stunServers.insert(stunAddress);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-01 03:59:32 +02:00
|
|
|
_portAllocator->SetConfiguration(stunServers, turnServers, 2, webrtc::NO_PRUNE, _turnCustomizer.get());
|
2020-08-14 18:58:22 +02:00
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
_asyncResolverFactory = std::make_unique<webrtc::BasicAsyncResolverFactory>();
|
2023-02-18 22:24:25 +01:00
|
|
|
|
|
|
|
webrtc::IceTransportInit iceTransportInit;
|
|
|
|
iceTransportInit.set_port_allocator(_portAllocator.get());
|
|
|
|
iceTransportInit.set_async_resolver_factory(_asyncResolverFactory.get());
|
|
|
|
|
|
|
|
_transportChannel = cricket::P2PTransportChannel::Create("transport", 0, std::move(iceTransportInit));
|
2020-08-14 18:58:22 +02:00
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
cricket::IceConfig iceConfig;
|
|
|
|
iceConfig.continual_gathering_policy = cricket::GATHER_CONTINUALLY;
|
2020-08-14 18:58:22 +02:00
|
|
|
iceConfig.prioritize_most_likely_candidate_pairs = true;
|
|
|
|
iceConfig.regather_on_failed_networks_interval = 8000;
|
2020-08-15 23:06:36 +02:00
|
|
|
_transportChannel->SetIceConfig(iceConfig);
|
2020-08-14 18:58:22 +02:00
|
|
|
|
|
|
|
cricket::IceParameters localIceParameters(
|
|
|
|
_localIceParameters.ufrag,
|
|
|
|
_localIceParameters.pwd,
|
|
|
|
false
|
|
|
|
);
|
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
_transportChannel->SetIceParameters(localIceParameters);
|
|
|
|
_transportChannel->SetIceRole(_isOutgoing ? cricket::ICEROLE_CONTROLLING : cricket::ICEROLE_CONTROLLED);
|
2020-08-14 18:58:22 +02:00
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
_transportChannel->SignalCandidateGathered.connect(this, &NetworkManager::candidateGathered);
|
|
|
|
_transportChannel->SignalGatheringState.connect(this, &NetworkManager::candidateGatheringState);
|
|
|
|
_transportChannel->SignalIceTransportStateChanged.connect(this, &NetworkManager::transportStateChanged);
|
|
|
|
_transportChannel->SignalReadPacket.connect(this, &NetworkManager::transportPacketReceived);
|
2020-08-14 18:58:22 +02:00
|
|
|
_transportChannel->SignalNetworkRouteChanged.connect(this, &NetworkManager::transportRouteChanged);
|
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
_transportChannel->MaybeStartGathering();
|
2020-08-14 18:58:22 +02:00
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
_transportChannel->SetRemoteIceMode(cricket::ICEMODE_FULL);
|
2020-08-14 18:58:22 +02:00
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
_lastNetworkActivityMs = rtc::TimeMillis();
|
|
|
|
|
|
|
|
checkConnectionTimeout();
|
2020-08-14 18:58:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkManager::receiveSignalingMessage(DecryptedMessage &&message) {
|
|
|
|
const auto list = absl::get_if<CandidatesListMessage>(&message.message.data);
|
|
|
|
assert(list != nullptr);
|
|
|
|
|
|
|
|
if (!_remoteIceParameters.has_value()) {
|
2022-04-21 20:03:20 +02:00
|
|
|
PeerIceParameters parameters(list->iceParameters.ufrag, list->iceParameters.pwd, false);
|
2020-08-14 18:58:22 +02:00
|
|
|
_remoteIceParameters = parameters;
|
|
|
|
|
|
|
|
cricket::IceParameters remoteIceParameters(
|
|
|
|
parameters.ufrag,
|
|
|
|
parameters.pwd,
|
|
|
|
false
|
|
|
|
);
|
|
|
|
|
|
|
|
_transportChannel->SetRemoteIceParameters(remoteIceParameters);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto &candidate : list->candidates) {
|
|
|
|
_transportChannel->AddRemoteCandidate(candidate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t NetworkManager::sendMessage(const Message &message) {
|
|
|
|
if (const auto prepared = _transport.prepareForSending(message)) {
|
|
|
|
rtc::PacketOptions packetOptions;
|
|
|
|
_transportChannel->SendPacket((const char *)prepared->bytes.data(), prepared->bytes.size(), packetOptions, 0);
|
2020-08-15 23:06:36 +02:00
|
|
|
addTrafficStats(prepared->bytes.size(), false);
|
2020-08-14 18:58:22 +02:00
|
|
|
return prepared->counter;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkManager::sendTransportService(int cause) {
|
|
|
|
if (const auto prepared = _transport.prepareForSendingService(cause)) {
|
|
|
|
rtc::PacketOptions packetOptions;
|
|
|
|
_transportChannel->SendPacket((const char *)prepared->bytes.data(), prepared->bytes.size(), packetOptions, 0);
|
2020-08-15 23:06:36 +02:00
|
|
|
addTrafficStats(prepared->bytes.size(), false);
|
2020-08-14 18:58:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
void NetworkManager::setIsLocalNetworkLowCost(bool isLocalNetworkLowCost) {
|
|
|
|
_isLocalNetworkLowCost = isLocalNetworkLowCost;
|
2020-10-01 03:59:32 +02:00
|
|
|
|
|
|
|
logCurrentNetworkState();
|
2020-08-15 23:06:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
TrafficStats NetworkManager::getNetworkStats() {
|
|
|
|
TrafficStats stats;
|
|
|
|
stats.bytesSentWifi = _trafficStatsWifi.outgoing;
|
|
|
|
stats.bytesReceivedWifi = _trafficStatsWifi.incoming;
|
|
|
|
stats.bytesSentMobile = _trafficStatsCellular.outgoing;
|
|
|
|
stats.bytesReceivedMobile = _trafficStatsCellular.incoming;
|
|
|
|
return stats;
|
|
|
|
}
|
|
|
|
|
2020-10-01 03:59:32 +02:00
|
|
|
void NetworkManager::fillCallStats(CallStats &callStats) {
|
|
|
|
callStats.networkRecords = std::move(_networkRecords);
|
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkManager::logCurrentNetworkState() {
|
|
|
|
if (!_currentEndpointType.has_value()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
CallStatsNetworkRecord record;
|
|
|
|
record.timestamp = (int32_t)(rtc::TimeMillis() / 1000);
|
|
|
|
record.endpointType = *_currentEndpointType;
|
|
|
|
record.isLowCost = _isLocalNetworkLowCost;
|
|
|
|
_networkRecords.push_back(std::move(record));
|
|
|
|
}
|
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
void NetworkManager::checkConnectionTimeout() {
|
|
|
|
const auto weak = std::weak_ptr<NetworkManager>(shared_from_this());
|
2023-02-18 22:24:25 +01:00
|
|
|
_thread->PostDelayedTask([weak]() {
|
2020-08-15 23:06:36 +02:00
|
|
|
auto strong = weak.lock();
|
|
|
|
if (!strong) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t currentTimestamp = rtc::TimeMillis();
|
|
|
|
const int64_t maxTimeout = 20000;
|
|
|
|
|
|
|
|
if (strong->_lastNetworkActivityMs + maxTimeout < currentTimestamp) {
|
|
|
|
NetworkManager::State emitState;
|
|
|
|
emitState.isReadyToSendData = false;
|
|
|
|
emitState.isFailed = true;
|
|
|
|
strong->_stateUpdated(emitState);
|
|
|
|
}
|
|
|
|
|
|
|
|
strong->checkConnectionTimeout();
|
2023-02-18 22:24:25 +01:00
|
|
|
}, webrtc::TimeDelta::Millis(1000));
|
2020-08-15 23:06:36 +02:00
|
|
|
}
|
|
|
|
|
2020-08-14 18:58:22 +02:00
|
|
|
void NetworkManager::candidateGathered(cricket::IceTransportInternal *transport, const cricket::Candidate &candidate) {
|
|
|
|
assert(_thread->IsCurrent());
|
|
|
|
_sendSignalingMessage({ CandidatesListMessage{ { 1, candidate }, _localIceParameters } });
|
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkManager::candidateGatheringState(cricket::IceTransportInternal *transport) {
|
|
|
|
assert(_thread->IsCurrent());
|
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkManager::transportStateChanged(cricket::IceTransportInternal *transport) {
|
|
|
|
assert(_thread->IsCurrent());
|
|
|
|
|
|
|
|
auto state = transport->GetIceTransportState();
|
|
|
|
bool isConnected = false;
|
|
|
|
switch (state) {
|
|
|
|
case webrtc::IceTransportState::kConnected:
|
|
|
|
case webrtc::IceTransportState::kCompleted:
|
|
|
|
isConnected = true;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
NetworkManager::State emitState;
|
|
|
|
emitState.isReadyToSendData = isConnected;
|
|
|
|
_stateUpdated(emitState);
|
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkManager::transportReadyToSend(cricket::IceTransportInternal *transport) {
|
|
|
|
assert(_thread->IsCurrent());
|
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkManager::transportPacketReceived(rtc::PacketTransportInternal *transport, const char *bytes, size_t size, const int64_t ×tamp, int unused) {
|
|
|
|
assert(_thread->IsCurrent());
|
2020-08-15 23:06:36 +02:00
|
|
|
|
|
|
|
_lastNetworkActivityMs = rtc::TimeMillis();
|
|
|
|
|
|
|
|
addTrafficStats(size, true);
|
2020-08-14 18:58:22 +02:00
|
|
|
|
|
|
|
if (auto decrypted = _transport.handleIncomingPacket(bytes, size)) {
|
|
|
|
if (_transportMessageReceived) {
|
|
|
|
_transportMessageReceived(std::move(decrypted->main));
|
|
|
|
for (auto &message : decrypted->additional) {
|
|
|
|
_transportMessageReceived(std::move(message));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void NetworkManager::transportRouteChanged(absl::optional<rtc::NetworkRoute> route) {
|
|
|
|
assert(_thread->IsCurrent());
|
|
|
|
|
|
|
|
if (route.has_value()) {
|
|
|
|
RTC_LOG(LS_INFO) << "NetworkManager route changed: " << route->DebugString();
|
|
|
|
|
|
|
|
bool localIsWifi = route->local.adapter_type() == rtc::AdapterType::ADAPTER_TYPE_WIFI;
|
|
|
|
bool remoteIsWifi = route->remote.adapter_type() == rtc::AdapterType::ADAPTER_TYPE_WIFI;
|
|
|
|
|
|
|
|
RTC_LOG(LS_INFO) << "NetworkManager is wifi: local=" << localIsWifi << ", remote=" << remoteIsWifi;
|
2020-10-01 03:59:32 +02:00
|
|
|
|
|
|
|
CallStatsConnectionEndpointType endpointType;
|
|
|
|
if (route->local.uses_turn()) {
|
|
|
|
endpointType = CallStatsConnectionEndpointType::ConnectionEndpointTURN;
|
|
|
|
} else {
|
|
|
|
endpointType = CallStatsConnectionEndpointType::ConnectionEndpointP2P;
|
|
|
|
}
|
|
|
|
if (!_currentEndpointType.has_value() || _currentEndpointType != endpointType) {
|
|
|
|
_currentEndpointType = endpointType;
|
|
|
|
logCurrentNetworkState();
|
|
|
|
}
|
2020-08-14 18:58:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-15 23:06:36 +02:00
|
|
|
void NetworkManager::addTrafficStats(int64_t byteCount, bool isIncoming) {
|
|
|
|
if (_isLocalNetworkLowCost) {
|
|
|
|
if (isIncoming) {
|
|
|
|
_trafficStatsWifi.incoming += byteCount;
|
|
|
|
} else {
|
|
|
|
_trafficStatsWifi.outgoing += byteCount;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (isIncoming) {
|
|
|
|
_trafficStatsCellular.incoming += byteCount;
|
|
|
|
} else {
|
|
|
|
_trafficStatsCellular.outgoing += byteCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-14 18:58:22 +02:00
|
|
|
} // namespace tgcalls
|