2019-02-25 14:48:22 +01:00
|
|
|
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
2013-03-10 15:11:32 +01:00
|
|
|
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "livestatus/attributefilter.hpp"
|
|
|
|
#include "base/convert.hpp"
|
|
|
|
#include "base/array.hpp"
|
|
|
|
#include "base/objectlock.hpp"
|
2014-10-19 14:21:12 +02:00
|
|
|
#include "base/logger.hpp"
|
2013-11-05 17:05:41 +01:00
|
|
|
#include <boost/regex.hpp>
|
2015-02-04 21:52:02 +01:00
|
|
|
#include <boost/algorithm/string/predicate.hpp>
|
2013-03-10 15:11:32 +01:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
2018-01-04 08:54:18 +01:00
|
|
|
AttributeFilter::AttributeFilter(String column, String op, String operand)
|
|
|
|
: m_Column(std::move(column)), m_Operator(std::move(op)), m_Operand(std::move(operand))
|
2013-03-10 15:11:32 +01:00
|
|
|
{ }
|
|
|
|
|
2013-07-10 16:11:40 +02:00
|
|
|
bool AttributeFilter::Apply(const Table::Ptr& table, const Value& row)
|
2013-03-10 15:11:32 +01:00
|
|
|
{
|
2013-03-10 18:49:14 +01:00
|
|
|
Column column = table->GetColumn(m_Column);
|
2013-03-10 15:11:32 +01:00
|
|
|
|
2013-07-10 16:11:40 +02:00
|
|
|
Value value = column.ExtractValue(row);
|
2013-03-10 15:11:32 +01:00
|
|
|
|
|
|
|
if (value.IsObjectType<Array>()) {
|
2013-07-22 14:26:18 +02:00
|
|
|
Array::Ptr array = value;
|
|
|
|
|
2014-06-25 11:30:27 +02:00
|
|
|
if (m_Operator == ">=" || m_Operator == "<") {
|
|
|
|
bool negate = (m_Operator == "<");
|
|
|
|
|
2013-07-16 15:38:59 +02:00
|
|
|
ObjectLock olock(array);
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const String& item : array) {
|
2013-03-10 15:11:32 +01:00
|
|
|
if (item == m_Operand)
|
2014-06-25 11:30:27 +02:00
|
|
|
return !negate; /* Item found in list. */
|
2013-03-10 15:11:32 +01:00
|
|
|
}
|
|
|
|
|
2014-06-25 11:30:27 +02:00
|
|
|
return negate; /* Item not found in list. */
|
2013-07-22 14:26:18 +02:00
|
|
|
} else if (m_Operator == "=") {
|
|
|
|
return (array->GetLength() == 0);
|
2013-03-10 15:11:32 +01:00
|
|
|
} else {
|
2013-07-22 14:26:18 +02:00
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid operator for column '" + m_Column + "': " + m_Operator + " (expected '>=' or '=')."));
|
2013-03-10 15:11:32 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (m_Operator == "=") {
|
2014-12-10 09:04:49 +01:00
|
|
|
if (value.GetType() == ValueNumber || value.GetType() == ValueBoolean)
|
2013-03-10 15:11:32 +01:00
|
|
|
return (static_cast<double>(value) == Convert::ToDouble(m_Operand));
|
|
|
|
else
|
|
|
|
return (static_cast<String>(value) == m_Operand);
|
|
|
|
} else if (m_Operator == "~") {
|
2014-06-05 16:13:28 +02:00
|
|
|
bool ret;
|
|
|
|
try {
|
|
|
|
boost::regex expr(m_Operand.GetData());
|
|
|
|
String operand = value;
|
|
|
|
boost::smatch what;
|
|
|
|
ret = boost::regex_search(operand.GetData(), what, expr);
|
|
|
|
} catch (boost::exception&) {
|
2014-10-19 17:52:17 +02:00
|
|
|
Log(LogWarning, "AttributeFilter")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Regex '" << m_Operand << " " << m_Operator << " " << value << "' error.";
|
2014-06-05 16:13:28 +02:00
|
|
|
ret = false;
|
|
|
|
}
|
2013-11-05 17:05:41 +01:00
|
|
|
|
2014-10-19 17:52:17 +02:00
|
|
|
//Log(LogDebug, "LivestatusListener/AttributeFilter")
|
|
|
|
// << "Attribute filter '" << m_Operand + " " << m_Operator << " "
|
|
|
|
// << value << "' " << (ret ? "matches" : "doesn't match") << ".";
|
2013-11-05 17:05:41 +01:00
|
|
|
|
|
|
|
return ret;
|
2013-03-10 15:11:32 +01:00
|
|
|
} else if (m_Operator == "=~") {
|
2015-02-04 21:52:02 +01:00
|
|
|
bool ret;
|
|
|
|
try {
|
|
|
|
String operand = value;
|
|
|
|
ret = boost::iequals(operand, m_Operand.GetData());
|
|
|
|
} catch (boost::exception&) {
|
|
|
|
Log(LogWarning, "AttributeFilter")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Case-insensitive equality '" << m_Operand << " " << m_Operator << " " << value << "' error.";
|
2015-02-04 21:52:02 +01:00
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2013-03-10 15:11:32 +01:00
|
|
|
} else if (m_Operator == "~~") {
|
2014-06-05 16:13:28 +02:00
|
|
|
bool ret;
|
|
|
|
try {
|
|
|
|
boost::regex expr(m_Operand.GetData(), boost::regex::icase);
|
|
|
|
String operand = value;
|
|
|
|
boost::smatch what;
|
|
|
|
ret = boost::regex_search(operand.GetData(), what, expr);
|
|
|
|
} catch (boost::exception&) {
|
2014-10-19 17:52:17 +02:00
|
|
|
Log(LogWarning, "AttributeFilter")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Regex '" << m_Operand << " " << m_Operator << " " << value << "' error.";
|
2014-06-05 16:13:28 +02:00
|
|
|
ret = false;
|
|
|
|
}
|
2013-11-05 17:05:41 +01:00
|
|
|
|
2014-10-19 17:52:17 +02:00
|
|
|
//Log(LogDebug, "LivestatusListener/AttributeFilter")
|
|
|
|
// << "Attribute filter '" << m_Operand << " " << m_Operator << " "
|
|
|
|
// << value << "' " << (ret ? "matches" : "doesn't match") << ".";
|
2013-03-10 15:11:32 +01:00
|
|
|
|
2013-11-05 17:05:41 +01:00
|
|
|
return ret;
|
2013-03-10 15:11:32 +01:00
|
|
|
} else if (m_Operator == "<") {
|
|
|
|
if (value.GetType() == ValueNumber)
|
|
|
|
return (static_cast<double>(value) < Convert::ToDouble(m_Operand));
|
|
|
|
else
|
|
|
|
return (static_cast<String>(value) < m_Operand);
|
|
|
|
} else if (m_Operator == ">") {
|
|
|
|
if (value.GetType() == ValueNumber)
|
|
|
|
return (static_cast<double>(value) > Convert::ToDouble(m_Operand));
|
|
|
|
else
|
|
|
|
return (static_cast<String>(value) > m_Operand);
|
|
|
|
} else if (m_Operator == "<=") {
|
|
|
|
if (value.GetType() == ValueNumber)
|
|
|
|
return (static_cast<double>(value) <= Convert::ToDouble(m_Operand));
|
|
|
|
else
|
|
|
|
return (static_cast<String>(value) <= m_Operand);
|
|
|
|
} else if (m_Operator == ">=") {
|
|
|
|
if (value.GetType() == ValueNumber)
|
|
|
|
return (static_cast<double>(value) >= Convert::ToDouble(m_Operand));
|
|
|
|
else
|
|
|
|
return (static_cast<String>(value) >= m_Operand);
|
|
|
|
} else {
|
2013-03-16 21:18:53 +01:00
|
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Unknown operator for column '" + m_Column + "': " + m_Operator));
|
2013-03-10 15:11:32 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|