1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-12 00:54:02 -05:00

fix(runtime/permissions): Resolve executable specifiers in allowlists and queries (#14130)

Closes #14122.

Adds two extensions to `--allow-run` behaviour:
- When `--allow-run=foo` is specified and `foo` is found in the `PATH`
at startup, `RunDescriptor::Path(which("foo"))` is added to the
allowlist alongside `RunDescriptor::Name("foo")`. Currently only the
latter is.
- When run permission for `foo` is queried and `foo` is found in the
`PATH` at runtime, either `RunDescriptor::Path(which("foo"))` or
`RunDescriptor::Name("foo")` would qualify in the allowlist. Currently
only the latter does.
This commit is contained in:
Nayeem Rahman 2023-08-30 18:52:01 +01:00 committed by GitHub
parent d28384c3de
commit 1cce306022
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 186 additions and 48 deletions

1
Cargo.lock generated
View file

@ -1515,6 +1515,7 @@ dependencies = [
"tokio", "tokio",
"tokio-metrics", "tokio-metrics",
"uuid", "uuid",
"which",
"winapi", "winapi",
"winres", "winres",
] ]

View file

@ -3617,6 +3617,11 @@ itest!(followup_dyn_import_resolved {
output: "run/followup_dyn_import_resolves/main.ts.out", output: "run/followup_dyn_import_resolves/main.ts.out",
}); });
itest!(allow_run_allowlist_resolution {
args: "run --quiet --unstable -A allow_run_allowlist_resolution.ts",
output: "allow_run_allowlist_resolution.ts.out",
});
itest!(unhandled_rejection { itest!(unhandled_rejection {
args: "run --check run/unhandled_rejection.ts", args: "run --check run/unhandled_rejection.ts",
output: "run/unhandled_rejection.ts.out", output: "run/unhandled_rejection.ts.out",

View file

@ -0,0 +1,66 @@
// Testing the following (but with `deno` instead of `echo`):
// | `deno run --allow-run=echo` | `which path == "/usr/bin/echo"` at startup | `which path != "/usr/bin/echo"` at startup |
// |-------------------------------------|--------------------------------------------|--------------------------------------------|
// | **`Deno.Command("echo")`** | ✅ | ✅ |
// | **`Deno.Command("/usr/bin/echo")`** | ✅ | ❌ |
// | `deno run --allow-run=/usr/bin/echo | `which path == "/usr/bin/echo"` at runtime | `which path != "/usr/bin/echo"` at runtime |
// |-------------------------------------|--------------------------------------------|--------------------------------------------|
// | **`Deno.Command("echo")`** | ✅ | ❌ |
// | **`Deno.Command("/usr/bin/echo")`** | ✅ | ✅ |
const execPath = Deno.execPath();
const execPathParent = execPath.replace(/[/\\][^/\\]+$/, "");
const testUrl = `data:application/typescript;base64,${
btoa(`
console.log(await Deno.permissions.query({ name: "run", command: "deno" }));
console.log(await Deno.permissions.query({ name: "run", command: "${
execPath.replaceAll("\\", "\\\\")
}" }));
Deno.env.set("PATH", "");
console.log(await Deno.permissions.query({ name: "run", command: "deno" }));
console.log(await Deno.permissions.query({ name: "run", command: "${
execPath.replaceAll("\\", "\\\\")
}" }));
`)
}`;
const process1 = await new Deno.Command(Deno.execPath(), {
args: [
"run",
"--quiet",
"--allow-env",
"--allow-run=deno",
testUrl,
],
stderr: "null",
env: { "PATH": execPathParent },
}).output();
console.log(new TextDecoder().decode(process1.stdout));
const process2 = await new Deno.Command(Deno.execPath(), {
args: [
"run",
"--quiet",
"--allow-env",
"--allow-run=deno",
testUrl,
],
stderr: "null",
env: { "PATH": "" },
}).output();
console.log(new TextDecoder().decode(process2.stdout));
const process3 = await new Deno.Command(Deno.execPath(), {
args: [
"run",
"--quiet",
"--allow-env",
`--allow-run=${execPath}`,
testUrl,
],
stderr: "null",
env: { "PATH": execPathParent },
}).output();
console.log(new TextDecoder().decode(process3.stdout));

View file

@ -0,0 +1,15 @@
PermissionStatus { state: "granted", onchange: null }
PermissionStatus { state: "granted", onchange: null }
PermissionStatus { state: "granted", onchange: null }
PermissionStatus { state: "granted", onchange: null }
PermissionStatus { state: "granted", onchange: null }
PermissionStatus { state: "prompt", onchange: null }
PermissionStatus { state: "granted", onchange: null }
PermissionStatus { state: "prompt", onchange: null }
PermissionStatus { state: "granted", onchange: null }
PermissionStatus { state: "granted", onchange: null }
PermissionStatus { state: "prompt", onchange: null }
PermissionStatus { state: "granted", onchange: null }

View file

@ -109,6 +109,7 @@ termcolor = "1.1.3"
tokio.workspace = true tokio.workspace = true
tokio-metrics.workspace = true tokio-metrics.workspace = true
uuid.workspace = true uuid.workspace = true
which = "4.2.5"
[target.'cfg(windows)'.dependencies] [target.'cfg(windows)'.dependencies]
fwdansi.workspace = true fwdansi.workspace = true

View file

@ -27,6 +27,7 @@ use std::path::PathBuf;
use std::str::FromStr; use std::str::FromStr;
use std::string::ToString; use std::string::ToString;
use std::sync::Arc; use std::sync::Arc;
use which::which;
mod prompter; mod prompter;
use prompter::permission_prompt; use prompter::permission_prompt;
@ -261,6 +262,9 @@ pub trait Descriptor: Eq + Clone {
fn stronger_than(&self, other: &Self) -> bool { fn stronger_than(&self, other: &Self) -> bool {
self == other self == other
} }
fn aliases(&self) -> Vec<Self> {
vec![]
}
} }
#[derive(Clone, Debug, Eq, PartialEq)] #[derive(Clone, Debug, Eq, PartialEq)]
@ -326,33 +330,43 @@ impl<T: Descriptor + Hash> UnaryPermission<T> {
desc: &Option<T>, desc: &Option<T>,
allow_partial: AllowPartial, allow_partial: AllowPartial,
) -> PermissionState { ) -> PermissionState {
if self.is_flag_denied(desc) || self.is_prompt_denied(desc) { let aliases = desc.as_ref().map_or(vec![], T::aliases);
PermissionState::Denied for desc in [desc]
} else if self.is_granted(desc) { .into_iter()
match allow_partial { .chain(&aliases.into_iter().map(Some).collect::<Vec<_>>())
AllowPartial::TreatAsGranted => PermissionState::Granted,
AllowPartial::TreatAsDenied => {
if self.is_partial_flag_denied(desc) {
PermissionState::Denied
} else {
PermissionState::Granted
}
}
AllowPartial::TreatAsPartialGranted => {
if self.is_partial_flag_denied(desc) {
PermissionState::GrantedPartial
} else {
PermissionState::Granted
}
}
}
} else if matches!(allow_partial, AllowPartial::TreatAsDenied)
&& self.is_partial_flag_denied(desc)
{ {
PermissionState::Denied let state = if self.is_flag_denied(desc) || self.is_prompt_denied(desc) {
} else { PermissionState::Denied
PermissionState::Prompt } else if self.is_granted(desc) {
match allow_partial {
AllowPartial::TreatAsGranted => PermissionState::Granted,
AllowPartial::TreatAsDenied => {
if self.is_partial_flag_denied(desc) {
PermissionState::Denied
} else {
PermissionState::Granted
}
}
AllowPartial::TreatAsPartialGranted => {
if self.is_partial_flag_denied(desc) {
PermissionState::GrantedPartial
} else {
PermissionState::Granted
}
}
}
} else if matches!(allow_partial, AllowPartial::TreatAsDenied)
&& self.is_partial_flag_denied(desc)
{
PermissionState::Denied
} else {
PermissionState::Prompt
};
if state != PermissionState::Prompt {
return state;
}
} }
PermissionState::Prompt
} }
fn request_desc( fn request_desc(
@ -402,7 +416,12 @@ impl<T: Descriptor + Hash> UnaryPermission<T> {
fn revoke_desc(&mut self, desc: &Option<T>) -> PermissionState { fn revoke_desc(&mut self, desc: &Option<T>) -> PermissionState {
match desc.as_ref() { match desc.as_ref() {
Some(desc) => self.granted_list.retain(|v| !v.stronger_than(desc)), Some(desc) => {
self.granted_list.retain(|v| !v.stronger_than(desc));
for alias in desc.aliases() {
self.granted_list.retain(|v| !v.stronger_than(&alias));
}
}
None => { None => {
self.granted_global = false; self.granted_global = false;
// Revoke global is a special case where the entire granted list is // Revoke global is a special case where the entire granted list is
@ -469,7 +488,11 @@ impl<T: Descriptor + Hash> UnaryPermission<T> {
) { ) {
match desc { match desc {
Some(desc) => { Some(desc) => {
let aliases = desc.aliases();
list.insert(desc); list.insert(desc);
for alias in aliases {
list.insert(alias);
}
} }
None => *list_global = true, None => *list_global = true,
} }
@ -580,7 +603,11 @@ impl AsRef<str> for EnvDescriptor {
#[derive(Clone, Eq, PartialEq, Hash, Debug)] #[derive(Clone, Eq, PartialEq, Hash, Debug)]
pub enum RunDescriptor { pub enum RunDescriptor {
/// Warning: You may want to construct with `RunDescriptor::from()` for case
/// handling.
Name(String), Name(String),
/// Warning: You may want to construct with `RunDescriptor::from()` for case
/// handling.
Path(PathBuf), Path(PathBuf),
} }
@ -592,19 +619,41 @@ impl Descriptor for RunDescriptor {
fn name(&self) -> Cow<str> { fn name(&self) -> Cow<str> {
Cow::from(self.to_string()) Cow::from(self.to_string())
} }
fn aliases(&self) -> Vec<Self> {
match self {
RunDescriptor::Name(name) => match which(name) {
Ok(path) => vec![RunDescriptor::Path(path)],
Err(_) => vec![],
},
RunDescriptor::Path(_) => vec![],
}
}
} }
impl FromStr for RunDescriptor { impl From<String> for RunDescriptor {
type Err = (); fn from(s: String) -> Self {
#[cfg(windows)]
fn from_str(s: &str) -> Result<Self, Self::Err> { let s = s.to_lowercase();
let is_path = s.contains('/'); let is_path = s.contains('/');
#[cfg(windows)] #[cfg(windows)]
let is_path = is_path || s.contains('\\') || Path::new(s).is_absolute(); let is_path = is_path || s.contains('\\') || Path::new(&s).is_absolute();
if is_path { if is_path {
Ok(Self::Path(resolve_from_cwd(Path::new(s)).unwrap())) Self::Path(resolve_from_cwd(Path::new(&s)).unwrap())
} else { } else {
Ok(Self::Name(s.to_string())) Self::Name(s)
}
}
}
impl From<PathBuf> for RunDescriptor {
fn from(p: PathBuf) -> Self {
#[cfg(windows)]
let p = PathBuf::from(p.to_string_lossy().to_string().to_lowercase());
if p.is_absolute() {
Self::Path(p)
} else {
Self::Path(resolve_from_cwd(&p).unwrap())
} }
} }
} }
@ -905,19 +954,19 @@ impl UnaryPermission<SysDescriptor> {
impl UnaryPermission<RunDescriptor> { impl UnaryPermission<RunDescriptor> {
pub fn query(&self, cmd: Option<&str>) -> PermissionState { pub fn query(&self, cmd: Option<&str>) -> PermissionState {
self.query_desc( self.query_desc(
&cmd.map(|c| RunDescriptor::from_str(c).unwrap()), &cmd.map(|c| RunDescriptor::from(c.to_string())),
AllowPartial::TreatAsPartialGranted, AllowPartial::TreatAsPartialGranted,
) )
} }
pub fn request(&mut self, cmd: Option<&str>) -> PermissionState { pub fn request(&mut self, cmd: Option<&str>) -> PermissionState {
self.request_desc(&cmd.map(|c| RunDescriptor::from_str(c).unwrap()), || { self.request_desc(&cmd.map(|c| RunDescriptor::from(c.to_string())), || {
Some(cmd?.to_string()) Some(cmd?.to_string())
}) })
} }
pub fn revoke(&mut self, cmd: Option<&str>) -> PermissionState { pub fn revoke(&mut self, cmd: Option<&str>) -> PermissionState {
self.revoke_desc(&cmd.map(|c| RunDescriptor::from_str(c).unwrap())) self.revoke_desc(&cmd.map(|c| RunDescriptor::from(c.to_string())))
} }
pub fn check( pub fn check(
@ -926,7 +975,7 @@ impl UnaryPermission<RunDescriptor> {
api_name: Option<&str>, api_name: Option<&str>,
) -> Result<(), AnyError> { ) -> Result<(), AnyError> {
self.check_desc( self.check_desc(
&Some(RunDescriptor::from_str(cmd).unwrap()), &Some(RunDescriptor::from(cmd.to_string())),
false, false,
api_name, api_name,
|| Some(format!("\"{}\"", cmd)), || Some(format!("\"{}\"", cmd)),
@ -1594,19 +1643,20 @@ fn parse_sys_list(
fn parse_run_list( fn parse_run_list(
list: &Option<Vec<String>>, list: &Option<Vec<String>>,
) -> Result<HashSet<RunDescriptor>, AnyError> { ) -> Result<HashSet<RunDescriptor>, AnyError> {
let mut result = HashSet::new();
if let Some(v) = list { if let Some(v) = list {
v.iter() for s in v {
.map(|x| { if s.is_empty() {
if x.is_empty() { return Err(AnyError::msg("Empty path is not allowed"));
Err(AnyError::msg("Empty path is not allowed")) } else {
} else { let desc = RunDescriptor::from(s.to_string());
Ok(RunDescriptor::from_str(x).unwrap()) let aliases = desc.aliases();
} result.insert(desc);
}) result.extend(aliases);
.collect() }
} else { }
Ok(HashSet::new())
} }
Ok(result)
} }
fn escalation_error() -> AnyError { fn escalation_error() -> AnyError {