Skip to main content

Read from a Service

To allow for communication, services can read from each others outputs. The service.yaml file and a service's source code are tightly coupled to allow for this.

Elias Groot

Elias Groot

Software Lead, Project Administrator

Prerequisites

How to Read

Suppose that we have a simple pipeline of two services: a producer service that generates data and outputs it, and a consumer service that reads this data and uses it for its functionality.

For the consumer to know where and how to read, it must define its dependencies as input in its service.yaml definition. Turn to the service.yal of your template service:

service.yaml
...
inputs:
- service: imaging
streams:
- path
...

The above declaration tells us that our example-service depends on data from an "imaging" service. This imaging service should write this data to its "path" stream.

Then in your service's source code, you can open a read stream and read from it using the APIs provided by roverlib for your language.

Open the main program in src/main.go.

roverlib-go exposes the GetReadStream() method, that can be used to open a read stream that corresponds to the one defined in the service.yaml file, like so:

src/main.go
func run(service roverlib.Service, configuration *roverlib.ServiceConfiguration) error {
readStream := service.GetReadStream("imaging", "path")
if readStream == nil {
return fmt.Errorf("Failed to get read stream")
}

data, err := readStream.Read()
if data == nil || err != nil {
return fmt.Errorf("Failed to read")
}
}

What to Read

When you know where to read from, you must also know what format the data you are reading is in. Because services can be written in many different languages, they must agree to all "speak" the same wire format, such as JSON.

While JSON could work, it is rather slow and lacks type safety. Hence, we have opted for the Protobuf wire format with a fixed collection of messages that you can send and receive, as defined in our rovercom package.

Protobuf definitions can be transpiled to serialization and deserialization code for virtually any programming language. Each roverlib already takes care of this for you. You can switch or match on the message type when reading.

For example, you can read an CameraSensorOutput message like so:

src/main.go
func run(service roverlib.Service, configuration *roverlib.ServiceConfiguration) error {
readStream := service.GetReadStream("imaging", "path")
if readStream == nil {
return fmt.Errorf("Failed to get read stream")
}

data, err := readStream.Read()
if data == nil || err != nil {
return fmt.Errorf("Failed to read")
}

imagingData := data.GetCameraOutput()
if imagingData == nil {
return fmt.Errorf("Message does not contain camera output. What did imaging do??")
}
}

Then, you can use the properties of the CameraSensorOutput message to find the track edges - if that is what the imaging service provides.

Bring Your Own Messages

We strongly recommend to use our messaging definitions, as you will enjoy first-class type safety and debugging support. However, if it does not fit your needs, you can always resort to reading and writing raw bytes using the roverlib, so that you can use your own serialization and deserialization logic.

src/main.go
func run(service roverlib.Service, configuration *roverlib.ServiceConfiguration) error {
readStream := service.GetReadStream("imaging", "path")
if readStream == nil {
return fmt.Errorf("Failed to get read stream")
}

data, err := readStream.ReadBytes()
if err != nil {
return fmt.Errorf("Failed to read bytes")
}
}

Completing the Pipeline

Open roverctl-web again and upload your service using roverctl. Enable only your service and the ASE "imaging" service. (This service can be installed from https://github.com/VU-ASE/imaging/releases/latest).

Press "start execution". The pipeline should start running successfully, but the Rover does not seem to drive. Oh no.

The Rover does not drive because there is no service enabled yet that takes in the outputs from your service and turns it into hardware signals to steer the servo and spin the motors. ASE provides the actuator service to do this for you, it can be installed from https://github.com/VU-ASE/actuator/releases/latest.

Once installed, enable your service, together with the ASE "imaging" and "actuator" services. Put the Rover on the ground and press "start execution".

The Rover will start driving in a circle while logging input it receives from the imaging service in the process. You can view the logs through roverctl-web, or by using the roverctl logs command.

# View the logs of my-example-service on Rover 12
roverctl logs elias my-example-service 1.0.0 --rover 12

Press "stop execution", so that we can alter the service's behavior.