15 #ifndef RCLCPP__SERVICE_HPP_
16 #define RCLCPP__SERVICE_HPP_
26 #include "rcl/error_handling.h"
27 #include "rcl/event_callback.h"
30 #include "rmw/error_handling.h"
31 #include "rmw/impl/cpp/demangle.hpp"
34 #include "tracetools/tracetools.h"
36 #include "rclcpp/any_service_callback.hpp"
37 #include "rclcpp/detail/cpp_callback_trampoline.hpp"
38 #include "rclcpp/exceptions.hpp"
39 #include "rclcpp/expand_topic_or_service_name.hpp"
40 #include "rclcpp/logging.hpp"
41 #include "rclcpp/macros.hpp"
42 #include "rclcpp/qos.hpp"
43 #include "rclcpp/type_support_decl.hpp"
44 #include "rclcpp/visibility_control.hpp"
52 RCLCPP_SMART_PTR_DEFINITIONS_NOT_COPYABLE(
ServiceBase)
55 explicit ServiceBase(std::shared_ptr<rcl_node_t> node_handle);
72 std::shared_ptr<rcl_service_t>
81 std::shared_ptr<const rcl_service_t>
104 std::shared_ptr<void>
105 create_request() = 0;
108 std::shared_ptr<rmw_request_id_t>
109 create_request_header() = 0;
114 std::shared_ptr<rmw_request_id_t> request_header,
115 std::shared_ptr<void> request) = 0;
194 throw std::invalid_argument(
195 "The callback passed to set_on_new_request_callback "
200 [callback,
this](
size_t number_of_requests) {
202 callback(number_of_requests);
203 }
catch (
const std::exception & exception) {
206 "rclcpp::ServiceBase@" <<
this <<
207 " caught " << rmw::impl::cpp::demangle(exception) <<
208 " exception in user-provided callback for the 'on new request' callback: " <<
213 "rclcpp::ServiceBase@" <<
this <<
214 " caught unhandled exception in user-provided callback " <<
215 "for the 'on new request' callback");
219 std::lock_guard<std::recursive_mutex> lock(callback_mutex_);
225 rclcpp::detail::cpp_callback_trampoline<const void *, size_t>,
226 static_cast<const void *
>(&new_callback));
229 on_new_request_callback_ = new_callback;
233 rclcpp::detail::cpp_callback_trampoline<const void *, size_t>,
234 static_cast<const void *
>(&on_new_request_callback_));
241 std::lock_guard<std::recursive_mutex> lock(callback_mutex_);
242 if (on_new_request_callback_) {
244 on_new_request_callback_ =
nullptr;
253 get_rcl_node_handle();
257 get_rcl_node_handle()
const;
263 std::shared_ptr<rcl_node_t> node_handle_;
265 std::shared_ptr<rcl_service_t> service_handle_;
266 bool owns_rcl_handle_ =
true;
270 std::atomic<bool> in_use_by_wait_set_{
false};
272 std::recursive_mutex callback_mutex_;
273 std::function<void(
size_t)> on_new_request_callback_{
nullptr};
276 template<
typename ServiceT>
279 public std::enable_shared_from_this<Service<ServiceT>>
282 using CallbackType = std::function<
284 const std::shared_ptr<typename ServiceT::Request>,
285 std::shared_ptr<typename ServiceT::Response>)>;
287 using CallbackWithHeaderType = std::function<
289 const std::shared_ptr<rmw_request_id_t>,
290 const std::shared_ptr<typename ServiceT::Request>,
291 std::shared_ptr<typename ServiceT::Response>)>;
292 RCLCPP_SMART_PTR_DEFINITIONS(
Service)
307 const std::
string & service_name,
310 :
ServiceBase(node_handle), any_callback_(any_callback)
312 using rosidl_typesupport_cpp::get_service_type_support_handle;
313 auto service_type_support_handle = get_service_type_support_handle<ServiceT>();
316 service_handle_ = std::shared_ptr<rcl_service_t>(
321 rclcpp::get_node_logger(handle.get()).get_child(
"rclcpp"),
322 "Error in destruction of rcl service handle: %s",
323 rcl_get_error_string().str);
331 service_handle_.get(),
333 service_type_support_handle,
334 service_name.c_str(),
338 auto rcl_node_handle = get_rcl_node_handle();
348 rclcpp::exceptions::throw_from_rcl_error(ret,
"could not create service");
351 rclcpp_service_callback_added,
353 static_cast<const void *
>(&any_callback_));
354 #ifndef TRACETOOLS_DISABLED
355 any_callback_.register_callback_for_tracing();
370 std::shared_ptr<rcl_node_t> node_handle,
371 std::shared_ptr<rcl_service_t> service_handle,
374 any_callback_(any_callback)
379 throw std::runtime_error(
380 std::string(
"rcl_service_t in constructor argument must be initialized beforehand."));
384 service_handle_ = service_handle;
386 rclcpp_service_callback_added,
388 static_cast<const void *
>(&any_callback_));
389 #ifndef TRACETOOLS_DISABLED
390 any_callback_.register_callback_for_tracing();
405 std::shared_ptr<rcl_node_t> node_handle,
409 any_callback_(any_callback)
414 throw std::runtime_error(
415 std::string(
"rcl_service_t in constructor argument must be initialized beforehand."));
420 service_handle_ = std::shared_ptr<rcl_service_t>(
new rcl_service_t);
421 service_handle_->impl = service_handle->
impl;
423 rclcpp_service_callback_added,
425 static_cast<const void *
>(&any_callback_));
426 #ifndef TRACETOOLS_DISABLED
427 any_callback_.register_callback_for_tracing();
450 take_request(
typename ServiceT::Request & request_out, rmw_request_id_t & request_id_out)
455 std::shared_ptr<void>
456 create_request()
override
458 return std::make_shared<typename ServiceT::Request>();
461 std::shared_ptr<rmw_request_id_t>
462 create_request_header()
override
464 return std::make_shared<rmw_request_id_t>();
469 std::shared_ptr<rmw_request_id_t> request_header,
470 std::shared_ptr<void> request)
override
472 auto typed_request = std::static_pointer_cast<typename ServiceT::Request>(request);
473 auto response = any_callback_.dispatch(this->shared_from_this(), request_header, typed_request);
475 send_response(*request_header, *response);
480 send_response(rmw_request_id_t & req_id,
typename ServiceT::Response & response)
487 "failed to send response to %s (timeout): %s",
488 this->get_service_name(), rcl_get_error_string().str);
493 rclcpp::exceptions::throw_from_rcl_error(ret,
"failed to send response");
498 RCLCPP_DISABLE_COPY(Service)
500 AnyServiceCallback<ServiceT> any_callback_;
RCLCPP_PUBLIC Logger get_child(const std::string &suffix)
Return a logger that is a descendant of this logger.
Encapsulation of Quality of Service settings.
RCLCPP_PUBLIC bool exchange_in_use_by_wait_set_state(bool in_use_state)
Exchange the "in use by wait set" state for this service.
RCLCPP_PUBLIC rclcpp::QoS get_request_subscription_actual_qos() const
Get the actual request subscription QoS settings, after the defaults have been determined.
RCLCPP_PUBLIC rclcpp::QoS get_response_publisher_actual_qos() const
Get the actual response publisher QoS settings, after the defaults have been determined.
void set_on_new_request_callback(std::function< void(size_t)> callback)
Set a callback to be called when each new request is received.
void clear_on_new_request_callback()
Unset the callback registered for new requests, if any.
RCLCPP_PUBLIC std::shared_ptr< rcl_service_t > get_service_handle()
Return the rcl_service_t service handle in a std::shared_ptr.
RCLCPP_PUBLIC bool take_type_erased_request(void *request_out, rmw_request_id_t &request_id_out)
Take the next request from the service as a type erased pointer.
RCLCPP_PUBLIC const char * get_service_name()
Return the name of the service.
bool take_request(typename ServiceT::Request &request_out, rmw_request_id_t &request_id_out)
Take the next request from the service.
Service(std::shared_ptr< rcl_node_t > node_handle, std::shared_ptr< rcl_service_t > service_handle, AnyServiceCallback< ServiceT > any_callback)
Default constructor.
Service(std::shared_ptr< rcl_node_t > node_handle, rcl_service_t *service_handle, AnyServiceCallback< ServiceT > any_callback)
Default constructor.
Versions of rosidl_typesupport_cpp::get_message_type_support_handle that handle adapted types.
RCLCPP_PUBLIC std::string expand_topic_or_service_name(const std::string &name, const std::string &node_name, const std::string &namespace_, bool is_service=false)
Expand a topic or service name and throw if it is not valid.
RCL_PUBLIC RCL_WARN_UNUSED const char * rcl_node_get_name(const rcl_node_t *node)
Return the name of the node.
RCL_PUBLIC RCL_WARN_UNUSED const char * rcl_node_get_namespace(const rcl_node_t *node)
Return the namespace of the node.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_service_init(rcl_service_t *service, const rcl_node_t *node, const rosidl_service_type_support_t *type_support, const char *service_name, const rcl_service_options_t *options)
Initialize a rcl service.
RCL_PUBLIC bool rcl_service_is_valid(const rcl_service_t *service)
Check that the service is valid.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_service_fini(rcl_service_t *service, rcl_node_t *node)
Finalize a rcl_service_t.
RCL_PUBLIC RCL_WARN_UNUSED rcl_service_t rcl_get_zero_initialized_service(void)
Return a rcl_service_t struct with members set to NULL.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_send_response(const rcl_service_t *service, rmw_request_id_t *response_header, void *ros_response)
Send a ROS response to a client using a service.
Structure which encapsulates a ROS Node.
Options available for a rcl service.
Structure which encapsulates a ROS Service.
rcl_service_impl_t * impl
Pointer to the service implementation.
#define RCL_RET_SERVICE_NAME_INVALID
Service name (same as topic name) does not pass validation.
#define RCL_RET_OK
Success return code.
#define RCL_RET_TIMEOUT
Timeout occurred return code.
rmw_ret_t rcl_ret_t
The type that holds an rcl return code.