blob: 31163d180925684f74e0aba3d3e277e65dd4662e [file] [log] [blame]
/*
// Copyright (c) 2018 Intel Corporation
//
// 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.
*/
#include "EntityManager.hpp"
#include "VariantVisitors.hpp"
#include <Overlay.hpp>
#include <Utils.hpp>
#include <VariantVisitors.hpp>
#include <boost/algorithm/string/case_conv.hpp>
#include <boost/algorithm/string/classification.hpp>
#include <boost/algorithm/string/predicate.hpp>
#include <boost/algorithm/string/replace.hpp>
#include <boost/algorithm/string/split.hpp>
#include <boost/container/flat_map.hpp>
#include <boost/container/flat_set.hpp>
#include <boost/range/iterator_range.hpp>
#include <filesystem>
#include <fstream>
#include <iostream>
#include <nlohmann/json.hpp>
#include <regex>
#include <sdbusplus/asio/connection.hpp>
#include <sdbusplus/asio/object_server.hpp>
#include <variant>
constexpr const char* configurationDirectory = PACKAGE_DIR "configurations";
constexpr const char* schemaDirectory = PACKAGE_DIR "configurations/schemas";
constexpr const char* tempConfigDir = "/tmp/configuration/";
constexpr const char* lastConfiguration = "/tmp/configuration/last.json";
constexpr const char* currentConfiguration = "/var/configuration/system.json";
constexpr const char* globalSchema = "global.json";
constexpr const int32_t MAX_MAPPER_DEPTH = 0;
constexpr const bool DEBUG = false;
struct cmp_str
{
bool operator()(const char* a, const char* b) const
{
return std::strcmp(a, b) < 0;
}
};
struct PerformProbe;
// underscore T for collison with dbus c api
enum class probe_type_codes
{
FALSE_T,
TRUE_T,
AND,
OR,
FOUND,
MATCH_ONE
};
const static boost::container::flat_map<const char*, probe_type_codes, cmp_str>
PROBE_TYPES{{{"FALSE", probe_type_codes::FALSE_T},
{"TRUE", probe_type_codes::TRUE_T},
{"AND", probe_type_codes::AND},
{"OR", probe_type_codes::OR},
{"FOUND", probe_type_codes::FOUND},
{"MATCH_ONE", probe_type_codes::MATCH_ONE}}};
static constexpr std::array<const char*, 5> settableInterfaces = {
"FanProfile", "Pid", "Pid.Zone", "Stepwise", "Thresholds"};
using JsonVariantType =
std::variant<std::vector<std::string>, std::vector<double>, std::string,
int64_t, uint64_t, double, int32_t, uint32_t, int16_t,
uint16_t, uint8_t, bool>;
using GetSubTreeType = std::vector<
std::pair<std::string,
std::vector<std::pair<std::string, std::vector<std::string>>>>>;
using ManagedObjectType = boost::container::flat_map<
sdbusplus::message::object_path,
boost::container::flat_map<
std::string,
boost::container::flat_map<std::string, BasicVariantType>>>;
using FoundDeviceT =
std::vector<boost::container::flat_map<std::string, BasicVariantType>>;
boost::container::flat_map<
std::string,
std::vector<boost::container::flat_map<std::string, BasicVariantType>>>
DBUS_PROBE_OBJECTS;
std::vector<std::string> PASSED_PROBES;
// todo: pass this through nicer
std::shared_ptr<sdbusplus::asio::connection> SYSTEM_BUS;
static nlohmann::json lastJson;
const std::regex ILLEGAL_DBUS_PATH_REGEX("[^A-Za-z0-9_.]");
const std::regex ILLEGAL_DBUS_MEMBER_REGEX("[^A-Za-z0-9_]");
void registerCallbacks(boost::asio::io_service& io,
std::vector<sdbusplus::bus::match::match>& dbusMatches,
nlohmann::json& systemConfiguration,
sdbusplus::asio::object_server& objServer);
// calls the mapper to find all exposed objects of an interface type
// and creates a vector<flat_map> that contains all the key value pairs
// getManagedObjects
void findDbusObjects(std::shared_ptr<PerformProbe> probe,
std::shared_ptr<sdbusplus::asio::connection> connection,
std::string& interface)
{
// store reference to pending callbacks so we don't overwhelm services
static boost::container::flat_map<
std::string, std::vector<std::shared_ptr<PerformProbe>>>
pendingProbes;
if (DBUS_PROBE_OBJECTS[interface].size())
{
return;
}
// add shared_ptr to vector of Probes waiting for callback from a specific
// interface to keep alive while waiting for response
std::array<const char*, 1> objects = {interface.c_str()};
std::vector<std::shared_ptr<PerformProbe>>& pending =
pendingProbes[interface];
auto iter = pending.emplace(pending.end(), probe);
// only allow first call to run to not overwhelm processes
if (iter != pending.begin())
{
return;
}
// find all connections in the mapper that expose a specific type
connection->async_method_call(
[connection, interface, probe](boost::system::error_code& ec,
const GetSubTreeType& interfaceSubtree) {
boost::container::flat_set<std::string> interfaceConnections;
if (ec)
{
pendingProbes[interface].clear();
if (ec.value() == ENOENT)
{
return; // wasn't found by mapper
}
std::cerr << "Error communicating to mapper.\n";
// if we can't communicate to the mapper something is very wrong
std::exit(EXIT_FAILURE);
}
else
{
for (auto& object : interfaceSubtree)
{
for (auto& connPair : object.second)
{
interfaceConnections.insert(connPair.first);
}
}
}
if (interfaceConnections.empty())
{
pendingProbes[interface].clear();
return;
}
// get managed objects for all interfaces
for (const auto& conn : interfaceConnections)
{
connection->async_method_call(
[conn,
interface](boost::system::error_code& errc,
const ManagedObjectType& managedInterface) {
if (errc)
{
std::cerr
<< "error getting managed object for device "
<< conn << "\n";
pendingProbes[interface].clear();
return;
}
for (auto& interfaceManagedObj : managedInterface)
{
auto ifaceObjFind =
interfaceManagedObj.second.find(interface);
if (ifaceObjFind !=
interfaceManagedObj.second.end())
{
std::vector<boost::container::flat_map<
std::string, BasicVariantType>>&
dbusObject = DBUS_PROBE_OBJECTS[interface];
dbusObject.emplace_back(ifaceObjFind->second);
}
}
pendingProbes[interface].clear();
},
conn.c_str(), "/", "org.freedesktop.DBus.ObjectManager",
"GetManagedObjects");
}
},
"xyz.openbmc_project.ObjectMapper",
"/xyz/openbmc_project/object_mapper",
"xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", MAX_MAPPER_DEPTH,
objects);
}
// probes dbus interface dictionary for a key with a value that matches a regex
bool probeDbus(const std::string& interface,
const std::map<std::string, nlohmann::json>& matches,
FoundDeviceT& devices, bool& foundProbe)
{
std::vector<boost::container::flat_map<std::string, BasicVariantType>>&
dbusObject = DBUS_PROBE_OBJECTS[interface];
if (dbusObject.empty())
{
foundProbe = false;
return false;
}
foundProbe = true;
bool foundMatch = false;
for (auto& device : dbusObject)
{
bool deviceMatches = true;
for (auto& match : matches)
{
auto deviceValue = device.find(match.first);
if (deviceValue != device.end())
{
switch (match.second.type())
{
case nlohmann::json::value_t::string:
{
std::regex search(match.second.get<std::string>());
std::smatch regMatch;
// convert value to string respresentation
std::string probeValue = std::visit(
VariantToStringVisitor(), deviceValue->second);
if (!std::regex_search(probeValue, regMatch, search))
{
deviceMatches = false;
break;
}
break;
}
case nlohmann::json::value_t::boolean:
case nlohmann::json::value_t::number_unsigned:
{
unsigned int probeValue = std::visit(
VariantToUnsignedIntVisitor(), deviceValue->second);
if (probeValue != match.second.get<unsigned int>())
{
deviceMatches = false;
}
break;
}
case nlohmann::json::value_t::number_integer:
{
int probeValue = std::visit(VariantToIntVisitor(),
deviceValue->second);
if (probeValue != match.second.get<int>())
{
deviceMatches = false;
}
break;
}
case nlohmann::json::value_t::number_float:
{
float probeValue = std::visit(VariantToFloatVisitor(),
deviceValue->second);
if (probeValue != match.second.get<float>())
{
deviceMatches = false;
}
break;
}
default:
{
std::cerr << "unexpected dbus probe type "
<< match.second.type_name() << "\n";
}
}
}
else
{
deviceMatches = false;
break;
}
}
if (deviceMatches)
{
devices.emplace_back(device);
foundMatch = true;
deviceMatches = false; // for next iteration
}
}
return foundMatch;
}
// default probe entry point, iterates a list looking for specific types to
// call specific probe functions
bool probe(
const std::vector<std::string>& probeCommand,
std::vector<boost::container::flat_map<std::string, BasicVariantType>>&
foundDevs)
{
const static std::regex command(R"(\((.*)\))");
std::smatch match;
bool ret = false;
bool matchOne = false;
bool cur = true;
probe_type_codes lastCommand = probe_type_codes::FALSE_T;
bool first = true;
for (auto& probe : probeCommand)
{
bool foundProbe = false;
boost::container::flat_map<const char*, probe_type_codes,
cmp_str>::const_iterator probeType;
for (probeType = PROBE_TYPES.begin(); probeType != PROBE_TYPES.end();
probeType++)
{
if (probe.find(probeType->first) != std::string::npos)
{
foundProbe = true;
break;
}
}
if (foundProbe)
{
switch (probeType->second)
{
case probe_type_codes::FALSE_T:
{
cur = false;
break;
}
case probe_type_codes::TRUE_T:
{
cur = true;
break;
}
case probe_type_codes::MATCH_ONE:
{
// set current value to last, this probe type shouldn't
// affect the outcome
cur = ret;
matchOne = true;
break;
}
/*case probe_type_codes::AND:
break;
case probe_type_codes::OR:
break;
// these are no-ops until the last command switch
*/
case probe_type_codes::FOUND:
{
if (!std::regex_search(probe, match, command))
{
std::cerr << "found probe syntax error " << probe
<< "\n";
return false;
}
std::string commandStr = *(match.begin() + 1);
boost::replace_all(commandStr, "'", "");
cur = (std::find(PASSED_PROBES.begin(), PASSED_PROBES.end(),
commandStr) != PASSED_PROBES.end());
break;
}
default:
{
break;
}
}
}
// look on dbus for object
else
{
if (!std::regex_search(probe, match, command))
{
std::cerr << "dbus probe syntax error " << probe << "\n";
return false;
}
std::string commandStr = *(match.begin() + 1);
// convert single ticks and single slashes into legal json
boost::replace_all(commandStr, "'", "\"");
boost::replace_all(commandStr, R"(\)", R"(\\)");
auto json = nlohmann::json::parse(commandStr, nullptr, false);
if (json.is_discarded())
{
std::cerr << "dbus command syntax error " << commandStr << "\n";
return false;
}
// we can match any (string, variant) property. (string, string)
// does a regex
std::map<std::string, nlohmann::json> dbusProbeMap =
json.get<std::map<std::string, nlohmann::json>>();
auto findStart = probe.find("(");
if (findStart == std::string::npos)
{
return false;
}
std::string probeInterface = probe.substr(0, findStart);
cur =
probeDbus(probeInterface, dbusProbeMap, foundDevs, foundProbe);
}
// some functions like AND and OR only take affect after the
// fact
if (lastCommand == probe_type_codes::AND)
{
ret = cur && ret;
}
else if (lastCommand == probe_type_codes::OR)
{
ret = cur || ret;
}
if (first)
{
ret = cur;
first = false;
}
lastCommand = probeType != PROBE_TYPES.end()
? probeType->second
: probe_type_codes::FALSE_T;
}
// probe passed, but empty device
if (ret && foundDevs.size() == 0)
{
foundDevs.emplace_back(
boost::container::flat_map<std::string, BasicVariantType>{});
}
if (matchOne && ret)
{
// match the last one
auto last = foundDevs.back();
foundDevs.clear();
foundDevs.emplace_back(std::move(last));
}
return ret;
}
// this class finds the needed dbus fields and on destruction runs the probe
struct PerformProbe : std::enable_shared_from_this<PerformProbe>
{
PerformProbe(const std::vector<std::string>& probeCommand,
std::function<void(FoundDeviceT&)>&& callback) :
_probeCommand(probeCommand),
_callback(std::move(callback))
{
}
~PerformProbe()
{
FoundDeviceT foundDevs;
if (probe(_probeCommand, foundDevs))
{
_callback(foundDevs);
}
}
void run()
{
// parse out dbus probes by discarding other probe types
for (std::string& probe : _probeCommand)
{
bool found = false;
boost::container::flat_map<const char*, probe_type_codes,
cmp_str>::const_iterator probeType;
for (probeType = PROBE_TYPES.begin();
probeType != PROBE_TYPES.end(); probeType++)
{
if (probe.find(probeType->first) != std::string::npos)
{
found = true;
break;
}
}
if (found)
{
continue;
}
// syntax requires probe before first open brace
auto findStart = probe.find("(");
std::string interface = probe.substr(0, findStart);
findDbusObjects(shared_from_this(), SYSTEM_BUS, interface);
}
}
std::vector<std::string> _probeCommand;
std::function<void(FoundDeviceT&)> _callback;
};
// writes output files to persist data
bool writeJsonFiles(const nlohmann::json& systemConfiguration)
{
std::filesystem::create_directory(configurationOutDir);
std::ofstream output(currentConfiguration);
if (!output.good())
{
return false;
}
output << systemConfiguration.dump(4);
output.close();
return true;
}
template <typename JsonType>
bool setJsonFromPointer(const std::string& ptrStr, const JsonType& value,
nlohmann::json& systemConfiguration)
{
try
{
nlohmann::json::json_pointer ptr(ptrStr);
nlohmann::json& ref = systemConfiguration[ptr];
ref = value;
return true;
}
catch (const std::out_of_range&)
{
return false;
}
}
// template function to add array as dbus property
template <typename PropertyType>
void addArrayToDbus(const std::string& name, const nlohmann::json& array,
sdbusplus::asio::dbus_interface* iface,
sdbusplus::asio::PropertyPermission permission,
nlohmann::json& systemConfiguration,
const std::string& jsonPointerString)
{
std::vector<PropertyType> values;
for (const auto& property : array)
{
auto ptr = property.get_ptr<const PropertyType*>();
if (ptr != nullptr)
{
values.emplace_back(*ptr);
}
}
if (permission == sdbusplus::asio::PropertyPermission::readOnly)
{
iface->register_property(name, values);
}
else
{
iface->register_property(
name, values,
[&systemConfiguration,
jsonPointerString{std::string(jsonPointerString)}](
const std::vector<PropertyType>& newVal,
std::vector<PropertyType>& val) {
val = newVal;
if (!setJsonFromPointer(jsonPointerString, val,
systemConfiguration))
{
std::cerr << "error setting json field\n";
return -1;
}
if (!writeJsonFiles(systemConfiguration))
{
std::cerr << "error setting json file\n";
return -1;
}
return 1;
});
}
}
template <typename PropertyType>
void addProperty(const std::string& propertyName, const PropertyType& value,
sdbusplus::asio::dbus_interface* iface,
nlohmann::json& systemConfiguration,
const std::string& jsonPointerString,
sdbusplus::asio::PropertyPermission permission)
{
if (permission == sdbusplus::asio::PropertyPermission::readOnly)
{
iface->register_property(propertyName, value);
return;
}
iface->register_property(
propertyName, value,
[&systemConfiguration,
jsonPointerString{std::string(jsonPointerString)}](
const PropertyType& newVal, PropertyType& val) {
val = newVal;
if (!setJsonFromPointer(jsonPointerString, val,
systemConfiguration))
{
std::cerr << "error setting json field\n";
return -1;
}
if (!writeJsonFiles(systemConfiguration))
{
std::cerr << "error setting json file\n";
return -1;
}
return 1;
});
}
void createDeleteObjectMethod(
const std::string& jsonPointerPath,
const std::shared_ptr<sdbusplus::asio::dbus_interface>& iface,
sdbusplus::asio::object_server& objServer,
nlohmann::json& systemConfiguration)
{
std::weak_ptr<sdbusplus::asio::dbus_interface> interface = iface;
iface->register_method(
"Delete", [&objServer, &systemConfiguration, interface,
jsonPointerPath{std::string(jsonPointerPath)}]() {
std::shared_ptr<sdbusplus::asio::dbus_interface> dbusInterface =
interface.lock();
if (!dbusInterface)
{
// this technically can't happen as the pointer is pointing to
// us
throw DBusInternalError();
}
nlohmann::json::json_pointer ptr(jsonPointerPath);
if (!objServer.remove_interface(dbusInterface))
{
std::cerr << "Can't delete interface " << jsonPointerPath
<< "\n";
throw DBusInternalError();
}
systemConfiguration[ptr] = nullptr;
if (!writeJsonFiles(systemConfiguration))
{
std::cerr << "error setting json file\n";
throw DBusInternalError();
}
return -1;
});
}
// adds simple json types to interface's properties
void populateInterfaceFromJson(
nlohmann::json& systemConfiguration, const std::string& jsonPointerPath,
std::shared_ptr<sdbusplus::asio::dbus_interface>& iface,
nlohmann::json& dict, sdbusplus::asio::object_server& objServer,
sdbusplus::asio::PropertyPermission permission =
sdbusplus::asio::PropertyPermission::readOnly)
{
for (auto& dictPair : dict.items())
{
auto type = dictPair.value().type();
bool array = false;
if (dictPair.value().type() == nlohmann::json::value_t::array)
{
array = true;
if (!dictPair.value().size())
{
continue;
}
type = dictPair.value()[0].type();
bool isLegal = true;
for (const auto& arrayItem : dictPair.value())
{
if (arrayItem.type() != type)
{
isLegal = false;
break;
}
}
if (!isLegal)
{
std::cerr << "dbus format error" << dictPair.value() << "\n";
continue;
}
}
if (type == nlohmann::json::value_t::object)
{
continue; // handled elsewhere
}
std::string key = jsonPointerPath + "/" + dictPair.key();
if (permission == sdbusplus::asio::PropertyPermission::readWrite)
{
// all setable numbers are doubles as it is difficult to always
// create a configuration file with all whole numbers as decimals
// i.e. 1.0
if (array)
{
if (dictPair.value()[0].is_number())
{
type = nlohmann::json::value_t::number_float;
}
}
else if (dictPair.value().is_number())
{
type = nlohmann::json::value_t::number_float;
}
}
switch (type)
{
case (nlohmann::json::value_t::boolean):
{
if (array)
{
// todo: array of bool isn't detected correctly by
// sdbusplus, change it to numbers
addArrayToDbus<uint64_t>(dictPair.key(), dictPair.value(),
iface.get(), permission,
systemConfiguration, key);
}
else
{
addProperty(dictPair.key(), dictPair.value().get<bool>(),
iface.get(), systemConfiguration, key,
permission);
}
break;
}
case (nlohmann::json::value_t::number_integer):
{
if (array)
{
addArrayToDbus<int64_t>(dictPair.key(), dictPair.value(),
iface.get(), permission,
systemConfiguration, key);
}
else
{
addProperty(dictPair.key(), dictPair.value().get<int64_t>(),
iface.get(), systemConfiguration, key,
sdbusplus::asio::PropertyPermission::readOnly);
}
break;
}
case (nlohmann::json::value_t::number_unsigned):
{
if (array)
{
addArrayToDbus<uint64_t>(dictPair.key(), dictPair.value(),
iface.get(), permission,
systemConfiguration, key);
}
else
{
addProperty(dictPair.key(),
dictPair.value().get<uint64_t>(), iface.get(),
systemConfiguration, key,
sdbusplus::asio::PropertyPermission::readOnly);
}
break;
}
case (nlohmann::json::value_t::number_float):
{
if (array)
{
addArrayToDbus<double>(dictPair.key(), dictPair.value(),
iface.get(), permission,
systemConfiguration, key);
}
else
{
addProperty(dictPair.key(), dictPair.value().get<double>(),
iface.get(), systemConfiguration, key,
permission);
}
break;
}
case (nlohmann::json::value_t::string):
{
if (array)
{
addArrayToDbus<std::string>(
dictPair.key(), dictPair.value(), iface.get(),
permission, systemConfiguration, key);
}
else
{
addProperty(
dictPair.key(), dictPair.value().get<std::string>(),
iface.get(), systemConfiguration, key, permission);
}
break;
}
default:
{
std::cerr << "Unexpected json type in system configuration "
<< dictPair.key() << ": "
<< dictPair.value().type_name() << "\n";
break;
}
}
}
if (permission == sdbusplus::asio::PropertyPermission::readWrite)
{
createDeleteObjectMethod(jsonPointerPath, iface, objServer,
systemConfiguration);
}
iface->initialize();
}
sdbusplus::asio::PropertyPermission getPermission(const std::string& interface)
{
return std::find(settableInterfaces.begin(), settableInterfaces.end(),
interface) != settableInterfaces.end()
? sdbusplus::asio::PropertyPermission::readWrite
: sdbusplus::asio::PropertyPermission::readOnly;
}
void createAddObjectMethod(const std::string& jsonPointerPath,
const std::string& path,
nlohmann::json& systemConfiguration,
sdbusplus::asio::object_server& objServer)
{
auto iface = objServer.add_interface(path, "xyz.openbmc_project.AddObject");
iface->register_method(
"AddObject",
[&systemConfiguration, &objServer,
jsonPointerPath{std::string(jsonPointerPath)},
path{std::string(path)}](
const boost::container::flat_map<std::string, JsonVariantType>&
data) {
nlohmann::json::json_pointer ptr(jsonPointerPath);
nlohmann::json& base = systemConfiguration[ptr];
auto findExposes = base.find("Exposes");
if (findExposes == base.end())
{
throw std::invalid_argument("Entity must have children.");
}
// this will throw invalid-argument to sdbusplus if invalid json
nlohmann::json newData{};
for (const auto& item : data)
{
nlohmann::json& newJson = newData[item.first];
std::visit([&newJson](auto&& val) { newJson = std::move(val); },
item.second);
}
auto findName = newData.find("Name");
auto findType = newData.find("Type");
if (findName == newData.end() || findType == newData.end())
{
throw std::invalid_argument("AddObject missing Name or Type");
}
const std::string* type = findType->get_ptr<const std::string*>();
const std::string* name = findName->get_ptr<const std::string*>();
if (type == nullptr || name == nullptr)
{
throw std::invalid_argument("Type and Name must be a string.");
}
size_t lastIndex = 0;
// we add in the "exposes"
for (; lastIndex < findExposes->size(); lastIndex++)
{
if (findExposes->at(lastIndex)["Name"] == *name &&
findExposes->at(lastIndex)["Type"] == *type)
{
throw std::invalid_argument(
"Field already in JSON, not adding");
}
lastIndex++;
}
std::ifstream schemaFile(std::string(schemaDirectory) + "/" +
*type + ".json");
// todo(james) we might want to also make a list of 'can add'
// interfaces but for now I think the assumption if there is a
// schema avaliable that it is allowed to update is fine
if (!schemaFile.good())
{
throw std::invalid_argument(
"No schema avaliable, cannot validate.");
}
nlohmann::json schema =
nlohmann::json::parse(schemaFile, nullptr, false);
if (schema.is_discarded())
{
std::cerr << "Schema not legal" << *type << ".json\n";
throw DBusInternalError();
}
if (!validateJson(schema, newData))
{
throw std::invalid_argument("Data does not match schema");
}
findExposes->push_back(newData);
if (!writeJsonFiles(systemConfiguration))
{
std::cerr << "Error writing json files\n";
throw DBusInternalError();
}
std::string dbusName = *name;
std::regex_replace(dbusName.begin(), dbusName.begin(),
dbusName.end(), ILLEGAL_DBUS_MEMBER_REGEX, "_");
auto interface = objServer.add_interface(
path + "/" + dbusName,
"xyz.openbmc_project.Configuration." + *type);
// permission is read-write, as since we just created it, must be
// runtime modifiable
populateInterfaceFromJson(
systemConfiguration,
jsonPointerPath + "/Exposes/" + std::to_string(lastIndex),
interface, newData, objServer,
sdbusplus::asio::PropertyPermission::readWrite);
});
iface->initialize();
}
void postToDbus(const nlohmann::json& newConfiguration,
nlohmann::json& systemConfiguration,
sdbusplus::asio::object_server& objServer)
{
// iterate through boards
for (auto& boardPair : newConfiguration.items())
{
std::string boardKey = boardPair.value()["Name"];
std::string jsonPointerPath = "/" + boardPair.key();
// loop through newConfiguration, but use values from system
// configuration to be able to modify via dbus later
auto boardValues = systemConfiguration[boardPair.key()];
auto findBoardType = boardValues.find("Type");
std::string boardType;
if (findBoardType != boardValues.end() &&
findBoardType->type() == nlohmann::json::value_t::string)
{
boardType = findBoardType->get<std::string>();
std::regex_replace(boardType.begin(), boardType.begin(),
boardType.end(), ILLEGAL_DBUS_MEMBER_REGEX, "_");
}
else
{
std::cerr << "Unable to find type for " << boardKey
<< " reverting to Chassis.\n";
boardType = "Chassis";
}
std::string boardtypeLower = boost::algorithm::to_lower_copy(boardType);
std::regex_replace(boardKey.begin(), boardKey.begin(), boardKey.end(),
ILLEGAL_DBUS_MEMBER_REGEX, "_");
std::string boardName = "/xyz/openbmc_project/inventory/system/" +
boardtypeLower + "/" + boardKey;
auto inventoryIface = objServer.add_interface(
boardName, "xyz.openbmc_project.Inventory.Item");
auto boardIface = objServer.add_interface(
boardName, "xyz.openbmc_project.Inventory.Item." + boardType);
createAddObjectMethod(jsonPointerPath, boardName, systemConfiguration,
objServer);
populateInterfaceFromJson(systemConfiguration, jsonPointerPath,
boardIface, boardValues, objServer);
jsonPointerPath += "/";
// iterate through board properties
for (auto& boardField : boardValues.items())
{
if (boardField.value().type() == nlohmann::json::value_t::object)
{
auto iface =
objServer.add_interface(boardName, boardField.key());
populateInterfaceFromJson(systemConfiguration,
jsonPointerPath + boardField.key(),
iface, boardField.value(), objServer);
}
}
auto exposes = boardValues.find("Exposes");
if (exposes == boardValues.end())
{
continue;
}
// iterate through exposes
jsonPointerPath += "Exposes/";
// store the board level pointer so we can modify it on the way down
std::string jsonPointerPathBoard = jsonPointerPath;
size_t exposesIndex = -1;
for (auto& item : *exposes)
{
exposesIndex++;
jsonPointerPath = jsonPointerPathBoard;
jsonPointerPath += std::to_string(exposesIndex);
auto findName = item.find("Name");
if (findName == item.end())
{
std::cerr << "cannot find name in field " << item << "\n";
continue;
}
auto findStatus = item.find("Status");
// if status is not found it is assumed to be status = 'okay'
if (findStatus != item.end())
{
if (*findStatus == "disabled")
{
continue;
}
}
auto findType = item.find("Type");
std::string itemType;
if (findType != item.end())
{
itemType = findType->get<std::string>();
std::regex_replace(itemType.begin(), itemType.begin(),
itemType.end(), ILLEGAL_DBUS_PATH_REGEX,
"_");
}
else
{
itemType = "unknown";
}
std::string itemName = findName->get<std::string>();
std::regex_replace(itemName.begin(), itemName.begin(),
itemName.end(), ILLEGAL_DBUS_MEMBER_REGEX, "_");
auto itemIface = objServer.add_interface(
boardName + "/" + itemName,
"xyz.openbmc_project.Configuration." + itemType);
populateInterfaceFromJson(systemConfiguration, jsonPointerPath,
itemIface, item, objServer,
getPermission(itemType));
for (auto& objectPair : item.items())
{
jsonPointerPath = jsonPointerPathBoard +
std::to_string(exposesIndex) + "/" +
objectPair.key();
if (objectPair.value().type() ==
nlohmann::json::value_t::object)
{
auto objectIface = objServer.add_interface(
boardName + "/" + itemName,
"xyz.openbmc_project.Configuration." + itemType + "." +
objectPair.key());
populateInterfaceFromJson(
systemConfiguration, jsonPointerPath, objectIface,
objectPair.value(), objServer, getPermission(itemType));
}
else if (objectPair.value().type() ==
nlohmann::json::value_t::array)
{
size_t index = 0;
if (!objectPair.value().size())
{
continue;
}
bool isLegal = true;
auto type = objectPair.value()[0].type();
if (type != nlohmann::json::value_t::object)
{
continue;
}
// verify legal json
for (const auto& arrayItem : objectPair.value())
{
if (arrayItem.type() != type)
{
isLegal = false;
break;
}
}
if (!isLegal)
{
std::cerr << "dbus format error" << objectPair.value()
<< "\n";
break;
}
for (auto& arrayItem : objectPair.value())
{
auto objectIface = objServer.add_interface(
boardName + "/" + itemName,
"xyz.openbmc_project.Configuration." + itemType +
"." + objectPair.key() + std::to_string(index));
populateInterfaceFromJson(
systemConfiguration,
jsonPointerPath + "/" + std::to_string(index),
objectIface, arrayItem, objServer,
getPermission(objectPair.key()));
index++;
}
}
}
}
}
}
// reads json files out of the filesystem
bool findJsonFiles(std::list<nlohmann::json>& configurations)
{
// find configuration files
std::vector<std::filesystem::path> jsonPaths;
if (!findFiles(std::filesystem::path(configurationDirectory), R"(.*\.json)",
jsonPaths))
{
std::cerr << "Unable to find any configuration files in "
<< configurationDirectory << "\n";
return false;
}
std::ifstream schemaStream(std::string(schemaDirectory) + "/" +
globalSchema);
if (!schemaStream.good())
{
std::cerr
<< "Cannot open schema file, cannot validate JSON, exiting\n\n";
std::exit(EXIT_FAILURE);
return false;
}
nlohmann::json schema = nlohmann::json::parse(schemaStream, nullptr, false);
if (schema.is_discarded())
{
std::cerr
<< "Illegal schema file detected, cannot validate JSON, exiting\n";
std::exit(EXIT_FAILURE);
return false;
}
for (auto& jsonPath : jsonPaths)
{
std::ifstream jsonStream(jsonPath.c_str());
if (!jsonStream.good())
{
std::cerr << "unable to open " << jsonPath.string() << "\n";
continue;
}
auto data = nlohmann::json::parse(jsonStream, nullptr, false);
if (data.is_discarded())
{
std::cerr << "syntax error in " << jsonPath.string() << "\n";
continue;
}
/*
* todo(james): reenable this once less things are in flight
*
if (!validateJson(schema, data))
{
std::cerr << "Error validating " << jsonPath.string() << "\n";
continue;
}
*/
if (data.type() == nlohmann::json::value_t::array)
{
for (auto& d : data)
{
configurations.emplace_back(d);
}
}
else
{
configurations.emplace_back(data);
}
}
return true;
}
struct PerformScan : std::enable_shared_from_this<PerformScan>
{
PerformScan(nlohmann::json& systemConfiguration,
std::list<nlohmann::json>& configurations,
std::function<void(void)>&& callback) :
_systemConfiguration(systemConfiguration),
_configurations(configurations), _callback(std::move(callback))
{
}
void run()
{
for (auto it = _configurations.begin(); it != _configurations.end();)
{
auto findProbe = it->find("Probe");
auto findName = it->find("Name");
nlohmann::json probeCommand;
// check for poorly formatted fields, probe must be an array
if (findProbe == it->end())
{
std::cerr << "configuration file missing probe:\n " << *it
<< "\n";
it = _configurations.erase(it);
continue;
}
else if ((*findProbe).type() != nlohmann::json::value_t::array)
{
probeCommand = nlohmann::json::array();
probeCommand.push_back(*findProbe);
}
else
{
probeCommand = *findProbe;
}
if (findName == it->end())
{
std::cerr << "configuration file missing name:\n " << *it
<< "\n";
it = _configurations.erase(it);
continue;
}
std::string probeName = *findName;
if (std::find(PASSED_PROBES.begin(), PASSED_PROBES.end(),
probeName) != PASSED_PROBES.end())
{
it = _configurations.erase(it);
continue;
}
nlohmann::json* recordPtr = &(*it);
// store reference to this to children to makes sure we don't get
// destroyed too early
auto thisRef = shared_from_this();
auto p = std::make_shared<
PerformProbe>(probeCommand, [&, recordPtr, probeName, thisRef](
FoundDeviceT& foundDevices) {
_passed = true;
PASSED_PROBES.push_back(probeName);
size_t foundDeviceIdx = 1;
for (auto& foundDevice : foundDevices)
{
nlohmann::json record = *recordPtr;
std::string recordName;
size_t hash = 0;
if (foundDevice.size())
{
// use an array so alphabetical order from the
// flat_map is maintained
auto device = nlohmann::json::array();
for (auto& devPair : foundDevice)
{
device.push_back(devPair.first);
std::visit(
[&device](auto&& v) { device.push_back(v); },
devPair.second);
}
hash =
std::hash<std::string>{}(probeName + device.dump());
// hashes are hard to distinguish, use the
// non-hashed version if we want debug
if constexpr (DEBUG)
{
recordName = probeName + device.dump();
}
else
{
recordName = std::to_string(hash);
}
}
else
{
recordName = probeName;
}
auto fromLastJson = lastJson.find(recordName);
if (fromLastJson != lastJson.end())
{
// keep user changes
_systemConfiguration[recordName] = *fromLastJson;
continue;
}
// insert into configuration temporarily to be able to
// reference ourselves
_systemConfiguration[recordName] = record;
for (auto keyPair = record.begin(); keyPair != record.end();
keyPair++)
{
templateCharReplace(keyPair, foundDevice,
foundDeviceIdx);
}
auto findExpose = record.find("Exposes");
if (findExpose == record.end())
{
_systemConfiguration[recordName] = record;
logDeviceAdded(record);
foundDeviceIdx++;
continue;
}
for (auto& expose : *findExpose)
{
for (auto keyPair = expose.begin();
keyPair != expose.end(); keyPair++)
{
templateCharReplace(keyPair, foundDevice,
foundDeviceIdx);
// special case bind
if (boost::starts_with(keyPair.key(), "Bind"))
{
if (keyPair.value().type() !=
nlohmann::json::value_t::string)
{
std::cerr << "bind_ value must be of "
"type string "
<< keyPair.key() << "\n";
continue;
}
bool foundBind = false;
std::string bind =
keyPair.key().substr(sizeof("Bind") - 1);
for (auto& configurationPair :
_systemConfiguration.items())
{
auto configListFind =
configurationPair.value().find(
"Exposes");
if (configListFind ==
configurationPair.value().end() ||
configListFind->type() !=
nlohmann::json::value_t::array)
{
continue;
}
for (auto& exposedObject : *configListFind)
{
std::string foundObjectName =
(exposedObject)["Name"];
if (boost::iequals(
foundObjectName,
keyPair.value()
.get<std::string>()))
{
exposedObject["Status"] = "okay";
expose[bind] = exposedObject;
foundBind = true;
break;
}
}
if (foundBind)
{
break;
}
}
if (!foundBind)
{
std::cerr << "configuration file "
"dependency error, "
"could not find bind "
<< keyPair.value() << "\n";
}
}
}
}
// overwrite ourselves with cleaned up version
_systemConfiguration[recordName] = record;
logDeviceAdded(record);
foundDeviceIdx++;
}
});
p->run();
it++;
}
}
~PerformScan()
{
if (_passed)
{
auto nextScan = std::make_shared<PerformScan>(
_systemConfiguration, _configurations, std::move(_callback));
nextScan->run();
}
else
{
_callback();
}
}
nlohmann::json& _systemConfiguration;
std::list<nlohmann::json> _configurations;
std::function<void(void)> _callback;
std::vector<std::shared_ptr<PerformProbe>> _probes;
bool _passed = false;
bool powerWasOn = isPowerOn();
};
void startRemovedTimer(boost::asio::deadline_timer& timer,
nlohmann::json& systemConfiguration)
{
static bool scannedPowerOff = false;
static bool scannedPowerOn = false;
if (systemConfiguration.empty() || lastJson.empty())
{
return; // not ready yet
}
if (scannedPowerOn)
{
return;
}
if (!isPowerOn() && scannedPowerOff)
{
return;
}
timer.expires_from_now(boost::posix_time::seconds(10));
timer.async_wait(
[&systemConfiguration](const boost::system::error_code& ec) {
if (ec == boost::asio::error::operation_aborted)
{
// we were cancelled
return;
}
bool powerOff = !isPowerOn();
for (const auto& item : lastJson.items())
{
if (systemConfiguration.find(item.key()) ==
systemConfiguration.end())
{
bool isDetectedPowerOn = false;
auto powerState = item.value().find("PowerState");
if (powerState != item.value().end())
{
auto ptr = powerState->get_ptr<const std::string*>();
if (ptr)
{
if (*ptr == "On" || *ptr == "BiosPost")
{
isDetectedPowerOn = true;
}
}
}
if (powerOff && isDetectedPowerOn)
{
// power not on yet, don't know if it's there or not
continue;
}
if (!powerOff && scannedPowerOff && isDetectedPowerOn)
{
// already logged it when power was off
continue;
}
logDeviceRemoved(item.value());
}
}
scannedPowerOff = true;
if (!powerOff)
{
scannedPowerOn = true;
}
});
}
// main properties changed entry
void propertiesChangedCallback(
boost::asio::io_service& io,
std::vector<sdbusplus::bus::match::match>& dbusMatches,
nlohmann::json& systemConfiguration,
sdbusplus::asio::object_server& objServer)
{
static boost::asio::deadline_timer timer(io);
static bool timerRunning;
timerRunning = true;
timer.expires_from_now(boost::posix_time::seconds(1));
// setup an async wait as we normally get flooded with new requests
timer.async_wait([&](const boost::system::error_code& ec) {
if (ec == boost::asio::error::operation_aborted)
{
// we were cancelled
return;
}
else if (ec)
{
std::cerr << "async wait error " << ec << "\n";
return;
}
timerRunning = false;
nlohmann::json oldConfiguration = systemConfiguration;
DBUS_PROBE_OBJECTS.clear();
std::list<nlohmann::json> configurations;
if (!findJsonFiles(configurations))
{
std::cerr << "cannot find json files\n";
return;
}
auto perfScan = std::make_shared<PerformScan>(
systemConfiguration, configurations, [&, oldConfiguration]() {
nlohmann::json newConfiguration = systemConfiguration;
for (auto it = newConfiguration.begin();
it != newConfiguration.end();)
{
auto findKey = oldConfiguration.find(it.key());
if (findKey != oldConfiguration.end())
{
it = newConfiguration.erase(it);
}
else
{
it++;
}
}
registerCallbacks(io, dbusMatches, systemConfiguration,
objServer);
io.post([&, newConfiguration]() {
loadOverlays(newConfiguration);
io.post([&]() {
if (!writeJsonFiles(systemConfiguration))
{
std::cerr << "Error writing json files\n";
}
});
io.post([&, newConfiguration]() {
postToDbus(newConfiguration, systemConfiguration,
objServer);
if (!timerRunning)
{
startRemovedTimer(timer, systemConfiguration);
}
});
});
});
perfScan->run();
});
}
void registerCallbacks(boost::asio::io_service& io,
std::vector<sdbusplus::bus::match::match>& dbusMatches,
nlohmann::json& systemConfiguration,
sdbusplus::asio::object_server& objServer)
{
static boost::container::flat_set<std::string> watchedObjects;
for (const auto& objectMap : DBUS_PROBE_OBJECTS)
{
auto findObject = watchedObjects.find(objectMap.first);
if (findObject != watchedObjects.end())
{
continue;
}
std::function<void(sdbusplus::message::message & message)>
eventHandler =
[&](sdbusplus::message::message&) {
propertiesChangedCallback(io, dbusMatches,
systemConfiguration, objServer);
};
sdbusplus::bus::match::match match(
static_cast<sdbusplus::bus::bus&>(*SYSTEM_BUS),
"type='signal',member='PropertiesChanged',arg0='" +
objectMap.first + "'",
eventHandler);
dbusMatches.emplace_back(std::move(match));
}
}
int main()
{
// setup connection to dbus
boost::asio::io_service io;
SYSTEM_BUS = std::make_shared<sdbusplus::asio::connection>(io);
SYSTEM_BUS->request_name("xyz.openbmc_project.EntityManager");
sdbusplus::asio::object_server objServer(SYSTEM_BUS);
std::shared_ptr<sdbusplus::asio::dbus_interface> entityIface =
objServer.add_interface("/xyz/openbmc_project/EntityManager",
"xyz.openbmc_project.EntityManager");
std::shared_ptr<sdbusplus::asio::dbus_interface> inventoryIface =
objServer.add_interface("/xyz/openbmc_project/inventory",
"xyz.openbmc_project.Inventory.Manager");
// to keep reference to the match / filter objects so they don't get
// destroyed
std::vector<sdbusplus::bus::match::match> dbusMatches;
nlohmann::json systemConfiguration = nlohmann::json::object();
inventoryIface->register_method(
"Notify",
[](const boost::container::flat_map<
std::string,
boost::container::flat_map<std::string, BasicVariantType>>&) {
return;
});
inventoryIface->initialize();
io.post([&]() {
#if OVERLAYS
unloadAllOverlays();
#endif
propertiesChangedCallback(io, dbusMatches, systemConfiguration,
objServer);
});
entityIface->register_method("ReScan", [&]() {
propertiesChangedCallback(io, dbusMatches, systemConfiguration,
objServer);
});
entityIface->initialize();
if (fwVersionIsSame())
{
if (std::filesystem::is_regular_file(currentConfiguration))
{
// this file could just be deleted, but it's nice for debug
std::filesystem::create_directory(tempConfigDir);
std::filesystem::remove(lastConfiguration);
std::filesystem::copy(currentConfiguration, lastConfiguration);
std::filesystem::remove(currentConfiguration);
std::ifstream jsonStream(lastConfiguration);
if (jsonStream.good())
{
auto data = nlohmann::json::parse(jsonStream, nullptr, false);
if (data.is_discarded())
{
std::cerr << "syntax error in " << lastConfiguration
<< "\n";
}
else
{
lastJson = std::move(data);
}
}
else
{
std::cerr << "unable to open " << lastConfiguration << "\n";
}
}
}
else
{
// not an error, just logging at this level to make it in the journal
std::cerr << "Clearing previous configuration\n";
std::filesystem::remove(currentConfiguration);
}
// some boards only show up after power is on, we want to not say they are
// removed until the same state happens
setupPowerMatch(SYSTEM_BUS);
io.run();
return 0;
}