| #include "association_manager.hpp" |
| |
| #include <nlohmann/json.hpp> |
| #include <phosphor-logging/log.hpp> |
| |
| #include <filesystem> |
| #include <fstream> |
| |
| namespace phosphor |
| { |
| namespace inventory |
| { |
| namespace manager |
| { |
| namespace associations |
| { |
| using namespace phosphor::logging; |
| using sdbusplus::exception::SdBusError; |
| |
| Manager::Manager(sdbusplus::bus::bus& bus, const std::string& jsonPath) : |
| _bus(bus), _jsonFile(jsonPath) |
| { |
| load(); |
| } |
| |
| /** |
| * @brief Throws an exception if 'num' is zero. Used for JSON |
| * sanity checking. |
| * |
| * @param[in] num - the number to check |
| */ |
| void throwIfZero(int num) |
| { |
| if (!num) |
| { |
| throw std::invalid_argument("Invalid empty field in JSON"); |
| } |
| } |
| |
| void Manager::load() |
| { |
| // Load the contents of _jsonFile into _associations and throw |
| // an exception on any problem. |
| |
| std::ifstream file{_jsonFile}; |
| |
| auto json = nlohmann::json::parse(file, nullptr, true); |
| |
| const std::string root{INVENTORY_ROOT}; |
| |
| for (const auto& jsonAssoc : json) |
| { |
| // Only add the slash if necessary |
| std::string path = jsonAssoc.at("path"); |
| throwIfZero(path.size()); |
| if (path.front() != '/') |
| { |
| path = root + "/" + path; |
| } |
| else |
| { |
| path = root + path; |
| } |
| |
| auto& assocEndpoints = _associations[path]; |
| |
| for (const auto& endpoint : jsonAssoc.at("endpoints")) |
| { |
| std::string ftype = endpoint.at("types").at("fType"); |
| std::string rtype = endpoint.at("types").at("rType"); |
| throwIfZero(ftype.size()); |
| throwIfZero(rtype.size()); |
| Types types{std::move(ftype), std::move(rtype)}; |
| |
| Paths paths = endpoint.at("paths"); |
| throwIfZero(paths.size()); |
| assocEndpoints.emplace_back(std::move(types), std::move(paths)); |
| } |
| } |
| } |
| |
| void Manager::createAssociations(const std::string& objectPath, |
| bool deferSignal) |
| { |
| auto endpoints = _associations.find(objectPath); |
| if (endpoints == _associations.end()) |
| { |
| return; |
| } |
| |
| if (std::find(_handled.begin(), _handled.end(), objectPath) != |
| _handled.end()) |
| { |
| return; |
| } |
| |
| _handled.push_back(objectPath); |
| |
| for (const auto& endpoint : endpoints->second) |
| { |
| const auto& types = std::get<typesPos>(endpoint); |
| const auto& paths = std::get<pathsPos>(endpoint); |
| |
| for (const auto& endpointPath : paths) |
| { |
| const auto& forwardType = std::get<forwardTypePos>(types); |
| const auto& reverseType = std::get<reverseTypePos>(types); |
| |
| createAssociation(objectPath, forwardType, endpointPath, |
| reverseType, deferSignal); |
| } |
| } |
| } |
| |
| void Manager::createAssociation(const std::string& forwardPath, |
| const std::string& forwardType, |
| const std::string& reversePath, |
| const std::string& reverseType, |
| bool deferSignal) |
| { |
| auto object = _associationIfaces.find(forwardPath); |
| if (object == _associationIfaces.end()) |
| { |
| auto a = std::make_unique<AssociationObject>(_bus, forwardPath.c_str(), |
| true); |
| |
| using AssociationProperty = |
| std::vector<std::tuple<std::string, std::string, std::string>>; |
| AssociationProperty prop; |
| |
| prop.emplace_back(forwardType, reverseType, reversePath); |
| a->associations(std::move(prop)); |
| if (!deferSignal) |
| { |
| a->emit_object_added(); |
| } |
| _associationIfaces.emplace(forwardPath, std::move(a)); |
| } |
| else |
| { |
| // Interface exists, just update the property |
| auto prop = object->second->associations(); |
| prop.emplace_back(forwardType, reverseType, reversePath); |
| object->second->associations(std::move(prop), deferSignal); |
| } |
| } |
| } // namespace associations |
| } // namespace manager |
| } // namespace inventory |
| } // namespace phosphor |