|
| 1 | +use std::env; |
| 2 | +use std::sync::{Arc, RwLock}; |
| 3 | + |
| 4 | +use anyhow::{Error, Result}; |
| 5 | + |
| 6 | +struct MinimalSubscriberInner { |
| 7 | + num_messages: u32, |
| 8 | + node: Arc<rclrs::Node>, |
| 9 | + subscription: Option<Arc<rclrs::Subscription<std_msgs::msg::String>>>, |
| 10 | +} |
| 11 | + |
| 12 | +struct MinimalSubscriber { |
| 13 | + inner: RwLock<MinimalSubscriberInner>, |
| 14 | +} |
| 15 | + |
| 16 | +impl MinimalSubscriber { |
| 17 | + pub fn new(name: &str, topic: &str) -> Result<Arc<Self>, rclrs::RclrsError> { |
| 18 | + let context = rclrs::Context::new(env::args())?; |
| 19 | + let node = rclrs::create_node(&context, name)?; |
| 20 | + let minimal_subscriber = Arc::new(MinimalSubscriber { |
| 21 | + inner: RwLock::new(MinimalSubscriberInner { |
| 22 | + num_messages: 0, |
| 23 | + node: Arc::clone(&node), |
| 24 | + subscription: None, |
| 25 | + }), |
| 26 | + }); |
| 27 | + |
| 28 | + let minimal_subscriber_aux = Arc::clone(&minimal_subscriber); |
| 29 | + let subscription = node.create_subscription::<std_msgs::msg::String, _>( |
| 30 | + topic, |
| 31 | + rclrs::QOS_PROFILE_DEFAULT, |
| 32 | + move |msg: std_msgs::msg::String| { |
| 33 | + minimal_subscriber_aux.callback(msg); |
| 34 | + }, |
| 35 | + )?; |
| 36 | + minimal_subscriber.inner.write().unwrap().subscription = Some(subscription); |
| 37 | + Ok(minimal_subscriber) |
| 38 | + } |
| 39 | + |
| 40 | + fn callback(&self, msg: std_msgs::msg::String) { |
| 41 | + self.inner.write().unwrap().num_messages += 1; |
| 42 | + println!("[{}] I heard: '{}'", self.node().name(), msg.data); |
| 43 | + println!( |
| 44 | + "[{}] (Got {} messages so far)", |
| 45 | + self.node().name(), |
| 46 | + self.inner.read().unwrap().num_messages |
| 47 | + ); |
| 48 | + } |
| 49 | + |
| 50 | + pub fn node(&self) -> Arc<rclrs::Node> { |
| 51 | + Arc::clone(&self.inner.read().unwrap().node) |
| 52 | + } |
| 53 | +} |
| 54 | + |
| 55 | +fn main() -> Result<(), Error> { |
| 56 | + let publisher_context = rclrs::Context::new(env::args())?; |
| 57 | + let publisher_node = rclrs::create_node(&publisher_context, "minimal_publisher")?; |
| 58 | + |
| 59 | + let subscriber_node_one = MinimalSubscriber::new("minimal_subscriber_one", "topic")?; |
| 60 | + let subscriber_node_two = MinimalSubscriber::new("minimal_subscriber_two", "topic")?; |
| 61 | + |
| 62 | + let publisher = publisher_node |
| 63 | + .create_publisher::<std_msgs::msg::String>("topic", rclrs::QOS_PROFILE_DEFAULT)?; |
| 64 | + |
| 65 | + std::thread::spawn(move || -> Result<(), rclrs::RclrsError> { |
| 66 | + let mut message = std_msgs::msg::String::default(); |
| 67 | + let mut publish_count: u32 = 1; |
| 68 | + loop { |
| 69 | + message.data = format!("Hello, world! {}", publish_count); |
| 70 | + println!("Publishing: [{}]", message.data); |
| 71 | + publisher.publish(&message)?; |
| 72 | + publish_count += 1; |
| 73 | + std::thread::sleep(std::time::Duration::from_millis(500)); |
| 74 | + } |
| 75 | + }); |
| 76 | + |
| 77 | + let executor = rclrs::SingleThreadedExecutor::new(); |
| 78 | + |
| 79 | + executor.add_node(&publisher_node)?; |
| 80 | + executor.add_node(&subscriber_node_one.node())?; |
| 81 | + executor.add_node(&subscriber_node_two.node())?; |
| 82 | + |
| 83 | + executor.spin().map_err(|err| err.into()) |
| 84 | +} |
0 commit comments