naiveproxy/net/nqe/network_quality_store.cc

138 lines
4.7 KiB
C++
Raw Permalink Normal View History

2018-01-28 21:32:06 +03:00
// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "net/nqe/network_quality_store.h"
#include "base/location.h"
#include "base/threading/thread_task_runner_handle.h"
#include "net/base/network_change_notifier.h"
namespace net {
namespace nqe {
namespace internal {
NetworkQualityStore::NetworkQualityStore()
: disable_offline_check_(false), weak_ptr_factory_(this) {
static_assert(kMaximumNetworkQualityCacheSize > 0,
"Size of the network quality cache must be > 0");
// This limit should not be increased unless the logic for removing the
// oldest cache entry is rewritten to use a doubly-linked-list LRU queue.
static_assert(kMaximumNetworkQualityCacheSize <= 10,
"Size of the network quality cache must <= 10");
}
NetworkQualityStore::~NetworkQualityStore() {
DCHECK(thread_checker_.CalledOnValidThread());
}
void NetworkQualityStore::Add(
const nqe::internal::NetworkID& network_id,
const nqe::internal::CachedNetworkQuality& cached_network_quality) {
DCHECK(thread_checker_.CalledOnValidThread());
DCHECK_LE(cached_network_qualities_.size(),
static_cast<size_t>(kMaximumNetworkQualityCacheSize));
if (cached_network_quality.effective_connection_type() ==
EFFECTIVE_CONNECTION_TYPE_UNKNOWN) {
return;
}
if (!EligibleForCaching(network_id))
return;
// Remove the entry from the map, if it is already present.
cached_network_qualities_.erase(network_id);
if (cached_network_qualities_.size() == kMaximumNetworkQualityCacheSize) {
// Remove the oldest entry.
CachedNetworkQualities::iterator oldest_entry_iterator =
cached_network_qualities_.begin();
for (CachedNetworkQualities::iterator it =
cached_network_qualities_.begin();
it != cached_network_qualities_.end(); ++it) {
if ((it->second).OlderThan(oldest_entry_iterator->second))
oldest_entry_iterator = it;
}
cached_network_qualities_.erase(oldest_entry_iterator);
}
cached_network_qualities_.insert(
std::make_pair(network_id, cached_network_quality));
DCHECK_LE(cached_network_qualities_.size(),
static_cast<size_t>(kMaximumNetworkQualityCacheSize));
for (auto& observer : network_qualities_cache_observer_list_)
observer.OnChangeInCachedNetworkQuality(network_id, cached_network_quality);
}
bool NetworkQualityStore::GetById(
const nqe::internal::NetworkID& network_id,
nqe::internal::CachedNetworkQuality* cached_network_quality) {
DCHECK(thread_checker_.CalledOnValidThread());
CachedNetworkQualities::const_iterator it =
cached_network_qualities_.find(network_id);
if (it == cached_network_qualities_.end())
return false;
*cached_network_quality = it->second;
return true;
}
void NetworkQualityStore::AddNetworkQualitiesCacheObserver(
NetworkQualitiesCacheObserver* observer) {
DCHECK(thread_checker_.CalledOnValidThread());
network_qualities_cache_observer_list_.AddObserver(observer);
// Notify the |observer| on the next message pump since |observer| may not
// be completely set up for receiving the callbacks.
base::ThreadTaskRunnerHandle::Get()->PostTask(
FROM_HERE, base::Bind(&NetworkQualityStore::NotifyCacheObserverIfPresent,
weak_ptr_factory_.GetWeakPtr(), observer));
}
void NetworkQualityStore::RemoveNetworkQualitiesCacheObserver(
NetworkQualitiesCacheObserver* observer) {
DCHECK(thread_checker_.CalledOnValidThread());
network_qualities_cache_observer_list_.RemoveObserver(observer);
}
bool NetworkQualityStore::EligibleForCaching(
const NetworkID& network_id) const {
DCHECK(thread_checker_.CalledOnValidThread());
// |disable_offline_check_| forces caching of the network quality even if
// the network is set to offline.
return network_id.type == NetworkChangeNotifier::CONNECTION_ETHERNET ||
!network_id.id.empty() ||
(network_id.type == NetworkChangeNotifier::CONNECTION_NONE &&
disable_offline_check_);
}
void NetworkQualityStore::DisableOfflineCheckForTesting(
bool disable_offline_check) {
DCHECK(thread_checker_.CalledOnValidThread());
disable_offline_check_ = disable_offline_check;
}
void NetworkQualityStore::NotifyCacheObserverIfPresent(
NetworkQualitiesCacheObserver* observer) const {
DCHECK(thread_checker_.CalledOnValidThread());
if (!network_qualities_cache_observer_list_.HasObserver(observer))
return;
for (const auto it : cached_network_qualities_)
observer->OnChangeInCachedNetworkQuality(it.first, it.second);
}
} // namespace internal
} // namespace nqe
} // namespace net