rcl/rcl_action/test/rcl_action/test_action_interaction.cpp
Chris Lalancette 885fa21a89 Make sure to check the return value of rcl APIs. (#838)
This is just a further check to ensure the test is correct,
and also gets rid of a slew of dead store warnings from
clang static analysis.

Signed-off-by: Chris Lalancette <clalancette@openrobotics.org>
2020-11-02 16:22:11 +01:00

857 lines
35 KiB
C++

// Copyright 2018 Open Source Robotics Foundation, Inc.
//
// 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 "osrf_testing_tools_cpp/scope_exit.hpp"
#include "rcl_action/action_client.h"
#include "rcl_action/action_server.h"
#include "rcl_action/wait.h"
#include "rcl/error_handling.h"
#include "rcl/rcl.h"
#include "rosidl_runtime_c/primitives_sequence_functions.h"
#include "test_msgs/action/fibonacci.h"
#ifdef RMW_IMPLEMENTATION
# define CLASSNAME_(NAME, SUFFIX) NAME ## __ ## SUFFIX
# define CLASSNAME(NAME, SUFFIX) CLASSNAME_(NAME, SUFFIX)
#else
# define CLASSNAME(NAME, SUFFIX) NAME
#endif
class CLASSNAME (TestActionClientServerInteraction, RMW_IMPLEMENTATION) : public ::testing::Test
{
protected:
void SetUp() override
{
test_msgs__action__Fibonacci_SendGoal_Request__init(&this->outgoing_goal_request);
test_msgs__action__Fibonacci_SendGoal_Request__init(&this->incoming_goal_request);
test_msgs__action__Fibonacci_SendGoal_Response__init(&this->outgoing_goal_response);
test_msgs__action__Fibonacci_SendGoal_Response__init(&this->incoming_goal_response);
test_msgs__action__Fibonacci_GetResult_Request__init(&this->outgoing_result_request);
test_msgs__action__Fibonacci_GetResult_Request__init(&this->incoming_result_request);
test_msgs__action__Fibonacci_GetResult_Response__init(&this->outgoing_result_response);
test_msgs__action__Fibonacci_GetResult_Response__init(&this->incoming_result_response);
test_msgs__action__Fibonacci_FeedbackMessage__init(&this->outgoing_feedback);
test_msgs__action__Fibonacci_FeedbackMessage__init(&this->incoming_feedback);
rcl_allocator_t allocator = rcl_get_default_allocator();
rcl_ret_t ret;
rcl_init_options_t init_options = rcl_get_zero_initialized_init_options();
ret = rcl_init_options_init(&init_options, allocator);
ASSERT_EQ(RCL_RET_OK, ret) << rcl_get_error_string().str;
OSRF_TESTING_TOOLS_CPP_SCOPE_EXIT(
{
EXPECT_EQ(RCL_RET_OK, rcl_init_options_fini(&init_options)) << rcl_get_error_string().str;
});
context = rcl_get_zero_initialized_context();
ret = rcl_init(0, nullptr, &init_options, &context);
ASSERT_EQ(RCL_RET_OK, ret) << rcl_get_error_string().str;
this->node = rcl_get_zero_initialized_node();
rcl_node_options_t node_options = rcl_node_get_default_options();
ret = rcl_node_init(&this->node, "test_action_communication_node", "", &context, &node_options);
ASSERT_EQ(RCL_RET_OK, ret) << rcl_get_error_string().str;
ret = rcl_clock_init(RCL_STEADY_TIME, &this->clock, &allocator);
ASSERT_EQ(RCL_RET_OK, ret) << rcl_get_error_string().str;
const rosidl_action_type_support_t * ts = ROSIDL_GET_ACTION_TYPE_SUPPORT(
test_msgs, Fibonacci);
const char * action_name = "test_action_commmunication_name";
const rcl_action_server_options_t server_options = rcl_action_server_get_default_options();
this->action_server = rcl_action_get_zero_initialized_server();
ret = rcl_action_server_init(
&this->action_server, &this->node, &this->clock, ts, action_name, &server_options);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
const rcl_action_client_options_t client_options = rcl_action_client_get_default_options();
this->action_client = rcl_action_get_zero_initialized_client();
ret = rcl_action_client_init(
&this->action_client, &this->node, ts, action_name, &client_options);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
size_t num_subscriptions_server;
size_t num_guard_conditions_server;
size_t num_timers_server;
size_t num_clients_server;
size_t num_services_server;
size_t num_subscriptions_client;
size_t num_guard_conditions_client;
size_t num_timers_client;
size_t num_clients_client;
size_t num_services_client;
this->wait_set = rcl_get_zero_initialized_wait_set();
ret = rcl_action_server_wait_set_get_num_entities(
&this->action_server,
&num_subscriptions_server,
&num_guard_conditions_server,
&num_timers_server,
&num_clients_server,
&num_services_server);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_client_wait_set_get_num_entities(
&this->action_client,
&num_subscriptions_client,
&num_guard_conditions_client,
&num_timers_client,
&num_clients_client,
&num_services_client);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_init(
&this->wait_set,
num_subscriptions_server + num_subscriptions_client,
num_guard_conditions_server + num_guard_conditions_client,
num_timers_server + num_timers_client,
num_clients_server + num_clients_client,
num_services_server + num_services_client,
0 /* num_events_server + num_events_client */,
&context,
rcl_get_default_allocator());
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
}
void TearDown() override
{
// Finalize
test_msgs__action__Fibonacci_SendGoal_Request__fini(&this->outgoing_goal_request);
test_msgs__action__Fibonacci_SendGoal_Request__fini(&this->incoming_goal_request);
test_msgs__action__Fibonacci_SendGoal_Response__fini(&this->incoming_goal_response);
test_msgs__action__Fibonacci_SendGoal_Response__fini(&this->outgoing_goal_response);
test_msgs__action__Fibonacci_GetResult_Request__fini(&this->incoming_result_request);
test_msgs__action__Fibonacci_GetResult_Request__fini(&this->outgoing_result_request);
test_msgs__action__Fibonacci_GetResult_Response__fini(&this->incoming_result_response);
test_msgs__action__Fibonacci_GetResult_Response__fini(&this->outgoing_result_response);
test_msgs__action__Fibonacci_FeedbackMessage__fini(&this->outgoing_feedback);
test_msgs__action__Fibonacci_FeedbackMessage__fini(&this->incoming_feedback);
rcl_ret_t ret = rcl_action_server_fini(&this->action_server, &this->node);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_clock_fini(&this->clock);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_client_fini(&this->action_client, &this->node);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_node_fini(&this->node);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_fini(&this->wait_set);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_shutdown(&context);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_context_fini(&this->context);
EXPECT_EQ(RCL_RET_OK, ret) << rcl_get_error_string().str;
}
void init_test_uuid0(uint8_t * uuid)
{
for (uint8_t i = 0; i < UUID_SIZE; ++i) {
uuid[i] = i;
}
}
void init_test_uuid1(uint8_t * uuid)
{
for (uint8_t i = 0; i < UUID_SIZE; ++i) {
uuid[i] = 15 - i;
}
}
test_msgs__action__Fibonacci_SendGoal_Request outgoing_goal_request;
test_msgs__action__Fibonacci_SendGoal_Request incoming_goal_request;
test_msgs__action__Fibonacci_SendGoal_Response outgoing_goal_response;
test_msgs__action__Fibonacci_SendGoal_Response incoming_goal_response;
test_msgs__action__Fibonacci_GetResult_Request outgoing_result_request;
test_msgs__action__Fibonacci_GetResult_Request incoming_result_request;
test_msgs__action__Fibonacci_GetResult_Response outgoing_result_response;
test_msgs__action__Fibonacci_GetResult_Response incoming_result_response;
test_msgs__action__Fibonacci_FeedbackMessage outgoing_feedback;
test_msgs__action__Fibonacci_FeedbackMessage incoming_feedback;
rcl_action_client_t action_client;
rcl_action_server_t action_server;
rcl_node_t node;
rcl_clock_t clock;
rcl_context_t context;
rcl_wait_set_t wait_set;
bool is_goal_request_ready;
bool is_cancel_request_ready;
bool is_result_request_ready;
bool is_goal_expired;
bool is_feedback_ready;
bool is_status_ready;
bool is_goal_response_ready;
bool is_cancel_response_ready;
bool is_result_response_ready;
}; // class TestActionClientServerInteraction
// Exercises the "Example 1" sequence diagram found in the actions_proposal document.
// In this example, the action client request a goal and gets a response from
// the server accepting the goal (synchronous). Upon accepting the goal, the
// action server starts the required tasks for completing the goal.
// Following the goal request, the client makes an asynchronous request for the
// result. The feedback is published to the action client as it executes the goal.
// Ultimately, a result message is populated which is then used as part of the result response.
TEST_F(CLASSNAME(TestActionClientServerInteraction, RMW_IMPLEMENTATION), test_interaction)
{
// Initialize goal request
init_test_uuid0(this->outgoing_goal_request.goal_id.uuid);
this->outgoing_goal_request.goal.order = 10;
// Send goal request with valid arguments
int64_t sequence_number;
rcl_ret_t ret = rcl_action_send_goal_request(
&this->action_client, &this->outgoing_goal_request, &sequence_number);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_server(&this->wait_set, &this->action_server, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Wait for goal request to be ready
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_server_wait_set_get_entities_ready(
&this->wait_set,
&this->action_server,
&this->is_goal_request_ready,
&this->is_cancel_request_ready,
&this->is_result_request_ready,
&this->is_goal_expired);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_goal_request_ready) << rcl_get_error_string().str;
EXPECT_FALSE(this->is_cancel_request_ready) << rcl_get_error_string().str;
EXPECT_FALSE(this->is_result_request_ready) << rcl_get_error_string().str;
// Take goal request with valid arguments
rmw_request_id_t request_header;
ret = rcl_action_take_goal_request(
&this->action_server, &request_header, &this->incoming_goal_request);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that the goal request was received correctly
EXPECT_EQ(this->outgoing_goal_request.goal.order, this->incoming_goal_request.goal.order);
EXPECT_TRUE(
uuidcmp(
this->outgoing_goal_request.goal_id.uuid,
this->incoming_goal_request.goal_id.uuid));
// Initialize goal response
this->outgoing_goal_response.accepted = true;
this->outgoing_goal_response.stamp.sec = 123;
this->outgoing_goal_response.stamp.nanosec = 456789u;
// Send goal response with valid arguments
ret = rcl_action_send_goal_response(
&this->action_server, &request_header, &this->outgoing_goal_response);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_client(
&this->wait_set, &this->action_client, NULL, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Wait for goal response to be ready
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_client_wait_set_get_entities_ready(
&this->wait_set,
&this->action_client,
&this->is_feedback_ready,
&this->is_status_ready,
&this->is_goal_response_ready,
&this->is_cancel_response_ready,
&this->is_result_response_ready);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_goal_response_ready);
EXPECT_FALSE(this->is_cancel_response_ready);
EXPECT_FALSE(this->is_feedback_ready);
EXPECT_FALSE(this->is_status_ready);
EXPECT_FALSE(this->is_result_response_ready);
// Take goal response with valid arguments
ret = rcl_action_take_goal_response(
&this->action_client, &request_header, &this->incoming_goal_response);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that the goal response was received correctly
EXPECT_EQ(this->outgoing_goal_response.accepted, this->incoming_goal_response.accepted);
EXPECT_EQ(this->outgoing_goal_response.stamp.sec, this->incoming_goal_response.stamp.sec);
EXPECT_EQ(this->outgoing_goal_response.stamp.nanosec, this->incoming_goal_response.stamp.nanosec);
// Initialize result request
init_test_uuid0(this->outgoing_result_request.goal_id.uuid);
// Send result request with valid arguments
ret = rcl_action_send_result_request(
&this->action_client, &this->outgoing_result_request, &sequence_number);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Initialize feedback
ASSERT_TRUE(
rosidl_runtime_c__int32__Sequence__init(
&this->outgoing_feedback.feedback.sequence, 3));
this->outgoing_feedback.feedback.sequence.data[0] = 0;
this->outgoing_feedback.feedback.sequence.data[1] = 1;
this->outgoing_feedback.feedback.sequence.data[2] = 2;
init_test_uuid0(this->outgoing_feedback.goal_id.uuid);
// Publish feedback with valid arguments
ret = rcl_action_publish_feedback(&this->action_server, &this->outgoing_feedback);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_client(
&this->wait_set, &this->action_client, NULL, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Wait for feedback to be ready
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_client_wait_set_get_entities_ready(
&this->wait_set,
&this->action_client,
&this->is_feedback_ready,
&this->is_status_ready,
&this->is_goal_response_ready,
&this->is_cancel_response_ready,
&this->is_result_response_ready);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_feedback_ready);
EXPECT_FALSE(this->is_status_ready);
EXPECT_FALSE(this->is_result_response_ready);
EXPECT_FALSE(this->is_cancel_response_ready);
EXPECT_FALSE(this->is_goal_response_ready);
// Take feedback with valid arguments
ret = rcl_action_take_feedback(&this->action_client, &this->incoming_feedback);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that feedback was received correctly
EXPECT_TRUE(
uuidcmp(
this->outgoing_feedback.goal_id.uuid,
this->incoming_feedback.goal_id.uuid));
ASSERT_EQ(
this->outgoing_feedback.feedback.sequence.size,
this->incoming_feedback.feedback.sequence.size);
EXPECT_TRUE(
!memcmp(
this->outgoing_feedback.feedback.sequence.data,
this->incoming_feedback.feedback.sequence.data,
this->outgoing_feedback.feedback.sequence.size));
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_server(&this->wait_set, &this->action_server, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Wait for result request to be ready
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_server_wait_set_get_entities_ready(
&this->wait_set,
&this->action_server,
&this->is_goal_request_ready,
&this->is_cancel_request_ready,
&this->is_result_request_ready,
&this->is_goal_expired);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_result_request_ready);
EXPECT_FALSE(this->is_cancel_request_ready);
EXPECT_FALSE(this->is_goal_request_ready);
// Take result request with valid arguments
// rmw_request_id_t request_header;
ret = rcl_action_take_result_request(
&this->action_server, &request_header, &this->incoming_result_request);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that the result request was received correctly
EXPECT_TRUE(
uuidcmp(
this->outgoing_result_request.goal_id.uuid,
this->incoming_result_request.goal_id.uuid));
// Initialize result response
ASSERT_TRUE(
rosidl_runtime_c__int32__Sequence__init(
&this->outgoing_result_response.result.sequence, 4));
this->outgoing_result_response.result.sequence.data[0] = 0;
this->outgoing_result_response.result.sequence.data[1] = 1;
this->outgoing_result_response.result.sequence.data[2] = 2;
this->outgoing_result_response.result.sequence.data[3] = 6;
this->outgoing_result_response.status =
action_msgs__msg__GoalStatus__STATUS_SUCCEEDED;
// Send result response with valid arguments
ret = rcl_action_send_result_response(
&this->action_server, &request_header, &this->outgoing_result_response);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_client(
&this->wait_set, &this->action_client, NULL, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Wait for result response to be ready
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_client_wait_set_get_entities_ready(
&this->wait_set,
&this->action_client,
&this->is_feedback_ready,
&this->is_status_ready,
&this->is_goal_response_ready,
&this->is_cancel_response_ready,
&this->is_result_response_ready);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_result_response_ready);
EXPECT_FALSE(this->is_cancel_response_ready);
EXPECT_FALSE(this->is_feedback_ready);
EXPECT_FALSE(this->is_status_ready);
EXPECT_FALSE(this->is_goal_response_ready);
// Take result response with valid arguments
ret = rcl_action_take_result_response(
&this->action_client, &request_header, &this->incoming_result_response);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that the result response was received correctly
EXPECT_EQ(
this->outgoing_result_response.status,
this->incoming_result_response.status);
ASSERT_EQ(
this->outgoing_result_response.result.sequence.size,
this->incoming_result_response.result.sequence.size);
EXPECT_TRUE(
!memcmp(
this->outgoing_result_response.result.sequence.data,
this->incoming_result_response.result.sequence.data,
this->outgoing_result_response.result.sequence.size));
}
// Exercises the "Example 2" sequence diagram found in the actions_proposal document.
// This example is almost identical to the first, but this time the action client requests
// for the goal to be canceled mid-execution. Note that it is allowed to perform any shutdown
// operations after the cancel request before returning with the cancellation result.
TEST_F(
CLASSNAME(TestActionClientServerInteraction, RMW_IMPLEMENTATION), test_interaction_with_cancel)
{
action_msgs__srv__CancelGoal_Request outgoing_cancel_request;
action_msgs__srv__CancelGoal_Request incoming_cancel_request;
action_msgs__srv__CancelGoal_Response outgoing_cancel_response;
action_msgs__srv__CancelGoal_Response incoming_cancel_response;
action_msgs__srv__CancelGoal_Request__init(&outgoing_cancel_request);
action_msgs__srv__CancelGoal_Request__init(&incoming_cancel_request);
action_msgs__srv__CancelGoal_Response__init(&outgoing_cancel_response);
action_msgs__srv__CancelGoal_Response__init(&incoming_cancel_response);
// Initialize goal request
init_test_uuid0(this->outgoing_goal_request.goal_id.uuid);
this->outgoing_goal_request.goal.order = 10;
// Send goal request with valid arguments
int64_t sequence_number;
rcl_ret_t ret = rcl_action_send_goal_request(
&this->action_client, &this->outgoing_goal_request, &sequence_number);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_server(&this->wait_set, &this->action_server, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Wait for goal request to be ready
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_server_wait_set_get_entities_ready(
&this->wait_set,
&this->action_server,
&this->is_goal_request_ready,
&this->is_cancel_request_ready,
&this->is_result_request_ready,
&this->is_goal_expired);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_goal_request_ready) << rcl_get_error_string().str;
EXPECT_FALSE(this->is_cancel_request_ready) << rcl_get_error_string().str;
EXPECT_FALSE(this->is_result_request_ready) << rcl_get_error_string().str;
// Take goal request with valid arguments
rmw_request_id_t request_header;
ret = rcl_action_take_goal_request(
&this->action_server, &request_header, &this->incoming_goal_request);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that the goal request was received correctly
EXPECT_EQ(this->outgoing_goal_request.goal.order, this->incoming_goal_request.goal.order);
EXPECT_TRUE(
uuidcmp(
this->outgoing_goal_request.goal_id.uuid,
this->incoming_goal_request.goal_id.uuid));
// Initialize goal response
this->outgoing_goal_response.accepted = true;
this->outgoing_goal_response.stamp.sec = 123;
this->outgoing_goal_response.stamp.nanosec = 456789u;
// Send goal response with valid arguments
ret = rcl_action_send_goal_response(
&this->action_server, &request_header, &this->outgoing_goal_response);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_client(
&this->wait_set, &this->action_client, NULL, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Wait for goal response to be ready
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_client_wait_set_get_entities_ready(
&this->wait_set,
&this->action_client,
&this->is_feedback_ready,
&this->is_status_ready,
&this->is_goal_response_ready,
&this->is_cancel_response_ready,
&this->is_result_response_ready);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_goal_response_ready);
EXPECT_FALSE(this->is_cancel_response_ready);
EXPECT_FALSE(this->is_feedback_ready);
EXPECT_FALSE(this->is_status_ready);
EXPECT_FALSE(this->is_result_response_ready);
// Take goal response with valid arguments
ret = rcl_action_take_goal_response(
&this->action_client, &request_header, &this->incoming_goal_response);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that the goal response was received correctly
EXPECT_EQ(this->outgoing_goal_response.accepted, this->incoming_goal_response.accepted);
EXPECT_EQ(this->outgoing_goal_response.stamp.sec, this->incoming_goal_response.stamp.sec);
EXPECT_EQ(this->outgoing_goal_response.stamp.nanosec, this->incoming_goal_response.stamp.nanosec);
// Initialize result request
init_test_uuid0(this->outgoing_result_request.goal_id.uuid);
// Send result request with valid arguments
ret = rcl_action_send_result_request(
&this->action_client, &this->outgoing_result_request, &sequence_number);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Initialize feedback
ASSERT_TRUE(
rosidl_runtime_c__int32__Sequence__init(
&this->outgoing_feedback.feedback.sequence, 3));
this->outgoing_feedback.feedback.sequence.data[0] = 0;
this->outgoing_feedback.feedback.sequence.data[1] = 1;
this->outgoing_feedback.feedback.sequence.data[2] = 2;
init_test_uuid0(this->outgoing_feedback.goal_id.uuid);
// Publish feedback with valid arguments
ret = rcl_action_publish_feedback(&this->action_server, &this->outgoing_feedback);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_client(
&this->wait_set, &this->action_client, NULL, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Wait for feedback to be ready
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_client_wait_set_get_entities_ready(
&this->wait_set,
&this->action_client,
&this->is_feedback_ready,
&this->is_status_ready,
&this->is_goal_response_ready,
&this->is_cancel_response_ready,
&this->is_result_response_ready);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_feedback_ready);
EXPECT_FALSE(this->is_status_ready);
EXPECT_FALSE(this->is_result_response_ready);
EXPECT_FALSE(this->is_cancel_response_ready);
EXPECT_FALSE(this->is_goal_response_ready);
// Take feedback with valid arguments
ret = rcl_action_take_feedback(&this->action_client, &this->incoming_feedback);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that feedback was received correctly
EXPECT_TRUE(
uuidcmp(
this->outgoing_feedback.goal_id.uuid,
this->incoming_feedback.goal_id.uuid));
ASSERT_EQ(
this->outgoing_feedback.feedback.sequence.size,
this->incoming_feedback.feedback.sequence.size);
EXPECT_TRUE(
!memcmp(
this->outgoing_feedback.feedback.sequence.data,
this->incoming_feedback.feedback.sequence.data,
this->outgoing_feedback.feedback.sequence.size));
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_server(&this->wait_set, &this->action_server, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Wait for result request to be ready
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_server_wait_set_get_entities_ready(
&this->wait_set,
&this->action_server,
&this->is_goal_request_ready,
&this->is_cancel_request_ready,
&this->is_result_request_ready,
&this->is_goal_expired);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_result_request_ready);
EXPECT_FALSE(this->is_cancel_request_ready);
EXPECT_FALSE(this->is_goal_request_ready);
// Take result request with valid arguments
// rmw_request_id_t request_header;
ret = rcl_action_take_result_request(
&this->action_server, &request_header, &this->incoming_result_request);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that the result request was received correctly
EXPECT_TRUE(
uuidcmp(
this->outgoing_result_request.goal_id.uuid,
this->incoming_result_request.goal_id.uuid));
// Initialize result response
ASSERT_TRUE(
rosidl_runtime_c__int32__Sequence__init(
&this->outgoing_result_response.result.sequence, 4));
this->outgoing_result_response.result.sequence.data[0] = 0;
this->outgoing_result_response.result.sequence.data[1] = 1;
this->outgoing_result_response.result.sequence.data[2] = 2;
this->outgoing_result_response.result.sequence.data[3] = 6;
this->outgoing_result_response.status =
action_msgs__msg__GoalStatus__STATUS_SUCCEEDED;
// Initialize cancel request
rmw_request_id_t cancel_request_header;
init_test_uuid0(outgoing_cancel_request.goal_info.goal_id.uuid);
outgoing_cancel_request.goal_info.stamp.sec = 321;
outgoing_cancel_request.goal_info.stamp.nanosec = 987654u;
// Send cancel request with valid arguments
ret = rcl_action_send_cancel_request(
&this->action_client, &outgoing_cancel_request, &sequence_number);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_server(&this->wait_set, &this->action_server, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_server_wait_set_get_entities_ready(
&this->wait_set,
&this->action_server,
&this->is_goal_request_ready,
&this->is_cancel_request_ready,
&this->is_result_request_ready,
&this->is_goal_expired);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_cancel_request_ready);
EXPECT_FALSE(this->is_goal_request_ready);
EXPECT_FALSE(this->is_result_request_ready);
// Take cancel request with valid arguments
ret = rcl_action_take_cancel_request(
&this->action_server, &cancel_request_header, &incoming_cancel_request);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that the cancel request was received correctly
EXPECT_TRUE(
uuidcmp(
outgoing_cancel_request.goal_info.goal_id.uuid,
incoming_cancel_request.goal_info.goal_id.uuid));
EXPECT_EQ(
outgoing_cancel_request.goal_info.stamp.sec,
incoming_cancel_request.goal_info.stamp.sec);
EXPECT_EQ(
outgoing_cancel_request.goal_info.stamp.nanosec,
incoming_cancel_request.goal_info.stamp.nanosec);
// Get a list of goal info that should be attempted to be cancelled
rcl_action_cancel_response_t cancel_response = rcl_action_get_zero_initialized_cancel_response();
ret = rcl_action_process_cancel_request(
&this->action_server, &incoming_cancel_request, &cancel_response);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Send cancel response with valid arguments
// rmw_request_id_t response_header;
ret = rcl_action_send_cancel_response(
&this->action_server, &cancel_request_header, &outgoing_cancel_response);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_client(
&this->wait_set, &this->action_client, NULL, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_client_wait_set_get_entities_ready(
&this->wait_set,
&this->action_client,
&this->is_feedback_ready,
&this->is_status_ready,
&this->is_goal_response_ready,
&this->is_cancel_response_ready,
&this->is_result_response_ready);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_cancel_response_ready);
EXPECT_FALSE(this->is_feedback_ready);
EXPECT_FALSE(this->is_status_ready);
EXPECT_FALSE(this->is_goal_response_ready);
EXPECT_FALSE(this->is_result_response_ready);
// Take cancel response with valid arguments
ret = rcl_action_take_cancel_response(
&this->action_client, &cancel_request_header, &incoming_cancel_response);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that the cancel response was received correctly
ASSERT_EQ(
outgoing_cancel_response.goals_canceling.size,
incoming_cancel_response.goals_canceling.size);
for (size_t i = 0; i < outgoing_cancel_response.goals_canceling.size; ++i) {
const action_msgs__msg__GoalInfo * outgoing_goal_info =
&outgoing_cancel_response.goals_canceling.data[i];
const action_msgs__msg__GoalInfo * incoming_goal_info =
&incoming_cancel_response.goals_canceling.data[i];
EXPECT_TRUE(uuidcmp(outgoing_goal_info->goal_id.uuid, incoming_goal_info->goal_id.uuid));
EXPECT_EQ(outgoing_goal_info->stamp.sec, incoming_goal_info->stamp.sec);
EXPECT_EQ(outgoing_goal_info->stamp.nanosec, incoming_goal_info->stamp.nanosec);
}
// Send result response with valid arguments
ret = rcl_action_send_result_response(
&this->action_server, &request_header, &this->outgoing_result_response);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_wait_set_clear(&this->wait_set);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_wait_set_add_action_client(
&this->wait_set, &this->action_client, NULL, NULL);
ASSERT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Wait for result response to be ready
ret = rcl_wait(&this->wait_set, RCL_S_TO_NS(1));
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
ret = rcl_action_client_wait_set_get_entities_ready(
&this->wait_set,
&this->action_client,
&this->is_feedback_ready,
&this->is_status_ready,
&this->is_goal_response_ready,
&this->is_cancel_response_ready,
&this->is_result_response_ready);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
EXPECT_TRUE(this->is_result_response_ready);
EXPECT_FALSE(this->is_cancel_response_ready);
EXPECT_FALSE(this->is_feedback_ready);
EXPECT_FALSE(this->is_status_ready);
EXPECT_FALSE(this->is_goal_response_ready);
// Take result response with valid arguments
ret = rcl_action_take_result_response(
&this->action_client, &request_header, &this->incoming_result_response);
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
// Check that the result response was received correctly
EXPECT_EQ(
this->outgoing_result_response.status,
this->incoming_result_response.status);
ASSERT_EQ(
this->outgoing_result_response.result.sequence.size,
this->incoming_result_response.result.sequence.size);
EXPECT_TRUE(
!memcmp(
this->outgoing_result_response.result.sequence.data,
this->incoming_result_response.result.sequence.data,
this->outgoing_result_response.result.sequence.size));
test_msgs__action__Fibonacci_SendGoal_Request__fini(&this->outgoing_goal_request);
test_msgs__action__Fibonacci_SendGoal_Request__fini(&this->incoming_goal_request);
test_msgs__action__Fibonacci_SendGoal_Response__fini(&this->incoming_goal_response);
test_msgs__action__Fibonacci_SendGoal_Response__fini(&this->outgoing_goal_response);
test_msgs__action__Fibonacci_GetResult_Request__fini(&this->incoming_result_request);
test_msgs__action__Fibonacci_GetResult_Request__fini(&this->outgoing_result_request);
test_msgs__action__Fibonacci_GetResult_Response__fini(&this->incoming_result_response);
test_msgs__action__Fibonacci_GetResult_Response__fini(&this->outgoing_result_response);
action_msgs__srv__CancelGoal_Request__fini(&incoming_cancel_request);
action_msgs__srv__CancelGoal_Request__fini(&outgoing_cancel_request);
action_msgs__srv__CancelGoal_Response__fini(&incoming_cancel_response);
action_msgs__srv__CancelGoal_Response__fini(&outgoing_cancel_response);
}