Skip to main content

Write to a Service

Similar to reading, services can output data on write streams for other services to read. We will write "decision" outputs to steer the Rover.

Elias Groot

Elias Groot

Software Lead, Project Administrator

Prerequisites

How to Write

Any service can output data to 0 or more output/write streams. Just like with input/read streams, these must be declared in the service.yaml file. Open your template service's declaration.

service.yaml
...
outputs:
- decision
...

The above tells us that our service outputs data to the "decision" stream. All outputs stream follow a one-to-many principle: 0 or more other services could "subscribe" to our output stream and read from it.

To write data to this stream in your service's source code, you can open a write stream and write using the APIs provided by roverlib for your language. Again, it is recommended to communicate using our standard messaging definitions but you can also send raw bytes if necessary.

roverlib-go exposes the GetWriteStream() method, that can be used to open a write 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 {
writeStream := service.GetWriteStream("decision")
if writeStream == nil {
return fmt.Errorf("Failed to create write stream 'decision'")
}

actuatorMsg := pb_outputs.SensorOutput{
Timestamp: uint64(time.Now().UnixMilli()), // milliseconds since epoch
Status: 0, // all is well
SensorId: 1,
SensorOutput: &pb_outputs.SensorOutput_ControllerOutput{
ControllerOutput: &pb_outputs.ControllerOutput{
SteeringAngle: steerPosition,
LeftThrottle: float32(tunableSpeed),
RightThrottle: float32(tunableSpeed),
FanSpeed: 0,
FrontLights: false,
},
},
}

err = writeStream.Write(&actuatorMsg)
if err != nil {
log.Warn().Err(err).Msg("Could not write")
}
}

Or, to write bytes:

src/main.go
func run(service roverlib.Service, configuration *roverlib.ServiceConfiguration) error {
writeStream := service.GetWriteStream("decision")
if writeStream == nil {
return fmt.Errorf("Failed to create write stream 'decision'")
}

err = writeStream.WriteBytes([]byte{"hello, world"})
if err != nil {
log.Warn().Err(err).Msg("Could not write")
}
}

Alter Steering

In your service template, try to alter the steering and throttle values that you write. Then, open your service.yaml file and bump the version number to 0.0.2.

Now, reupload your service and open roverctl-web again. Notice that you can choose between multiple versions of the same service now. We recommend versioning your services wisely, so that you can quickly A/B-test and revert to old code.

Finally, start the pipeline and see how the changes in your source code impact how the Rover drives.

Debugging

Just like with the official ASE services, you can debug your output data if you use our communication definitions. Open roverctl-web in debug mode and enable debug mode for your pipeline.

Start the pipeline again and notice that the output data by your template service is plotted there. If you do not use our communication definitions, you will lose this debugging functionality.