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

Adding Io gripper controller #1439

Draft
wants to merge 50 commits into
base: master
Choose a base branch
from

Conversation

sachinkum0009
Copy link

@sachinkum0009 sachinkum0009 commented Dec 21, 2024

This Pull Request introduces the IO Gripper Controller, an implementation for controlling a gripper using IOs. The controller supports functionalities such as open, close, and reconfigure, which can be triggered either through an Action Server or a Service Server. Additionally, the controller publishes the gripper’s state via joint_states and provides dynamic_interfaces to expose all command and state interfaces.

Features

  • Action and Service Integration: Offers flexible control of gripper operations through action or service calls.
  • State Publishing: Continuously publishes the gripper's joint states and dynamic interface values for real-time monitoring.

Interfaces

Published Topics

  1. joint_states [sensor_msgs::msg::JointState]:

    • Publishes the state of the gripper joint and its configuration joint.
  2. dynamic_interfaces [control_msgs::msg::DynamicInterfaceValues]:

    • Publishes all command and state interfaces related to the gripper's IOs and sensors.

This addition enables better modular control of robotic grippers, paving the way for seamless integration in complex robotic systems.

Contributions via pull requests are much appreciated. Before sending us a pull request, please ensure that:

  1. Limited scope. Your PR should do one thing or one set of things. Avoid adding “random fixes” to PRs. Put those on separate PRs.
  2. Give your PR a descriptive title. Add a short summary, if required.
  3. Make sure the pipeline is green.
  4. Don’t be afraid to request reviews from maintainers.
  5. New code = new tests. If you are adding new functionality, always make sure to add some tests exercising the code and serving as live documentation of your original intention.

To send us a pull request, please:

  • Fork the repository.
  • Modify the source; please focus on the specific change you are contributing. If you also reformat all the code, it will be hard for us to focus on your change.
  • Ensure local tests pass. (colcon test and pre-commit run (requires you to install pre-commit by pip3 install pre-commit)
  • Commit to your fork using clear commit messages.
  • Send a pull request, answering any default questions in the pull request interface.
  • Pay attention to any automated CI failures reported in the pull request, and stay involved in the conversation.

@sachinkum0009
Copy link
Author

This PR depends on this PR

Comment on lines 1 to 9
# IO Gripper Controller

The IO Gripper Controller is provides implementation to control the gripper using IOs. It provides functionalities like open, close and reconfigure which can be used either though action server or service server and also publishes `joint_states` of gripper and also `dynamic_interfaces` for all command and state interfaces.

## Description of controller's interfaces

- `joint_states` [`sensor_msgs::msg::JointState`]: Publishes the state of gripper joint and configuration joint
- `dynamic_interfaces` [`control_msgs::msg::DynamicInterfaceValues`]: Publishes all command and state interface of the IOs and sensors of gripper.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this should go into the doc folder in the rst format. For an example please check the joint_trajectory_controller.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please also write docs extensively about the internal state machine and also internal functionalities and external interface. As template, use the docs from another controller and adjust the content.

Copy link
Member

@destogl destogl left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

First set of comments. The second one comes tomorrow.

io_gripper_controller/CMakeLists.txt Outdated Show resolved Hide resolved
io_gripper_controller/CMakeLists.txt Outdated Show resolved Hide resolved
io_gripper_controller/CMakeLists.txt Outdated Show resolved Hide resolved
Comment on lines 1 to 9
# IO Gripper Controller

The IO Gripper Controller is provides implementation to control the gripper using IOs. It provides functionalities like open, close and reconfigure which can be used either though action server or service server and also publishes `joint_states` of gripper and also `dynamic_interfaces` for all command and state interfaces.

## Description of controller's interfaces

- `joint_states` [`sensor_msgs::msg::JointState`]: Publishes the state of gripper joint and configuration joint
- `dynamic_interfaces` [`control_msgs::msg::DynamicInterfaceValues`]: Publishes all command and state interface of the IOs and sensors of gripper.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please also write docs extensively about the internal state machine and also internal functionalities and external interface. As template, use the docs from another controller and adjust the content.

io_gripper_controller/io_gripper_controller.xml Outdated Show resolved Hide resolved
io_gripper_controller/package.xml Outdated Show resolved Hide resolved
io_gripper_controller/package.xml Outdated Show resolved Hide resolved
io_gripper_controller/package.xml Outdated Show resolved Hide resolved
sachinkum0009 and others added 9 commits January 2, 2025 13:06
- removed the template from license
- added one variable per line
- documented the enums
- updated the doc folder as per `joint_trajectory_controller`
Co-authored-by: Dr. Denis <denis@stoglrobotics.de>
Co-authored-by: Dr. Denis <denis@stoglrobotics.de>
Co-authored-by: Dr. Denis <denis@stoglrobotics.de>
Co-authored-by: Dr. Denis <denis@stoglrobotics.de>
Co-authored-by: Dr. Denis <denis@stoglrobotics.de>
Comment on lines 913 to 914
"~/gripper_close", close_service_callback, rmw_qos_profile_services_hist_keep_all,
close_service_callback_group_);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
"~/gripper_close", close_service_callback, rmw_qos_profile_services_hist_keep_all,
close_service_callback_group_);
"~/gripper_close", close_service_callback, qos_services, close_service_callback_group_);

}
}

void IOGripperController::handle_reconfigure_state_transition(const reconfigure_state_type & state)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Proposal here. Can we do this general check only when activating the controller, i.e., on_activate because later when we're switching states we know already which states do we expect, and then we check only those.

The main idea is to avoid so many iterations in every control step is this is not even necessary.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we also please add tests with different initial states of the gripper? To make sure that we recognize it correctly.

io_gripper_controller/src/io_gripper_controller.cpp Outdated Show resolved Hide resolved
io_gripper_controller/src/io_gripper_controller.cpp Outdated Show resolved Hide resolved
}
}

void IOGripperController::handle_gripper_state_transition_open(const gripper_state_type & state)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I find generally this function somewhat inconsistent with the closing state checks. Can we unify this a bit. Especially the part when setting low and high states. I would except there more or less copy-paste code, or better yet calling subfunctions as we constantly do the same.

}
}

controller_interface::CallbackReturn IOGripperController::check_parameters()
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please update the tests according to these checks of the parameters I have changed below. We should be able to work with parameter size 0.

Comment on lines 310 to 330
closed_state_values_ = params_.close.state.possible_closed_states_map.at(closed_state_name_);

for (const auto & high_val : closed_state_values_.set_after_command_high)
{
setResult = find_and_set_command(high_val, 1.0);
if (!setResult)
{
RCLCPP_ERROR(
get_node()->get_logger(), "Failed to set the command state for %s", high_val.c_str());
}
}

for (const auto & low_val : closed_state_values_.set_after_command_low)
{
setResult = find_and_set_command(low_val, 0.0);
if (!setResult)
{
RCLCPP_ERROR(
get_node()->get_logger(), "Failed to set the command state for %s", low_val.c_str());
}
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This code has two parts that are 98% identical. This should go into a function. Also this should be updated for all before/after, or if even possible setting commands in both function. We can reduce at least 50 lines of code doing that.

@destogl
Copy link
Member

destogl commented Jan 17, 2025

Some major things to fix:

  • fix compilation
  • use a new handle_gripper_transition method for all transitions:
    • update how close data is stored — use struct and lambda function to manage this in the code to make the code shorter and nicer (see example for open ios)
    • update how parameters for configurations are used and stored using the new structure
    • use only one method with the same transitions, this means that all transitions should have before and after states, i.e., this should be added for configurations.
  • Add support for also getting feedback from before and after states (if those exist, or are set)
  • react by aborting action if timeout happens and HALTED is set
  • Fill out the new IOGripperControllerState message with the names of the overall states we are in and transitions if we are in the transition at the moment. Check the message.

Let me know if you have any questions.

Comment on lines 301 to 307
<<<<<<< Updated upstream
void handle_gripper_state_transition_open(const gripper_state_type & state);
=======
void handle_gripper_state_transition(
const rclcpp::Time & current_time, const GripperTransitionIOs & ios, const uint & state,
const std::string & transition_name, std::vector<double> after_joint_states);
>>>>>>> Stashed changes
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've notices a few unresolved merge conflicts. I can grab them if you want.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks a lot @Raivias
But I have fixed the merge conflicts in local commits. Will push the changes later.

<depend>rclcpp</depend>
<depend>rclcpp_lifecycle</depend>
<depend>realtime_tools</depend>
<depend>ros2_control_test_assets</depend>
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not familiar with this package but I suspect it's best to set this as a <test_depend>.

Comment on lines +271 to +289
setResult = find_and_get_state(high_val, state_value_);
if (!setResult)
{
RCLCPP_ERROR(
get_node()->get_logger(),
"CLOSE - CHECK_GRIPPER_STATE: Failed to get the state for %s", high_val.c_str());
}
else
{
if (abs(state_value_ - 1.0) < std::numeric_limits<double>::epsilon())
{
check_state_ios_ = true;
}
else
{
check_state_ios_ = false;
break;
}
}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This if tree is getting a bit intense. Can we simplify it down to something more like below. Same with the similar following if tree:

          setResult = find_and_get_state(high_val, state_value_);
          if (!setResult)
          {
            RCLCPP_ERROR(
              get_node()->get_logger(),
              "CLOSE - CHECK_GRIPPER_STATE: Failed to get the state for %s", high_val.c_str());
            continue;
          }

          check_state_ios_ = abs(state_value_ - 1.0) < std::numeric_limits<double>::epsilon();
          if(!check_state_ios_)
          {
            break;
          }

@destogl
Copy link
Member

destogl commented Jan 20, 2025

@Raivias thanks for your review. @sachinkum0009 and I have few more things to clean, some of those are related to your comments.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants