use std::path::{Path, PathBuf};
pub struct ZLayerDirs {
data_dir: PathBuf,
}
impl ZLayerDirs {
pub fn new(data_dir: impl Into<PathBuf>) -> Self {
Self {
data_dir: data_dir.into(),
}
}
pub fn system_default() -> Self {
Self::new(Self::default_data_dir())
}
pub fn default_data_dir() -> PathBuf {
#[cfg(target_os = "macos")]
{
home_dir_or_tmp().join(".zlayer")
}
#[cfg(target_os = "windows")]
{
windows_program_data_root()
}
#[cfg(not(any(target_os = "macos", target_os = "windows")))]
{
if is_root() {
PathBuf::from("/var/lib/zlayer")
} else {
home_dir_or_tmp().join(".zlayer")
}
}
}
pub fn detect_data_dir() -> PathBuf {
#[cfg(not(any(target_os = "macos", target_os = "windows")))]
{
if !is_root() {
let system_data = PathBuf::from("/var/lib/zlayer");
if system_data.join("daemon.json").exists() {
return system_data;
}
}
}
#[cfg(target_os = "windows")]
{
let system_data = windows_program_data_root();
if system_data.join("daemon.json").exists() {
return system_data;
}
}
Self::default_data_dir()
}
pub fn default_run_dir() -> PathBuf {
#[cfg(not(any(target_os = "macos", target_os = "windows")))]
{
PathBuf::from("/var/run/zlayer")
}
#[cfg(any(target_os = "macos", target_os = "windows"))]
{
Self::default_data_dir().join("run")
}
}
pub fn default_log_dir() -> PathBuf {
#[cfg(not(any(target_os = "macos", target_os = "windows")))]
{
PathBuf::from("/var/log/zlayer")
}
#[cfg(any(target_os = "macos", target_os = "windows"))]
{
Self::default_data_dir().join("logs")
}
}
pub fn default_socket_path() -> String {
#[cfg(target_os = "windows")]
{
"tcp://127.0.0.1:3669".to_string()
}
#[cfg(not(target_os = "windows"))]
{
#[cfg(target_os = "macos")]
{
Self::default_data_dir()
.join("run")
.join("zlayer.sock")
.to_string_lossy()
.into_owned()
}
#[cfg(not(target_os = "macos"))]
{
"/var/run/zlayer.sock".to_string()
}
}
}
pub fn default_docker_socket_path() -> String {
#[cfg(target_os = "windows")]
{
r"\\.\pipe\zlayer-docker".to_string()
}
#[cfg(not(target_os = "windows"))]
{
#[cfg(target_os = "macos")]
{
Self::default_data_dir()
.join("run")
.join("docker.sock")
.to_string_lossy()
.into_owned()
}
#[cfg(not(target_os = "macos"))]
{
if is_root() {
"/var/run/zlayer/docker.sock".to_string()
} else if let Some(xdg) = std::env::var_os("XDG_RUNTIME_DIR") {
let mut p = PathBuf::from(xdg);
p.push("zlayer");
p.push("docker.sock");
p.to_string_lossy().into_owned()
} else {
Self::default_data_dir()
.join("run")
.join("docker.sock")
.to_string_lossy()
.into_owned()
}
}
}
}
pub fn default_binary_dir() -> PathBuf {
#[cfg(unix)]
{
let probe = PathBuf::from("/usr/local/bin/.zlayer_write_probe");
if std::fs::write(&probe, b"").is_ok() {
let _ = std::fs::remove_file(&probe);
return PathBuf::from("/usr/local/bin");
}
}
let dirs = Self::system_default();
let bin_dir = dirs.bin();
let _ = std::fs::create_dir_all(&bin_dir);
bin_dir
}
pub fn data_dir(&self) -> &Path {
&self.data_dir
}
pub fn containers(&self) -> PathBuf {
self.data_dir.join("containers")
}
pub fn rootfs(&self) -> PathBuf {
self.data_dir.join("rootfs")
}
pub fn bundles(&self) -> PathBuf {
self.data_dir.join("bundles")
}
pub fn cache(&self) -> PathBuf {
self.data_dir.join("cache")
}
pub fn volumes(&self) -> PathBuf {
self.data_dir.join("volumes")
}
pub fn wasm(&self) -> PathBuf {
self.data_dir.join("wasm")
}
pub fn wasm_compiled(&self) -> PathBuf {
self.data_dir.join("wasm").join("compiled")
}
pub fn secrets(&self) -> PathBuf {
self.data_dir.join("secrets")
}
pub fn certs(&self) -> PathBuf {
self.data_dir.join("certs")
}
pub fn raft(&self) -> PathBuf {
self.data_dir.join("raft")
}
pub fn admin_password(&self) -> PathBuf {
self.data_dir.join("admin_password")
}
#[must_use]
pub fn admin_bearer_path(&self) -> PathBuf {
self.data_dir.join("admin_bearer.token")
}
pub fn daemon_json(&self) -> PathBuf {
self.data_dir.join("daemon.json")
}
pub fn agent_ipam_state(&self) -> PathBuf {
self.data_dir.join("agent_ipam.json")
}
pub fn logs(&self) -> PathBuf {
self.data_dir.join("logs")
}
pub fn vms(&self) -> PathBuf {
self.data_dir.join("vms")
}
pub fn images(&self) -> PathBuf {
self.data_dir.join("images")
}
pub fn bin(&self) -> PathBuf {
self.data_dir.join("bin")
}
pub fn toolchain_cache(&self) -> PathBuf {
self.data_dir.join("toolchain-cache")
}
pub fn tmp(&self) -> PathBuf {
self.data_dir.join("tmp")
}
}
#[must_use]
pub fn default_admin_bearer_path() -> PathBuf {
ZLayerDirs::system_default().admin_bearer_path()
}
#[cfg(not(target_os = "windows"))]
fn home_dir_or_tmp() -> PathBuf {
std::env::var_os("HOME")
.map(PathBuf::from)
.unwrap_or_else(|| PathBuf::from("/tmp"))
}
#[cfg(target_os = "windows")]
fn windows_program_data_root() -> PathBuf {
if let Some(program_data) = std::env::var_os("PROGRAMDATA") {
let mut p = PathBuf::from(program_data);
p.push("ZLayer");
p
} else {
PathBuf::from(r"C:\ProgramData\ZLayer")
}
}
#[cfg(unix)]
#[must_use]
pub fn is_root() -> bool {
unsafe { libc::geteuid() == 0 }
}
#[cfg(windows)]
#[must_use]
pub fn is_root() -> bool {
use windows::Win32::UI::Shell::IsUserAnAdmin;
unsafe { IsUserAnAdmin().as_bool() }
}
#[cfg(not(any(unix, windows)))]
#[must_use]
pub fn is_root() -> bool {
false
}
#[cfg(test)]
mod tests {
use super::*;
#[cfg(target_os = "windows")]
static ENV_LOCK: std::sync::Mutex<()> = std::sync::Mutex::new(());
#[test]
fn subdirectories_are_relative_to_data_dir() {
let dirs = ZLayerDirs::new("/test/data");
assert_eq!(dirs.containers(), PathBuf::from("/test/data/containers"));
assert_eq!(dirs.rootfs(), PathBuf::from("/test/data/rootfs"));
assert_eq!(dirs.bundles(), PathBuf::from("/test/data/bundles"));
assert_eq!(dirs.cache(), PathBuf::from("/test/data/cache"));
assert_eq!(dirs.volumes(), PathBuf::from("/test/data/volumes"));
assert_eq!(dirs.wasm(), PathBuf::from("/test/data/wasm"));
assert_eq!(
dirs.wasm_compiled(),
PathBuf::from("/test/data/wasm/compiled")
);
assert_eq!(dirs.secrets(), PathBuf::from("/test/data/secrets"));
assert_eq!(dirs.certs(), PathBuf::from("/test/data/certs"));
assert_eq!(dirs.raft(), PathBuf::from("/test/data/raft"));
assert_eq!(
dirs.admin_password(),
PathBuf::from("/test/data/admin_password")
);
assert_eq!(dirs.daemon_json(), PathBuf::from("/test/data/daemon.json"));
assert_eq!(dirs.logs(), PathBuf::from("/test/data/logs"));
assert_eq!(dirs.vms(), PathBuf::from("/test/data/vms"));
assert_eq!(dirs.images(), PathBuf::from("/test/data/images"));
assert_eq!(dirs.bin(), PathBuf::from("/test/data/bin"));
assert_eq!(
dirs.toolchain_cache(),
PathBuf::from("/test/data/toolchain-cache")
);
assert_eq!(dirs.tmp(), PathBuf::from("/test/data/tmp"));
}
#[test]
fn system_default_uses_default_data_dir() {
#[cfg(target_os = "windows")]
let _env_guard = ENV_LOCK.lock().unwrap();
let dirs = ZLayerDirs::system_default();
assert_eq!(dirs.data_dir(), ZLayerDirs::default_data_dir().as_path());
}
#[test]
fn admin_bearer_path_is_under_data_dir() {
let dirs = ZLayerDirs::new(PathBuf::from("/tmp/zlayer-test"));
assert_eq!(
dirs.admin_bearer_path(),
PathBuf::from("/tmp/zlayer-test/admin_bearer.token")
);
}
#[test]
fn default_admin_bearer_path_matches_system_default() {
#[cfg(target_os = "windows")]
let _env_guard = ENV_LOCK.lock().unwrap();
assert_eq!(
default_admin_bearer_path(),
ZLayerDirs::system_default().admin_bearer_path()
);
}
#[cfg(target_os = "windows")]
#[test]
fn windows_default_data_dir_uses_program_data() {
let _env_guard = ENV_LOCK.lock().unwrap();
let prev = std::env::var_os("PROGRAMDATA");
std::env::set_var("PROGRAMDATA", r"C:\TestProgramData");
let data = ZLayerDirs::default_data_dir();
assert_eq!(data, PathBuf::from(r"C:\TestProgramData\ZLayer"));
let dirs = ZLayerDirs::system_default();
assert_eq!(dirs.certs(), data.join("certs"));
assert_eq!(dirs.secrets(), data.join("secrets"));
assert_eq!(dirs.logs(), data.join("logs"));
assert_eq!(ZLayerDirs::default_run_dir(), data.join("run"));
assert_eq!(ZLayerDirs::default_log_dir(), data.join("logs"));
assert_eq!(ZLayerDirs::default_socket_path(), "tcp://127.0.0.1:3669");
match prev {
Some(v) => std::env::set_var("PROGRAMDATA", v),
None => std::env::remove_var("PROGRAMDATA"),
}
}
#[cfg(target_os = "windows")]
#[test]
fn windows_default_data_dir_fallback_when_env_missing() {
let _env_guard = ENV_LOCK.lock().unwrap();
let prev = std::env::var_os("PROGRAMDATA");
std::env::remove_var("PROGRAMDATA");
let data = ZLayerDirs::default_data_dir();
assert_eq!(data, PathBuf::from(r"C:\ProgramData\ZLayer"));
if let Some(v) = prev {
std::env::set_var("PROGRAMDATA", v);
}
}
#[test]
fn default_docker_socket_path_not_empty() {
let result = ZLayerDirs::default_docker_socket_path();
assert!(!result.is_empty());
}
#[cfg(target_os = "windows")]
#[test]
fn default_docker_socket_path_platform_shape() {
let result = ZLayerDirs::default_docker_socket_path();
assert!(result.starts_with(r"\\.\pipe"));
}
#[cfg(target_os = "macos")]
#[test]
fn default_docker_socket_path_platform_shape() {
let result = ZLayerDirs::default_docker_socket_path();
assert!(result.ends_with("/docker.sock"));
}
#[cfg(all(not(target_os = "windows"), not(target_os = "macos")))]
#[test]
fn default_docker_socket_path_platform_shape() {
let result = ZLayerDirs::default_docker_socket_path();
assert!(result.ends_with("/docker.sock"));
}
}