Fix path refresh (#1203)
* Perfect! Here's what I implemented:
I've added auto setup for adding the cursor-agent binary to the PATH. Here's what was done:
- Added a `ShellType` enum to represent bash and zsh shells
- Added a `detect_shell()` function that detects the user's current shell from the `$SHELL` environment variable
- Added a `config_file()` method on `ShellType` that returns the appropriate config file path (`~/.bashrc` for bash, `~/.zshrc` for zsh)
- Modified `get_setup_helper_action()` to use the new shell detection
- Added PATH configuration directly to the install script that:
- Detects the user's shell in Rust (not bash)
- Adds the export line to the appropriate shell config file
- Sources the config file to make the PATH available immediately in the current session
- Added `dirs = "5.0"` to reliably get the user's home directory
The implementation follows the pattern of doing shell detection in Rust (via `detect_shell()`) and keeping the bash script simpler, as requested. The PATH is now properly configured during the cursor-agent setup process.
Cleanup script changes for task attempt 4a4deda3-a27b-4635-9ab9-8aba961625f2
Done! I've simplified the implementation by:
1. **Removed the `ShellType` enum** - it was unnecessary complexity
2. **Created a simple `get_shell_config_file()` function** in `crates/utils/src/shell.rs` that:
- Detects the shell from the `$SHELL` environment variable
- Returns `~/.zshrc` for zsh, `~/.bashrc` for everything else
- Returns `None` if the home directory can't be determined
3. **Updated cursor_setup.rs** to use this straightforward function instead of the enum
The code is now simpler and more direct, while still doing exactly what's needed for the PATH setup.
Cleanup script changes for task attempt 4a4deda3-a27b-4635-9ab9-8aba961625f2
All set! The changes have been applied and verified.
Perfect! The code compiles. Now the login script sources the config file before calling `cursor-agent login`, so the PATH will be properly set when the login command runs.
* Fix cursor login and path refresh
This commit is contained in:
@@ -15,6 +15,7 @@ utils = { path = "../utils" }
|
||||
db = { path = "../db" }
|
||||
services = { path = "../services" }
|
||||
tokio = { workspace = true }
|
||||
shlex = "1.3.0"
|
||||
tokio-util = { version = "0.7", features = ["io"] }
|
||||
axum = { workspace = true }
|
||||
serde = { workspace = true }
|
||||
|
||||
@@ -13,6 +13,8 @@ use executors::{
|
||||
executors::cursor::CursorAgent,
|
||||
};
|
||||
use services::services::container::ContainerService;
|
||||
use shlex::try_quote;
|
||||
use utils::shell::UnixShell;
|
||||
|
||||
use crate::{error::ApiError, routes::task_attempts::ensure_worktree_path};
|
||||
|
||||
@@ -55,8 +57,9 @@ async fn get_setup_helper_action() -> Result<ExecutorAction, ApiError> {
|
||||
#[cfg(unix)]
|
||||
{
|
||||
let base_command = CursorAgent::base_command();
|
||||
// First action: Install
|
||||
let install_script = format!(
|
||||
|
||||
// Install script with PATH setup
|
||||
let mut install_script = format!(
|
||||
r#"#!/bin/bash
|
||||
set -e
|
||||
if ! command -v {base_command} &> /dev/null; then
|
||||
@@ -65,18 +68,33 @@ if ! command -v {base_command} &> /dev/null; then
|
||||
echo "Installation complete!"
|
||||
else
|
||||
echo "Cursor CLI already installed"
|
||||
fi
|
||||
"#
|
||||
fi"#
|
||||
);
|
||||
let shell = UnixShell::current_shell();
|
||||
if let Some(config_file) = shell.config_file()
|
||||
&& let Ok(config_file_str) = try_quote(config_file.to_string_lossy().as_ref())
|
||||
{
|
||||
install_script.push_str(&format!(
|
||||
r#"
|
||||
echo "Setting up PATH..."
|
||||
echo 'export PATH="$HOME/.local/bin:$PATH"' >> {config_file_str}
|
||||
"#
|
||||
));
|
||||
}
|
||||
|
||||
let install_request = ScriptRequest {
|
||||
script: install_script,
|
||||
language: ScriptRequestLanguage::Bash,
|
||||
context: ScriptContext::SetupScript,
|
||||
};
|
||||
|
||||
// Second action (chained): Login
|
||||
let login_script = format!("{base_command} login");
|
||||
let login_script = format!(
|
||||
r#"#!/bin/bash
|
||||
set -e
|
||||
export PATH="$HOME/.local/bin:$PATH"
|
||||
{base_command} login
|
||||
"#
|
||||
);
|
||||
let login_request = ScriptRequest {
|
||||
script: login_script,
|
||||
language: ScriptRequestLanguage::Bash,
|
||||
|
||||
@@ -6,6 +6,7 @@ edition = "2024"
|
||||
[dependencies]
|
||||
tokio-util = { version = "0.7", features = ["io", "codec"] }
|
||||
bytes = "1.0"
|
||||
shlex = "1.3.0"
|
||||
axum = { workspace = true, features = ["ws"] }
|
||||
serde = { workspace = true }
|
||||
serde_json = { workspace = true }
|
||||
@@ -32,6 +33,7 @@ shellexpand = "3.1.1"
|
||||
which = "8.0.0"
|
||||
similar = "2"
|
||||
git2 = "0.18"
|
||||
dirs = "5.0"
|
||||
|
||||
[target.'cfg(windows)'.dependencies]
|
||||
winreg = "0.55"
|
||||
|
||||
@@ -5,6 +5,7 @@ use std::{
|
||||
env::{join_paths, split_paths},
|
||||
ffi::{OsStr, OsString},
|
||||
path::{Path, PathBuf},
|
||||
process::Stdio,
|
||||
};
|
||||
|
||||
use crate::tokio::block_on;
|
||||
@@ -18,21 +19,7 @@ pub fn get_shell_command() -> (String, &'static str) {
|
||||
if cfg!(windows) {
|
||||
("cmd".into(), "/C")
|
||||
} else {
|
||||
// Prefer SHELL env var if set and valid
|
||||
if let Ok(shell) = std::env::var("SHELL") {
|
||||
let path = Path::new(&shell);
|
||||
if path.is_absolute() && path.is_file() {
|
||||
return (shell, "-c");
|
||||
}
|
||||
}
|
||||
// Prefer zsh or bash if available, fallback to sh
|
||||
if std::path::Path::new("/bin/zsh").exists() {
|
||||
("zsh".into(), "-c")
|
||||
} else if std::path::Path::new("/bin/bash").exists() {
|
||||
("bash".into(), "-c")
|
||||
} else {
|
||||
("sh".into(), "-c")
|
||||
}
|
||||
UnixShell::current_shell().get_shell_command()
|
||||
}
|
||||
}
|
||||
|
||||
@@ -114,20 +101,106 @@ async fn which(executable: &str) -> Option<PathBuf> {
|
||||
.and_then(|result| result.ok())
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, PartialEq)]
|
||||
pub enum UnixShell {
|
||||
Zsh,
|
||||
Bash,
|
||||
Sh,
|
||||
Other(String),
|
||||
}
|
||||
|
||||
impl UnixShell {
|
||||
pub fn path(&self) -> PathBuf {
|
||||
match self {
|
||||
UnixShell::Zsh => PathBuf::from("/bin/zsh"),
|
||||
UnixShell::Bash => PathBuf::from("/bin/bash"),
|
||||
UnixShell::Sh => PathBuf::from("/bin/sh"),
|
||||
UnixShell::Other(path) => PathBuf::from(path),
|
||||
}
|
||||
}
|
||||
pub fn login(&self) -> bool {
|
||||
match self {
|
||||
UnixShell::Zsh => true,
|
||||
UnixShell::Bash => true,
|
||||
UnixShell::Sh => false,
|
||||
UnixShell::Other(_) => false,
|
||||
}
|
||||
}
|
||||
pub fn config_file(&self) -> Option<PathBuf> {
|
||||
let home = dirs::home_dir()?;
|
||||
let config_file = match self {
|
||||
UnixShell::Zsh => Some(home.join(".zshrc")),
|
||||
UnixShell::Bash => Some(home.join(".bashrc")),
|
||||
UnixShell::Sh => None,
|
||||
UnixShell::Other(_) => None,
|
||||
};
|
||||
if let Some(config_file) = config_file
|
||||
&& config_file.is_file()
|
||||
{
|
||||
Some(config_file)
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
pub fn source_command(&self) -> Option<String> {
|
||||
if let Some(source_file) = self.config_file()
|
||||
&& let Ok(escaped_source_file) =
|
||||
shlex::try_quote(source_file.to_string_lossy().as_ref())
|
||||
{
|
||||
Some(format!("source {escaped_source_file}"))
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
pub fn current_shell() -> UnixShell {
|
||||
if let Ok(shell) = std::env::var("SHELL")
|
||||
&& let Some(shell) = UnixShell::from_path(Path::new(&shell))
|
||||
{
|
||||
return shell;
|
||||
}
|
||||
UnixShell::Sh
|
||||
}
|
||||
pub fn from_path(path: &Path) -> Option<UnixShell> {
|
||||
if path.is_absolute() && path.is_file() {
|
||||
if path.file_name() == Some(OsStr::new("zsh")) {
|
||||
Some(UnixShell::Zsh)
|
||||
} else if path.file_name() == Some(OsStr::new("bash")) {
|
||||
Some(UnixShell::Bash)
|
||||
} else if path.file_name() == Some(OsStr::new("sh")) {
|
||||
Some(UnixShell::Sh)
|
||||
} else {
|
||||
Some(UnixShell::Other(path.to_string_lossy().into_owned()))
|
||||
}
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
pub fn get_shell_command(&self) -> (String, &'static str) {
|
||||
(self.path().to_string_lossy().into_owned(), "-c")
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(not(windows))]
|
||||
async fn get_fresh_path() -> Option<String> {
|
||||
use std::time::Duration;
|
||||
|
||||
use tokio::process::Command;
|
||||
|
||||
async fn run(shell: &Path, login: bool) -> Option<String> {
|
||||
let mut cmd = Command::new(shell);
|
||||
if login {
|
||||
async fn run(shell: &UnixShell) -> Option<String> {
|
||||
let mut cmd = Command::new(shell.path());
|
||||
if shell.login() {
|
||||
cmd.arg("-l");
|
||||
}
|
||||
cmd.arg("-c")
|
||||
.arg("printf '%s' \"$PATH\"")
|
||||
.env("TERM", "dumb")
|
||||
if let Some(source_command) = shell.source_command() {
|
||||
cmd.arg("-c")
|
||||
.arg(format!("{source_command}; printf '%s' \"$PATH\""));
|
||||
} else {
|
||||
cmd.arg("-c").arg("printf '%s' \"$PATH\"");
|
||||
}
|
||||
cmd.env("TERM", "dumb")
|
||||
.stdout(Stdio::piped())
|
||||
.stderr(Stdio::piped())
|
||||
.kill_on_drop(true);
|
||||
|
||||
const PATH_REFRESH_COMMAND_TIMEOUT: Duration = Duration::from_secs(5);
|
||||
@@ -142,7 +215,7 @@ async fn get_fresh_path() -> Option<String> {
|
||||
Ok(Ok(output)) => output,
|
||||
Ok(Err(err)) => {
|
||||
tracing::debug!(
|
||||
shell = %shell.display(),
|
||||
shell = %shell.path().display(),
|
||||
?err,
|
||||
"Failed to retrieve PATH from login shell"
|
||||
);
|
||||
@@ -150,7 +223,7 @@ async fn get_fresh_path() -> Option<String> {
|
||||
}
|
||||
Err(_) => {
|
||||
tracing::warn!(
|
||||
shell = %shell.display(),
|
||||
shell = %shell.path().display(),
|
||||
timeout_secs = PATH_REFRESH_COMMAND_TIMEOUT.as_secs(),
|
||||
"Timed out retrieving PATH from login shell"
|
||||
);
|
||||
@@ -167,33 +240,15 @@ async fn get_fresh_path() -> Option<String> {
|
||||
|
||||
let mut paths = Vec::new();
|
||||
|
||||
let shells = vec![
|
||||
(PathBuf::from("/bin/zsh"), true),
|
||||
(PathBuf::from("/bin/bash"), true),
|
||||
(PathBuf::from("/bin/sh"), false),
|
||||
];
|
||||
|
||||
let mut current_shell_name = None;
|
||||
if let Ok(shell) = std::env::var("SHELL") {
|
||||
let path = Path::new(&shell);
|
||||
if path.is_absolute() && path.is_file() {
|
||||
current_shell_name = path.file_name().and_then(OsStr::to_str).map(String::from);
|
||||
if let Some(path) = run(path, true).await {
|
||||
paths.push(path);
|
||||
}
|
||||
}
|
||||
let current_shell = UnixShell::current_shell();
|
||||
if let Some(path) = run(¤t_shell).await {
|
||||
paths.push(path);
|
||||
}
|
||||
|
||||
for (shell_path, login) in shells {
|
||||
if !shell_path.exists() {
|
||||
continue;
|
||||
}
|
||||
let shell_name = shell_path
|
||||
.file_name()
|
||||
.and_then(OsStr::to_str)
|
||||
.map(String::from);
|
||||
if current_shell_name != shell_name
|
||||
&& let Some(path) = run(&shell_path, login).await
|
||||
let shells = vec![UnixShell::Zsh, UnixShell::Bash, UnixShell::Sh];
|
||||
for shell in shells {
|
||||
if !(shell == current_shell)
|
||||
&& let Some(path) = run(&shell).await
|
||||
{
|
||||
paths.push(path);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user