Part 4: Services
Introduction¶
Exercises: 6
Estimated Completion Time: 2 hours
Difficulty Level: Intermediate
Aims¶
In this part you'll learn about Services: an alternative communication method that can be used to transmit data/information or invoke actions on a ROS Network. You'll learn how this works, and why it might be useful. You'll also look at some practical applications of this.
Intended Learning Outcomes¶
By the end of this session you will be able to:
- Recognise how ROS Services differ from the standard topic-based publisher-subscriber approach, and identify appropriate use-cases for this type of messaging system.
- Identify the services that are available on a ROS network, and use ROS command-line tools to interrogate and call them.
- Develop Python Service Client Nodes.
- Invoke different services using various service-type interfaces.
Quick Links¶
- Exercise 1: Using Command-line Tools to Interrogate a Service and its Interface
- Exercise 2: Playing the Number Game (from the Command-line)
- Exercise 3: Creating a Service Interface
- Exercise 4: Adapting the Number Game Server
- Exercise 5: Creating a Python Service Client
- Exercise 6: Developing A Map Saver Service Client
Additional Resources¶
Getting Started¶
Step 1: Launch your ROS2 Environment
If you haven't done so already, launch your ROS environment now. Having done this, you should now have access to a Linux terminal instance (aka TERMINAL 1).
Step 2: Restore your work (WSL-ROS2 Managed Desktop Users ONLY)
Remember that any work you do within the WSL-ROS2 Environment will not be preserved between sessions or across different University computers, so you should be backing up your work to your U:\
drive regularly. When prompted (on first launch of WSL-ROS2 in TERMINAL 1) enter Y+Enter to restore your data1.
It looks like you already have a backup from a previous session:
U:\wsl-ros\ros2-backup-XXX.tar.gz
Do you want to restore this now? [y/n]
Step 3: Launch VS Code
It's also worth launching VS Code now. WSL users remember to check for this:
Step 4: Make Sure The Course Repo is Up-To-Date
Once again, it's worth quickly checking that the Course Repo is up-to-date before you start on the Part 4 exercises. Go back to Part 1 if you haven't installed it yet (really?!) or - alternatively - see here for how to update.
An Introduction to Services¶
So far, we've learnt about ROS topics and the message-type interfaces that we use to transmit data on them. We've also learnt how individual nodes can access data on a robot by simply subscribing to topics that another node on the ROS network is publishing messages to. In addition to this, we know that any node can publish messages to any topic, which broadcasts data across the ROS network, making it available to any other node on the network that may wish to access it.
Another way to pass data between ROS Nodes is by using Services. These are based on a call and response type of communication:
- A Service Client sends a Request to a Service Server.
- The Service Server processes that request and sends back a Response.
This is a bit like a transaction: one node requests something, and another node fulfils that request and responds. This is good for quick, short duration tasks, e.g.:
- Turning a device on or off.
- Grabbing some data and saving it to a file (a map for example).
- Performing a calculation and returning a result.
- Making a sound2.
A single service can have many clients, but you can only have a single Server providing that particular service at any one time.
Let's see how this all works in practice now, by playing a number game! We don't need a simulation up and running for this one, so in TERMINAL 1 use the following command to launch the Guess the Number Service:
TERMINAL 1:
Having launched the service successfully, you should be presented with the following:
[INFO] [#####] [number_game_service]: The '/guess_the_number' service is active.
[INFO] [#####] [number_game_service]: A secret number has been set... Game on!
We need to interrogate this now, in order to work out how to play the game...
Interrogating a Service¶
Exercise 1: Using Command-line Tools to Interrogate a Service and its Interface¶
-
Open up a new terminal instance (TERMINAL 2) and use the
ros2 service
command to list all active ROS services:
TERMINAL 2:
There'll be a few items in this list, most of them with the prefix:
/number_game_service
. This is the name of the node that is providing the service (i.e. the Server) and these items are all automatically generated by ROS. What we're really interested in is the service itself, which should be listed as/guess_the_number
. -
Next, we need to find the interface type used by this service, which we can do a couple of ways:
TERMINAL 2:
-
Use the
type
sub-command: -
Use the
list
sub-command again, but with the-t
flag:The latter will provide the same list of services as before, but each one will now have its interface type provided alongside it.
-
-
Regardless of the method that you used above, you should have discovered that the interface type used by the
/guess_the_number
service is:Notice how (much like with message interfaces used by topics), there are three fields to this type definition:
tuos_interfaces
: the name of the ROS package that this interface belongs to.srv
: that this is a service interface, the second interface type we've covered now (we'll learn about the third and final one in Part 5).NumberGame
: the data structure.
We need to know the data structure in order to make a call to the service, so let's identify this next.
-
We can use the
ros2 interface list
command to list all interface types available to us on our ROS system, but this will provide us with a long list!
TERMINAL 2:
-
We can use the
-m
flag to filter for message interfaces, or the-s
flag to filter for service interfaces. Try the latter: -
Still quite a lot there, right!? Let's filter this further with Grep to identify only interfaces that belong to the
tuos_interfaces
package:Hopefully, the
srv/NumberGame
interface is now listed. -
Use the
show
sub-command to show the message structure:
The interface structure should be shown as follows:
-
The Format of Service Interfaces¶
Service interfaces have two parts to them, separated by three hyphens (---
). Above the separator is the Service Request, and below it is the Service Response:
int32 guess <-- Request
---
int32 guesses <-- Response (1 of 3)
string hint <-- Response (2 of 3)
bool success <-- Response (3 of 3)
In order to Call a service, we need to provide data to it in the format specified in the Request section of the interface. A service Server will then send data back to the caller in the format specified in the Response section of the interface.
The tuos_interfaces/srv/NumberGame
service interface has only one request parameter:
guess
: aint32
(32-bit integer)
...which is the only thing we need to send to the/number_game_service
Service Server in order to call it.
There are then three response parameters:
- A 32-bit integer called
guesses
- A text string called
hint
-
A boolean flag called
success
...all of which will be returned by the server, once it has processed our request.
Exercise 2: Playing the Number Game (from the Command-line)¶
We're now ready to make a call to the service, and we can do this using the ros2 service
command again (from TERMINAL 2):
-
To start, let's send an initial guess of
0
and see what happens:The request will be echoed back to us, followed by a response, which will likely look something like this, and which shows us the value of the three response parameters that we identified above:
guesses
: tells us how many times we've tried to guess the number in total (just once so far)hint
: tells us if we should go "higher" or "lower" on our next guess in order to get closer to the secret numbersuccess
: tells us if we guessed the right number or not (unlikely on the first attempt!)
-
Make another service call, this time changing the value of your
guess
, e.g.: -
Try making a guess of 500 next.
The service should respond with the hint
'Error'
now. Have a look back in TERMINAL 1 (where the Server is running) to get more information on this. -
Keep going until you guess the magic number, how many guesses does it take you?!
-
Stop the server, by entering Ctrl+C in TERMINAL 1.
Creating Our Own Services¶
Over the next three exercises, we'll learn how to create a service interface of our own, and build a Server and Client (in Python) that use this.
First though, we need to create a new package, so follow the same procedure as you have in the previous parts of this course to create one called part4_services
.
TERMINAL 1:
Exercise 3: Creating a Service Interface¶
Let's create a service interface now which has a similar structure to the one used by the /guess_the_number
service, but this time with two request parameters, rather than just one...
-
In TERMINAL 1, navigate into the root of your
part4_services
package directory: -
Create a new directory there called
srv
: -
Create a new file in this directory called
MyNumberGame.srv
:In here is where we will define the structure of our own
MyNumberGame
service interface. -
Open up this file in VS Code, enter the following content and save the file:
The Request will therefore have two fields now:
# Field Name Data Type 1 guess
int32
2 cheat
bool
-
The rest of the process now is very similar to creating a message interface, like we did in Part 1.
First, we need to declare the interface in our
CMakeLists.txt
file, by adding the following above theament_package()
line: -
Next, we need to modify the
package.xml
file. Add the following lines to this one, just above the<export>
line: -
And finally, we use Colcon to generate the necessary source code for the service interface:
-
Navigate to the root of the ROS2 Workspace:
-
Run
colcon build
: -
And re-source the
.bashrc
:
-
-
Let's verify that this worked, using the
ros2
CLI (the same way as we did earlier when interrogatingtuos_interfaces/srv/NumberGame
):-
First, list all the ROS service interfaces that are available on the system (
-s
to filter for service interface types remember!):Scroll through this list and see if you can find
part4_services/srv/MyNumberGame
(or, usegrep
again). -
If it's there, use the
show
sub-command to show the data structure:
Does it match with the definition in our
MyNumberGame.srv
file? -
Exercise 4: Adapting the Number Game Server¶
We're going to take a copy of the tuos_examples/number_game.py
server node now, and adapt it to use the service interface that we created above.
-
In TERMINAL 1 still, navigate into the
part4_services/scripts
directory: -
Copy the
number_game.py
script from the course repo into here, renaming it tomy_number_game.py
at the same time:This file should already have execute permissions, but it's always worth checking...
-
Declare this as a package executable by going back to the
CMakeLists.txt
file, and addingmy_number_game.py
belowminimal_node.py
(or just replacingminimal_node.py
entirely): -
Now, let's look at the code, and see what needs to be adapted:
-
Open up the
my_number_game.py
node in VS Code and review it. -
As it stands, the node imports the
NumberGame
service interface fromtuos_interfaces
, so you'll need to change this to use the interface from our own package:You'll also need to change the
srv_type
definition, when the service is created in the__init__()
: -
Everything that this service does (when a Request is sent to it), is contained within the
srv_callback()
method.In here,
request
parameters are processed,response
parameters are defined and the overallresponse
is returned once the callback tasks have been completed. -
You may have noticed that when we created our
MyNumberGame
interface in the previous exercise, the Response parameters were the same as the originals from Exercise 1 & 2, except that some of their names were changed slightly!Work through the
srv_callback()
method and make sure that allresponse
attributes are renamed to match the new names that we've given them inMyNumberGame.srv
. -
Remember that our
MyNumberGame
interface has an additional Request parameter too:... i.e. a boolean flag with the attribute name
cheat
.Adapt the
srv_callback()
method further now to read this value as well.-
If (when a request is made to the server) the value of
cheat
isTrue
the hint that the server returns should contain the actual secret number! E.g.: -
In such situations, the value of
response.num_guesses
should still go up by one, and thecorrect
flag should still returnFalse
.
-
-
-
Once you've adapted the node, test it out by running it:
TERMINAL 1:
You should then be able to make calls to this from TERMINAL 2 using the
ros2 service call
sub-command, as we did in Exercise 2.
Exercise 5: Creating a Python Service Client¶
So far we've been making service calls from the command-line, but we can also call services from within Python Nodes. When a node calls (i.e. requests) a service, it becomes a Service "Client".
-
Make sure your
my_number_game.py
node is still active in TERMINAL 1 for this exercise. -
In TERMINAL 2, create a new file in the
part4_services/scripts
directory callednumber_game_client.py
. -
Add this to your package's Python executables list in
CMakeLists.txt
: -
Re-build your package (as before), remembering that there are three steps to this:
- Navigate to the root of the ROS2 workspace,
- Run
colcon build
(with the necessary additional arguments), - Re-source your
~/.bashrc
.
-
Now, open up the
number_game_client.py
file in VS Code. Have a look at the code here, review it (including all the annotations), then copy and paste it and save the file. -
You should now be able to run the code with
ros2 run
. To begin with, run it without supplying any additional command-line arguments:
TERMINAL 2:
You'll probably then get an output like this:
[INFO] [#####] [number_game_client]: Sending the request: - guess: 0 - cheat: False Awaiting response... [INFO] [#####] [number_game_client]: The server has responded with: - Incorrect guess :( - Number of attempts so far: 1 - A hint: 'Higher'.
Notice how the request parameters
guess
andcheat
have defaulted to0
andFalse
respectively? -
Supply a guess now, using our node's CLI:
TERMINAL 2:
... replacing
GUESS
with an actual number! -
Have a go at cheating now too:
TERMINAL 2:
... notice how we only need to supply the
--cheat
flag, no actual value is required.
The Map Saver Service¶
Clearly the examples that we've been working with here so far have been fairly trivial: it's unlikely that you'll ever need to program a robot to play the number game! The aim however has been to illustrate how ROS Services work, and how to develop your own.
One application that you might find useful however is map saving. In Part 3 we learnt about SLAM, where we drove a robot around in an environment while SLAM algorithms were working in the background to generate a map of the world using data from the robot's LiDAR sensor and its odometry system:
Having mapped out the environment, we called up a node called map_saver_cli
from the nav2_map_server
package, to save a copy of that map to a file:
... wouldn't it be nice if there was a way to be able to do this programmatically (i.e. from within a Python node, for example) rather than having to run the above command manually? Well, there is a way, and guess what - it involves Services!
Exercise 6: Developing A Map Saver Service Client¶
-
Make sure everything in TERMINALS 1 and 2 from the previous exercises are closed down now.
-
In TERMINAL 1, let's fire up the Nav World again, like we did in Part 3:
TERMINAL 1:
-
In TERMINAL 2, let's also fire up Cartographer again (the SLAM algorithms):
TERMINAL 2:
-
Open up another terminal instance now (TERMINAL 3), and use this one to fire up the Map Saver Service (wouldn't it be nice if we could launch all of these launch files at once?!):
TERMINAL 3:
This will add a number of
/map_saver
services to our ROS network. -
Use a
ros2 service
sub-command to identify all the/map_saver
services (like we did in Exercise 1).Question
Do you see an item in this list that could be related to saving a map? (It has a
/map_saver
prefix3!) -
Use another
ros2 service
sub-command to determine the type of interface used by this service (again, like we did in Exercise 1). -
Next, use a
ros2 interface
command to discover the structure of this service interface.Questions
- How many Request parameters does this interface have?
- How many Response parameters are there too?4
- What are their data types?
-
Develop a Python Service Client to make calls to this service:
-
Create a new Node in your
part4_services
package calledmap_saver_client.py
for this.Things to remember when doing this:
- Create this in your
part4_services/scripts
directory. - Make sure it has execute permissions.
- Declare it as a package executable in your
CMakeLists.txt
. - Re-build your package with
colcon
, making sure you follow the full three-step process
- Create this in your
-
Use the
number_game_client.py
Node from Exercise 5 as a starting point when building your newmap_saver_client.py
node... all the same principals will apply here, you are just applying them to a different service (and therefore you need to account for a different service interface). -
Use
argparse
again to build a CLI for yourmap_saver_client.py
node.In this case, your CLI should only need one argument though (where, in
number_game_client.py
there were two). This should be used to pass in a filename for a map, e.g.:You should ensure that this argument is optional though, i.e. if the argument isn't provided when your node is called, then a default value is applied instead.
-
When constructing service requests, consider the following tips:
- The SLAM map data (as generated by Cartographer) is published to a topic called
/map
. -
When providing a name for the map file:
- You don't need to include a file extension
-
File names are interpreted relative to your home directory, so:
my_amazing_map
would result in a map file at~/my_amazing_map.yaml
my/amazing/map
would result in a map file at~/my/amazing/map.yaml
(assuming the directory structure already exists!)
-
For further guidance see here for a usage example.
- The server will apply its own defaults to certain parameters, if they aren't set in the request.
- The SLAM map data (as generated by Cartographer) is published to a topic called
-
Summary of the Map Saver Service¶
Back in Part 3 we saved our SLAM map once via a command-line call after we had fully explored and mapped out the environment:
... which resulted in something like:
MAP_NAME.pgm
fileIn real-world tasks however (i.e. tasks that you might need to complete in Assignment #2 for example), your robot might be exploring an environment autonomously, and you don't necessarily know when the full environment has been explored, nor are you always going to be there to run the map_saver_cli
node manually! You might therefore want to program your robot with the ability to save a map incrementally and periodically as more and more of the environment is explored.
The process that we explored in the previous exercise allows you to do just that! In the example, our client node was programmed to make only one request to the server and then stop. It could however be programmed to make regular service requests (say, once every 5 or 10 seconds) in order to continuously update its map as the robot explores further and further.
Think about how you might adapt the map_saver_client.py
node to achieve this, drawing upon other exercises that you have worked through in previous parts of this course.
Wrapping Up¶
In Part 4 we've learnt about ROS Services and why they might be useful for robot applications:
- Services differ from standard topic-based communication methods in ROS in that they are a call and response type of communication, taking place between one node and another.
- Typically, a service Caller will request a service, and then wait for a response (although it is possible to do other things in the meantime).
- In general however, Services are useful for controlling quick, short-duration tasks or calculations.
WSL-ROS2 Managed Desktop Users: Save your work!¶
Remember, to save the work you have done in WSL-ROS2 during this session so that you can restore it on a different machine at a later date. Run the following script in any idle WSL-ROS2 Terminal Instance now:
You'll then be able to restore it to a fresh WSL-ROS2 environment next time you fire one up (wsl_ros restore
).
-
Remember: you can also use the
wsl_ros restore
command at any time. ↩ -
On the real Waffles, there's a service called
/sound
. Have a look at this next time you're in the lab... Once you've worked through the whole of Part 4 you'll know exactly how to interrogate this service and leverage the functionality that it provides! ↩ -
There should be one in the list called
/map_saver/save_map
↩ -
The
nav2_msgs/srv/SaveMap
Service Interface has 6 Requests (map_topic
,map_url
,image_format
,map_mode
,free_thresh
,occupied_thresh
) and 1 Response (result
). ↩