use std::{collections::HashSet, io::IsTerminal};
use anyhow::Context as _;
use itertools::Either;
use re_build_info::CrateVersion;
use re_chunk::{external::crossbeam, TransportChunk};
use crate::commands::read_rrd_streams_from_file_or_stdin;
#[derive(Debug, Clone, clap::Parser)]
pub struct FilterCommand {
path_to_input_rrds: Vec<String>,
#[arg(short = 'o', long = "output", value_name = "dst.(rrd|rbl)")]
path_to_output_rrd: Option<String>,
#[clap(long = "drop-timeline")]
dropped_timelines: Vec<String>,
#[clap(long = "drop-entity")]
dropped_entity_paths: Vec<String>,
#[clap(long = "continue-on-error", default_value_t = false)]
continue_on_error: bool,
}
impl FilterCommand {
pub fn run(&self) -> anyhow::Result<()> {
let Self {
path_to_input_rrds,
path_to_output_rrd,
dropped_timelines,
dropped_entity_paths,
continue_on_error,
} = self;
let path_to_output_rrd = path_to_output_rrd.clone();
if path_to_output_rrd.is_none() {
anyhow::ensure!(
!std::io::stdout().is_terminal(),
"you must redirect the output to a file and/or stream"
);
}
let now = std::time::Instant::now();
re_log::info!(srcs = ?path_to_input_rrds, ?dropped_timelines, "filter started");
let dropped_timelines: HashSet<_> = dropped_timelines.iter().collect();
let dropped_entity_paths: HashSet<EntityPath> = dropped_entity_paths
.iter()
.map(|s| EntityPath::parse_forgiving(s))
.collect();
let version_policy = re_log_encoding::VersionPolicy::Warn;
let (rx_decoder, rx_size_bytes) =
read_rrd_streams_from_file_or_stdin(version_policy, path_to_input_rrds);
let (tx_encoder, rx_encoder) = crossbeam::channel::bounded(100);
let encoding_handle = std::thread::Builder::new()
.name("rerun-rrd-filter-out".to_owned())
.spawn(move || -> anyhow::Result<u64> {
use std::io::Write as _;
let mut rrd_out = if let Some(path) = path_to_output_rrd.as_ref() {
Either::Left(std::io::BufWriter::new(
std::fs::File::create(path).with_context(|| format!("{path:?}"))?,
))
} else {
Either::Right(std::io::BufWriter::new(std::io::stdout().lock()))
};
let mut encoder = {
let version = CrateVersion::LOCAL;
let options = re_log_encoding::EncodingOptions::MSGPACK_COMPRESSED;
re_log_encoding::encoder::DroppableEncoder::new(version, options, &mut rrd_out)
.context("couldn't init encoder")?
};
let mut size_bytes = 0;
for msg in rx_encoder {
size_bytes += encoder.append(&msg).context("encoding failure")?;
}
drop(encoder);
rrd_out.flush().context("couldn't flush output")?;
Ok(size_bytes)
});
for res in rx_decoder {
let mut is_success = true;
match res {
Ok(msg) => {
let msg = match msg {
re_log_types::LogMsg::ArrowMsg(store_id, mut msg) => {
if !should_keep_entity_path(&dropped_entity_paths, &msg.schema) {
None
} else {
let (fields, columns): (Vec<_>, Vec<_>) =
itertools::izip!(msg.schema.fields.iter(), msg.chunk.iter())
.filter(|(field, _col)| {
should_keep_timeline(&dropped_timelines, field)
})
.map(|(field, col)| (field.clone(), col.clone()))
.unzip();
msg.schema.fields = fields;
msg.chunk =
re_log_types::external::arrow2::chunk::Chunk::new(columns);
Some(re_log_types::LogMsg::ArrowMsg(store_id, msg))
}
}
msg => Some(msg),
};
if let Some(msg) = msg {
tx_encoder.send(msg).ok();
}
}
Err(err) => {
re_log::error!(err = re_error::format(err));
is_success = false;
}
}
if !*continue_on_error && !is_success {
anyhow::bail!(
"one or more IO and/or decoding failures in the input stream (check logs)"
)
}
}
std::mem::drop(tx_encoder);
let rrd_out_size = encoding_handle
.context("couldn't spawn IO thread")?
.join()
.unwrap()?;
let rrds_in_size = rx_size_bytes.recv().ok();
let size_reduction =
if let (Some(rrds_in_size), rrd_out_size) = (rrds_in_size, rrd_out_size) {
format!(
"-{:3.3}%",
100.0 - rrd_out_size as f64 / (rrds_in_size as f64 + f64::EPSILON) * 100.0
)
} else {
"N/A".to_owned()
};
let file_size_to_string = |size: Option<u64>| {
size.map_or_else(
|| "<unknown>".to_owned(),
|size| re_format::format_bytes(size as _),
)
};
re_log::info!(
dst_size_bytes = %file_size_to_string(Some(rrd_out_size)),
time = ?now.elapsed(),
size_reduction,
srcs = ?path_to_input_rrds,
srcs_size_bytes = %file_size_to_string(rrds_in_size),
"filter finished"
);
Ok(())
}
}
use re_sdk::{
external::arrow2::{datatypes::Field as ArrowField, datatypes::Schema as Arrow2Schema},
EntityPath,
};
fn should_keep_timeline(dropped_timelines: &HashSet<&String>, field: &ArrowField) -> bool {
let is_timeline = field
.metadata
.get(TransportChunk::FIELD_METADATA_KEY_KIND)
.map(|s| s.as_str())
== Some(TransportChunk::FIELD_METADATA_VALUE_KIND_TIME);
let is_dropped = dropped_timelines.contains(&field.name);
!is_timeline || !is_dropped
}
fn should_keep_entity_path(
dropped_entity_paths: &HashSet<EntityPath>,
schema: &Arrow2Schema,
) -> bool {
let Some(entity_path) = schema
.metadata
.get(TransportChunk::CHUNK_METADATA_KEY_ENTITY_PATH)
.map(|s| EntityPath::parse_forgiving(s))
else {
return true;
};
let is_dropped = dropped_entity_paths.contains(&entity_path);
!is_dropped
}