#[path = "../common/logsetup.rs"] mod logsetup; use std::{fs, io::BufWriter}; use anyhow::{Context, Result}; use camino::{Utf8Path, Utf8PathBuf}; use clap::Parser; use log::*; use mcap_rs as mcap; use memmap::Mmap; #[derive(Parser, Debug)] struct Args { /// Verbosity (-v, -vv, -vvv, etc.) #[clap(short, long, parse(from_occurrences))] verbose: u8, #[clap(short, long, arg_enum, default_value = "auto")] color: logsetup::Color, mcap: Utf8PathBuf, } fn map_mcap(p: &Utf8Path) -> Result { let fd = fs::File::open(p).context("Couldn't open MCAP file")?; unsafe { Mmap::map(&fd) }.context("Couldn't map MCAP file") } fn run() -> Result<()> { let args = Args::parse(); logsetup::init_logger(args.verbose, args.color); let mapped = map_mcap(&args.mcap)?; let mut out = mcap::Writer::new(BufWriter::new(fs::File::create("out.mcap")?))?; for message in mcap::MessageStream::new(&mapped)? { let message = message?; let ts = message.publish_time; info!( "{} {} [{}] [{}]...", ts, message.channel.topic, message .channel .schema .as_ref() .map(|s| s.name.as_str()) .unwrap_or_default(), message .data .iter() .take(10) .map(|b| b.to_string()) .collect::>() .join(" ") ); // We can easily take each Message and write it as a quick and dirty example, // but in real code, we'd be much better off adding each channel to the writer, // then calling `write_to_known_channel()`. // This avoids having to rehash the channel (and its schema) on each `write()` // to figure out what its ID is. out.write(&message)?; } Ok(()) } fn main() { run().unwrap_or_else(|e| { error!("{:?}", e); std::process::exit(1); }); }