use crate::config::{SandboxFilesystem, SandboxNetwork, SandboxWindows};
#[cfg(not(target_os = "macos"))]
use crate::paths::normalize_path;
use crate::paths::{normalize_path_lexically, normalize_roots};
use crate::traversal::subtract_denied_roots;
use anyhow::Result;
use rayon::prelude::*;
use std::env;
use std::fs;
use std::io;
#[cfg(target_os = "macos")]
use std::os::unix::fs::FileTypeExt;
use std::path::{Path, PathBuf};
#[derive(Debug, PartialEq, Eq, Hash)]
pub(crate) struct AccessPolicy {
pub(crate) write_roots: Vec<PathBuf>,
pub(crate) write_denied_roots: Vec<PathBuf>,
pub(crate) write_denied_links: Vec<PathBuf>,
pub(crate) read_access: ReadAccess,
pub(crate) read_denied_roots: Vec<PathBuf>,
pub(crate) network_access: NetworkAccess,
pub(crate) windows: WindowsPolicy,
}
#[cfg(target_os = "linux")]
impl AccessPolicy {
pub(crate) fn is_write_denied(&self, canonical: &Path, lexical: &Path) -> bool {
self.write_denied_roots
.iter()
.any(|root| canonical == root || canonical.starts_with(root))
|| self
.write_denied_links
.iter()
.any(|root| lexical == root || lexical.starts_with(root))
}
pub(crate) fn to_reason(
&self,
canonical: &Path,
lexical: &Path,
surface_allow_miss: bool,
) -> Option<&'static str> {
if self.is_write_denied(canonical, lexical) {
Some("deny_match")
} else if surface_allow_miss
&& !self
.write_roots
.iter()
.any(|root| canonical == root || canonical.starts_with(root))
{
Some("allow_miss")
} else {
None
}
}
}
#[derive(Debug, strum_macros::Display)]
#[strum(serialize_all = "SCREAMING_SNAKE_CASE")]
#[allow(dead_code)]
pub(crate) enum AccessPolicyError {
PartialRead,
UnrestrictedRead,
TcpPolicy,
UnixSocketPolicy,
UnixSocketPath,
DenyWriteSymlinkAncestor,
InvalidPort,
EmptyPath,
HomeUnavailable,
TraversalDepth,
}
impl std::error::Error for AccessPolicyError {}
#[cfg(target_os = "macos")]
impl AccessPolicy {
pub(crate) fn validate(&self) -> std::result::Result<(), AccessPolicyError> {
if let ReadAccess::AllowRoots(roots) = &self.read_access {
if !roots.iter().any(|root| root == Path::new("/")) {
return Err(AccessPolicyError::PartialRead);
}
}
if let UnixSocketAccess::AllowPaths(paths) = &self.network_access.unix_socket_access {
for path in paths {
match fs::symlink_metadata(path) {
Ok(metadata) if metadata.file_type().is_socket() => {}
_ => return Err(AccessPolicyError::UnixSocketPath),
}
}
}
let has_writable_symlink_ancestor = self.write_denied_links.iter().any(|link| {
self.write_roots
.iter()
.any(|root| link == root || link.starts_with(root))
});
if has_writable_symlink_ancestor {
return Err(AccessPolicyError::DenyWriteSymlinkAncestor);
}
Ok(())
}
}
#[cfg(target_os = "windows")]
impl AccessPolicy {
pub(crate) fn validate(&self) -> std::result::Result<(), AccessPolicyError> {
if matches!(self.read_access, ReadAccess::Unrestricted) {
return Err(AccessPolicyError::UnrestrictedRead);
}
let network = &self.network_access;
if network.is_unrestricted() {
return Ok(());
}
if network.local_tcp_bind || !network.connect_tcp_ports.is_empty() {
return Err(AccessPolicyError::TcpPolicy);
}
if !matches!(&network.unix_socket_access, UnixSocketAccess::AllowPaths(paths) if paths.is_empty())
{
return Err(AccessPolicyError::UnixSocketPolicy);
}
Ok(())
}
}
#[cfg(not(any(target_os = "macos", target_os = "windows")))]
impl AccessPolicy {
#[allow(clippy::unused_self, clippy::unnecessary_wraps)]
pub(crate) fn validate(&self) -> std::result::Result<(), AccessPolicyError> {
Ok(())
}
}
#[derive(Debug, PartialEq, Eq, Hash)]
pub(crate) enum ReadAccess {
Unrestricted,
AllowRoots(Vec<PathBuf>),
}
#[derive(Debug, PartialEq, Eq, Hash)]
pub(crate) struct NetworkAccess {
pub(crate) restrict_connect_tcp: bool,
pub(crate) connect_tcp_ports: Vec<u16>,
pub(crate) restrict_bind_tcp: bool,
pub(crate) local_tcp_bind: bool,
pub(crate) unix_socket_access: UnixSocketAccess,
}
impl NetworkAccess {
pub(crate) fn unrestricted() -> Self {
Self {
restrict_connect_tcp: false,
connect_tcp_ports: Vec::new(),
restrict_bind_tcp: false,
local_tcp_bind: false,
unix_socket_access: UnixSocketAccess::Unrestricted,
}
}
pub(crate) fn is_unrestricted(&self) -> bool {
!self.restrict_connect_tcp
&& self.connect_tcp_ports.is_empty()
&& !self.restrict_bind_tcp
&& !self.local_tcp_bind
&& matches!(self.unix_socket_access, UnixSocketAccess::Unrestricted)
}
}
#[derive(Debug, PartialEq, Eq, Hash)]
pub(crate) enum UnixSocketAccess {
Unrestricted,
AllowPaths(Vec<PathBuf>),
}
#[derive(Clone, Debug, Default, PartialEq, Eq, Hash)]
pub(crate) struct WindowsPolicy {
pub(crate) mitigation_policy: u64,
}
pub(crate) fn resolve_policy(
filesystem: &SandboxFilesystem,
network: &SandboxNetwork,
windows: &SandboxWindows,
policy_base: &Path,
) -> Result<AccessPolicy> {
let home_dir = dirs::home_dir();
let home = home_dir.as_deref();
let policy_base = if policy_base.is_absolute() {
policy_base.to_path_buf()
} else {
env::current_dir()?.join(policy_base)
};
let policy_base = normalize_path_lexically(&policy_base);
let write_allow = resolve_paths(&filesystem.allow_write, &policy_base, home)?;
let write_deny = resolve_paths(&filesystem.deny_write, &policy_base, home)?;
let write_denied_links = collect_symlink_ancestors(&filesystem.deny_write, &policy_base, home)?;
let read_allow = resolve_paths(&filesystem.allow_read, &policy_base, home)?;
let read_deny = resolve_paths(&filesystem.deny_read, &policy_base, home)?;
let read_access = if read_deny.is_empty() {
ReadAccess::Unrestricted
} else {
let mut read_roots = subtract_denied_roots(vec![PathBuf::from("/")], &read_deny)?;
for allow in &read_allow {
let nested: Vec<PathBuf> = read_deny
.iter()
.filter(|deny| deny.as_path() != allow.as_path() && deny.starts_with(allow))
.cloned()
.collect();
if nested.is_empty() {
read_roots.push(allow.clone());
} else {
read_roots.extend(subtract_denied_roots(vec![allow.clone()], &nested)?);
}
}
normalize_roots(&mut read_roots);
ReadAccess::AllowRoots(read_roots)
};
let policy = AccessPolicy {
write_roots: write_allow,
write_denied_roots: write_deny,
write_denied_links,
read_access,
read_denied_roots: read_deny,
network_access: lower_network_policy(network, &policy_base, home)?,
windows: lower_windows_policy(windows),
};
policy.validate()?;
Ok(policy)
}
fn lower_windows_policy(windows: &SandboxWindows) -> WindowsPolicy {
let mut mitigation_policy = 0;
if windows.disable_win32k {
mitigation_policy |= 0x1 << 28;
}
if windows.disable_extension_points {
mitigation_policy |= 0x1_u64 << 32;
}
if windows.strict_handle_checks {
mitigation_policy |= 0x1 << 24;
}
if windows.image_load_no_remote {
mitigation_policy |= 0x1_u64 << 52;
}
if windows.image_load_no_low_label {
mitigation_policy |= 0x1_u64 << 56;
}
if windows.image_load_prefer_system32 {
mitigation_policy |= 0x1_u64 << 60;
}
WindowsPolicy { mitigation_policy }
}
fn lower_network_policy(
network: &SandboxNetwork,
policy_base: &Path,
home: Option<&Path>,
) -> Result<NetworkAccess> {
if network.allow_network {
return Ok(NetworkAccess::unrestricted());
}
let mut connect_tcp_ports = Vec::new();
push_proxy_port(&mut connect_tcp_ports, network.http_proxy_port)?;
push_proxy_port(&mut connect_tcp_ports, network.socks_proxy_port)?;
connect_tcp_ports.sort_unstable();
connect_tcp_ports.dedup();
let unix_socket_paths = resolve_paths(&network.allow_unix_sockets, policy_base, home)?;
let unix_socket_access = if network.allow_all_unix_sockets {
UnixSocketAccess::Unrestricted
} else {
UnixSocketAccess::AllowPaths(unix_socket_paths)
};
Ok(NetworkAccess {
restrict_connect_tcp: true,
connect_tcp_ports,
restrict_bind_tcp: !network.allow_local_binding,
local_tcp_bind: network.allow_local_binding,
unix_socket_access,
})
}
fn push_proxy_port(ports: &mut Vec<u16>, port: Option<u16>) -> Result<()> {
let Some(port) = port else {
return Ok(());
};
if port == 0 {
return Err(AccessPolicyError::InvalidPort.into());
}
ports.push(port);
Ok(())
}
fn resolve_paths(
paths: &[String],
policy_base: &Path,
home: Option<&Path>,
) -> Result<Vec<PathBuf>> {
let mut resolved: Vec<PathBuf> = paths
.par_iter()
.map(|path| {
let path = resolve_sandbox_path(path, policy_base, home)?;
let candidates = if path.to_string_lossy().bytes().any(is_glob_byte) {
expand_glob_path(&path)?
} else {
vec![path]
};
let mut resolved = Vec::new();
for candidate in &candidates {
push_path_variants(&mut resolved, candidate);
}
Ok(resolved)
})
.collect::<Result<Vec<_>>>()?
.into_iter()
.flatten()
.collect();
normalize_roots(&mut resolved);
Ok(resolved)
}
fn collect_symlink_ancestors(
paths: &[String],
policy_base: &Path,
home: Option<&Path>,
) -> Result<Vec<PathBuf>> {
let mut links = Vec::new();
for path in paths {
let resolved = resolve_sandbox_path(path, policy_base, home)?;
let mut current = PathBuf::new();
for component in resolved.components() {
current.push(component);
match fs::symlink_metadata(¤t) {
Ok(metadata) if metadata.file_type().is_symlink() => {
links.push(normalize_path_lexically(¤t));
}
_ => {}
}
}
}
links.sort_unstable();
links.dedup();
Ok(links)
}
#[cfg(target_os = "macos")]
fn push_path_variants(paths: &mut Vec<PathBuf>, path: &Path) {
paths.push(normalize_path_lexically(path));
if let Ok(canonical) = fs::canonicalize(path) {
paths.push(normalize_path_lexically(&canonical));
}
}
#[cfg(not(target_os = "macos"))]
fn push_path_variants(paths: &mut Vec<PathBuf>, path: &Path) {
paths.push(normalize_path(path));
}
fn resolve_sandbox_path(path: &str, base: &Path, home: Option<&Path>) -> Result<PathBuf> {
if path.is_empty() {
return Err(AccessPolicyError::EmptyPath.into());
}
let raw = Path::new(path);
let resolved = if raw.has_root() {
raw.to_path_buf()
} else if path == "~" {
home.map(Path::to_path_buf)
.ok_or(AccessPolicyError::HomeUnavailable)?
} else if let Some(rest) = path.strip_prefix("~/") {
home.map(|home| home.join(rest))
.ok_or(AccessPolicyError::HomeUnavailable)?
} else {
base.join(raw)
};
Ok(normalize_path_lexically(&resolved))
}
fn expand_glob_path(pattern: &Path) -> Result<Vec<PathBuf>> {
let pattern = pattern.to_string_lossy();
let base = glob_base(&pattern);
let mut matches = Vec::new();
match fs::symlink_metadata(&base) {
Ok(_) => collect_glob_matches(&base, &pattern, &mut matches, 0)?,
Err(error)
if error.kind() == io::ErrorKind::NotFound
|| error.kind() == io::ErrorKind::PermissionDenied => {}
Err(source) => return Err(source.into()),
}
Ok(matches)
}
fn is_glob_byte(byte: u8) -> bool {
matches!(byte, b'*' | b'?' | b'[' | b']')
}
fn glob_base(pattern: &str) -> PathBuf {
let Some(glob_at) = pattern.bytes().position(is_glob_byte) else {
return PathBuf::from(pattern);
};
let prefix = &pattern[..glob_at];
let base = if prefix.ends_with('/') {
Path::new(prefix.trim_end_matches('/'))
} else {
Path::new(prefix).parent().unwrap_or(Path::new("/"))
};
if base.as_os_str().is_empty() {
PathBuf::from("/")
} else {
base.to_path_buf()
}
}
fn collect_glob_matches(
path: &Path,
pattern: &str,
matches: &mut Vec<PathBuf>,
depth: u32,
) -> Result<()> {
const LIMIT: u32 = 40;
if depth >= LIMIT {
return Ok(());
}
let candidate = normalize_path_lexically(path);
let candidate_text = candidate.to_string_lossy();
let pattern_bytes = pattern.as_bytes();
let candidate_bytes = candidate_text.as_bytes();
let mut memo = vec![vec![None; candidate_bytes.len() + 1]; pattern_bytes.len() + 1];
if glob_matches_at(pattern_bytes, candidate_bytes, 0, 0, &mut memo) {
matches.push(candidate.clone());
}
let metadata = match fs::symlink_metadata(path) {
Ok(metadata) => metadata,
Err(error)
if error.kind() == io::ErrorKind::NotFound
|| error.kind() == io::ErrorKind::PermissionDenied =>
{
return Ok(());
}
Err(source) => return Err(source.into()),
};
if !metadata.is_dir() || metadata.file_type().is_symlink() {
return Ok(());
}
let entries = match fs::read_dir(path) {
Ok(entries) => entries,
Err(error) if error.kind() == io::ErrorKind::PermissionDenied => return Ok(()),
Err(source) => return Err(source.into()),
};
for entry in entries {
collect_glob_matches(&entry?.path(), pattern, matches, depth + 1)?;
}
Ok(())
}
fn glob_matches_at(
pattern: &[u8],
text: &[u8],
pattern_at: usize,
text_at: usize,
memo: &mut [Vec<Option<bool>>],
) -> bool {
if let Some(result) = memo[pattern_at][text_at] {
return result;
}
let result = if pattern_at == pattern.len() {
text_at == text.len()
} else if pattern[pattern_at..].starts_with(b"**/") {
globstar_slash_matches(pattern, text, pattern_at, text_at, memo)
} else if pattern[pattern_at..].starts_with(b"**") {
globstar_matches(pattern, text, pattern_at, text_at, memo)
} else {
match pattern[pattern_at] {
b'*' => star_matches(pattern, text, pattern_at, text_at, memo),
b'?' => {
text_at < text.len()
&& text[text_at] != b'/'
&& glob_matches_at(pattern, text, pattern_at + 1, text_at + 1, memo)
}
b'[' => class_matches(pattern, text, pattern_at, text_at, memo),
byte => {
text_at < text.len()
&& text[text_at] == byte
&& glob_matches_at(pattern, text, pattern_at + 1, text_at + 1, memo)
}
}
};
memo[pattern_at][text_at] = Some(result);
result
}
fn globstar_slash_matches(
pattern: &[u8],
text: &[u8],
pattern_at: usize,
text_at: usize,
memo: &mut [Vec<Option<bool>>],
) -> bool {
if glob_matches_at(pattern, text, pattern_at + 3, text_at, memo) {
return true;
}
for next in text_at..text.len() {
if text[next] == b'/' && glob_matches_at(pattern, text, pattern_at + 3, next + 1, memo) {
return true;
}
}
false
}
fn globstar_matches(
pattern: &[u8],
text: &[u8],
pattern_at: usize,
text_at: usize,
memo: &mut [Vec<Option<bool>>],
) -> bool {
for next in text_at..=text.len() {
if glob_matches_at(pattern, text, pattern_at + 2, next, memo) {
return true;
}
}
false
}
fn star_matches(
pattern: &[u8],
text: &[u8],
pattern_at: usize,
text_at: usize,
memo: &mut [Vec<Option<bool>>],
) -> bool {
let mut next = text_at;
while next <= text.len() {
if glob_matches_at(pattern, text, pattern_at + 1, next, memo) {
return true;
}
if next == text.len() || text[next] == b'/' {
break;
}
next += 1;
}
false
}
fn class_matches(
pattern: &[u8],
text: &[u8],
pattern_at: usize,
text_at: usize,
memo: &mut [Vec<Option<bool>>],
) -> bool {
let Some(class_end) = pattern[pattern_at + 1..]
.iter()
.position(|byte| *byte == b']')
.map(|offset| pattern_at + 1 + offset)
else {
return text_at < text.len()
&& text[text_at] == b'['
&& glob_matches_at(pattern, text, pattern_at + 1, text_at + 1, memo);
};
text_at < text.len()
&& text[text_at] != b'/'
&& byte_in_class(text[text_at], &pattern[pattern_at + 1..class_end])
&& glob_matches_at(pattern, text, class_end + 1, text_at + 1, memo)
}
fn byte_in_class(byte: u8, class: &[u8]) -> bool {
let mut at = 0;
while at < class.len() {
if at + 2 < class.len() && class[at + 1] == b'-' {
if byte >= class[at] && byte <= class[at + 2] {
return true;
}
at += 3;
} else {
if byte == class[at] {
return true;
}
at += 1;
}
}
false
}