From cbc19fa4471ecfa4729c0e8285ae43677a4da89e Mon Sep 17 00:00:00 2001 From: Emerson Knapp Date: Thu, 2 Dec 2021 11:58:44 -0800 Subject: [PATCH] Bag rewriter CLI and README Signed-off-by: Emerson Knapp --- README.md | 91 ++++++++++++++++++++++++ ros2bag/ros2bag/verb/convert.py | 51 +++++++++++++ ros2bag/setup.py | 1 + rosbag2_py/rosbag2_py/__init__.py | 2 + rosbag2_py/src/rosbag2_py/_storage.cpp | 1 + rosbag2_py/src/rosbag2_py/_transport.cpp | 31 ++++++++ 6 files changed, 177 insertions(+) create mode 100644 ros2bag/ros2bag/verb/convert.py diff --git a/README.md b/README.md index 22b3dde14d..b35c35d85f 100644 --- a/README.md +++ b/README.md @@ -174,6 +174,97 @@ Topic information: Topic: /chatter | Type: std_msgs/String | Count: 9 | Serializ Topic: /my_chatter | Type: std_msgs/String | Count: 18 | Serialization Format: cdr ``` +### Converting bags + +Rosbag2 provides a tool `ros2 bag convert` (or, `rosbag2_transport::bag_rewrite` in the C++ API). +This allows the user to take one or more input bags, and write them out to one or more output bags with new settings. +This flexible feature enables the following features: +* Merge (multiple input bags, one output bag) +* Split top-level bags (one input bag, multiple output bags) +* Split internal files (by time or size - one input bag with fewer internal files, one output bag with more, smaller, internal files) +* Compress/Decompress (output bag(s) with different compression settings than the input(s)) +* Serialization format conversion +* ... and more! + +Here is an example command: + +``` +ros2 bag convert --input /path/to/bag1 --input /path/to/bag2 storage_id --output-options output_options.yaml +``` + +The `--input` argument may be specified any number of times, and takes 1 or 2 values. +The first value is the URI of the input bag. +If a second value is supplied, it specifies the storage implementation of the bag. +If no storage implementation is specified, rosbag2 will try to determine it automatically from the bag. + +The `--output-options` argument must point to the URI of a YAML file specifying the full recording configuration for each bag to output (`StorageOptions` + `RecordOptions`). +This file must contain a top-level key `output_bags`, which contains a list of these objects. + +The only required value in the output bags is `uri` and `storage_id`. All other values are options (however, if no topic selection is specified, this output bag will be empty!). + +This example notes all fields that can have an effect, with a comment on the required ones. + +``` +output_bags +- uri: /output/bag1 # required + storage_id: sqlite3 # required + max_bagfile_size: 0 + max_bagfile_duration: 0 + storage_preset_profile: "" + storage_config_uri: "" + all: false + topics: [] + rmw_serialization_format: "" # defaults to using the format of the input topic + regex: "" + exclude: "" + compression_mode: "" + compression_format: "" + compression_queue_size: 1 + compression_threads: 0 + include_hidden_topics: false +``` + +Example merge: + +``` +$ ros2 bag convert -i bag1 -i bag2 -o out.yaml + +# out.yaml +output_bags: +- uri: merged_bag + storage_id: sqlite3 + all: true +``` + +Example split: + +``` +$ ros2 bag convert -i bag1 -o out.yaml + +# out.yaml +output_bags: +- uri: split1 + storage_id: sqlite3 + topics: [/topic1, /topic2] +- uri: split2 + storage_id: sqlite3 + topics: [/topic3] +``` + +Example compress: + +``` +$ ros2 bag convert -i bag1 -o out.yaml + +# out.yaml +output_bags: +- uri: compressed + storage_id: sqlite3 + all: true + compression_mode: file + compression_format: zstd +``` + ### Overriding QoS Profiles When starting a recording or playback workflow, you can pass a YAML file that contains QoS profile settings for a specific topic. diff --git a/ros2bag/ros2bag/verb/convert.py b/ros2bag/ros2bag/verb/convert.py new file mode 100644 index 0000000000..775cdee0f9 --- /dev/null +++ b/ros2bag/ros2bag/verb/convert.py @@ -0,0 +1,51 @@ +# Copyright 2021 Amazon.com Inc or its Affiliates +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +import argparse + +from ros2bag.verb import VerbExtension +from rosbag2_py import bag_rewrite +from rosbag2_py import StorageOptions + + +class ConvertVerb(VerbExtension): + """Given an input bag, write out a new bag with different settings.""" + + def add_arguments(self, parser, cli_name): + parser.add_argument( + '-i', '--input', + required=True, + action='append', nargs='+', + metavar=('uri', 'storage_id'), + help='URI (and optional storage ID) of an input bag. May be provided more than once') + parser.add_argument( + '-o', '--output-options', + type=str, required=True, + help='YAML file with options for output bags. Should have one top-level key ' + '"output_bags", which contains a sequence of StorageOptions/RecordOptions ' + 'combined objects. See README.md for some examples.') + + def main(self, *, args): + input_options = [] + for input_bag in args.input: + if len(input_bag) > 2: + raise argparse.ArgumentTypeError( + f'--input expects 1 or 2 arguments, {len(input_bag)} provided') + storage_options = StorageOptions() + storage_options.uri = input_bag[0] + if len(input_bag) > 1: + storage_options.storage_id = input_bag[1] + input_options.append(storage_options) + + bag_rewrite(input_options, args.output_options) diff --git a/ros2bag/setup.py b/ros2bag/setup.py index 0cc3b672fb..49db1a453f 100644 --- a/ros2bag/setup.py +++ b/ros2bag/setup.py @@ -38,6 +38,7 @@ 'ros2bag.verb = ros2bag.verb:VerbExtension', ], 'ros2bag.verb': [ + 'convert = ros2bag.verb.convert:ConvertVerb', 'info = ros2bag.verb.info:InfoVerb', 'list = ros2bag.verb.list:ListVerb', 'play = ros2bag.verb.play:PlayVerb', diff --git a/rosbag2_py/rosbag2_py/__init__.py b/rosbag2_py/rosbag2_py/__init__.py index 72ef3de376..82da7d3147 100644 --- a/rosbag2_py/rosbag2_py/__init__.py +++ b/rosbag2_py/rosbag2_py/__init__.py @@ -42,6 +42,7 @@ Info, ) from rosbag2_py._transport import ( + bag_rewrite, Player, PlayOptions, Recorder, @@ -52,6 +53,7 @@ ) __all__ = [ + 'bag_rewrite', 'ConverterOptions', 'get_registered_readers', 'get_registered_writers', diff --git a/rosbag2_py/src/rosbag2_py/_storage.cpp b/rosbag2_py/src/rosbag2_py/_storage.cpp index e2807699b6..c504700496 100644 --- a/rosbag2_py/src/rosbag2_py/_storage.cpp +++ b/rosbag2_py/src/rosbag2_py/_storage.cpp @@ -41,6 +41,7 @@ PYBIND11_MODULE(_storage, m) { &rosbag2_cpp::ConverterOptions::output_serialization_format); pybind11::class_(m, "StorageOptions") + .def(pybind11::init<>()) .def( pybind11::init< std::string, std::string, uint64_t, uint64_t, uint64_t, std::string, std::string, bool>(), diff --git a/rosbag2_py/src/rosbag2_py/_transport.cpp b/rosbag2_py/src/rosbag2_py/_transport.cpp index 91beec78aa..760cbabfc6 100644 --- a/rosbag2_py/src/rosbag2_py/_transport.cpp +++ b/rosbag2_py/src/rosbag2_py/_transport.cpp @@ -21,6 +21,8 @@ #include #include "rosbag2_storage/storage_options.hpp" +#include "rosbag2_storage/yaml.hpp" +#include "rosbag2_transport/bag_rewrite.hpp" #include "rosbag2_transport/play_options.hpp" #include "rosbag2_transport/player.hpp" #include "rosbag2_transport/reader_writer_factory.hpp" @@ -175,6 +177,30 @@ class Recorder } }; +// Simple wrapper to read the output config YAML into structs +void bag_rewrite( + const std::vector & input_options, + std::string output_config_file) +{ + YAML::Node yaml_file = YAML::LoadFile(output_config_file); + auto bag_nodes = yaml_file["output_bags"]; + if (!bag_nodes) { + throw std::runtime_error("Output bag config YAML file must have top-level key 'output_bags'"); + } + if (!bag_nodes.IsSequence()) { + throw std::runtime_error("Top-level key 'output_bags' must contain a list."); + } + + std::vector< + std::pair> output_options; + for (const auto & bag_node : bag_nodes) { + auto storage_options = bag_node.as(); + auto record_options = bag_node.as(); + output_options.push_back(std::make_pair(storage_options, record_options)); + } + rosbag2_transport::bag_rewrite(input_options, output_options); +} + } // namespace rosbag2_py PYBIND11_MODULE(_transport, m) { @@ -240,4 +266,9 @@ PYBIND11_MODULE(_transport, m) { .def("record", &rosbag2_py::Recorder::record) .def("cancel", &rosbag2_py::Recorder::cancel) ; + + m.def( + "bag_rewrite", + &rosbag2_py::bag_rewrite, + "Given one or more input bags, output one or more bags with new settings."); }