rerun 0.31.2

Log images, point clouds, etc, and visualize them effortlessly
Documentation
//! Integration with integration with the [`clap`](https://crates.io/crates/clap) command line argument parser.

use std::path::PathBuf;

use re_sdk::{RecordingStream, RecordingStreamBuilder};

// ---

#[derive(Debug, Clone, PartialEq, Eq)]
enum RerunBehavior {
    Connect(String),

    Save(PathBuf),

    Stdout,

    #[cfg(feature = "web_viewer")]
    Serve,

    Spawn,
}

/// This struct implements a `clap::Parser` that defines all the arguments that a typical Rerun
/// application might use, and provides helpers to evaluate those arguments and behave
/// consequently.
///
/// Integrate it into your own `clap::Parser` by flattening it:
/// ```no_run
/// #[derive(Debug, clap::Parser)]
/// #[clap(author, version, about)]
/// struct MyArgs {
///     #[command(flatten)]
///     rerun: rerun::clap::RerunArgs,
///
///     #[clap(long)]
///     my_arg: bool,
/// }
/// ```
///
/// Checkout the official examples to see it used in practice.
#[derive(Clone, Debug, clap::Args)]
#[clap(author, version, about)]
pub struct RerunArgs {
    /// Start a new Rerun Viewer process and feed it data in real-time.
    #[clap(long, default_value = "true")]
    spawn: bool,

    /// Saves the data to an rrd file rather than visualizing it immediately.
    #[clap(long)]
    save: Option<PathBuf>,

    /// Log data to standard output, to be piped into a Rerun Viewer.
    #[clap(long, short = 'o')]
    stdout: bool,

    /// Connects and sends the logged data to a remote Rerun viewer.
    ///
    /// Optionally takes a URL to connect to.
    ///
    /// The scheme must be one of `rerun://`, `rerun+http://`, or `rerun+https://`,
    /// and the pathname must be `/proxy`.
    ///
    /// The default is `rerun+http://127.0.0.1:9876/proxy`.
    #[clap(long)]
    #[expect(clippy::option_option)]
    connect: Option<Option<String>>,

    /// Connects and sends the logged data to a web-based Rerun viewer.
    #[cfg(feature = "web_viewer")]
    #[clap(long)]
    serve: bool,

    /// An upper limit on how much memory the gRPC server should use.
    ///
    /// The server buffers log messages for the benefit of late-arriving viewers.
    ///
    /// When this limit is reached, Rerun will drop the oldest data.
    /// Example: `16GB` or `50%` (of system total).
    ///
    /// Defaults to `25%`.
    #[clap(long, default_value = "25%")]
    server_memory_limit: String,

    /// If true, play back the most recent data first when new clients connect.
    #[clap(long)]
    newest_first: bool,

    /// What bind address IP to use.
    #[clap(long, default_value = "0.0.0.0")]
    bind: String,
}

/// [`RerunArgs::init`] might have to spawn a bunch of background tasks depending on what arguments
/// were passed in.
/// This object makes sure they live long enough and get polled as needed.
#[doc(hidden)]
#[derive(Default)]
pub struct ServeGuard {
    block_on_drop: bool,
}

impl Drop for ServeGuard {
    fn drop(&mut self) {
        if self.block_on_drop {
            eprintln!("Sleeping indefinitely while serving web viewer… Press ^C when done.");
            // TODO(andreas): It would be a lot better if we had a handle to the web server and could call `block_until_shutdown` on it.
            std::thread::sleep(std::time::Duration::from_secs(u64::MAX));
        }
    }
}

impl RerunArgs {
    /// Creates a new [`RecordingStream`] according to the CLI parameters.
    #[track_caller] // track_caller so that we can see if we are being called from an official example.
    pub fn init(&self, application_id: &str) -> anyhow::Result<(RecordingStream, ServeGuard)> {
        self.init_with_blueprint_opts(application_id, None)
    }

    /// Creates a new [`RecordingStream`] with a [`re_sdk::blueprint::Blueprint`] that activates immediately.
    ///
    /// For a default blueprint that only activates on reset, see [`Self::init_with_default_blueprint`].
    #[track_caller]
    pub fn init_with_blueprint(
        &self,
        application_id: &str,
        blueprint: re_sdk::blueprint::Blueprint,
    ) -> anyhow::Result<(RecordingStream, ServeGuard)> {
        let activation = re_sdk::blueprint::BlueprintActivation {
            make_active: true,
            make_default: true,
        };
        self.init_with_blueprint_opts(
            application_id,
            Some(re_sdk::blueprint::BlueprintOpts {
                blueprint,
                activation,
            }),
        )
    }

    /// Creates a new [`RecordingStream`] with a default [`re_sdk::blueprint::Blueprint`].
    ///
    /// The blueprint activates only when the user resets. For immediate activation, see [`Self::init_with_blueprint`].
    #[track_caller]
    pub fn init_with_default_blueprint(
        &self,
        application_id: &str,
        blueprint: re_sdk::blueprint::Blueprint,
    ) -> anyhow::Result<(RecordingStream, ServeGuard)> {
        let activation = re_sdk::blueprint::BlueprintActivation {
            make_active: false,
            make_default: true,
        };
        self.init_with_blueprint_opts(
            application_id,
            Some(re_sdk::blueprint::BlueprintOpts {
                blueprint,
                activation,
            }),
        )
    }

    /// Internal helper that handles blueprint options.
    #[track_caller]
    fn init_with_blueprint_opts(
        &self,
        application_id: &str,
        blueprint_opts: Option<re_sdk::blueprint::BlueprintOpts>,
    ) -> anyhow::Result<(RecordingStream, ServeGuard)> {
        let mut builder = RecordingStreamBuilder::new(application_id);

        // Add blueprint to builder if provided
        if let Some(re_sdk::blueprint::BlueprintOpts {
            blueprint,
            activation,
        }) = blueprint_opts
        {
            builder = if activation.make_active {
                builder.with_blueprint(blueprint)
            } else {
                builder.with_default_blueprint(blueprint)
            };
        }

        match self.to_behavior()? {
            RerunBehavior::Stdout => Ok((builder.stdout()?, Default::default())),

            RerunBehavior::Connect(url) => {
                Ok((builder.connect_grpc_opts(url)?, Default::default()))
            }

            RerunBehavior::Save(path) => Ok((builder.save(path)?, Default::default())),

            RerunBehavior::Spawn => Ok((builder.spawn()?, Default::default())),

            #[cfg(feature = "web_viewer")]
            RerunBehavior::Serve => {
                let server_options = re_sdk::ServerOptions {
                    playback_behavior: re_sdk::PlaybackBehavior::from_newest_first(
                        self.newest_first,
                    ),

                    memory_limit: re_sdk::MemoryLimit::parse(&self.server_memory_limit)
                        .map_err(|err| anyhow::format_err!("Bad --server-memory-limit: {err}"))?,
                };

                let rec = builder.serve_grpc_opts(
                    &self.bind,
                    crate::DEFAULT_SERVER_PORT,
                    server_options,
                )?;

                crate::serve_web_viewer(crate::web_viewer::WebViewerConfig {
                    open_browser: true,
                    connect_to: vec!["rerun+http://localhost:9876/proxy".to_owned()],
                    ..Default::default()
                })?
                .detach();

                // Ensure the server stays alive until the end of the program.
                let sleep_guard = ServeGuard {
                    block_on_drop: true,
                };

                Ok((rec, sleep_guard))
            }
        }
    }

    #[expect(clippy::unnecessary_wraps)] // False positive on some feature flags
    fn to_behavior(&self) -> anyhow::Result<RerunBehavior> {
        if self.stdout {
            return Ok(RerunBehavior::Stdout);
        }

        if let Some(path) = self.save.as_ref() {
            return Ok(RerunBehavior::Save(path.clone()));
        }

        #[cfg(feature = "web_viewer")]
        if self.serve {
            return Ok(RerunBehavior::Serve);
        }

        match &self.connect {
            Some(Some(url)) => return Ok(RerunBehavior::Connect(url.clone())),
            Some(None) => {
                return Ok(RerunBehavior::Connect(
                    re_sdk::DEFAULT_CONNECT_URL.to_owned(),
                ));
            }
            None => {}
        }

        Ok(RerunBehavior::Spawn)
    }
}