Realsense Record and Playback =================== Overview ------------- In addition to streaming video and other data from devices and sensors, the realsense SDK provides the ability to record a live session of streaming to a file. The recorded files can later be loaded by the SDK and to create a device with "read-only" abilities of the recorded device ( we will explain what "read-only" abilities mean later on). The SDK is recording a single device to a single [rosbag file](http://wiki.ros.org/rosbag), using mostly standard ROS messages. This allows files recorded by the SDK to be replayed using any ROS tools \ application. > For example recording files, please see [Sample Data](../../doc/sample-data.md) Quick Start ------------- #### `rs2::recorder` > :exclamation: If you are not familiar with the basic streaming [examples](../../examples), please follow them before moving on To enable recording of any device, simply create a **rs2::recorder** from it and provide a path to the desired output file: ```cpp //Create a context and get the first device rs2::context ctx; auto devices = ctx.query_devices(); if (devices.size() > 0) { //Create a rs2::recorder from the first device, and desired file name //'.bag' is the common extension for rosbag files rs2::recorder device("my_file_name.bag", devices[0]); //recorder "is a" device, so just use it like any other device now } ``` A `recorder` has the same functionality as a "real" device, with additional control for recording, such as pausing and resuming record. #### `rs2::playback` > :exclamation: If you are not familiar with the basic streaming [examples](../../examples), please follow them before moving on Recorded files can be loaded and used to create a playback device by simply loading a file to the context: ```cpp //Create a context rs2::context ctx; //Load the recorded file to the context rs2::playback device = ctx.load_device("my_file_name.bag"); //playback "is a" device, so just use it like any other device now ``` The above code creates a playback device, which can be used as any device, but has the obvious limitation of only playing the recorded streams. Playback devices can be used to query information on the device and it sensors, and can be extended to which ever extension the "real" device could. A `playback` provides additional functionalities such as seek, pause, resume and playback speed. ### Using `rs2::config` with `rs2::pipeline` The `rs2::pipeline` can be configured to record or play a streaming session by providing it with a `rs2::config` of your choice: Recording to file: ```cpp rs2::config cfg; cfg.enable_record_to_file("path_to_output_file.bag"); rs2::pipeline pipe; pipe.start(cfg); //File will be opened in write mode at this point for (int i = 0; i < 30; i++) { auto frames = pipe.wait_for_frames(); //use frames here } pipe.stop(); //File will be closed at this point ``` Playing from file: ```cpp rs2::config cfg; cfg.enable_device_from_file("path_to_input_file.bag"); rs2::pipeline pipe; pipe.start(cfg); //File will be opened in read mode at this point for (int i = 0; i < 30; i++) { rs2::frameset frames; if (pipe.poll_for_frames(&frames)) { //use frames here } //Note that we use poll_for_frames instead of wait_for_frames //This is because the file only contains a finite amount of frames // and if we use wait_for_frames then after reading the entire file // we will fail to wait for the next frame (and get an exception) } pipe.stop(); //File will be closed at this point ``` Playback and Record in RealSense Viewer ------------- Among its many features, the [RealSense Viewer](../../tools/realsense-viewer) allows recording a device, and loading a file to playback. To record a streaming session, simply click the "bars" icon next to the device name, choose "Record to File...", and select the destination for the file. ![Recording a device](../../doc/img/record_screenshot.png) After choosing a file destination, a red dot will appear next to the device's name, indicating that it is recording. Starting a stream will save its frames to the file, and once all streams are stopped, recording will complete automatically. To replay a file, click "Add Source", choose "Load Recorded Sequence" and select the file you want to play. Once you select the file, the Viewer will automatically add it the the list of source, and a popup should appear stating that the file was loaded: ![Loading Recorded Sequence](../../doc/img/playback_screenshot.png) >Notice that devices that were loaded from file have a "movie" icon next to their name. After loading the file, you can start streaming its streams, view its controls (with the values at time of record), pause the playback, choose speed, and use the seek bar to navigate trough frames. Under the Hood ------ #### Basics Terminology A **Device** is a container of Sensors with some correlation between them (e.g - all sensors are on a single board, sensors are mounted on a robot and share calibration information, etc.). A **Sensor** is a data streaming object, that provides one or more Streams. **Stream** is a sequence of data items of a single data type, which are ordered according to their time of creation or arrival. The Sensor provides the Streams frames to the user. We call the device's sensors and stream, the **topology** of the device. Devices and Sensors can have **Extensions** that provide additional functionalities. A **Snapshot** of an Extension is a snapshot of the data that is available by the extension at some point of time, it is a sort of "read-only" version of the extension. For example, say we have a `DigitalClockExtension`, that can set and show the time. If we take a snapshot of that extension at noon, then whenever we ask the snapshot to show the time it will show "12:00", and trying to set its time will fail. Finally, we will refer to a an actual implementation of devices and sensors as "live" or "real" devices and sensors. #### Rosbag ##### Overview on ROSBAG file structure The Rosbag file consists of data records, each data that is written to the file is represented by a topic that represents the data content, message that is the data definition, and capture time. Any topic can be coupled with any message, in our file each topic will be represented by a single message. However, we will have different topics with the same message such as property. A bag file consists of a list of records, each record can be one of the following types: - Bag header. Stores information about the entire bag, such as the offset to the first index data record, and the number of chunks and connections. - Chunk. Stores (possibly compressed) connection and message records. - Connection. Stores the header of a ROS connection, including topic name and full text of the message definition (message structure). - Message data. Stores the serialized message data (which can be zero-length) with the ID of the connection. - Index data. Stores an index of messages in a single connection of the preceding chunk. - Chunk info. Stores information about messages in a chunk. Each message that we write is stored in a chunk, each chunk consists of a number of messages based on the chunk size. In addition to the message data, the chunk stores also the connections of the messages, the connection describes the topic name and message definition to be able to serialize the message data. The flow to find a message on topic “A”: - Check in all chunk info’s what kind of connections can be found in the chunk. - Access the chunk with the wanted connection, chunk info contains an offset to the paired chunk. - Access the message from the relevant index data. ##### Dependencies The SDK depends on ROS packages that are used for recording and playing the bag file. The dependencies are: - rosbag_storage - roscpp_serialization - cpp_common - rostime - boost_system - lz4 > **Note:** The above dependencies are embedded into the source files, there is no need for additional installations. ##### Topics The rosbag file requires messages to be associates with Topics > Topics are [named](http://wiki.ros.org/Names) buses over which [nodes](http://wiki.ros.org/Nodes) exchange [messages](http://wiki.ros.org/Messages). [[ROS Wiki\Topics]](http://wiki.ros.org/Topics) The following table depicts the Topics that are supported by RealSense file format:
Topic | Published Name | Message Type | Description | ||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
File Version | /file_version | std_msgs/UInt32 | Version of file format. A single message for entire file |
||||||||||||
Device Information | /device_<device_id>/info | diagnostic_msgs/KeyValue | Device information. Many messages to a single topic |
||||||||||||
Sensor Information | /source_<source_id>/info | diagnostic_msgs/KeyValue | Sensor information. Many messages to a single topic |
||||||||||||
Stream Information | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/info | realsense_msg/StreamInfo | generic-stream information A single messages to a single topic |
||||||||||||
Video Stream Info | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/info/camera_info | sensor_msgs/camera_info | Image information. A single messages to a single topic |
||||||||||||
IMU Intrinsics | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/info/imu_intrinsic | realsense_msgs/ImuIntrinsic | Intrinsic of a motion stream. A single messages to a single topic |
||||||||||||
Options | /device_<device_id>/sensor_<sensor_id>/option/<option name>/value /device_<device_id>/sensor_<sensor_id>/option/<option name>/description |
std_msgs/Float32 std_msgs/String |
Options of a sensor. values represent the value of the option and has many messages to a single topic description is a human readable description of the option, with a single messages to a single topic |
||||||||||||
Image Data | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/image/data | sensor_msgs/Image | The data of a single image. A single messages to a single topic | ||||||||||||
Image Information | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/image/metadata | diagnostic_msgs/KeyValue | Additional information of a single image. Many message to a single topic | ||||||||||||
IMU Data | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/imu/data | sensor_msgs/Imu | The data of a single imu frame. A single messages to a single topic |
||||||||||||
IMU Information | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/imu/metadata | diagnostic_msgs/KeyValue | Additional information of a single imu frame. Many message to a single topic | ||||||||||||
Pose Data | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/pose/{transform, accel, twist}/data |
|
The Pose data is split into 3 messages (each under different topic [1]):
|
||||||||||||
Pose Information | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/pose/metadata | diagnostic_msgs/KeyValue | Additional information of a single pose frame. Many message to a single topic | ||||||||||||
Occupancy Map Data | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/occupancy_map/data | nav_msgs/OccupancyGrid | |||||||||||||
Occupancy Map Information | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/occupancy_map/metadata | diagnostic_msgs/KeyValue | Additional | ||||||||||||
Time Stream | /device_<device_id>/sensor_<sensor_id>/time | sensor_msgs/TimeReference | |||||||||||||
Log | /log | rosgraph_msgs/Log | Log messages | ||||||||||||
Stream Exntrinsics | /device_<device_id>/sensor_<sensor_id>/<stream_type>_<stream_id>/tf/<group_index> | geometry_msgs/Transform Message | Extrinsic transformation between some point of reference (indexed by group_index) to the stream in the topic | ||||||||||||
Additional Info | /additional_info | diagnostic_msgs/KeyValue | Additinal information of any kind. Can be useful for application that require additional metadata on the recorded file (such as program name, version etc...) | ||||||||||||
Sensor Notification | /device_<device_id>/sensor_<sensor_id>/notification/<rs2_notification_category> | realsense_msgs/Notification | Additinal information of any kind. Can be useful for application that require additional metadata on the recorded file (such as program name, version etc...) |
Name | Description | Format |
---|---|---|
is_recommended | Indicates if this stream is recommended by RealSense SDK | bool |
fps | Frame per second value | uint32 |
encoding | Stream's data format. Supported encodings are listed below | string |
Name | Description | Format | ||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
data |
Interpret data array values.
|
float32[12] | ||||||||||||
noise_variances | Variance of noise for X, Y, and Z axis | float32[3] | ||||||||||||
bias_variances | Variance of bias for X, Y, and Z axis | float32[3] |
Name | Description | Format |
---|---|---|
timestamp | The time of occurrence | std_msgs/Time |
category | Category of the notification, matching an rs2_notification_category | std_msgs/String |
severity | Severity of the notification, matching an rs2_log_severity | std_msgs/String |
description | Human readable description of the notification | std_msgs/String |
serialized_data | JSON string with additional data | std_msgs/String |