Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add custom forwarding routes feature [19249] #56

Merged
merged 8 commits into from
Sep 26, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@

#include <ddspipe_core/communication/Bridge.hpp>
#include <ddspipe_core/communication/dds/Track.hpp>
#include <ddspipe_core/configuration/RoutesConfiguration.hpp>
#include <ddspipe_core/types/topic/dds/DistributedTopic.hpp>

namespace eprosima {
Expand Down Expand Up @@ -55,7 +56,8 @@ class DdsBridge : public Bridge
const utils::Heritable<types::DistributedTopic>& topic,
const std::shared_ptr<ParticipantsDatabase>& participants_database,
const std::shared_ptr<PayloadPool>& payload_pool,
const std::shared_ptr<utils::SlotThreadPool>& thread_pool);
const std::shared_ptr<utils::SlotThreadPool>& thread_pool,
const RoutesConfiguration& routes_config);

DDSPIPE_CORE_DllAPI
~DdsBridge();
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
// Copyright 2023 Proyectos y Sistemas de Mantenimiento SL (eProsima).
//
// 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 <map>
#include <set>

#include <cpp_utils/Formatter.hpp>

#include <ddspipe_core/configuration/IConfiguration.hpp>
#include <ddspipe_core/types/participant/ParticipantId.hpp>

#include <ddspipe_core/library/library_dll.h>

namespace eprosima {
namespace ddspipe {
namespace core {

/**
* Configuration structure encapsulating the forwarding routes of a \c DdsPipe instance.
*/
struct RoutesConfiguration : public IConfiguration
{

using RoutesMap = std::map<types::ParticipantId, std::set<types::ParticipantId>>;

/////////////////////////
// CONSTRUCTORS
/////////////////////////

DDSPIPE_CORE_DllAPI RoutesConfiguration() = default;

/////////////////////////
// METHODS
/////////////////////////

DDSPIPE_CORE_DllAPI virtual bool is_valid(
Tempate marked this conversation as resolved.
Show resolved Hide resolved
utils::Formatter& error_msg) const noexcept override;

DDSPIPE_CORE_DllAPI bool is_valid(
utils::Formatter& error_msg,
std::map<types::ParticipantId, bool> participant_ids) const noexcept;

/////////////////////////
// OPERATORS
/////////////////////////

DDSPIPE_CORE_DllAPI RoutesMap operator () () const;

/////////////////////////
// VARIABLES
/////////////////////////

RoutesMap routes {};
};

} /* namespace core */
} /* namespace ddspipe */
} /* namespace eprosima */
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
// Copyright 2023 Proyectos y Sistemas de Mantenimiento SL (eProsima).
//
// 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 <set>

#include <cpp_utils/Formatter.hpp>
#include <cpp_utils/memory/Heritable.hpp>

#include <ddspipe_core/configuration/IConfiguration.hpp>
#include <ddspipe_core/configuration/RoutesConfiguration.hpp>
#include <ddspipe_core/types/topic/dds/DistributedTopic.hpp>

#include <ddspipe_core/library/library_dll.h>

namespace eprosima {
namespace ddspipe {
namespace core {

/**
* Configuration structure encapsulating the forwarding routes of a \c DdsPipe instance for a set of topics.
*/
struct TopicRoutesConfiguration : public IConfiguration
{

using TopicRoutesMap = std::map<utils::Heritable<types::DistributedTopic>, RoutesConfiguration>;

/////////////////////////
// CONSTRUCTORS
/////////////////////////

DDSPIPE_CORE_DllAPI TopicRoutesConfiguration() = default;

/////////////////////////
// METHODS
/////////////////////////

DDSPIPE_CORE_DllAPI virtual bool is_valid(
Tempate marked this conversation as resolved.
Show resolved Hide resolved
utils::Formatter& error_msg) const noexcept override;

DDSPIPE_CORE_DllAPI bool is_valid(
utils::Formatter& error_msg,
std::map<types::ParticipantId, bool> participant_ids) const noexcept;

/////////////////////////
// OPERATORS
/////////////////////////

DDSPIPE_CORE_DllAPI TopicRoutesMap operator () () const;

/////////////////////////
// VARIABLES
/////////////////////////

TopicRoutesMap topic_routes {};
};

} /* namespace core */
} /* namespace ddspipe */
} /* namespace eprosima */
22 changes: 19 additions & 3 deletions ddspipe_core/include/ddspipe_core/core/DdsPipe.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -21,10 +21,13 @@

#include <ddspipe_core/communication/dds/DdsBridge.hpp>
#include <ddspipe_core/communication/rpc/RpcBridge.hpp>
#include <ddspipe_core/dynamic/DiscoveryDatabase.hpp>
#include <ddspipe_core/configuration/RoutesConfiguration.hpp>
#include <ddspipe_core/configuration/TopicRoutesConfiguration.hpp>
#include <ddspipe_core/dynamic/AllowedTopicList.hpp>
#include <ddspipe_core/dynamic/DiscoveryDatabase.hpp>
#include <ddspipe_core/dynamic/ParticipantsDatabase.hpp>
#include <ddspipe_core/efficiency/payload/PayloadPool.hpp>

#include <ddspipe_core/library/library_dll.h>

namespace eprosima {
Expand Down Expand Up @@ -65,7 +68,9 @@ class DdsPipe
const std::shared_ptr<ParticipantsDatabase>& participants_database,
const std::shared_ptr<utils::SlotThreadPool>& thread_pool,
const std::set<utils::Heritable<types::DistributedTopic>>& builtin_topics = {},
bool start_enable = false);
bool start_enable = false,
const RoutesConfiguration& routes_config = {},
const TopicRoutesConfiguration& topic_routes_config = {});

/**
* @brief Destroy the DdsPipe object
Expand Down Expand Up @@ -316,8 +321,9 @@ class DdsPipe
*/
std::map<types::RpcTopic, bool> current_services_;

/////
/////////////////////////
// AUXILIAR VARIABLES
/////////////////////////

//! Whether the DdsPipe is currently communicating data or not
bool enabled_;
Expand All @@ -326,6 +332,16 @@ class DdsPipe
* @brief Internal mutex for concurrent calls
*/
mutable std::mutex mutex_;

/////////////////////////
// CONFIGURATION VARIABLES
/////////////////////////

//! Custom forwarding routes
RoutesConfiguration routes_config_;

//! Custom forwarding routes per topic
TopicRoutesConfiguration topic_routes_config_;
};

} /* namespace core */
Expand Down
90 changes: 70 additions & 20 deletions ddspipe_core/src/cpp/communication/dds/DdsBridge.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -27,41 +27,91 @@ DdsBridge::DdsBridge(
const utils::Heritable<DistributedTopic>& topic,
const std::shared_ptr<ParticipantsDatabase>& participants_database,
const std::shared_ptr<PayloadPool>& payload_pool,
const std::shared_ptr<utils::SlotThreadPool>& thread_pool)
const std::shared_ptr<utils::SlotThreadPool>& thread_pool,
const RoutesConfiguration& routes_config)
: Bridge(participants_database, payload_pool, thread_pool)
, topic_(topic)
{
logDebug(DDSPIPE_DDSBRIDGE, "Creating DdsBridge " << *this << ".");

std::set<ParticipantId> ids = participants_->get_participants_ids();
auto routes = routes_config();

std::map<types::ParticipantId, std::shared_ptr<IWriter>> writers;
std::map<types::ParticipantId, std::shared_ptr<IReader>> readers;
// Determine which endpoints need to be created
std::set<types::ParticipantId> writers_to_create;
std::set<types::ParticipantId> readers_to_create;
for (const ParticipantId& id: ids)
{
const auto& it = routes.find(id);
if (it != routes.end())
{
const auto& src_id = it->first;
const auto& dst_ids = it->second;
if (dst_ids.size() != 0) // Only create reader if there are any destination writers
{
readers_to_create.insert(src_id);
writers_to_create.insert(dst_ids.begin(), dst_ids.end());
}
}
else
{
// When no route is defined, forward to all other participants (+ itself if repeater)
auto dst_ids = ids;
if (!participants_->get_participant(id)->is_repeater())
{
// Do not add writer for this participant because it is not repeater
dst_ids.erase(id);
}
writers_to_create.insert(dst_ids.begin(), dst_ids.end());
readers_to_create.insert(id);
}
}

// Generate readers and writers for each participant
for (const auto& id: ids)
// Generate writers for each participant
std::map<types::ParticipantId, std::shared_ptr<IWriter>> writers;
for (const auto& id: writers_to_create)
{
std::shared_ptr<IParticipant> participant = participants_database->get_participant(id);

writers[id] = participant->create_writer(*topic);
}

// Generate readers for each participant
std::map<types::ParticipantId, std::shared_ptr<IReader>> readers;
for (const auto& id: readers_to_create)
{
std::shared_ptr<IParticipant> participant = participants_database->get_participant(id);
readers[id] = participant->create_reader(*topic);
}

// Generate tracks
for (ParticipantId id: ids)
for (const ParticipantId& id: ids)
{
// List of all Participants
std::map<ParticipantId, std::shared_ptr<IWriter>> writers_except_one =
writers; // Create a copy of the map

if (!participants_->get_participant(id)->is_repeater())
std::map<ParticipantId, std::shared_ptr<IWriter>> dst_writers;
auto it = routes.find(id);
if (it != routes.end())
{
// Remove this Track source participant because it is not repeater
writers_except_one.erase(id);

logDebug(
DDSPIPE_DDSBRIDGE,
"Not adding own Writer to Track in " << *this << " in Participant " << id << ".");
// Custom route available for this participant
for (const auto& writer_id : it->second)
{
dst_writers[writer_id] = writers[writer_id];
}

// Do not create track if no destination writers
if (dst_writers.size() == 0)
{
continue;
}
}
else
{
// Use default forwarding route (receiver participant to all others)
dst_writers = writers;

// Remove this Track source participant if not repeater
if (!participants_->get_participant(id)->is_repeater())
{
dst_writers.erase(id);
}
}

// This insert is required as there is no copy method for Track
Expand All @@ -70,8 +120,8 @@ DdsBridge::DdsBridge(
std::make_unique<Track>(
topic,
id,
readers[id],
std::move(writers_except_one),
std::move(readers[id]),
std::move(dst_writers),
payload_pool,
thread_pool);
}
Expand Down
2 changes: 1 addition & 1 deletion ddspipe_core/src/cpp/communication/dds/Track.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ Track::Track(
const std::shared_ptr<utils::SlotThreadPool>& thread_pool) noexcept
: topic_(topic)
, reader_participant_id_(reader_participant_id)
, reader_(reader)
, reader_(std::move(reader))
, writers_(std::move(writers))
, payload_pool_(payload_pool)
, enabled_(false)
Expand Down
Loading
Loading