2017-11-27 14:13:55 +01:00
|
|
|
/**
|
|
|
|
|
* (C) 2017 KISTLER INSTRUMENTE AG, Winterthur, Switzerland
|
|
|
|
|
*
|
|
|
|
|
* @file Message.cpp
|
|
|
|
|
*
|
|
|
|
|
* Created on: Nov 9, 2016
|
|
|
|
|
* Project: sdbus-c++
|
|
|
|
|
* Description: High-level D-Bus IPC C++ library based on sd-bus
|
|
|
|
|
*
|
|
|
|
|
* This file is part of sdbus-c++.
|
|
|
|
|
*
|
|
|
|
|
* sdbus-c++ is free software; you can redistribute it and/or modify it
|
|
|
|
|
* under the terms of the GNU Lesser General Public License as published by
|
|
|
|
|
* the Free Software Foundation, either version 2.1 of the License, or
|
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* sdbus-c++ is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
* GNU Lesser General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
|
|
|
* along with sdbus-c++. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <sdbus-c++/Message.h>
|
|
|
|
|
#include <sdbus-c++/Types.h>
|
|
|
|
|
#include <sdbus-c++/Error.h>
|
|
|
|
|
#include "MessageUtils.h"
|
2019-03-25 16:28:31 +01:00
|
|
|
#include "SdBus.h"
|
2017-11-27 14:13:55 +01:00
|
|
|
#include "ScopeGuard.h"
|
|
|
|
|
#include <systemd/sd-bus.h>
|
|
|
|
|
#include <cassert>
|
|
|
|
|
|
2018-07-02 11:22:00 +02:00
|
|
|
namespace sdbus {
|
2017-11-27 14:13:55 +01:00
|
|
|
|
2019-03-25 16:28:31 +01:00
|
|
|
Message::Message(internal::ISdBus* sdbus) noexcept
|
|
|
|
|
: sdbus_(sdbus)
|
|
|
|
|
{
|
|
|
|
|
assert(sdbus_ != nullptr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message::Message(void *msg, internal::ISdBus* sdbus) noexcept
|
|
|
|
|
: msg_(msg)
|
|
|
|
|
, sdbus_(sdbus)
|
|
|
|
|
{
|
|
|
|
|
assert(msg_ != nullptr);
|
|
|
|
|
assert(sdbus_ != nullptr);
|
|
|
|
|
sdbus_->sd_bus_message_ref((sd_bus_message*)msg_);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message::Message(void *msg, internal::ISdBus* sdbus, adopt_message_t) noexcept
|
2017-11-27 14:13:55 +01:00
|
|
|
: msg_(msg)
|
2019-03-25 16:28:31 +01:00
|
|
|
, sdbus_(sdbus)
|
2017-11-27 14:13:55 +01:00
|
|
|
{
|
|
|
|
|
assert(msg_ != nullptr);
|
2019-03-25 16:28:31 +01:00
|
|
|
assert(sdbus_ != nullptr);
|
2017-11-27 14:13:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message::Message(const Message& other) noexcept
|
|
|
|
|
{
|
|
|
|
|
*this = other;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator=(const Message& other) noexcept
|
|
|
|
|
{
|
2018-06-19 08:56:28 +02:00
|
|
|
if (msg_)
|
2019-03-25 16:28:31 +01:00
|
|
|
sdbus_->sd_bus_message_unref((sd_bus_message*)msg_);
|
2018-06-19 08:56:28 +02:00
|
|
|
|
2017-11-27 14:13:55 +01:00
|
|
|
msg_ = other.msg_;
|
2019-03-25 16:28:31 +01:00
|
|
|
sdbus_ = other.sdbus_;
|
2017-11-27 14:13:55 +01:00
|
|
|
ok_ = other.ok_;
|
|
|
|
|
|
2019-03-25 16:28:31 +01:00
|
|
|
sdbus_->sd_bus_message_ref((sd_bus_message*)msg_);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message::Message(Message&& other) noexcept
|
|
|
|
|
{
|
|
|
|
|
*this = std::move(other);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator=(Message&& other) noexcept
|
|
|
|
|
{
|
2018-06-19 08:56:28 +02:00
|
|
|
if (msg_)
|
2019-03-25 16:28:31 +01:00
|
|
|
sdbus_->sd_bus_message_unref((sd_bus_message*)msg_);
|
2018-06-19 08:56:28 +02:00
|
|
|
|
2017-11-27 14:13:55 +01:00
|
|
|
msg_ = other.msg_;
|
|
|
|
|
other.msg_ = nullptr;
|
2019-03-25 16:28:31 +01:00
|
|
|
sdbus_ = other.sdbus_;
|
|
|
|
|
other.sdbus_ = nullptr;
|
2017-11-27 14:13:55 +01:00
|
|
|
ok_ = other.ok_;
|
|
|
|
|
other.ok_ = true;
|
|
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message::~Message()
|
|
|
|
|
{
|
|
|
|
|
if (msg_)
|
2019-03-25 16:28:31 +01:00
|
|
|
sdbus_->sd_bus_message_unref((sd_bus_message*)msg_);
|
2017-11-27 14:13:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(bool item)
|
|
|
|
|
{
|
|
|
|
|
int intItem = item;
|
|
|
|
|
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_BOOLEAN, &intItem);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize a bool value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(int16_t item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_INT16, &item);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize a int16_t value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(int32_t item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_INT32, &item);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize a int32_t value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(int64_t item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_INT64, &item);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize a int64_t value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(uint8_t item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_BYTE, &item);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize a byte value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(uint16_t item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_UINT16, &item);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize a uint16_t value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(uint32_t item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_UINT32, &item);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize a uint32_t value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(uint64_t item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_UINT64, &item);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize a uint64_t value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(double item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_DOUBLE, &item);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize a double value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(const char* item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_STRING, item);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize a C-string value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(const std::string& item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_STRING, item.c_str());
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize a string value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(const Variant &item)
|
|
|
|
|
{
|
|
|
|
|
item.serializeTo(*this);
|
|
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(const ObjectPath &item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_OBJECT_PATH, item.c_str());
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize an ObjectPath value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator<<(const Signature &item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_append_basic((sd_bus_message*)msg_, SD_BUS_TYPE_SIGNATURE, item.c_str());
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to serialize an Signature value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(bool& item)
|
|
|
|
|
{
|
|
|
|
|
int intItem;
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_BOOLEAN, &intItem);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize a bool value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
item = static_cast<bool>(intItem);
|
|
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(int16_t& item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_INT16, &item);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize a int16_t value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(int32_t& item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_INT32, &item);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize a int32_t value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(int64_t& item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_INT64, &item);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize a bool value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(uint8_t& item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_BYTE, &item);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize a byte value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(uint16_t& item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_UINT16, &item);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize a uint16_t value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(uint32_t& item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_UINT32, &item);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize a uint32_t value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(uint64_t& item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_UINT64, &item);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize a uint64_t value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(double& item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_DOUBLE, &item);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize a double value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(char*& item)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_STRING, &item);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize a string value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(std::string& item)
|
|
|
|
|
{
|
|
|
|
|
char* str{};
|
|
|
|
|
(*this) >> str;
|
|
|
|
|
|
|
|
|
|
if (str != nullptr)
|
|
|
|
|
item = str;
|
|
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(Variant &item)
|
|
|
|
|
{
|
|
|
|
|
item.deserializeFrom(*this);
|
|
|
|
|
|
|
|
|
|
// Empty variant is normally prohibited. Users cannot send empty variants.
|
|
|
|
|
// Therefore in this context an empty variant means that we are at the end
|
|
|
|
|
// of deserializing a container, and thus we shall set ok_ flag to false.
|
|
|
|
|
if (item.isEmpty())
|
|
|
|
|
ok_ = false;
|
|
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(ObjectPath &item)
|
|
|
|
|
{
|
|
|
|
|
char* str{};
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_OBJECT_PATH, &str);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize an ObjectPath value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
if (str != nullptr)
|
|
|
|
|
item = str;
|
|
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::operator>>(Signature &item)
|
|
|
|
|
{
|
|
|
|
|
char* str{};
|
|
|
|
|
auto r = sd_bus_message_read_basic((sd_bus_message*)msg_, SD_BUS_TYPE_SIGNATURE, &str);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to deserialize a Signature value", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
if (str != nullptr)
|
|
|
|
|
item = str;
|
|
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Message& Message::openContainer(const std::string& signature)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_open_container((sd_bus_message*)msg_, SD_BUS_TYPE_ARRAY, signature.c_str());
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to open a container", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::closeContainer()
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_close_container((sd_bus_message*)msg_);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to close a container", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::openDictEntry(const std::string& signature)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_open_container((sd_bus_message*)msg_, SD_BUS_TYPE_DICT_ENTRY, signature.c_str());
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to open a dictionary entry", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::closeDictEntry()
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_close_container((sd_bus_message*)msg_);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to close a dictionary entry", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::openVariant(const std::string& signature)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_open_container((sd_bus_message*)msg_, SD_BUS_TYPE_VARIANT, signature.c_str());
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to open a variant", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::closeVariant()
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_close_container((sd_bus_message*)msg_);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to close a variant", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::openStruct(const std::string& signature)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_open_container((sd_bus_message*)msg_, SD_BUS_TYPE_STRUCT, signature.c_str());
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to open a struct", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::closeStruct()
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_close_container((sd_bus_message*)msg_);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to close a struct", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Message& Message::enterContainer(const std::string& signature)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_enter_container((sd_bus_message*)msg_, SD_BUS_TYPE_ARRAY, signature.c_str());
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to enter a container", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::exitContainer()
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_exit_container((sd_bus_message*)msg_);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to exit a container", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::enterDictEntry(const std::string& signature)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_enter_container((sd_bus_message*)msg_, SD_BUS_TYPE_DICT_ENTRY, signature.c_str());
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to enter a dictionary entry", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::exitDictEntry()
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_exit_container((sd_bus_message*)msg_);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to exit a dictionary entry", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::enterVariant(const std::string& signature)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_enter_container((sd_bus_message*)msg_, SD_BUS_TYPE_VARIANT, signature.c_str());
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to enter a variant", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::exitVariant()
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_exit_container((sd_bus_message*)msg_);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to exit a variant", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::enterStruct(const std::string& signature)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_enter_container((sd_bus_message*)msg_, SD_BUS_TYPE_STRUCT, signature.c_str());
|
|
|
|
|
if (r == 0)
|
|
|
|
|
ok_ = false;
|
2018-05-25 20:48:20 +02:00
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to enter a struct", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message& Message::exitStruct()
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_exit_container((sd_bus_message*)msg_);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to exit a struct", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Message::operator bool() const
|
|
|
|
|
{
|
|
|
|
|
return ok_;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Message::clearFlags()
|
|
|
|
|
{
|
|
|
|
|
ok_ = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Message::copyTo(Message& destination, bool complete) const
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_copy((sd_bus_message*)destination.msg_, (sd_bus_message*)msg_, complete);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to copy the message", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Message::seal()
|
|
|
|
|
{
|
|
|
|
|
const auto messageCookie = 1;
|
|
|
|
|
const auto sealTimeout = 0;
|
|
|
|
|
auto r = sd_bus_message_seal((sd_bus_message*)msg_, messageCookie, sealTimeout);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to seal the message", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Message::rewind(bool complete)
|
|
|
|
|
{
|
|
|
|
|
auto r = sd_bus_message_rewind((sd_bus_message*)msg_, complete);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to rewind the message", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::string Message::getInterfaceName() const
|
|
|
|
|
{
|
|
|
|
|
return sd_bus_message_get_interface((sd_bus_message*)msg_);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::string Message::getMemberName() const
|
|
|
|
|
{
|
|
|
|
|
return sd_bus_message_get_member((sd_bus_message*)msg_);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Message::peekType(std::string& type, std::string& contents) const
|
|
|
|
|
{
|
|
|
|
|
char typeSig;
|
|
|
|
|
const char* contentsSig;
|
|
|
|
|
auto r = sd_bus_message_peek_type((sd_bus_message*)msg_, &typeSig, &contentsSig);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to peek message type", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
type = typeSig;
|
|
|
|
|
contents = contentsSig;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Message::isValid() const
|
|
|
|
|
{
|
2019-03-25 16:28:31 +01:00
|
|
|
return msg_ != nullptr && sdbus_ != nullptr;
|
2017-11-27 14:13:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Message::isEmpty() const
|
|
|
|
|
{
|
|
|
|
|
return sd_bus_message_is_empty((sd_bus_message*)msg_);
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-10 08:47:59 +01:00
|
|
|
void MethodCall::dontExpectReply()
|
|
|
|
|
{
|
2019-03-25 16:28:31 +01:00
|
|
|
auto r = sd_bus_message_set_expect_reply((sd_bus_message*)msg_, 0);
|
2019-01-10 08:47:59 +01:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to set the dont-expect-reply flag", -r);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MethodCall::doesntExpectReply() const
|
|
|
|
|
{
|
2019-03-25 16:28:31 +01:00
|
|
|
auto r = sd_bus_message_get_expect_reply((sd_bus_message*)msg_);
|
2019-01-10 08:47:59 +01:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to get the dont-expect-reply flag", -r);
|
|
|
|
|
return r > 0 ? false : true;
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-02 11:22:00 +02:00
|
|
|
MethodReply MethodCall::send() const
|
2019-01-10 08:47:59 +01:00
|
|
|
{
|
|
|
|
|
if (!doesntExpectReply())
|
|
|
|
|
return sendWithReply();
|
|
|
|
|
else
|
|
|
|
|
return sendWithNoReply();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MethodReply MethodCall::sendWithReply() const
|
2018-07-02 11:22:00 +02:00
|
|
|
{
|
|
|
|
|
sd_bus_error sdbusError = SD_BUS_ERROR_NULL;
|
|
|
|
|
SCOPE_EXIT{ sd_bus_error_free(&sdbusError); };
|
|
|
|
|
|
2019-03-25 16:28:31 +01:00
|
|
|
sd_bus_message* sdbusReply{};
|
|
|
|
|
auto r = sdbus_->sd_bus_call(nullptr, (sd_bus_message*)msg_, 0, &sdbusError, &sdbusReply);
|
2018-07-02 11:22:00 +02:00
|
|
|
|
|
|
|
|
if (sd_bus_error_is_set(&sdbusError))
|
|
|
|
|
throw sdbus::Error(sdbusError.name, sdbusError.message);
|
|
|
|
|
|
|
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to call method", -r);
|
|
|
|
|
|
2019-03-25 16:28:31 +01:00
|
|
|
return MethodReply{sdbusReply, sdbus_, adopt_message};
|
2018-07-02 11:22:00 +02:00
|
|
|
}
|
|
|
|
|
|
2019-01-10 08:47:59 +01:00
|
|
|
MethodReply MethodCall::sendWithNoReply() const
|
|
|
|
|
{
|
2019-03-25 16:28:31 +01:00
|
|
|
auto r = sdbus_->sd_bus_send(nullptr, (sd_bus_message*)msg_, nullptr);
|
2019-01-10 08:47:59 +01:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to call method with no reply", -r);
|
2019-03-25 16:28:31 +01:00
|
|
|
|
2019-01-10 08:47:59 +01:00
|
|
|
return MethodReply{}; // No reply
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-02 11:22:00 +02:00
|
|
|
MethodReply MethodCall::createReply() const
|
|
|
|
|
{
|
2019-03-25 16:28:31 +01:00
|
|
|
sd_bus_message* sdbusReply{};
|
|
|
|
|
auto r = sdbus_->sd_bus_message_new_method_return((sd_bus_message*)msg_, &sdbusReply);
|
2018-07-02 11:22:00 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to create method reply", -r);
|
|
|
|
|
|
2019-03-25 16:28:31 +01:00
|
|
|
return MethodReply{sdbusReply, sdbus_, adopt_message};
|
2018-07-02 11:22:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MethodReply MethodCall::createErrorReply(const Error& error) const
|
2017-11-27 14:13:55 +01:00
|
|
|
{
|
2018-07-02 11:22:00 +02:00
|
|
|
sd_bus_error sdbusError = SD_BUS_ERROR_NULL;
|
|
|
|
|
SCOPE_EXIT{ sd_bus_error_free(&sdbusError); };
|
|
|
|
|
sd_bus_error_set(&sdbusError, error.getName().c_str(), error.getMessage().c_str());
|
|
|
|
|
|
2019-03-25 16:28:31 +01:00
|
|
|
sd_bus_message* sdbusErrorReply{};
|
|
|
|
|
auto r = sdbus_->sd_bus_message_new_method_error((sd_bus_message*)msg_, &sdbusErrorReply, &sdbusError);
|
2018-07-02 11:22:00 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to create method error reply", -r);
|
|
|
|
|
|
2019-03-25 16:28:31 +01:00
|
|
|
return MethodReply{sdbusErrorReply, sdbus_, adopt_message};
|
2018-07-02 11:22:00 +02:00
|
|
|
}
|
|
|
|
|
|
2019-03-25 14:45:48 +01:00
|
|
|
AsyncMethodCall::AsyncMethodCall(MethodCall&& call) noexcept
|
2019-03-25 16:28:31 +01:00
|
|
|
: Message(std::move(call))
|
2019-03-25 14:45:48 +01:00
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void AsyncMethodCall::send(void* callback, void* userData) const
|
|
|
|
|
{
|
2019-03-25 16:28:31 +01:00
|
|
|
auto r = sdbus_->sd_bus_call_async(nullptr, nullptr, (sd_bus_message*)msg_, (sd_bus_message_handler_t)callback, userData, 0);
|
2019-03-25 14:45:48 +01:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to call method asynchronously", -r);
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-02 11:22:00 +02:00
|
|
|
void MethodReply::send() const
|
|
|
|
|
{
|
2019-03-25 16:28:31 +01:00
|
|
|
auto r = sdbus_->sd_bus_send(nullptr, (sd_bus_message*)msg_, nullptr);
|
2018-07-02 11:22:00 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to send reply", -r);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Signal::send() const
|
|
|
|
|
{
|
2019-03-25 16:28:31 +01:00
|
|
|
auto r = sdbus_->sd_bus_send(nullptr, (sd_bus_message*)msg_, nullptr);
|
2018-07-02 11:22:00 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to emit signal", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Message createPlainMessage()
|
|
|
|
|
{
|
|
|
|
|
int r;
|
|
|
|
|
|
|
|
|
|
sd_bus* bus{};
|
2019-03-25 16:28:31 +01:00
|
|
|
SCOPE_EXIT{ sd_bus_unref(bus); };
|
2017-11-27 14:13:55 +01:00
|
|
|
r = sd_bus_default_system(&bus);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to get default system bus", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
2019-03-25 16:28:31 +01:00
|
|
|
thread_local struct BusReferenceKeeper
|
|
|
|
|
{
|
|
|
|
|
BusReferenceKeeper(sd_bus* bus) : bus_(bus) {}
|
|
|
|
|
~BusReferenceKeeper() { sd_bus_unref(bus_); }
|
|
|
|
|
sd_bus* bus_{};
|
|
|
|
|
} busReferenceKeeper{bus};
|
|
|
|
|
|
2017-11-27 14:13:55 +01:00
|
|
|
sd_bus_message* sdbusMsg{};
|
|
|
|
|
r = sd_bus_message_new(bus, &sdbusMsg, _SD_BUS_MESSAGE_TYPE_INVALID);
|
2018-05-25 20:48:20 +02:00
|
|
|
SDBUS_THROW_ERROR_IF(r < 0, "Failed to create a new message", -r);
|
2017-11-27 14:13:55 +01:00
|
|
|
|
2019-03-25 16:28:31 +01:00
|
|
|
thread_local internal::SdBus sdbus;
|
|
|
|
|
|
|
|
|
|
return Message{sdbusMsg, &sdbus, adopt_message};
|
2017-11-27 14:13:55 +01:00
|
|
|
}
|
|
|
|
|
|
2018-07-02 11:22:00 +02:00
|
|
|
}
|