From 32aedc850d066ff559bf4c8592478cf371fac709 Mon Sep 17 00:00:00 2001 From: hardfau1t Date: Mon, 25 Nov 2024 05:12:18 +0530 Subject: [PATCH] fix(nu-command): `cd` on android fails to cd to `/sdcard` on android checking access to a directory by userid and group is not enough, it is also granted by android NDK, so to check if a user has permission just cd to that directory and check --- crates/nu-utils/src/filesystem.rs | 36 ++++++++++++++++++++++++++++++- 1 file changed, 35 insertions(+), 1 deletion(-) diff --git a/crates/nu-utils/src/filesystem.rs b/crates/nu-utils/src/filesystem.rs index 588f0fccff..a1bc299155 100644 --- a/crates/nu-utils/src/filesystem.rs +++ b/crates/nu-utils/src/filesystem.rs @@ -31,8 +31,13 @@ pub fn have_permission(dir: impl AsRef) -> PermissionResult<'static> { } } -#[cfg(unix)] +#[cfg(all(unix, not(target_os = "android")))] pub fn have_permission(dir: impl AsRef) -> PermissionResult<'static> { + check_unix_permission(dir) +} + +#[cfg(unix)] +fn check_unix_permission(dir: impl AsRef) -> PermissionResult<'static> { match dir.as_ref().metadata() { Ok(metadata) => { let mode = Mode::from_bits_truncate(metadata.mode() as mode_t); @@ -83,6 +88,35 @@ pub fn have_permission(dir: impl AsRef) -> PermissionResult<'static> { } } +#[cfg(all(unix, target_os = "android"))] +pub fn have_permission(dir: impl AsRef) -> PermissionResult<'static> { + // on android checking access to a directory by userid and group is not + // enough, it is also granted by android NDK, so to check if a user has + // permission just cd to that directory and check + let old_pwd = std::env::current_dir(); + + if let Err(_e) = std::env::set_current_dir(&dir) { + // get the possible reason of failure + let cause = check_unix_permission(dir); + + match &cause { + PermissionResult::PermissionDenied(_) => cause, + PermissionResult::PermissionOk => PermissionResult::PermissionDenied( + "You don't have permission to read this directory", + ), + } + } else { + // this could be Err() if the current_dir is deleted `std::env::current_dir()` will yield error + // if thats the case should we return to last pwd or keep the current_dir, or should we return to home dir? + if let Ok(old_pwd) = old_pwd { + // this is unlikely to happen, but if it errors + // should this error be ignored? + let _ = std::env::set_current_dir(old_pwd); + } + PermissionResult::PermissionOk + } +} + #[cfg(any(target_os = "linux", target_os = "freebsd", target_os = "android"))] fn any_group(_current_user_gid: Gid, owner_group: Gid) -> bool { users::current_user_groups()