blob: 2bd62a0ae14449cbda2f4d3546d11ffd20779055 [file] [log] [blame]
#include "Thresholds.hpp"
#include "VariantVisitors.hpp"
#include "sensor.hpp"
#include <array>
#include <boost/algorithm/string/replace.hpp>
#include <boost/container/flat_map.hpp>
#include <boost/lexical_cast.hpp>
#include <cmath>
#include <fstream>
#include <iostream>
#include <memory>
#include <stdexcept>
#include <string>
#include <utility>
#include <variant>
#include <vector>
static constexpr bool DEBUG = false;
namespace thresholds
{
unsigned int toBusValue(const Level& level)
{
switch (level)
{
case (Level::WARNING):
{
return 0;
}
case (Level::CRITICAL):
{
return 1;
}
default:
{
return -1;
}
}
}
std::string toBusValue(const Direction& direction)
{
switch (direction)
{
case (Direction::LOW):
{
return "less than";
}
case (Direction::HIGH):
{
return "greater than";
}
default:
{
return "err";
}
}
}
bool parseThresholdsFromConfig(
const SensorData& sensorData,
std::vector<thresholds::Threshold>& thresholdVector,
const std::string* matchLabel)
{
for (const auto& item : sensorData)
{
if (item.first.find("Thresholds") == std::string::npos)
{
continue;
}
if (matchLabel != nullptr)
{
auto labelFind = item.second.find("Label");
if (labelFind == item.second.end())
continue;
if (std::visit(VariantToStringVisitor(), labelFind->second) !=
*matchLabel)
continue;
}
auto directionFind = item.second.find("Direction");
auto severityFind = item.second.find("Severity");
auto valueFind = item.second.find("Value");
if (valueFind == item.second.end() ||
severityFind == item.second.end() ||
directionFind == item.second.end())
{
std::cerr << "Malformed threshold in configuration\n";
return false;
}
Level level;
Direction direction;
if (std::visit(VariantToUnsignedIntVisitor(), severityFind->second) ==
0)
{
level = Level::WARNING;
}
else
{
level = Level::CRITICAL;
}
if (std::visit(VariantToStringVisitor(), directionFind->second) ==
"less than")
{
direction = Direction::LOW;
}
else
{
direction = Direction::HIGH;
}
double val = std::visit(VariantToDoubleVisitor(), valueFind->second);
thresholdVector.emplace_back(level, direction, val);
}
return true;
}
void persistThreshold(const std::string& path, const std::string& baseInterface,
const thresholds::Threshold& threshold,
std::shared_ptr<sdbusplus::asio::connection>& conn,
size_t thresholdCount)
{
for (size_t ii = 0; ii < thresholdCount; ii++)
{
std::string thresholdInterface =
baseInterface + ".Thresholds" + std::to_string(ii);
conn->async_method_call(
[&, path, threshold, thresholdInterface](
const boost::system::error_code& ec,
const boost::container::flat_map<std::string, BasicVariantType>&
result) {
if (ec)
{
return; // threshold not supported
}
auto directionFind = result.find("Direction");
auto severityFind = result.find("Severity");
auto valueFind = result.find("Value");
if (valueFind == result.end() || severityFind == result.end() ||
directionFind == result.end())
{
std::cerr << "Malformed threshold in configuration\n";
return;
}
unsigned int level = std::visit(VariantToUnsignedIntVisitor(),
severityFind->second);
std::string dir =
std::visit(VariantToStringVisitor(), directionFind->second);
if ((toBusValue(threshold.level) != level) ||
(toBusValue(threshold.direction) != dir))
{
return; // not the droid we're looking for
}
std::variant<double> value(threshold.value);
conn->async_method_call(
[](const boost::system::error_code& ec) {
if (ec)
{
std::cerr << "Error setting threshold " << ec
<< "\n";
}
},
entityManagerName, path, "org.freedesktop.DBus.Properties",
"Set", thresholdInterface, "Value", value);
},
entityManagerName, path, "org.freedesktop.DBus.Properties",
"GetAll", thresholdInterface);
}
}
void updateThresholds(Sensor* sensor)
{
if (sensor->thresholds.empty())
{
return;
}
for (const auto& threshold : sensor->thresholds)
{
std::shared_ptr<sdbusplus::asio::dbus_interface> interface;
std::string property;
if (threshold.level == thresholds::Level::CRITICAL)
{
interface = sensor->thresholdInterfaceCritical;
if (threshold.direction == thresholds::Direction::HIGH)
{
property = "CriticalHigh";
}
else
{
property = "CriticalLow";
}
}
else if (threshold.level == thresholds::Level::WARNING)
{
interface = sensor->thresholdInterfaceWarning;
if (threshold.direction == thresholds::Direction::HIGH)
{
property = "WarningHigh";
}
else
{
property = "WarningLow";
}
}
else
{
continue;
}
if (!interface)
{
continue;
}
interface->set_property(property, threshold.value);
}
}
static std::vector<std::pair<Threshold, bool>> checkThresholds(Sensor* sensor,
double value)
{
std::vector<std::pair<Threshold, bool>> thresholdChanges;
if (sensor->thresholds.empty())
{
return thresholdChanges;
}
for (auto& threshold : sensor->thresholds)
{
if (threshold.direction == thresholds::Direction::HIGH)
{
if (value >= threshold.value)
{
thresholdChanges.push_back(std::make_pair(threshold, true));
}
else
{
thresholdChanges.push_back(std::make_pair(threshold, false));
}
}
else
{
if (value <= threshold.value)
{
thresholdChanges.push_back(std::make_pair(threshold, true));
}
else
{
thresholdChanges.push_back(std::make_pair(threshold, false));
}
}
}
return thresholdChanges;
}
bool checkThresholds(Sensor* sensor)
{
bool status = true;
std::vector<std::pair<Threshold, bool>> changes =
checkThresholds(sensor, sensor->value);
for (const auto& [threshold, asserted] : changes)
{
assertThresholds(sensor, threshold.level, threshold.direction,
asserted);
if (threshold.level == thresholds::Level::CRITICAL && asserted)
{
status = false;
}
}
return status;
}
void checkThresholdsPowerDelay(Sensor* sensor, ThresholdTimer& thresholdTimer)
{
std::vector<std::pair<Threshold, bool>> changes =
checkThresholds(sensor, sensor->value);
for (const auto& [threshold, asserted] : changes)
{
if (asserted)
{
thresholdTimer.startTimer(threshold);
}
else
{
assertThresholds(sensor, threshold.level, threshold.direction,
false);
}
}
}
void assertThresholds(Sensor* sensor, thresholds::Level level,
thresholds::Direction direction, bool assert)
{
std::string property;
std::shared_ptr<sdbusplus::asio::dbus_interface> interface;
if (level == thresholds::Level::WARNING &&
direction == thresholds::Direction::HIGH)
{
property = "WarningAlarmHigh";
interface = sensor->thresholdInterfaceWarning;
}
else if (level == thresholds::Level::WARNING &&
direction == thresholds::Direction::LOW)
{
property = "WarningAlarmLow";
interface = sensor->thresholdInterfaceWarning;
}
else if (level == thresholds::Level::CRITICAL &&
direction == thresholds::Direction::HIGH)
{
property = "CriticalAlarmHigh";
interface = sensor->thresholdInterfaceCritical;
}
else if (level == thresholds::Level::CRITICAL &&
direction == thresholds::Direction::LOW)
{
property = "CriticalAlarmLow";
interface = sensor->thresholdInterfaceCritical;
}
else
{
std::cerr << "Unknown threshold, level " << level << "direction "
<< direction << "\n";
return;
}
if (!interface)
{
std::cout << "trying to set uninitialized interface\n";
return;
}
interface->set_property(property, assert);
}
static constexpr std::array<const char*, 4> attrTypes = {"lcrit", "min", "max",
"crit"};
bool parseThresholdsFromAttr(
std::vector<thresholds::Threshold>& thresholdVector,
const std::string& inputPath, const double& scaleFactor,
const double& offset)
{
for (const std::string& type : attrTypes)
{
auto attrPath = boost::replace_all_copy(inputPath, "input", type);
std::ifstream attrFile(attrPath);
if (!attrFile.good())
{
continue;
}
std::string attr;
std::getline(attrFile, attr);
attrFile.close();
Level level;
Direction direction;
double val;
try
{
val = std::stod(attr) / scaleFactor;
}
catch (const std::invalid_argument&)
{
return false;
}
if (type == "min" || type == "max")
{
level = Level::WARNING;
}
else
{
level = Level::CRITICAL;
}
if (type == "min" || type == "lcrit")
{
direction = Direction::LOW;
}
else
{
direction = Direction::HIGH;
}
if (type == "crit")
{
val += offset;
}
if (DEBUG)
{
std::cout << "Threshold: " << attrPath << ": " << val << "\n";
}
thresholdVector.emplace_back(level, direction, val);
}
// no thresholds is allowed, not an error so return true.
return true;
}
bool hasCriticalInterface(
const std::vector<thresholds::Threshold>& thresholdVector)
{
for (auto& threshold : thresholdVector)
{
if (threshold.level == Level::CRITICAL)
{
return true;
}
}
return false;
}
bool hasWarningInterface(
const std::vector<thresholds::Threshold>& thresholdVector)
{
for (auto& threshold : thresholdVector)
{
if (threshold.level == Level::WARNING)
{
return true;
}
}
return false;
}
} // namespace thresholds