Skip to content

Commit

Permalink
Merge branch 'RJD-1057-remove-functions-forwarded-to-entity-base-midd…
Browse files Browse the repository at this point in the history
…le' into RJD-1057-remove-functions-forwarded-to-entity-base-with-middle

Signed-off-by: Mateusz Palczuk <mateusz.palczuk@robotec.ai>
  • Loading branch information
TauTheLepton committed Jul 31, 2024
2 parents 2ba6f86 + 8940b3e commit ff759fe
Show file tree
Hide file tree
Showing 11 changed files with 1,030 additions and 32 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,8 @@ class LoadDoNothingPluginScenario : public cpp_mock_scenarios::CppScenarioNode
api_.resetBehaviorPlugin(
"pedestrian_spawn_with_behavior_tree",
traffic_simulator::entity::PedestrianEntity::BuiltinBehavior::doNothing());

/// @note After the reset, the Entity objects are invalidated, so we need to obtain new ones.
if (
api_.getEntity("vehicle_spawn_with_behavior_tree")->getCurrentAction() != "do_nothing" ||
api_.getEntity("pedestrian_spawn_with_behavior_tree")->getCurrentAction() != "do_nothing") {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,7 @@ class DecelerateAndFollowScenario : public cpp_mock_scenarios::CppScenarioNode
traffic_simulator::helper::constructCanonicalizedLaneletPose(
34741, 0.0, 0.0, api_.getHdmapUtils()),
getVehicleParameters());
api_.getEntity("ego")->setLinearVelocity(15);
ego_entity->setLinearVelocity(15);

auto npc_entity = api_.spawn(
"npc",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,14 +44,14 @@ class SynchronizedAction : public cpp_mock_scenarios::CppScenarioNode

void onUpdate() override
{
auto npc = api_.getEntity("npc");
auto ego = api_.getEntity("ego");
auto ego_entity = api_.getEntity("ego");
auto npc_entity = api_.getEntity("npc");

// SUCCESS
if (
npc->requestSynchronize("ego", ego_target, npc_target, 0, 0.5) &&
ego->reachPosition(ego_target, 1.0) && npc->reachPosition(npc_target, 1.0) &&
npc->getCurrentTwist().linear.x < 0.5) {
npc_entity->requestSynchronize("ego", ego_target, npc_target, 0, 0.5) &&
ego_entity->reachPosition(ego_target, 1.0) && npc_entity->reachPosition(npc_target, 1.0) &&
npc_entity->getCurrentTwist().linear.x < 0.5) {
stop(cpp_mock_scenarios::Result::SUCCESS);
}

Expand All @@ -65,34 +65,31 @@ class SynchronizedAction : public cpp_mock_scenarios::CppScenarioNode
}
void onInitialize() override
{
api_.spawn(
auto ego_entity = api_.spawn(
"ego",
traffic_simulator::helper::constructCanonicalizedLaneletPose(
34976, 20, 0, api_.getHdmapUtils()),
getVehicleParameters());

auto ego = api_.getEntity("ego");
ego->setLinearVelocity(3);
ego->requestSpeedChange(3, true);
ego_entity->setLinearVelocity(3);
ego_entity->requestSpeedChange(3, true);

std::vector<geometry_msgs::msg::Pose> goal_poses;
goal_poses.emplace_back(traffic_simulator::helper::constructCanonicalizedLaneletPose(
34579, 20, 0, api_.getHdmapUtils()));
ego->requestAssignRoute(goal_poses);
ego_entity->requestAssignRoute(goal_poses);

api_.spawn(
auto npc_entity = api_.spawn(
"npc",
traffic_simulator::helper::constructCanonicalizedLaneletPose(
34576, 0, 0, api_.getHdmapUtils()),
getVehicleParameters());

auto npc = api_.getEntity("npc");

std::vector<geometry_msgs::msg::Pose> npc_goal_poses;
npc_goal_poses.emplace_back(traffic_simulator::helper::constructCanonicalizedLaneletPose(
34564, 20, 0, api_.getHdmapUtils()));
npc->requestAssignRoute(npc_goal_poses);
npc->setLinearVelocity(6);
npc_entity->requestAssignRoute(npc_goal_poses);
npc_entity->setLinearVelocity(6);
}

auto getSampleLaneletPose(const traffic_simulator::LaneletPose & lanelet_pose)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,14 +44,14 @@ class SynchronizedActionWithSpeed : public cpp_mock_scenarios::CppScenarioNode

void onUpdate() override
{
auto npc = api_.getEntity("npc");
auto ego = api_.getEntity("ego");
auto npc_entity = api_.getEntity("npc");
auto ego_entity = api_.getEntity("ego");

// SUCCESS
if (
npc->requestSynchronize("ego", ego_target, npc_target, 2, 0.5) &&
ego->reachPosition(ego_target, 1.0) && npc->reachPosition(npc_target, 1.0) &&
npc->getCurrentTwist().linear.x < 2.5) {
npc_entity->requestSynchronize("ego", ego_target, npc_target, 2, 0.5) &&
ego_entity->reachPosition(ego_target, 1.0) && npc_entity->reachPosition(npc_target, 1.0) &&
npc_entity->getCurrentTwist().linear.x < 2.5) {
stop(cpp_mock_scenarios::Result::SUCCESS);
}

Expand All @@ -63,36 +63,34 @@ class SynchronizedActionWithSpeed : public cpp_mock_scenarios::CppScenarioNode
stop(cpp_mock_scenarios::Result::FAILURE);
}
}

void onInitialize() override
{
api_.spawn(
auto ego_entity = api_.spawn(
"ego",
traffic_simulator::helper::constructCanonicalizedLaneletPose(
34976, 20, 0, api_.getHdmapUtils()),
getVehicleParameters());

auto ego = api_.getEntity("ego");
ego->setLinearVelocity(3);
ego->requestSpeedChange(3, true);
ego_entity->setLinearVelocity(3);
ego_entity->requestSpeedChange(3, true);

std::vector<geometry_msgs::msg::Pose> goal_poses;
goal_poses.emplace_back(traffic_simulator::helper::constructCanonicalizedLaneletPose(
34579, 20, 0, api_.getHdmapUtils()));
ego->requestAssignRoute(goal_poses);
ego_entity->requestAssignRoute(goal_poses);

api_.spawn(
auto npc_entity = api_.spawn(
"npc",
traffic_simulator::helper::constructCanonicalizedLaneletPose(
34576, 0, 0, api_.getHdmapUtils()),
getVehicleParameters());

auto npc = api_.getEntity("npc");

std::vector<geometry_msgs::msg::Pose> npc_goal_poses;
npc_goal_poses.emplace_back(traffic_simulator::helper::constructCanonicalizedLaneletPose(
34564, 20, 0, api_.getHdmapUtils()));
npc->requestAssignRoute(npc_goal_poses);
npc->setLinearVelocity(6);
npc_entity->requestAssignRoute(npc_goal_poses);
npc_entity->setLinearVelocity(6);
}

auto getSampleLaneletPose(const traffic_simulator::LaneletPose & lanelet_pose)
Expand Down
1 change: 1 addition & 0 deletions mock/cpp_mock_scenarios/src/traffic_simulation_demo.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -156,6 +156,7 @@ class TrafficSimulationDemoScenario : public cpp_mock_scenarios::CppScenarioNode
auto obstacle_entity = api_.spawn(
"obstacle", traffic_simulator::helper::constructPose(10, 5, 0, 0, 0, -1.57),
getMiscObjectParameters());

obstacle_entity->setStatus(
ego_entity->getMapPose(), traffic_simulator::helper::constructPose(10, 5, 0, 0, 0, -1.57),
traffic_simulator::helper::constructActionStatus());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,7 @@ class EntityBase : public std::enable_shared_from_this<EntityBase>
}
}

template <typename T>
template <typename EntityType>
/* */ auto as() const -> std::shared_ptr<const EntityType>
{
if (auto derived = std::dynamic_pointer_cast<const EntityType>(shared_from_this()); !derived) {
Expand Down
18 changes: 18 additions & 0 deletions simulation/traffic_simulator/test/src/expect_eq_macros.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,12 @@
EXPECT_DOUBLE_EQ(DATA0.z, DATA1.z); \
EXPECT_DOUBLE_EQ(DATA0.w, DATA1.w);

#define EXPECT_QUATERNION_EQ_STREAM(DATA0, DATA1, STREAM_MESSAGE) \
EXPECT_DOUBLE_EQ(DATA0.x, DATA1.x) << STREAM_MESSAGE; \
EXPECT_DOUBLE_EQ(DATA0.y, DATA1.y) << STREAM_MESSAGE; \
EXPECT_DOUBLE_EQ(DATA0.z, DATA1.z) << STREAM_MESSAGE; \
EXPECT_DOUBLE_EQ(DATA0.w, DATA1.w) << STREAM_MESSAGE;

#define EXPECT_QUATERNION_NEAR(DATA0, DATA1, EPS) \
EXPECT_NEAR(DATA0.x, DATA1.x, EPS); \
EXPECT_NEAR(DATA0.y, DATA1.y, EPS); \
Expand Down Expand Up @@ -130,4 +136,16 @@
EXPECT_EQ(DATA0.trajectory_shape, DATA1.trajectory_shape); \
EXPECT_LANE_CHANGE_CONSTRAINT_EQ(DATA0.constraint, DATA1.constraint);

#define EXPECT_COLOR_RGBA_EQ_STREAM(DATA0, DATA1, STREAM_MESSAGE) \
EXPECT_DOUBLE_EQ(DATA0.r, DATA1.r) << STREAM_MESSAGE; \
EXPECT_DOUBLE_EQ(DATA0.g, DATA1.g) << STREAM_MESSAGE; \
EXPECT_DOUBLE_EQ(DATA0.b, DATA1.b) << STREAM_MESSAGE; \
EXPECT_DOUBLE_EQ(DATA0.a, DATA1.a) << STREAM_MESSAGE;

#define EXPECT_COLOR_RGBA_NEAR_STREAM(DATA0, DATA1, EPS, STREAM_MESSAGE) \
EXPECT_NEAR(DATA0.r, DATA1.r, EPS) << STREAM_MESSAGE; \
EXPECT_NEAR(DATA0.g, DATA1.g, EPS) << STREAM_MESSAGE; \
EXPECT_NEAR(DATA0.b, DATA1.b, EPS) << STREAM_MESSAGE; \
EXPECT_NEAR(DATA0.a, DATA1.a, EPS) << STREAM_MESSAGE;

#endif // TRAFFIC_SIMULATOR__TEST__EXPECT_EQ_MACROS_HPP_
Original file line number Diff line number Diff line change
@@ -1,2 +1,8 @@
ament_add_gtest(test_traffic_light test_traffic_light.cpp)
target_link_libraries(test_traffic_light traffic_simulator)

ament_add_gtest(test_traffic_lights
test_traffic_lights_internal.cpp
test_traffic_lights.cpp
)
target_link_libraries(test_traffic_lights traffic_simulator)
44 changes: 44 additions & 0 deletions simulation/traffic_simulator/test/src/traffic_lights/helper.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
// Copyright 2024 TIER IV, Inc. All rights reserved.
//
// 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.

#ifndef TRAFFIC_SIMULATOR__TEST__TRAFFIC_LIGHTS__HELPER_HPP_
#define TRAFFIC_SIMULATOR__TEST__TRAFFIC_LIGHTS__HELPER_HPP_

#include <rclcpp/time.hpp>
#include <std_msgs/msg/header.hpp>

/// Helper functions
// clang-format off
inline auto stateFromColor(const std::string & color) -> std::string { return color + " solidOn circle"; }
inline auto stateFromStatus(const std::string & status) -> std::string { return "green " + status + " circle"; }
inline auto stateFromShape(const std::string & shape) -> std::string { return "green solidOn " + shape; }
// clang-format on

/// Returns time in nanoseconds
inline auto getTime(const builtin_interfaces::msg::Time & time) -> int
{
static constexpr int nanosecond_multiplier = static_cast<int>(1e+9);
return static_cast<int>(time.sec) * nanosecond_multiplier + static_cast<int>(time.nanosec);
}

/// Returns time in nanoseconds
inline auto getTime(const std_msgs::msg::Header & header) -> int { return getTime(header.stamp); }

/// Returns time in nanoseconds
inline auto getTime(const rclcpp::Time & time) -> int
{
return static_cast<int>(time.nanoseconds());
}

#endif // TRAFFIC_SIMULATOR__TEST__TRAFFIC_LIGHTS__HELPER_HPP_
Original file line number Diff line number Diff line change
@@ -0,0 +1,151 @@
// Copyright 2024 TIER IV, Inc. All rights reserved.
//
// 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 <gtest/gtest.h>

#include <ament_index_cpp/get_package_share_directory.hpp>
#include <scenario_simulator_exception/exception.hpp>
#include <std_msgs/msg/header.hpp>
#include <traffic_simulator/traffic_lights/traffic_lights.hpp>
#include <traffic_simulator/traffic_lights/traffic_lights_base.hpp>

#include "../expect_eq_macros.hpp"
#include "helper.hpp"

constexpr double timing_eps = 1e-3;
constexpr double frequency_eps = 0.5;

class TrafficLightsTest : public testing::Test
{
public:
const lanelet::Id id = 34836;
const lanelet::Id signal_id = 34806;

const rclcpp::Node::SharedPtr node_ptr = rclcpp::Node::make_shared("TrafficLightsTest");

const std::string path =
ament_index_cpp::get_package_share_directory("traffic_simulator") + "/map/lanelet2_map.osm";

const std::shared_ptr<hdmap_utils::HdMapUtils> hdmap_utils_ptr =
std::make_shared<hdmap_utils::HdMapUtils>(
path, geographic_msgs::build<geographic_msgs::msg::GeoPoint>()
.latitude(35.61836750154)
.longitude(139.78066608243)
.altitude(0.0));

const std::string red_state = stateFromColor("red");
const std::string yellow_state = "yellow flashing circle";

std::unique_ptr<traffic_simulator::TrafficLights> lights =
std::make_unique<traffic_simulator::TrafficLights>(node_ptr, hdmap_utils_ptr, "awf/universe");
};

TEST_F(TrafficLightsTest, isAnyTrafficLightChanged)
{
EXPECT_TRUE(lights->isAnyTrafficLightChanged());
}

TEST_F(TrafficLightsTest, getConventionalTrafficLights)
{
{
this->lights->getConventionalTrafficLights()->setTrafficLightsState(this->id, this->red_state);

const auto actual_state =
this->lights->getConventionalTrafficLights()->getTrafficLightsComposedState(this->id);

EXPECT_EQ(actual_state, this->red_state);
}
{
this->lights->getConventionalTrafficLights()->setTrafficLightsState(
this->id, this->yellow_state);

const auto actual_state =
this->lights->getConventionalTrafficLights()->getTrafficLightsComposedState(this->id);

EXPECT_EQ(actual_state, this->yellow_state);
}
}

TEST_F(TrafficLightsTest, getV2ITrafficLights)
{
{
this->lights->getV2ITrafficLights()->setTrafficLightsState(this->id, this->red_state);

const auto actual_state =
this->lights->getV2ITrafficLights()->getTrafficLightsComposedState(this->id);

EXPECT_EQ(actual_state, this->red_state);
}
{
this->lights->getV2ITrafficLights()->setTrafficLightsState(this->id, this->yellow_state);

const auto actual_state =
this->lights->getV2ITrafficLights()->getTrafficLightsComposedState(this->id);

EXPECT_EQ(actual_state, this->yellow_state);
}
}

TEST_F(TrafficLightsTest, startTrafficLightsUpdate)
{
this->lights->getConventionalTrafficLights()->setTrafficLightsState(this->id, this->red_state);
this->lights->getV2ITrafficLights()->setTrafficLightsState(this->id, this->red_state);

std::vector<visualization_msgs::msg::MarkerArray> markers;

rclcpp::Subscription<visualization_msgs::msg::MarkerArray>::SharedPtr subscriber =
this->node_ptr->template create_subscription<visualization_msgs::msg::MarkerArray>(
"traffic_light/marker", 10,
[&markers](const visualization_msgs::msg::MarkerArray::SharedPtr msg_in) {
markers.push_back(*msg_in);
});

this->lights->startTrafficLightsUpdate(20.0, 10.0);

// start time is required to be measured here and not from first message, because there are two publishers publishing to this topic at the same time
const auto start_time = node_ptr->now();

// spin for 1 second
const auto end = std::chrono::system_clock::now() + std::chrono::milliseconds(1020);
while (std::chrono::system_clock::now() < end) {
rclcpp::spin_some(this->node_ptr);
}

std::vector<std_msgs::msg::Header> headers;

// verify
for (std::size_t i = 0; i < markers.size(); ++i) {
const auto & one_marker = markers[i].markers;
EXPECT_EQ(one_marker.size(), static_cast<std::size_t>(1));

if (
one_marker.front().header.stamp.sec != 0 and one_marker.front().header.stamp.nanosec != 0u) {
headers.push_back(one_marker.front().header);
}
}

// verify message timing
const double expected_frequency = 30.0;
const double actual_frequency =
static_cast<double>(headers.size()) /
static_cast<double>(getTime(headers.back()) - getTime(start_time)) * 1e+9;
EXPECT_NEAR(actual_frequency, expected_frequency, frequency_eps);
}

int main(int argc, char ** argv)
{
testing::InitGoogleTest(&argc, argv);
rclcpp::init(argc, argv);
return RUN_ALL_TESTS();
}
Loading

0 comments on commit ff759fe

Please sign in to comment.