blob: a3a53c84eb21f8ddec5e5b1bdec92784c094cdfd [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.
*/
#pragma once
#include "dbus_singleton.hpp"
#include "logging.hpp"
#include <boost/system/error_code.hpp> // IWYU pragma: keep
#include <mapper/mapper.hpp>
#include <sdbusplus/asio/property.hpp>
#include <sdbusplus/message/native_types.hpp>
#include <array>
#include <cstddef>
#include <cstdint>
#include <filesystem>
#include <functional>
#include <regex>
#include <span>
#include <sstream>
#include <string>
#include <string_view>
#include <tuple>
#include <utility>
#include <variant>
#include <vector>
// IWYU pragma: no_include <stddef.h>
// IWYU pragma: no_include <stdint.h>
// IWYU pragma: no_include <boost/system/detail/error_code.hpp>
namespace dbus
{
namespace utility
{
// clang-format off
using DbusVariantType = std::variant<
std::vector<std::tuple<std::string, std::string, std::string>>,
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,
sdbusplus::message::unix_fd,
std::vector<uint32_t>,
std::vector<uint16_t>,
sdbusplus::message::object_path,
std::tuple<uint64_t, std::vector<std::tuple<std::string, std::string, double, uint64_t>>>,
std::vector<std::tuple<std::string, std::string>>,
std::vector<std::tuple<uint32_t, std::vector<uint32_t>>>,
std::vector<std::tuple<uint32_t, size_t>>,
std::vector<std::tuple<sdbusplus::message::object_path, std::string,
std::string, std::string>>
>;
// clang-format on
using DBusPropertiesMap = std::vector<std::pair<std::string, DbusVariantType>>;
using DBusInteracesMap = std::vector<std::pair<std::string, DBusPropertiesMap>>;
using ManagedObjectType =
std::vector<std::pair<sdbusplus::message::object_path, DBusInteracesMap>>;
// Map of service name to list of interfaces
using MapperServiceMap =
std::vector<std::pair<std::string, std::vector<std::string>>>;
// Map of object paths to MapperServiceMaps
using MapperGetSubTreeResponse =
std::vector<std::pair<std::string, MapperServiceMap>>;
using MapperGetObject =
std::vector<std::pair<std::string, std::vector<std::string>>>;
using MapperGetAncestorsResponse = std::vector<
std::pair<std::string,
std::vector<std::pair<std::string, std::vector<std::string>>>>>;
using MapperGetSubTreePathsResponse = std::vector<std::string>;
using MapperEndPoints = std::vector<std::string>;
inline void escapePathForDbus(std::string& path)
{
const static std::regex reg("[^A-Za-z0-9_/]");
std::regex_replace(path.begin(), path.begin(), path.end(), reg, "_");
}
inline void logError(const boost::system::error_code& ec)
{
if (ec)
{
BMCWEB_LOG_ERROR << "DBus error: " << ec << ", cannot call method";
}
}
// gets the string N strings deep into a path
// i.e. /0th/1st/2nd/3rd
inline bool getNthStringFromPath(const std::string& path, int index,
std::string& result)
{
if (index < 0)
{
return false;
}
std::filesystem::path p1(path);
int count = -1;
for (auto const& element : p1)
{
if (element.has_filename())
{
++count;
if (count == index)
{
result = element.stem().string();
break;
}
}
}
return count >= index;
}
template <typename Callback>
inline void checkDbusPathExists(const std::string& path, Callback&& callback)
{
crow::connections::systemBus->async_method_call(
[callback{std::forward<Callback>(callback)}](
const boost::system::error_code& ec,
const dbus::utility::MapperGetObject& objectNames) {
callback(!ec && !objectNames.empty());
},
"xyz.openbmc_project.ObjectMapper",
"/xyz/openbmc_project/object_mapper",
"xyz.openbmc_project.ObjectMapper", "GetObject", path,
std::array<std::string, 0>());
}
inline InterfaceMapType interfaceMap;
namespace dbus_mapper
{
inline void
getSubTree(const std::string& path, int32_t depth,
std::span<const std::string_view> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetSubTreeResponse&)>&& callback)
{
crow::connections::systemBus->async_method_call(
[callback{std::move(callback)}](
const boost::system::error_code& ec,
const MapperGetSubTreeResponse& subtree) { callback(ec, subtree); },
"xyz.openbmc_project.ObjectMapper",
"/xyz/openbmc_project/object_mapper",
"xyz.openbmc_project.ObjectMapper", "GetSubTree", path, depth,
interfaces);
}
// Overload to support span of strings for interface names.
inline void
getSubTree(const std::string& path, int32_t depth,
std::span<std::string> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetSubTreeResponse&)>&& callback)
{
crow::connections::systemBus->async_method_call(
[callback{std::move(callback)}](
const boost::system::error_code& ec,
const MapperGetSubTreeResponse& subtree) { callback(ec, subtree); },
"xyz.openbmc_project.ObjectMapper",
"/xyz/openbmc_project/object_mapper",
"xyz.openbmc_project.ObjectMapper", "GetSubTree", path, depth,
interfaces);
}
inline void getSubTreePaths(
const std::string& path, int32_t depth,
std::span<const std::string_view> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetSubTreePathsResponse&)>&& callback)
{
crow::connections::systemBus->async_method_call(
[callback{std::move(callback)}](
const boost::system::error_code& ec,
const MapperGetSubTreePathsResponse& subtreePaths) {
callback(ec, subtreePaths);
},
"xyz.openbmc_project.ObjectMapper",
"/xyz/openbmc_project/object_mapper",
"xyz.openbmc_project.ObjectMapper", "GetSubTreePaths", path, depth,
interfaces);
}
inline void getAssociatedSubTree(
const sdbusplus::message::object_path& associatedPath,
const sdbusplus::message::object_path& path, int32_t depth,
std::span<const std::string_view> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetSubTreeResponse&)>&& callback)
{
crow::connections::systemBus->async_method_call(
[callback{std::move(callback)}](
const boost::system::error_code& ec,
const MapperGetSubTreeResponse& subtree) { callback(ec, subtree); },
"xyz.openbmc_project.ObjectMapper",
"/xyz/openbmc_project/object_mapper",
"xyz.openbmc_project.ObjectMapper", "GetAssociatedSubTree",
associatedPath, path, depth, interfaces);
}
inline void getAssociatedSubTreePaths(
const sdbusplus::message::object_path& associatedPath,
const sdbusplus::message::object_path& path, int32_t depth,
std::span<const std::string_view> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetSubTreePathsResponse&)>&& callback)
{
crow::connections::systemBus->async_method_call(
[callback{std::move(callback)}](
const boost::system::error_code& ec,
const MapperGetSubTreePathsResponse& subtreePaths) {
callback(ec, subtreePaths);
},
"xyz.openbmc_project.ObjectMapper",
"/xyz/openbmc_project/object_mapper",
"xyz.openbmc_project.ObjectMapper", "GetAssociatedSubTreePaths",
associatedPath, path, depth, interfaces);
}
inline void
getDbusObject(const std::string& path,
std::span<const std::string_view> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetObject&)>&& callback)
{
crow::connections::systemBus->async_method_call(
[callback{std::move(callback)}](const boost::system::error_code& ec,
const MapperGetObject& object) {
callback(ec, object);
},
"xyz.openbmc_project.ObjectMapper",
"/xyz/openbmc_project/object_mapper",
"xyz.openbmc_project.ObjectMapper", "GetObject", path, interfaces);
}
} // namespace dbus_mapper
// Ideally this would be a class variable.
inline bool isMapperLiteEnabled()
{
return !interfaceMap.empty();
}
inline void
getSubTree(const std::string& path, int32_t depth,
std::span<const std::string_view> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetSubTreeResponse&)>&& callback)
{
if (!isMapperLiteEnabled())
{
dbus_mapper::getSubTree(path, depth, interfaces, std::move(callback));
return;
}
BMCWEB_LOG_DEBUG << "Mapper lite getSubTree called interfaceMap.size()="
<< interfaceMap.size();
std::vector<std::string> interfaces2;
for (auto& s : interfaces)
{
interfaces2.emplace_back(s);
}
std::vector<InterfaceMapType::value_type> out =
::getSubTree(interfaceMap, path, depth, interfaces2);
MapperGetSubTreeResponse out2;
for (const auto& connection : out)
{
auto& newConnections = out2.emplace_back(std::make_pair(
connection.first,
MapperGetSubTreeResponse::value_type::second_type{}));
for (const auto& path2 : connection.second)
{
auto& newPath = newConnections.second.emplace_back(
path2.first, MapperGetSubTreeResponse::value_type::second_type::
value_type::second_type{});
for (const auto& interface2 : path2.second)
{
newPath.second.emplace_back(interface2);
}
}
}
boost::system::error_code ec;
#ifdef BMCWEB_STATEFUL_CHECK_RESPONSE
dbus_mapper::getSubTree(path, depth, interfaces,
[out2](const boost::system::error_code& ec,
const MapperGetSubTreeResponse& out) {
if (ec)
{
BMCWEB_LOG_ERROR << "Mapper returned an error " << ec.message();
return;
}
if (out != out2)
{
BMCWEB_LOG_CRITICAL << "Dbus GetSubTree doesn't match mapper lite";
}
else
{
BMCWEB_LOG_DEBUG << "Dbus GetSubTree matched mapper lite";
}
});
#endif
callback(ec, out2);
}
// Overload to support span of strings for interface names.
inline void
getSubTree(const std::string& path, int32_t depth,
std::span<std::string> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetSubTreeResponse&)>&& callback)
{
std::vector<std::string_view> interfacesSpan;
for (const auto& interface : interfaces)
{
interfacesSpan.emplace_back(interface);
}
getSubTree(path, depth, interfacesSpan, std::move(callback));
}
inline void getSubTreePaths(
const std::string& path, int32_t depth,
std::span<const std::string_view> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetSubTreePathsResponse&)>&& callback)
{
if (!isMapperLiteEnabled())
{
dbus_mapper::getSubTreePaths(path, depth, interfaces,
std::move(callback));
return;
}
BMCWEB_LOG_DEBUG
<< "Mapper lite getSubTreePaths called interfaceMap.size()="
<< interfaceMap.size();
std::vector<std::string> interfaces2;
for (auto& s : interfaces)
{
interfaces2.emplace_back(s);
}
MapperGetSubTreePathsResponse out2 =
::getSubTreePaths(interfaceMap, path, depth, interfaces2);
#ifdef BMCWEB_STATEFUL_CHECK_RESPONSE
dbus_mapper::getSubTreePaths(
path, depth, interfaces,
[out2](const boost::system::error_code& ec,
const MapperGetSubTreePathsResponse& out) {
if (ec)
{
BMCWEB_LOG_ERROR << "Mapper returned an error " << ec.message();
return;
}
if (out != out2)
{
BMCWEB_LOG_CRITICAL
<< "Dbus GetSubTreePaths doesn't match mapper lite";
}
else
{
BMCWEB_LOG_DEBUG << "Dbus GetSubTreePaths matched mapper lite";
}
});
#endif
boost::system::error_code ec;
callback(ec, out2);
}
inline void getAssociatedSubTree(
const sdbusplus::message::object_path& associatedPath,
const sdbusplus::message::object_path& path, int32_t depth,
std::span<const std::string_view> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetSubTreeResponse&)>&& callback)
{
if (!isMapperLiteEnabled())
{
dbus_mapper::getAssociatedSubTree(associatedPath, path, depth,
interfaces, std::move(callback));
return;
}
BMCWEB_LOG_DEBUG
<< "Mapper lite getSubTreePaths called interfaceMap.size()="
<< interfaceMap.size();
std::vector<std::string> interfaces2;
for (auto& s : interfaces)
{
interfaces2.emplace_back(s);
}
std::vector<InterfaceMapType::value_type> out =
::getAssociatedSubTree(interfaceMap, associationMaps, associatedPath,
path, depth, interfaces2);
MapperGetSubTreeResponse out2;
for (const auto& connection : out)
{
auto& newConnections = out2.emplace_back(std::make_pair(
connection.first,
MapperGetSubTreeResponse::value_type::second_type{}));
for (const auto& path2 : connection.second)
{
auto& newPath = newConnections.second.emplace_back(
path2.first, MapperGetSubTreeResponse::value_type::second_type::
value_type::second_type{});
for (const auto& interface2 : path2.second)
{
newPath.second.emplace_back(interface2);
}
}
}
#ifdef BMCWEB_STATEFUL_CHECK_RESPONSE
dbus_mapper::getAssociatedSubTree(
associatedPath, path, depth, interfaces,
[out2](const boost::system::error_code& ec,
const MapperGetSubTreeResponse& out) {
if (ec)
{
BMCWEB_LOG_ERROR << "Mapper returned an error " << ec.message();
return;
}
if (out != out2)
{
BMCWEB_LOG_CRITICAL
<< "Dbus GetAssociatedSubTree doesn't match mapper lite";
}
else
{
BMCWEB_LOG_DEBUG << "Dbus GetAssociatedSubTree matched mapper lite";
}
});
#endif
boost::system::error_code ec;
callback(ec, out2);
}
inline void getAssociatedSubTreePaths(
const sdbusplus::message::object_path& associatedPath,
const sdbusplus::message::object_path& path, int32_t depth,
std::span<const std::string_view> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetSubTreePathsResponse&)>&& callback)
{
if (!isMapperLiteEnabled())
{
dbus_mapper::getAssociatedSubTreePaths(associatedPath, path, depth,
interfaces, std::move(callback));
return;
}
BMCWEB_LOG_DEBUG
<< "Mapper lite getAssociatedSubTreePaths called interfaceMap.size()="
<< interfaceMap.size();
std::vector<std::string> interfaces2;
for (auto& s : interfaces)
{
interfaces2.emplace_back(s);
}
MapperGetSubTreePathsResponse out2 =
::getAssociatedSubTreePaths(interfaceMap, associationMaps,
associatedPath, path, depth, interfaces2);
#ifdef BMCWEB_STATEFUL_CHECK_RESPONSE
dbus_mapper::getAssociatedSubTreePaths(
associatedPath, path, depth, interfaces,
[out2](const boost::system::error_code& ec,
const MapperGetSubTreePathsResponse& out) {
if (ec)
{
BMCWEB_LOG_ERROR << "Mapper returned an error " << ec.message();
return;
}
if (out != out2)
{
BMCWEB_LOG_CRITICAL
<< "Dbus GetAssociatedSubtreePaths doesn't match mapper lite";
}
else
{
BMCWEB_LOG_DEBUG
<< "Dbus GetAssociatedSubtreePaths matched mapper lite";
}
});
#endif
boost::system::error_code ec;
callback(ec, out2);
}
inline void
getDbusObject(const std::string& path,
std::span<const std::string_view> interfaces,
std::function<void(const boost::system::error_code&,
const MapperGetObject&)>&& callback)
{
if (!isMapperLiteEnabled())
{
dbus_mapper::getDbusObject(path, interfaces, std::move(callback));
return;
}
BMCWEB_LOG_DEBUG << "Mapper lite getDbusObject called interfaceMap.size()="
<< interfaceMap.size();
std::vector<std::string> interfaces2;
for (auto& s : interfaces)
{
interfaces2.emplace_back(s);
}
ConnectionNames out = ::getObject(interfaceMap, path, interfaces2);
MapperGetObject out2;
for (const auto& connection : out)
{
auto& newConnections = out2.emplace_back(std::make_pair(
connection.first, MapperGetObject::value_type::second_type{}));
for (const auto& path2 : connection.second)
{
newConnections.second.emplace_back(path2);
}
}
#ifdef BMCWEB_STATEFUL_CHECK_RESPONSE
dbus_mapper::getDbusObject(path, interfaces,
[out2](const boost::system::error_code& ec,
const MapperGetObject& out) {
if (ec)
{
BMCWEB_LOG_ERROR << "Mapper returned an error " << ec.message();
return;
}
if (out != out2)
{
BMCWEB_LOG_CRITICAL << "Dbus GetObject doesn't match mapper lite";
}
else
{
BMCWEB_LOG_DEBUG << "Dbus GetObject matched mapper lite";
}
});
#endif
boost::system::error_code ec;
callback(ec, out2);
}
// TODO: Deprecate the API to allow routing through managedStore via API in
// dbus_utils.hpp
template <typename PropertyType>
void getProperty(const std::string& service,
const sdbusplus::message::object_path& path,
const std::string& interface, const std::string& property,
std::function<void(const boost::system::error_code&,
const PropertyType&)>&& callback)
{
sdbusplus::asio::getProperty<PropertyType>(*crow::connections::systemBus,
service, path.str, interface,
property, std::move(callback));
}
inline void getProperty(const std::string& service,
const sdbusplus::message::object_path& path,
const std::string& interface,
const std::string& property,
std::function<void(const boost::system::error_code&,
const DbusVariantType&)>&& callback)
{
crow::connections::systemBus->async_method_call(
[callback{std::move(callback)}](const boost::system::error_code& ec,
const DbusVariantType& propertyValue) {
callback(ec, propertyValue);
},
service, path, "org.freedesktop.DBus.Properties", "Get", interface,
property);
}
inline void getAssociationEndPoints(
const std::string& path,
std::function<void(const boost::system::error_code&,
const MapperEndPoints&)>&& callback)
{
getProperty<MapperEndPoints>("xyz.openbmc_project.ObjectMapper", {path},
"xyz.openbmc_project.Association", "endpoints",
std::move(callback));
}
inline void
getManagedObjects(const std::string& service,
const sdbusplus::message::object_path& path,
std::function<void(const boost::system::error_code&,
const ManagedObjectType&)>&& callback)
{
crow::connections::systemBus->async_method_call(
[callback{std::move(callback)}](const boost::system::error_code& ec,
const ManagedObjectType& objects) {
callback(ec, objects);
},
service, path, "org.freedesktop.DBus.ObjectManager",
"GetManagedObjects");
}
} // namespace utility
} // namespace dbus