1
0
Fork 0
arangodb/arangod/MMFiles/MMFilesDatafileStatistics.cpp

276 lines
8.1 KiB
C++

////////////////////////////////////////////////////////////////////////////////
/// DISCLAIMER
///
/// Copyright 2014-2016 ArangoDB GmbH, Cologne, Germany
/// Copyright 2004-2014 triAGENS GmbH, Cologne, Germany
///
/// Licensed under the Apache License, Version 2.0 (the "License");
/// you may not use this file except in compliance with the License.
/// You may obtain a copy of the License at
///
/// http://www.apache.org/licenses/LICENSE-2.0
///
/// Unless required by applicable law or agreed to in writing, software
/// distributed under the License is distributed on an "AS IS" BASIS,
/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
/// See the License for the specific language governing permissions and
/// limitations under the License.
///
/// Copyright holder is ArangoDB GmbH, Cologne, Germany
///
/// @author Jan Steemann
////////////////////////////////////////////////////////////////////////////////
#include "MMFilesDatafileStatistics.h"
#include "Basics/Exceptions.h"
#include "Basics/ReadLocker.h"
#include "Basics/WriteLocker.h"
#include "Logger/Logger.h"
#include "MMFiles/MMFilesDatafile.h"
using namespace arangodb;
/// @brief create statistics manager for a collection
MMFilesDatafileStatistics::MMFilesDatafileStatistics()
: _lock(), _stats(), _statisticsLock(), _localStats({0, 0, 0, 0}) {}
/// @brief destroy statistics manager
MMFilesDatafileStatistics::~MMFilesDatafileStatistics() {
WRITE_LOCKER(writeLocker, _lock);
for (auto& it : _stats) {
delete it.second;
}
}
void MMFilesDatafileStatistics::compactionRun(uint64_t noCombined,
uint64_t read, uint64_t written) {
WRITE_LOCKER(writeLocker, _statisticsLock);
_localStats._compactionCount++;
if (noCombined > 1) {
_localStats._filesCombined += noCombined;
}
_localStats._compactionBytesRead += read;
_localStats._compactionBytesWritten += written;
}
MMFilesDatafileStatistics::CompactionStats MMFilesDatafileStatistics::getStats() {
READ_LOCKER(readLocker, _statisticsLock);
return _localStats;
}
/// @brief create an empty statistics container for a file
void MMFilesDatafileStatistics::create(TRI_voc_fid_t fid) {
auto stats = std::make_unique<MMFilesDatafileStatisticsContainer>();
WRITE_LOCKER(writeLocker, _lock);
auto it = _stats.find(fid);
if (it != _stats.end()) {
// already exists
return;
}
LOG_TOPIC("e63cd", TRACE, arangodb::Logger::DATAFILES)
<< "creating statistics for datafile " << fid;
_stats.emplace(fid, stats.get());
stats.release();
}
/// @brief create statistics for a datafile, using the stats provided
void MMFilesDatafileStatistics::create(TRI_voc_fid_t fid,
MMFilesDatafileStatisticsContainer const& src) {
auto stats = std::make_unique<MMFilesDatafileStatisticsContainer>();
*stats = src;
WRITE_LOCKER(writeLocker, _lock);
auto it = _stats.find(fid);
if (it != _stats.end()) {
// already exists
return;
}
LOG_TOPIC("82801", TRACE, arangodb::Logger::DATAFILES)
<< "creating statistics for datafile " << fid << " from initial data";
_stats.emplace(fid, stats.get());
stats.release();
}
/// @brief remove statistics for a file
void MMFilesDatafileStatistics::remove(TRI_voc_fid_t fid) {
LOG_TOPIC("2a42f", TRACE, arangodb::Logger::DATAFILES)
<< "removing statistics for datafile " << fid;
MMFilesDatafileStatisticsContainer* found = nullptr;
{
WRITE_LOCKER(writeLocker, _lock);
auto it = _stats.find(fid);
if (it != _stats.end()) {
found = (*it).second;
_stats.erase(it);
}
}
delete found;
}
/// @brief merge statistics for a file
void MMFilesDatafileStatistics::update(TRI_voc_fid_t fid,
MMFilesDatafileStatisticsContainer const& src,
bool warn) {
WRITE_LOCKER(writeLocker, _lock);
auto it = _stats.find(fid);
if (it == _stats.end()) {
if (warn) {
LOG_TOPIC("35926", WARN, arangodb::Logger::DATAFILES)
<< "did not find required statistics for datafile " << fid;
}
THROW_ARANGO_EXCEPTION_MESSAGE(TRI_ERROR_ARANGO_DATAFILE_STATISTICS_NOT_FOUND,
"datafile statistics not found on update");
}
auto& dst = (*it).second;
LOG_TOPIC("102a2", TRACE, arangodb::Logger::DATAFILES)
<< "updating statistics for datafile " << fid;
dst->update(src);
}
/// @brief merge statistics for a file, by copying the stats from another
void MMFilesDatafileStatistics::update(TRI_voc_fid_t fid, TRI_voc_fid_t src, bool warn) {
WRITE_LOCKER(writeLocker, _lock);
auto it = _stats.find(fid);
if (it == _stats.end()) {
if (warn) {
LOG_TOPIC("7d978", WARN, arangodb::Logger::DATAFILES)
<< "did not find required statistics for datafile " << fid;
}
THROW_ARANGO_EXCEPTION_MESSAGE(
TRI_ERROR_ARANGO_DATAFILE_STATISTICS_NOT_FOUND,
"datafile statistics not found for update target");
}
auto& dst = (*it).second;
it = _stats.find(src);
if (it == _stats.end()) {
if (warn) {
LOG_TOPIC("bc94b", WARN, arangodb::Logger::DATAFILES)
<< "did not find required statistics for source datafile " << src;
}
THROW_ARANGO_EXCEPTION_MESSAGE(
TRI_ERROR_ARANGO_DATAFILE_STATISTICS_NOT_FOUND,
"datafile statistics not found for update source");
}
LOG_TOPIC("3652a", TRACE, arangodb::Logger::DATAFILES)
<< "updating statistics for datafile " << fid;
dst->update(*(*it).second);
}
/// @brief replace statistics for a file
void MMFilesDatafileStatistics::replace(TRI_voc_fid_t fid,
MMFilesDatafileStatisticsContainer const& src,
bool warn) {
WRITE_LOCKER(writeLocker, _lock);
auto it = _stats.find(fid);
if (it == _stats.end()) {
if (warn) {
LOG_TOPIC("3ec85", WARN, arangodb::Logger::DATAFILES)
<< "did not find required statistics for datafile " << fid;
}
THROW_ARANGO_EXCEPTION_MESSAGE(TRI_ERROR_ARANGO_DATAFILE_STATISTICS_NOT_FOUND,
"datafile statistics not found on replace");
}
auto& dst = (*it).second;
*dst = src;
LOG_TOPIC("0205e", TRACE, arangodb::Logger::DATAFILES)
<< "replacing statistics for datafile " << fid;
}
/// @brief increase dead stats for a datafile, if it exists
void MMFilesDatafileStatistics::increaseDead(TRI_voc_fid_t fid, int64_t number, int64_t size) {
WRITE_LOCKER(writeLocker, _lock);
auto it = _stats.find(fid);
if (it == _stats.end()) {
// datafile not found. no problem
return;
}
auto& dst = (*it).second;
dst->numberDead += number;
dst->sizeDead += size;
dst->numberAlive -= number;
dst->sizeAlive -= size;
}
/// @brief increase number of uncollected entries
void MMFilesDatafileStatistics::increaseUncollected(TRI_voc_fid_t fid, int64_t number) {
WRITE_LOCKER(writeLocker, _lock);
auto it = _stats.find(fid);
if (it == _stats.end()) {
// datafile not found. no problem
return;
}
auto& dst = (*it).second;
dst->numberUncollected += number;
LOG_TOPIC("5410f", TRACE, arangodb::Logger::DATAFILES)
<< "increasing uncollected count for datafile " << fid;
}
/// @brief return a copy of the datafile statistics for a file
MMFilesDatafileStatisticsContainer MMFilesDatafileStatistics::get(TRI_voc_fid_t fid) {
MMFilesDatafileStatisticsContainer result;
{
READ_LOCKER(readLocker, _lock);
auto it = _stats.find(fid);
if (it == _stats.end()) {
LOG_TOPIC("4e682", WARN, arangodb::Logger::DATAFILES)
<< "did not find required statistics for datafile " << fid;
THROW_ARANGO_EXCEPTION_MESSAGE(
TRI_ERROR_ARANGO_DATAFILE_STATISTICS_NOT_FOUND,
"required datafile statistics not found on get");
}
result = *(*it).second;
}
return result;
}
/// @brief return a copy of the datafile statistics for a file
MMFilesDatafileStatisticsContainer MMFilesDatafileStatistics::all() {
MMFilesDatafileStatisticsContainer result;
{
READ_LOCKER(readLocker, _lock);
for (auto& it : _stats) {
result.update(*(it.second));
}
}
return result;
}