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 types to Action with rhel roscli fix #1361

Open
wants to merge 23 commits into
base: rolling
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 20 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
46a3488
Add types to Action objects
InvincibleRMC Aug 28, 2024
d8269e7
fix inheritance order
InvincibleRMC Aug 28, 2024
59b2665
move type into string
InvincibleRMC Aug 28, 2024
1230da4
string around type
InvincibleRMC Aug 28, 2024
13ccdc2
missed string type
InvincibleRMC Aug 28, 2024
332ce76
Merge branch 'rolling' into action-types
InvincibleRMC Aug 30, 2024
34b32bd
Merge branch 'rolling' into action-types
InvincibleRMC Aug 31, 2024
af90ef5
switch to 2 arg generics for clients/services
InvincibleRMC Aug 31, 2024
599a177
Merge branch 'rolling' into action-types
InvincibleRMC Sep 4, 2024
3e4ae31
Merge branch 'rolling' into action-types
InvincibleRMC Sep 4, 2024
e62e77f
move import into TYPE_CHECKING block
InvincibleRMC Sep 4, 2024
423976a
Type over type
InvincibleRMC Sep 5, 2024
85227ae
Add ClassVar
InvincibleRMC Sep 5, 2024
31908a4
Merge branch 'rolling' into action-types
InvincibleRMC Sep 20, 2024
4092a92
Hopefully fix rhel failure
InvincibleRMC Sep 20, 2024
55dbdf8
Merge branch 'ros2:rolling' into action-types
InvincibleRMC Oct 4, 2024
09c9429
Merge branch 'rolling' into action-types
mergify[bot] Nov 1, 2024
f1d3dd8
Merge branch 'rolling' into action-types
InvincibleRMC Nov 20, 2024
359af3d
re-run CI
InvincibleRMC Nov 20, 2024
7984bd4
re-run CI
InvincibleRMC Nov 20, 2024
f7fd296
Merge branch 'rolling' into action-types
InvincibleRMC Jan 5, 2025
195d364
Merge remote-tracking branch 'origin/rolling' into action-types
InvincibleRMC Jan 9, 2025
b6c13b6
Fix import
InvincibleRMC Jan 9, 2025
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
223 changes: 146 additions & 77 deletions rclpy/rclpy/action/client.py

Large diffs are not rendered by default.

221 changes: 148 additions & 73 deletions rclpy/rclpy/action/server.py

Large diffs are not rendered by default.

8 changes: 4 additions & 4 deletions rclpy/rclpy/client.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,20 +28,20 @@
from rclpy.qos import QoSProfile
from rclpy.service_introspection import ServiceIntrospectionState
from rclpy.task import Future
from rclpy.type_support import Srv, SrvEventT, SrvRequestT, SrvResponseT
from rclpy.type_support import Srv, SrvRequestT, SrvResponseT

# Left To Support Legacy TypeVars
SrvType = TypeVar('SrvType')
SrvTypeRequest = TypeVar('SrvTypeRequest')
SrvTypeResponse = TypeVar('SrvTypeResponse')


class Client(Generic[SrvRequestT, SrvResponseT, SrvEventT]):
class Client(Generic[SrvRequestT, SrvResponseT]):
def __init__(
self,
context: Context,
client_impl: _rclpy.Client,
srv_type: Type[Srv[SrvRequestT, SrvResponseT, SrvEventT]],
srv_type: Type[Srv[SrvRequestT, SrvResponseT]],
srv_name: str,
qos_profile: QoSProfile,
callback_group: CallbackGroup
Expand Down Expand Up @@ -231,7 +231,7 @@ def destroy(self) -> None:
"""
self.__client.destroy_when_not_in_use()

def __enter__(self) -> 'Client[SrvRequestT, SrvResponseT, SrvEventT]':
def __enter__(self) -> 'Client[SrvRequestT, SrvResponseT]':
return self

def __exit__(
Expand Down
194 changes: 192 additions & 2 deletions rclpy/rclpy/impl/_rclpy_pybind11.pyi
Original file line number Diff line number Diff line change
Expand Up @@ -18,12 +18,17 @@ from enum import Enum, IntEnum
from types import TracebackType
from typing import Any, Generic, Literal, overload, Sequence, TypeAlias, TypedDict


from action_msgs.msg import GoalInfo
from action_msgs.msg._goal_status_array import GoalStatusArray
from action_msgs.srv._cancel_goal import CancelGoal
from rclpy.clock import JumpHandle
from rclpy.clock_type import ClockType
from rclpy.duration import Duration
from rclpy.parameter import Parameter
from rclpy.subscription import MessageInfo
from rclpy.type_support import MsgT
from type_support import (MsgT, Action, GoalT, ResultT, FeedbackT, SendGoalServiceResponse,
GetResultServiceResponse, FeedbackMessage, SendGoalServiceRequest, GetResultServiceRequest)


def rclpy_remove_ros_args(pycli_args: Sequence[str]) -> list[str]:
Expand Down Expand Up @@ -512,6 +517,191 @@ class WaitSet(Destroyable):
"""Wait until timeout is reached or event happened."""


class ActionClient(Generic[GoalT, ResultT, FeedbackT], Destroyable):

def __init__(
self,
node: Node,
pyaction_type: type[Action[GoalT, ResultT, FeedbackT]],
goal_service_qos: rmw_qos_profile_t,
result_service_qos: rmw_qos_profile_t,
cancel_service_qos: rmw_qos_profile_t,
feedback_service_qos: rmw_qos_profile_t,
status_topci_qos: rmw_qos_profile_t
) -> None: ...

@property
def pointer(self) -> int:
"""Get the address of the entity as an integer."""

def take_goal_response(self, pymsg_type: type[SendGoalServiceResponse]
) -> tuple[int, SendGoalServiceResponse] | tuple[None, None]:
"""Take an action goal response."""

def send_result_request(self, pyrequest: GetResultServiceRequest) -> int:
"""Send an action result requst."""

def take_cancel_response(self, pymsg_type: type[CancelGoal.Response]
) -> tuple[int, CancelGoal.Response] | tuple[None, None]:
"""Take an action cancel response."""

def take_feedback(self, pymsg_type: type[FeedbackMessage[FeedbackT]]
) -> FeedbackMessage[FeedbackT] | None:
"""Take a feedback message from a given action client."""

def send_cancel_request(self: CancelGoal.Request) -> int:
"""Send an action cancel request."""

def send_goal_request(self: SendGoalServiceRequest[GoalT]) -> int:
"""Send an action goal request."""

def take_result_response(self, pymsg_type: type[GetResultServiceResponse[ResultT]]
) -> tuple[int, GetResultServiceResponse[ResultT]] | tuple[None, None]:
"""Take an action result response."""

def get_num_entities(self) -> tuple[int, int, int, int, int]:
"""Get the number of wait set entities that make up an action entity."""

def is_action_server_available(self) -> bool:
"""Check if an action server is available for the given action client."""

def add_to_waitset(self, waitset: WaitSet) -> None:
"""Add an action entity to a wait set."""

def is_ready(self) -> bool:
"""Check if an action entity has any ready wait set entities."""

def take_status(self, pymsg_type: type[GoalStatusArray]) -> GoalStatusArray | None:
"""Take an action status response."""


class GoalEvent(Enum):
_value_: int
EXECUTE = ...
CANCEL_GOAL = ...
SUCCEED = ...
ABORT = ...
CANCELED = ...


class rmw_request_id_t:
writer_guid: list[int]
sequence_number: int


class ActionServer(Generic[GoalT, ResultT, FeedbackT], Destroyable):

def __init__(
self,
node: Node,
rclpy_clock: Clock,
pyaction_type: type[Action[GoalT, ResultT, FeedbackT]],
action_name: str,
goal_service_qos: rmw_qos_profile_t,
result_service_qos: rmw_qos_profile_t,
cancel_service_qos: rmw_qos_profile_t,
feedback_topic_qos: rmw_qos_profile_t,
status_topic_qos: rmw_qos_profile_t,
result_timeout: float
) -> None: ...

@property
def pointer(self) -> int:
"""Get the address of the entity as an integer."""

def take_goal_request(
self,
pymsg_type: type[SendGoalServiceRequest[GoalT]]
) -> tuple[rmw_request_id_t, SendGoalServiceRequest[GoalT]] | tuple[None, None]:
"""Take an action goal request."""

def send_goal_response(
self,
header: rmw_request_id_t,
pyresponse: SendGoalServiceResponse
) -> None:
"""Send an action goal response."""

def send_result_response(
self,
header: rmw_request_id_t,
pyresponse: GetResultServiceResponse[ResultT]
) -> None:
"""Send an action result response."""

def take_cancel_request(
self,
pymsg_type: type[CancelGoal.Request]
) -> tuple[rmw_request_id_t, CancelGoal.Request] | tuple[None, None]:
"""Take an action cancel request."""

def take_result_request(
self,
pymsg_type: type[GetResultServiceRequest]
) -> tuple[rmw_request_id_t, GetResultServiceRequest] | tuple[None, None]:
"""Take an action result request."""

def send_cancel_response(
self,
header: rmw_request_id_t,
pyresponse: int
) -> None:
"""Send an action cancel response."""

def publish_feedback(
self,
pymsg: FeedbackMessage[FeedbackT]
) -> None:
"""Publish a feedback message from a given action server."""

def publish_status(self) -> None:
"""Publish a status message from a given action server."""

def notify_goal_done(self) -> None:
"""Notify goal is done."""

def goal_exists(self, pygoal_info: GoalInfo) -> bool:
"""Check is a goal exists in the server."""

def process_cancel_request(
self,
pycancel_request: CancelGoal.Request,
pycancel_response_tpye: type[CancelGoal.Response]
) -> CancelGoal.Response:
"""Process a cancel request"""

def expire_goals(self, max_num_goals: int) -> tuple[GoalInfo, ...]:
"""Expired goals."""

def get_num_entities(self) -> tuple[int, int, int, int, int]:
"""Get the number of wait set entities that make up an action entity."""

def is_ready(self, wait_set: WaitSet) -> tuple[bool, bool, bool, bool]:
"""Check if an action entity has any ready wait set entities."""

def add_to_waitset(self, wait_set: WaitSet) -> None:
"""Add an action entity to a wait set."""


class ActionGoalHandle:

def __init__(self, action_server: ActionServer, pygoal_info_msg: GoalInfo) -> None:
...

@property
def pointer(self) -> int:
"""Get the address of the entity as an integer."""

def get_status(self) -> GoalEvent:
"""Get the status of a goal."""

def update_goal_state(self, event: GoalEvent) -> None:
"""Update a goal state."""

def is_active(self) -> bool:
"""Check if a goal is active."""


class RCLError(RuntimeError):
pass

Expand All @@ -523,7 +713,7 @@ class NodeNameNonExistentError(RCLError):
class InvalidHandle(RuntimeError):
pass


class SignalHandlerOptions(Enum):
_value_: int
NO = ...
Expand Down
2 changes: 1 addition & 1 deletion rclpy/rclpy/impl/implementation_singleton.pyi
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,6 @@
# limitations under the License.


from rclpy.impl import _rclpy_pybind11
from impl import _rclpy_pybind11

rclpy_implementation = _rclpy_pybind11
21 changes: 10 additions & 11 deletions rclpy/rclpy/node.py
Original file line number Diff line number Diff line change
Expand Up @@ -89,7 +89,6 @@
from rclpy.type_support import check_is_valid_srv_type
from rclpy.type_support import MsgT
from rclpy.type_support import Srv
from rclpy.type_support import SrvEventT
from rclpy.type_support import SrvRequestT
from rclpy.type_support import SrvResponseT
from rclpy.utilities import get_default_context
Expand Down Expand Up @@ -175,8 +174,8 @@ def __init__(
self._parameters: Dict[str, Parameter[Any]] = {}
self._publishers: List[Publisher[Any]] = []
self._subscriptions: List[Subscription[Any]] = []
self._clients: List[Client[Any, Any, Any]] = []
self._services: List[Service[Any, Any, Any]] = []
self._clients: List[Client[Any, Any]] = []
self._services: List[Service[Any, Any]] = []
self._timers: List[Timer] = []
self._guards: List[GuardCondition] = []
self.__waitables: List[Waitable[Any]] = []
Expand Down Expand Up @@ -272,12 +271,12 @@ def subscriptions(self) -> Iterator[Subscription[Any]]:
yield from self._subscriptions

@property
def clients(self) -> Iterator[Client[Any, Any, Any]]:
def clients(self) -> Iterator[Client[Any, Any]]:
"""Get clients that have been created on this node."""
yield from self._clients

@property
def services(self) -> Iterator[Service[Any, Any, Any]]:
def services(self) -> Iterator[Service[Any, Any]]:
"""Get services that have been created on this node."""
yield from self._services

Expand Down Expand Up @@ -1738,12 +1737,12 @@ def create_subscription(

def create_client(
self,
srv_type: Type[Srv[SrvRequestT, SrvResponseT, SrvEventT]],
srv_type: Type[Srv[SrvRequestT, SrvResponseT]],
srv_name: str,
*,
qos_profile: QoSProfile = qos_profile_services_default,
callback_group: Optional[CallbackGroup] = None
) -> Client[SrvRequestT, SrvResponseT, SrvEventT]:
) -> Client[SrvRequestT, SrvResponseT]:
"""
Create a new service client.

Expand Down Expand Up @@ -1780,13 +1779,13 @@ def create_client(

def create_service(
self,
srv_type: Type[Srv[SrvRequestT, SrvResponseT, SrvEventT]],
srv_type: Type[Srv[SrvRequestT, SrvResponseT]],
srv_name: str,
callback: Callable[[SrvRequestT, SrvResponseT], SrvResponseT],
*,
qos_profile: QoSProfile = qos_profile_services_default,
callback_group: Optional[CallbackGroup] = None
) -> Service[SrvRequestT, SrvResponseT, SrvEventT]:
) -> Service[SrvRequestT, SrvResponseT]:
"""
Create a new service server.

Expand Down Expand Up @@ -1940,7 +1939,7 @@ def destroy_subscription(self, subscription: Subscription[Any]) -> bool:
return True
return False

def destroy_client(self, client: Client[Any, Any, Any]) -> bool:
def destroy_client(self, client: Client[Any, Any]) -> bool:
"""
Destroy a service client created by the node.

Expand All @@ -1956,7 +1955,7 @@ def destroy_client(self, client: Client[Any, Any, Any]) -> bool:
return True
return False

def destroy_service(self, service: Service[Any, Any, Any]) -> bool:
def destroy_service(self, service: Service[Any, Any]) -> bool:
"""
Destroy a service server created by the node.

Expand Down
8 changes: 4 additions & 4 deletions rclpy/rclpy/service.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,19 +24,19 @@
from rclpy.impl.implementation_singleton import rclpy_implementation as _rclpy
from rclpy.qos import QoSProfile
from rclpy.service_introspection import ServiceIntrospectionState
from rclpy.type_support import Srv, SrvEventT, SrvRequestT, SrvResponseT
from rclpy.type_support import Srv, SrvRequestT, SrvResponseT

# Used for documentation purposes only
SrvType = TypeVar('SrvType')
SrvTypeRequest = TypeVar('SrvTypeRequest')
SrvTypeResponse = TypeVar('SrvTypeResponse')


class Service(Generic[SrvRequestT, SrvResponseT, SrvEventT]):
class Service(Generic[SrvRequestT, SrvResponseT]):
def __init__(
self,
service_impl: _rclpy.Service,
srv_type: Type[Srv[SrvRequestT, SrvResponseT, SrvEventT]],
srv_type: Type[Srv[SrvRequestT, SrvResponseT]],
srv_name: str,
callback: Callable[[SrvRequestT, SrvResponseT], SrvResponseT],
callback_group: CallbackGroup,
Expand Down Expand Up @@ -121,7 +121,7 @@ def destroy(self) -> None:
"""
self.__service.destroy_when_not_in_use()

def __enter__(self) -> 'Service[SrvRequestT, SrvResponseT, SrvEventT]':
def __enter__(self) -> 'Service[SrvRequestT, SrvResponseT]':
return self

def __exit__(
Expand Down
Loading