2014-05-03 20:02:22 +02:00
|
|
|
/******************************************************************************
|
|
|
|
* Icinga 2 *
|
2018-01-02 12:06:00 +01:00
|
|
|
* Copyright (C) 2012-2018 Icinga Development Team (https://www.icinga.com/) *
|
2014-05-03 20:02:22 +02:00
|
|
|
* *
|
|
|
|
* 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 2 *
|
|
|
|
* 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, write to the Free Software Foundation *
|
|
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
|
|
|
|
******************************************************************************/
|
|
|
|
|
2015-06-22 11:11:21 +02:00
|
|
|
#include "remote/jsonrpcconnection.hpp"
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "remote/apilistener.hpp"
|
|
|
|
#include "remote/apifunction.hpp"
|
|
|
|
#include "remote/jsonrpc.hpp"
|
2015-08-15 20:28:05 +02:00
|
|
|
#include "base/configtype.hpp"
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "base/objectlock.hpp"
|
|
|
|
#include "base/utility.hpp"
|
2014-10-19 14:21:12 +02:00
|
|
|
#include "base/logger.hpp"
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "base/exception.hpp"
|
2016-06-14 08:19:13 +02:00
|
|
|
#include "base/convert.hpp"
|
2015-02-27 20:18:20 +01:00
|
|
|
#include <boost/thread/once.hpp>
|
2014-05-03 20:02:22 +02:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
2015-08-04 14:47:44 +02:00
|
|
|
static Value SetLogPositionHandler(const MessageOrigin::Ptr& origin, const Dictionary::Ptr& params);
|
2014-05-03 20:02:22 +02:00
|
|
|
REGISTER_APIFUNCTION(SetLogPosition, log, &SetLogPositionHandler);
|
|
|
|
|
2015-06-22 11:11:21 +02:00
|
|
|
static boost::once_flag l_JsonRpcConnectionOnceFlag = BOOST_ONCE_INIT;
|
|
|
|
static Timer::Ptr l_JsonRpcConnectionTimeoutTimer;
|
2016-01-27 15:45:58 +01:00
|
|
|
static WorkQueue *l_JsonRpcConnectionWorkQueues;
|
|
|
|
static size_t l_JsonRpcConnectionWorkQueueCount;
|
|
|
|
static int l_JsonRpcConnectionNextID;
|
2017-11-21 14:07:44 +01:00
|
|
|
static Timer::Ptr l_HeartbeatTimer;
|
2015-02-27 20:18:20 +01:00
|
|
|
|
2015-09-22 17:58:12 +02:00
|
|
|
JsonRpcConnection::JsonRpcConnection(const String& identity, bool authenticated,
|
2018-01-04 08:54:18 +01:00
|
|
|
TlsStream::Ptr stream, ConnectionRole role)
|
|
|
|
: m_ID(l_JsonRpcConnectionNextID++), m_Identity(identity), m_Authenticated(authenticated), m_Stream(std::move(stream)),
|
2017-12-19 15:50:05 +01:00
|
|
|
m_Role(role), m_Timestamp(Utility::GetTime()), m_Seen(Utility::GetTime()), m_NextHeartbeat(0), m_HeartbeatTimeout(0)
|
2014-05-08 15:00:09 +02:00
|
|
|
{
|
2015-06-22 11:11:21 +02:00
|
|
|
boost::call_once(l_JsonRpcConnectionOnceFlag, &JsonRpcConnection::StaticInitialize);
|
2015-02-27 20:18:20 +01:00
|
|
|
|
2014-10-16 09:01:18 +02:00
|
|
|
if (authenticated)
|
|
|
|
m_Endpoint = Endpoint::GetByName(identity);
|
2014-05-08 15:00:09 +02:00
|
|
|
}
|
2014-05-03 20:02:22 +02:00
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
void JsonRpcConnection::StaticInitialize()
|
2015-02-27 20:18:20 +01:00
|
|
|
{
|
2015-06-22 11:11:21 +02:00
|
|
|
l_JsonRpcConnectionTimeoutTimer = new Timer();
|
2017-11-21 11:52:55 +01:00
|
|
|
l_JsonRpcConnectionTimeoutTimer->OnTimerExpired.connect(std::bind(&JsonRpcConnection::TimeoutTimerHandler));
|
2015-06-22 11:11:21 +02:00
|
|
|
l_JsonRpcConnectionTimeoutTimer->SetInterval(15);
|
|
|
|
l_JsonRpcConnectionTimeoutTimer->Start();
|
2016-01-27 15:45:58 +01:00
|
|
|
|
|
|
|
l_JsonRpcConnectionWorkQueueCount = Application::GetConcurrency();
|
|
|
|
l_JsonRpcConnectionWorkQueues = new WorkQueue[l_JsonRpcConnectionWorkQueueCount];
|
2016-06-14 08:19:13 +02:00
|
|
|
|
2016-08-24 19:59:13 +02:00
|
|
|
for (size_t i = 0; i < l_JsonRpcConnectionWorkQueueCount; i++) {
|
2016-06-14 08:19:13 +02:00
|
|
|
l_JsonRpcConnectionWorkQueues[i].SetName("JsonRpcConnection, #" + Convert::ToString(i));
|
|
|
|
}
|
2017-11-21 14:07:44 +01:00
|
|
|
|
|
|
|
l_HeartbeatTimer = new Timer();
|
|
|
|
l_HeartbeatTimer->OnTimerExpired.connect(std::bind(&JsonRpcConnection::HeartbeatTimerHandler));
|
|
|
|
l_HeartbeatTimer->SetInterval(10);
|
|
|
|
l_HeartbeatTimer->Start();
|
2015-02-27 20:18:20 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
void JsonRpcConnection::Start()
|
2014-05-03 20:02:22 +02:00
|
|
|
{
|
2015-11-02 17:34:01 +01:00
|
|
|
/* the stream holds an owning reference to this object through the callback we're registering here */
|
2017-11-21 11:52:55 +01:00
|
|
|
m_Stream->RegisterDataHandler(std::bind(&JsonRpcConnection::DataAvailableHandler, JsonRpcConnection::Ptr(this)));
|
2015-06-24 09:44:59 +02:00
|
|
|
if (m_Stream->IsDataAvailable())
|
|
|
|
DataAvailableHandler();
|
2014-05-03 20:02:22 +02:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
double JsonRpcConnection::GetTimestamp() const
|
2016-01-25 10:57:06 +01:00
|
|
|
{
|
|
|
|
return m_Timestamp;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String JsonRpcConnection::GetIdentity() const
|
2014-05-08 15:00:09 +02:00
|
|
|
{
|
|
|
|
return m_Identity;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
bool JsonRpcConnection::IsAuthenticated() const
|
2014-10-16 09:01:18 +02:00
|
|
|
{
|
|
|
|
return m_Authenticated;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
Endpoint::Ptr JsonRpcConnection::GetEndpoint() const
|
2014-05-03 20:02:22 +02:00
|
|
|
{
|
|
|
|
return m_Endpoint;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
TlsStream::Ptr JsonRpcConnection::GetStream() const
|
2014-05-03 20:02:22 +02:00
|
|
|
{
|
|
|
|
return m_Stream;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
ConnectionRole JsonRpcConnection::GetRole() const
|
2014-05-03 20:02:22 +02:00
|
|
|
{
|
|
|
|
return m_Role;
|
|
|
|
}
|
|
|
|
|
2015-06-22 11:11:21 +02:00
|
|
|
void JsonRpcConnection::SendMessage(const Dictionary::Ptr& message)
|
2014-05-03 20:02:22 +02:00
|
|
|
{
|
|
|
|
try {
|
2014-07-01 08:57:57 +02:00
|
|
|
ObjectLock olock(m_Stream);
|
2014-09-09 15:12:39 +02:00
|
|
|
if (m_Stream->IsEof())
|
|
|
|
return;
|
2017-11-13 16:30:29 +01:00
|
|
|
size_t bytesSent = JsonRpc::SendMessage(m_Stream, message);
|
|
|
|
m_Endpoint->AddMessageSent(bytesSent);
|
2014-05-03 20:02:22 +02:00
|
|
|
} catch (const std::exception& ex) {
|
2014-10-19 17:52:17 +02:00
|
|
|
std::ostringstream info;
|
2014-05-22 16:07:14 +02:00
|
|
|
info << "Error while sending JSON-RPC message for identity '" << m_Identity << "'";
|
2015-06-22 11:11:21 +02:00
|
|
|
Log(LogWarning, "JsonRpcConnection")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< info.str() << "\n" << DiagnosticInformation(ex);
|
2014-05-03 20:02:22 +02:00
|
|
|
|
2015-11-02 17:34:01 +01:00
|
|
|
Disconnect();
|
2014-05-03 20:02:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
void JsonRpcConnection::Disconnect()
|
2014-05-03 20:02:22 +02:00
|
|
|
{
|
2015-06-22 11:11:21 +02:00
|
|
|
Log(LogWarning, "JsonRpcConnection")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "API client disconnected for identity '" << m_Identity << "'";
|
2014-05-08 15:00:09 +02:00
|
|
|
|
2016-08-22 16:35:16 +02:00
|
|
|
m_Stream->Close();
|
2015-09-29 16:03:38 +02:00
|
|
|
|
2014-05-08 15:00:09 +02:00
|
|
|
if (m_Endpoint)
|
2014-11-08 21:17:16 +01:00
|
|
|
m_Endpoint->RemoveClient(this);
|
2014-05-08 15:12:56 +02:00
|
|
|
else {
|
|
|
|
ApiListener::Ptr listener = ApiListener::GetInstance();
|
2014-11-08 21:17:16 +01:00
|
|
|
listener->RemoveAnonymousClient(this);
|
2014-05-08 15:12:56 +02:00
|
|
|
}
|
2014-05-03 20:02:22 +02:00
|
|
|
}
|
|
|
|
|
2016-01-27 16:43:23 +01:00
|
|
|
void JsonRpcConnection::MessageHandlerWrapper(const String& jsonString)
|
2014-05-03 20:02:22 +02:00
|
|
|
{
|
2016-01-27 16:43:23 +01:00
|
|
|
if (m_Stream->IsEof())
|
|
|
|
return;
|
|
|
|
|
|
|
|
try {
|
|
|
|
MessageHandler(jsonString);
|
|
|
|
} catch (const std::exception& ex) {
|
|
|
|
Log(LogWarning, "JsonRpcConnection")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Error while reading JSON-RPC message for identity '" << m_Identity
|
|
|
|
<< "': " << DiagnosticInformation(ex);
|
2016-01-27 16:43:23 +01:00
|
|
|
|
|
|
|
Disconnect();
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void JsonRpcConnection::MessageHandler(const String& jsonString)
|
|
|
|
{
|
|
|
|
Dictionary::Ptr message = JsonRpc::DecodeMessage(jsonString);
|
|
|
|
|
2015-02-26 12:41:47 +01:00
|
|
|
m_Seen = Utility::GetTime();
|
2014-05-03 20:02:22 +02:00
|
|
|
|
2015-03-11 12:53:43 +01:00
|
|
|
if (m_HeartbeatTimeout != 0)
|
|
|
|
m_NextHeartbeat = Utility::GetTime() + m_HeartbeatTimeout;
|
|
|
|
|
2014-05-08 15:00:09 +02:00
|
|
|
if (m_Endpoint && message->Contains("ts")) {
|
2014-05-03 20:02:22 +02:00
|
|
|
double ts = message->Get("ts");
|
|
|
|
|
|
|
|
/* ignore old messages */
|
|
|
|
if (ts < m_Endpoint->GetRemoteLogPosition())
|
2016-01-27 15:45:58 +01:00
|
|
|
return;
|
2014-05-03 20:02:22 +02:00
|
|
|
|
|
|
|
m_Endpoint->SetRemoteLogPosition(ts);
|
|
|
|
}
|
|
|
|
|
2015-08-04 14:47:44 +02:00
|
|
|
MessageOrigin::Ptr origin = new MessageOrigin();
|
|
|
|
origin->FromClient = this;
|
2014-05-03 20:02:22 +02:00
|
|
|
|
2014-05-08 15:00:09 +02:00
|
|
|
if (m_Endpoint) {
|
|
|
|
if (m_Endpoint->GetZone() != Zone::GetLocalZone())
|
2015-08-04 14:47:44 +02:00
|
|
|
origin->FromZone = m_Endpoint->GetZone();
|
2014-05-08 15:00:09 +02:00
|
|
|
else
|
2015-08-04 14:47:44 +02:00
|
|
|
origin->FromZone = Zone::GetByName(message->Get("originZone"));
|
2017-11-13 16:30:29 +01:00
|
|
|
|
|
|
|
m_Endpoint->AddMessageReceived(jsonString.GetLength());
|
2014-05-08 15:00:09 +02:00
|
|
|
}
|
2014-05-03 20:02:22 +02:00
|
|
|
|
2017-08-29 14:37:13 +02:00
|
|
|
Value vmethod;
|
|
|
|
|
|
|
|
if (!message->Get("method", &vmethod)) {
|
|
|
|
Value vid;
|
|
|
|
|
|
|
|
if (!message->Get("id", &vid))
|
|
|
|
return;
|
|
|
|
|
2017-09-07 10:39:00 +02:00
|
|
|
Log(LogWarning, "JsonRpcConnection",
|
2017-12-19 15:50:05 +01:00
|
|
|
"We received a JSON-RPC response message. This should never happen because we're only ever sending notifications.");
|
2017-08-29 14:37:13 +02:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
String method = vmethod;
|
2014-05-03 20:02:22 +02:00
|
|
|
|
2015-06-22 11:11:21 +02:00
|
|
|
Log(LogNotice, "JsonRpcConnection")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Received '" << method << "' message from '" << m_Identity << "'";
|
2014-05-03 20:02:22 +02:00
|
|
|
|
2014-11-08 21:17:16 +01:00
|
|
|
Dictionary::Ptr resultMessage = new Dictionary();
|
2014-05-03 20:02:22 +02:00
|
|
|
|
|
|
|
try {
|
|
|
|
ApiFunction::Ptr afunc = ApiFunction::GetByName(method);
|
|
|
|
|
2017-11-27 12:09:42 +01:00
|
|
|
if (!afunc) {
|
|
|
|
Log(LogNotice, "JsonRpcConnection")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Call to non-existent function '" << method << "' from endpoint '" << m_Identity << "'.";
|
2017-11-27 12:09:42 +01:00
|
|
|
} else {
|
2017-10-18 11:08:55 +02:00
|
|
|
resultMessage->Set("result", afunc->Invoke(origin, message->Get("params")));
|
2017-11-27 12:09:42 +01:00
|
|
|
}
|
2014-08-25 08:35:35 +02:00
|
|
|
} catch (const std::exception& ex) {
|
2015-09-22 17:58:12 +02:00
|
|
|
/* TODO: Add a user readable error message for the remote caller */
|
2017-08-30 15:12:24 +02:00
|
|
|
String diagInfo = DiagnosticInformation(ex);
|
|
|
|
resultMessage->Set("error", diagInfo);
|
2015-06-22 11:11:21 +02:00
|
|
|
Log(LogWarning, "JsonRpcConnection")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Error while processing message for identity '" << m_Identity << "'\n" << diagInfo;
|
2014-05-03 20:02:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (message->Contains("id")) {
|
|
|
|
resultMessage->Set("jsonrpc", "2.0");
|
|
|
|
resultMessage->Set("id", message->Get("id"));
|
2016-02-01 11:25:34 +01:00
|
|
|
SendMessage(resultMessage);
|
2014-05-03 20:02:22 +02:00
|
|
|
}
|
2016-01-27 15:45:58 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
bool JsonRpcConnection::ProcessMessage()
|
2016-01-27 15:45:58 +01:00
|
|
|
{
|
2016-01-27 16:43:23 +01:00
|
|
|
String message;
|
2016-01-27 15:45:58 +01:00
|
|
|
|
|
|
|
StreamReadStatus srs = JsonRpc::ReadMessage(m_Stream, &message, m_Context, false);
|
|
|
|
|
|
|
|
if (srs != StatusNewItem)
|
|
|
|
return false;
|
|
|
|
|
2017-11-21 11:52:55 +01:00
|
|
|
l_JsonRpcConnectionWorkQueues[m_ID % l_JsonRpcConnectionWorkQueueCount].Enqueue(std::bind(&JsonRpcConnection::MessageHandlerWrapper, JsonRpcConnection::Ptr(this), message));
|
2014-05-03 20:02:22 +02:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
void JsonRpcConnection::DataAvailableHandler()
|
2014-05-03 20:02:22 +02:00
|
|
|
{
|
2016-02-01 08:35:55 +01:00
|
|
|
bool close = false;
|
2015-06-24 09:44:59 +02:00
|
|
|
|
2016-08-19 20:35:20 +02:00
|
|
|
if (!m_Stream)
|
|
|
|
return;
|
|
|
|
|
2016-02-01 08:35:55 +01:00
|
|
|
if (!m_Stream->IsEof()) {
|
|
|
|
boost::mutex::scoped_lock lock(m_DataHandlerMutex);
|
2014-08-22 15:39:34 +02:00
|
|
|
|
2016-02-01 08:35:55 +01:00
|
|
|
try {
|
|
|
|
while (ProcessMessage())
|
|
|
|
; /* empty loop body */
|
|
|
|
} catch (const std::exception& ex) {
|
|
|
|
Log(LogWarning, "JsonRpcConnection")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Error while reading JSON-RPC message for identity '" << m_Identity
|
|
|
|
<< "': " << DiagnosticInformation(ex);
|
2016-01-13 10:30:38 +01:00
|
|
|
|
2016-02-01 08:35:55 +01:00
|
|
|
Disconnect();
|
2016-01-13 10:30:38 +01:00
|
|
|
|
2016-02-01 08:35:55 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
close = true;
|
|
|
|
|
|
|
|
if (close)
|
2016-01-13 10:30:38 +01:00
|
|
|
Disconnect();
|
2014-05-03 20:02:22 +02:00
|
|
|
}
|
|
|
|
|
2015-08-04 14:47:44 +02:00
|
|
|
Value SetLogPositionHandler(const MessageOrigin::Ptr& origin, const Dictionary::Ptr& params)
|
2014-05-03 20:02:22 +02:00
|
|
|
{
|
|
|
|
if (!params)
|
|
|
|
return Empty;
|
|
|
|
|
|
|
|
double log_position = params->Get("log_position");
|
2015-08-04 14:47:44 +02:00
|
|
|
Endpoint::Ptr endpoint = origin->FromClient->GetEndpoint();
|
2014-05-03 20:02:22 +02:00
|
|
|
|
2014-05-08 15:00:09 +02:00
|
|
|
if (!endpoint)
|
|
|
|
return Empty;
|
|
|
|
|
2014-05-03 20:02:22 +02:00
|
|
|
if (log_position > endpoint->GetLocalLogPosition())
|
|
|
|
endpoint->SetLocalLogPosition(log_position);
|
|
|
|
|
|
|
|
return Empty;
|
|
|
|
}
|
2014-10-16 12:27:09 +02:00
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
void JsonRpcConnection::CheckLiveness()
|
2015-02-26 12:41:47 +01:00
|
|
|
{
|
2015-02-27 14:09:46 +01:00
|
|
|
if (m_Seen < Utility::GetTime() - 60 && (!m_Endpoint || !m_Endpoint->GetSyncing())) {
|
2015-06-22 11:11:21 +02:00
|
|
|
Log(LogInformation, "JsonRpcConnection")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "No messages for identity '" << m_Identity << "' have been received in the last 60 seconds.";
|
2015-02-27 20:18:20 +01:00
|
|
|
Disconnect();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
void JsonRpcConnection::TimeoutTimerHandler()
|
2015-02-27 20:18:20 +01:00
|
|
|
{
|
|
|
|
ApiListener::Ptr listener = ApiListener::GetInstance();
|
|
|
|
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const JsonRpcConnection::Ptr& client : listener->GetAnonymousClients()) {
|
2015-02-27 20:18:20 +01:00
|
|
|
client->CheckLiveness();
|
|
|
|
}
|
|
|
|
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Endpoint::Ptr& endpoint : ConfigType::GetObjectsByType<Endpoint>()) {
|
|
|
|
for (const JsonRpcConnection::Ptr& client : endpoint->GetClients()) {
|
2015-02-27 20:18:20 +01:00
|
|
|
client->CheckLiveness();
|
|
|
|
}
|
2015-02-26 13:09:13 +01:00
|
|
|
}
|
2015-02-26 12:41:47 +01:00
|
|
|
}
|
2015-09-22 17:58:12 +02:00
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
size_t JsonRpcConnection::GetWorkQueueCount()
|
2017-05-11 17:30:20 +02:00
|
|
|
{
|
|
|
|
return l_JsonRpcConnectionWorkQueueCount;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
size_t JsonRpcConnection::GetWorkQueueLength()
|
2017-05-11 17:30:20 +02:00
|
|
|
{
|
2017-12-14 08:47:04 +01:00
|
|
|
size_t itemCount = 0;
|
2017-05-11 17:30:20 +02:00
|
|
|
|
2017-12-14 08:47:04 +01:00
|
|
|
for (size_t i = 0; i < GetWorkQueueCount(); i++)
|
2017-05-11 17:30:20 +02:00
|
|
|
itemCount += l_JsonRpcConnectionWorkQueues[i].GetLength();
|
|
|
|
|
|
|
|
return itemCount;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
double JsonRpcConnection::GetWorkQueueRate()
|
2017-05-11 17:30:20 +02:00
|
|
|
{
|
|
|
|
double rate = 0.0;
|
2017-12-14 08:47:04 +01:00
|
|
|
size_t count = GetWorkQueueCount();
|
2017-05-11 17:30:20 +02:00
|
|
|
|
2017-05-26 17:02:36 +02:00
|
|
|
/* If this is a standalone environment, we don't have any queues. */
|
|
|
|
if (count == 0)
|
|
|
|
return 0.0;
|
|
|
|
|
2017-12-14 08:47:04 +01:00
|
|
|
for (size_t i = 0; i < count; i++)
|
2017-05-11 17:30:20 +02:00
|
|
|
rate += l_JsonRpcConnectionWorkQueues[i].GetTaskCount(60) / 60.0;
|
|
|
|
|
2017-05-26 17:02:36 +02:00
|
|
|
return rate / count;
|
2017-05-11 17:30:20 +02:00
|
|
|
}
|
|
|
|
|