/************************************************************************
* EOS - the CERN Disk Storage System *
* Copyright (C) 2017 CERN/Switzerland *
* *
* This program is free software: you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation, either version 3 of the License, or *
* (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program. If not, see .*
************************************************************************/
#include
#include
#include
#include
#include
#include "qclient/BackgroundFlusher.hh"
#include "qclient/RocksDBPersistency.hh"
#include "namespace/ns_quarkdb/flusher/MetadataFlusher.hh"
#include "namespace/ns_quarkdb/QdbContactDetails.hh"
#include "common/Logging.hh"
#include
#include
#include "qclient/AssistedThread.hh"
#define __PRI64_PREFIX "l"
#define PRId64 __PRI64_PREFIX "d"
EOSNSNAMESPACE_BEGIN
//------------------------------------------------------------------------------
// Constructor
//------------------------------------------------------------------------------
MetadataFlusher::MetadataFlusher(const std::string& path,
const QdbContactDetails& contactDetails) :
id(basename(path.c_str())),
notifier(*this),
backgroundFlusher(contactDetails.members, contactDetails.constructOptions(),
notifier, new qclient::RocksDBPersistency(path)),
sizePrinter(&MetadataFlusher::queueSizeMonitoring, this)
{
synchronize();
}
//------------------------------------------------------------------------------
// Destructor
//------------------------------------------------------------------------------
MetadataFlusher::~MetadataFlusher()
{
sizePrinter.join();
synchronize();
}
//------------------------------------------------------------------------------
// Regularly print queue statistics
//------------------------------------------------------------------------------
void MetadataFlusher::queueSizeMonitoring(qclient::ThreadAssistant& assistant)
{
while (!assistant.terminationRequested()) {
if (backgroundFlusher.size()) {
eos_static_info("id=%s total-pending=%" PRId64 " enqueued=%" PRId64
" acknowledged=%" PRId64,
id.c_str(), backgroundFlusher.size(),
backgroundFlusher.getEnqueuedAndClear(),
backgroundFlusher.getAcknowledgedAndClear());
}
assistant.wait_for(std::chrono::seconds(10));
}
}
//------------------------------------------------------------------------------
// Queue an hset command
//------------------------------------------------------------------------------
void MetadataFlusher::hset(const std::string& key, const std::string& field,
const std::string& value)
{
backgroundFlusher.pushRequest({"HSET", key, field, value});
}
//------------------------------------------------------------------------------
// Queue an hincrby command
//------------------------------------------------------------------------------
void MetadataFlusher::hincrby(const std::string& key, const std::string& field,
int64_t value)
{
backgroundFlusher.pushRequest({"HINCRBY", key, field, std::to_string(value)});
}
//------------------------------------------------------------------------------
// Queue a del command
//------------------------------------------------------------------------------
void MetadataFlusher::del(const std::string& key)
{
backgroundFlusher.pushRequest({"DEL", key});
}
//------------------------------------------------------------------------------
// Queue an hdel command
//------------------------------------------------------------------------------
void MetadataFlusher::hdel(const std::string& key, const std::string& field)
{
backgroundFlusher.pushRequest({"HDEL", key, field});
}
//------------------------------------------------------------------------------
// Queue a sadd command
//------------------------------------------------------------------------------
void MetadataFlusher::sadd(const std::string& key, const std::string& field)
{
backgroundFlusher.pushRequest({"SADD", key, field});
}
//------------------------------------------------------------------------------
// Queue an srem command
//------------------------------------------------------------------------------
void MetadataFlusher::srem(const std::string& key, const std::string& field)
{
backgroundFlusher.pushRequest({"SREM", key, field});
}
//------------------------------------------------------------------------------
// Queue an srem command, use a list as contents
//------------------------------------------------------------------------------
void MetadataFlusher::srem(const std::string& key,
const std::list& items)
{
std::vector req = {"SREM", key};
for (auto it = items.begin(); it != items.end(); it++) {
req.emplace_back(*it);
}
backgroundFlusher.pushRequest(req);
}
//------------------------------------------------------------------------------
// Sleep until given index has been flushed to the backend
//------------------------------------------------------------------------------
void MetadataFlusher::synchronize(ItemIndex targetIndex)
{
if (targetIndex < 0) {
targetIndex = backgroundFlusher.getEndingIndex() - 1;
}
eos_static_info("starting-index=%" PRId64 " ending-index=%" PRId64
" msg=\"waiting until "
"queue item %" PRId64 " has been acknowledged..\"",
backgroundFlusher.getStartingIndex(),
backgroundFlusher.getEndingIndex(), targetIndex);
while (!backgroundFlusher.waitForIndex(targetIndex, std::chrono::seconds(1))) {
eos_static_warning("starting-index=%" PRId64 " ending-index=%" PRId64
" msg=\"queue item "
"%" PRId64 " has not been acknowledged yet..\"",
backgroundFlusher.getStartingIndex(),
backgroundFlusher.getEndingIndex(), targetIndex);
}
eos_static_info("starting-index=%" PRId64 " ending-index=%" PRId64
" msg=\"queue item %" PRId64
" has been acknowledged\"", backgroundFlusher.getStartingIndex(),
backgroundFlusher.getEndingIndex(), targetIndex);
}
//------------------------------------------------------------------------------
// Class to receive notifications from the BackgroundFlusher
//------------------------------------------------------------------------------
FlusherNotifier::FlusherNotifier(MetadataFlusher& flusher):
mFlusher(flusher)
{
(void) mFlusher; // avoid compilation warning
}
//------------------------------------------------------------------------------
// Record network events
//------------------------------------------------------------------------------
void FlusherNotifier::eventNetworkIssue(const std::string& err)
{
eos_static_notice("Network issue when contacting the redis backend: %s",
err.c_str());
}
//------------------------------------------------------------------------------
// Record unexpected responses
//------------------------------------------------------------------------------
void FlusherNotifier::eventUnexpectedResponse(const std::string& err)
{
eos_static_crit("Unexpected response when contacting the redis backend: %s",
err.c_str());
// Maybe we should just std::terminate now?
}
EOSNSNAMESPACE_END