aboutsummaryrefslogtreecommitdiffstats
path: root/crates/turtle/src/command/mod.rs
diff options
context:
space:
mode:
Diffstat (limited to 'crates/turtle/src/command/mod.rs')
-rw-r--r--crates/turtle/src/command/mod.rs156
1 files changed, 156 insertions, 0 deletions
diff --git a/crates/turtle/src/command/mod.rs b/crates/turtle/src/command/mod.rs
new file mode 100644
index 00000000..e58bfe72
--- /dev/null
+++ b/crates/turtle/src/command/mod.rs
@@ -0,0 +1,156 @@
+use clap::Subcommand;
+use eyre::Result;
+
+#[cfg(not(windows))]
+use rustix::{fs::Mode, process::umask};
+
+#[cfg(feature = "client")]
+mod client;
+
+mod contributors;
+
+mod gen_completions;
+
+mod external;
+
+#[derive(Subcommand)]
+#[command(infer_subcommands = true)]
+#[expect(clippy::large_enum_variant)]
+pub enum AtuinCmd {
+ #[cfg(feature = "client")]
+ #[command(flatten)]
+ Client(client::Cmd),
+
+ /// PTY proxy for atuin
+ #[cfg(feature = "pty-proxy")]
+ #[command(alias = "hex")]
+ PtyProxy(crate::atuin_pty_proxy::PtyProxy),
+
+ /// Generate a UUID
+ Uuid,
+
+ Contributors,
+
+ /// Generate shell completions
+ GenCompletions(gen_completions::Cmd),
+
+ #[command(external_subcommand)]
+ External(Vec<String>),
+}
+
+impl AtuinCmd {
+ pub fn run(self) -> Result<()> {
+ #[cfg(not(windows))]
+ {
+ // set umask before we potentially open/create files
+ // or in other words, 077. Do not allow any access to any other user
+ let mode = Mode::RWXG | Mode::RWXO;
+ umask(mode);
+ }
+
+ match self {
+ #[cfg(feature = "client")]
+ Self::Client(client) => client.run(),
+
+ #[cfg(feature = "pty-proxy")]
+ Self::PtyProxy(proxy) => {
+ run_pty_proxy(proxy);
+ Ok(())
+ }
+
+ Self::Contributors => {
+ contributors::run();
+ Ok(())
+ }
+ Self::Uuid => {
+ println!("{}", crate::atuin_common::utils::uuid_v7().as_simple());
+ Ok(())
+ }
+ Self::GenCompletions(gen_completions) => gen_completions.run(),
+ Self::External(args) => external::run(&args),
+ }
+ }
+}
+
+#[cfg(all(feature = "pty-proxy", unix))]
+fn run_pty_proxy(proxy: crate::atuin_pty_proxy::PtyProxy) {
+ #[cfg(feature = "daemon")]
+ proxy.run(semantic_command_capture_sink());
+
+ #[cfg(not(feature = "daemon"))]
+ proxy.run(None);
+}
+
+#[cfg(all(feature = "daemon", feature = "pty-proxy", unix))]
+fn semantic_command_capture_sink() -> Option<crate::atuin_pty_proxy::CommandCaptureSink> {
+ use std::sync::mpsc;
+ use std::time::Duration;
+
+ if is_truthy_env("ATUIN_TERMINAL") {
+ return None;
+ }
+
+ let settings = crate::atuin_client::settings::Settings::new().ok()?;
+ let (tx, rx) = mpsc::sync_channel::<crate::atuin_pty_proxy::CommandCapture>(128);
+
+ std::thread::spawn(move || {
+ let Ok(runtime) = tokio::runtime::Builder::new_current_thread()
+ .enable_all()
+ .build()
+ else {
+ return;
+ };
+
+ while let Ok(first) = rx.recv() {
+ let mut batch = vec![first];
+
+ while batch.len() < 64 {
+ match rx.recv_timeout(Duration::from_millis(25)) {
+ Ok(capture) => batch.push(capture),
+ Err(mpsc::RecvTimeoutError::Timeout | mpsc::RecvTimeoutError::Disconnected) => {
+ break;
+ }
+ }
+ }
+
+ runtime.block_on(send_semantic_command_captures(&settings, batch));
+ }
+ });
+
+ Some(Box::new(move |capture| {
+ let _ = tx.try_send(capture);
+ }))
+}
+
+#[cfg(all(feature = "daemon", feature = "pty-proxy", unix))]
+#[inline]
+fn is_truthy_env(name: &str) -> bool {
+ std::env::var(name)
+ .ok()
+ .as_ref()
+ .is_some_and(|value| !value.trim().is_empty() && value.trim() != "false")
+}
+
+#[cfg(all(feature = "daemon", feature = "pty-proxy", unix))]
+async fn send_semantic_command_captures(
+ settings: &crate::atuin_client::settings::Settings,
+ batch: Vec<crate::atuin_pty_proxy::CommandCapture>,
+) {
+ let captures = batch
+ .into_iter()
+ .map(|capture| crate::atuin_daemon::semantic::CommandCapture {
+ prompt: capture.prompt,
+ command: capture.command,
+ output: capture.output,
+ exit_code: capture.exit_code,
+ history_id: capture.history_id,
+ session_id: capture.session_id,
+ output_truncated: capture.output_truncated,
+ output_observed_bytes: capture.output_observed_bytes,
+ })
+ .collect();
+
+ if let Ok(mut client) = crate::atuin_daemon::SemanticClient::from_settings(settings).await {
+ let _ = client.record_commands(captures).await;
+ }
+}