use crate::home::find_default_config_dir;
use directories::ProjectDirs;
use include_dir::{include_dir, Dir};
use lazy_static::lazy_static;
use std::{path::PathBuf, sync::OnceLock};
use uuid::Uuid;
pub const ZELLIJ_CONFIG_FILE_ENV: &str = "ZELLIJ_CONFIG_FILE";
pub const ZELLIJ_CONFIG_DIR_ENV: &str = "ZELLIJ_CONFIG_DIR";
pub const ZELLIJ_LAYOUT_DIR_ENV: &str = "ZELLIJ_LAYOUT_DIR";
pub const VERSION: &str = env!("CARGO_PKG_VERSION");
pub const DEFAULT_SCROLL_BUFFER_SIZE: usize = 10_000;
pub static SCROLL_BUFFER_SIZE: OnceLock<usize> = OnceLock::new();
pub static DEBUG_MODE: OnceLock<bool> = OnceLock::new();
#[cfg(not(windows))]
pub const SYSTEM_DEFAULT_CONFIG_DIR: &str = "/etc/zellij";
#[cfg(windows)]
pub const SYSTEM_DEFAULT_CONFIG_DIR: &str = "C:\\ProgramData\\Zellij";
pub const SYSTEM_DEFAULT_DATA_DIR_PREFIX: &str = system_default_data_dir();
pub static ZELLIJ_DEFAULT_THEMES: Dir = include_dir!("$CARGO_MANIFEST_DIR/assets/themes");
pub const CLIENT_SERVER_CONTRACT_VERSION: usize = 1;
pub fn session_info_cache_file_name(session_name: &str) -> PathBuf {
session_info_folder_for_session(session_name).join("session-metadata.kdl")
}
pub fn session_layout_cache_file_name(session_name: &str) -> PathBuf {
session_info_folder_for_session(session_name).join("session-layout.kdl")
}
pub fn session_info_folder_for_session(session_name: &str) -> PathBuf {
ZELLIJ_SESSION_INFO_CACHE_DIR.join(session_name)
}
pub fn create_config_and_cache_folders() {
if let Err(e) = std::fs::create_dir_all(&ZELLIJ_CACHE_DIR.as_path()) {
log::error!("Failed to create cache dir: {:?}", e);
}
if let Some(config_dir) = find_default_config_dir() {
if let Err(e) = std::fs::create_dir_all(&config_dir.as_path()) {
log::error!("Failed to create config dir: {:?}", e);
}
}
if let Err(e) = std::fs::create_dir_all(&ZELLIJ_SESSION_INFO_CACHE_DIR.as_path()) {
log::error!("Failed to create session_info cache dir: {:?}", e);
}
}
const fn system_default_data_dir() -> &'static str {
if let Some(data_dir) = std::option_env!("PREFIX") {
data_dir
} else if cfg!(windows) {
"C:\\ProgramData\\Zellij"
} else {
"/usr"
}
}
lazy_static! {
pub static ref CLIENT_SERVER_CONTRACT_DIR: String =
format!("contract_version_{}", CLIENT_SERVER_CONTRACT_VERSION);
pub static ref ZELLIJ_PROJ_DIR: ProjectDirs = {
if cfg!(windows) {
ProjectDirs::from("", "", "Zellij").unwrap()
} else {
ProjectDirs::from("org", "Zellij Contributors", "Zellij").unwrap()
}
};
pub static ref ZELLIJ_CACHE_DIR: PathBuf = ZELLIJ_PROJ_DIR.cache_dir().to_path_buf();
pub static ref ZELLIJ_SESSION_CACHE_DIR: PathBuf = ZELLIJ_PROJ_DIR
.cache_dir()
.to_path_buf()
.join(format!("{}", Uuid::new_v4()));
pub static ref ZELLIJ_PLUGIN_PERMISSIONS_CACHE: PathBuf =
ZELLIJ_CACHE_DIR.join("permissions.kdl");
pub static ref ZELLIJ_SESSION_INFO_CACHE_DIR: PathBuf = ZELLIJ_CACHE_DIR
.join(CLIENT_SERVER_CONTRACT_DIR.clone())
.join("session_info");
pub static ref ZELLIJ_STDIN_CACHE_FILE: PathBuf =
ZELLIJ_CACHE_DIR.join(VERSION).join("stdin_cache");
pub static ref ZELLIJ_PLUGIN_ARTIFACT_DIR: PathBuf = ZELLIJ_CACHE_DIR.join(VERSION);
pub static ref ZELLIJ_SEEN_RELEASE_NOTES_CACHE_FILE: PathBuf =
ZELLIJ_CACHE_DIR.join(VERSION).join("seen_release_notes");
}
pub const FEATURES: &[&str] = &[
#[cfg(feature = "disable_automatic_asset_installation")]
"disable_automatic_asset_installation",
];
#[cfg(not(target_family = "wasm"))]
pub use not_wasm::*;
#[cfg(not(target_family = "wasm"))]
mod not_wasm {
use lazy_static::lazy_static;
use std::collections::HashMap;
use std::path::PathBuf;
macro_rules! add_plugin {
($assets:expr, $plugin:literal) => {
$assets.insert(
PathBuf::from("plugins").join($plugin),
#[cfg(any(not(feature = "plugins_from_target"), not(debug_assertions)))]
include_bytes!(concat!(
env!("CARGO_MANIFEST_DIR"),
"/assets/plugins/",
$plugin
))
.to_vec(),
#[cfg(all(feature = "plugins_from_target", debug_assertions))]
include_bytes!(concat!(
env!("CARGO_MANIFEST_DIR"),
"/../target/wasm32-wasip1/debug/",
$plugin
))
.to_vec(),
);
};
}
lazy_static! {
pub static ref ASSET_MAP: HashMap<PathBuf, Vec<u8>> = {
let mut assets = std::collections::HashMap::new();
add_plugin!(assets, "compact-bar.wasm");
add_plugin!(assets, "status-bar.wasm");
add_plugin!(assets, "tab-bar.wasm");
add_plugin!(assets, "strider.wasm");
add_plugin!(assets, "session-manager.wasm");
add_plugin!(assets, "configuration.wasm");
add_plugin!(assets, "plugin-manager.wasm");
add_plugin!(assets, "about.wasm");
add_plugin!(assets, "share.wasm");
add_plugin!(assets, "multiple-select.wasm");
add_plugin!(assets, "layout-manager.wasm");
add_plugin!(assets, "link.wasm");
assets
};
}
}
#[cfg(unix)]
pub fn is_ipc_socket(file_type: &std::fs::FileType) -> bool {
use std::os::unix::fs::FileTypeExt;
file_type.is_socket()
}
#[cfg(not(unix))]
pub fn is_ipc_socket(file_type: &std::fs::FileType) -> bool {
file_type.is_file()
}
#[cfg(unix)]
pub fn ipc_connect(path: &std::path::Path) -> std::io::Result<interprocess::local_socket::Stream> {
use interprocess::local_socket::{prelude::*, GenericFilePath, Stream as LocalSocketStream};
let fs_name = path.to_fs_name::<GenericFilePath>()?;
LocalSocketStream::connect(fs_name)
}
#[cfg(windows)]
pub fn ipc_connect(path: &std::path::Path) -> std::io::Result<interprocess::local_socket::Stream> {
use interprocess::local_socket::{prelude::*, GenericNamespaced, Stream as LocalSocketStream};
let name = path.to_string_lossy().to_string();
let ns_name = name.to_ns_name::<GenericNamespaced>()?;
LocalSocketStream::connect(ns_name)
}
#[cfg(unix)]
pub fn ipc_bind(path: &std::path::Path) -> std::io::Result<interprocess::local_socket::Listener> {
use interprocess::local_socket::{prelude::*, GenericFilePath, ListenerOptions};
let fs_name = path.to_fs_name::<GenericFilePath>()?;
ListenerOptions::new().name(fs_name).create_sync()
}
#[cfg(windows)]
pub fn ipc_bind(path: &std::path::Path) -> std::io::Result<interprocess::local_socket::Listener> {
use interprocess::local_socket::{prelude::*, GenericNamespaced, ListenerOptions};
let name = path.to_string_lossy().to_string();
let ns_name = name.to_ns_name::<GenericNamespaced>()?;
let listener = ListenerOptions::new().name(ns_name).create_sync()?;
std::fs::write(path, std::process::id().to_string())?;
Ok(listener)
}
#[cfg(unix)]
pub fn ipc_bind_async(
path: &std::path::Path,
) -> std::io::Result<interprocess::local_socket::tokio::Listener> {
use interprocess::local_socket::{prelude::*, GenericFilePath, ListenerOptions};
let fs_name = path.to_fs_name::<GenericFilePath>()?;
ListenerOptions::new().name(fs_name).create_tokio()
}
#[cfg(windows)]
pub fn ipc_bind_async(
path: &std::path::Path,
) -> std::io::Result<interprocess::local_socket::tokio::Listener> {
use interprocess::local_socket::{prelude::*, GenericNamespaced, ListenerOptions};
let name = path.to_string_lossy().to_string();
let ns_name = name.to_ns_name::<GenericNamespaced>()?;
let listener = ListenerOptions::new().name(ns_name).create_tokio()?;
std::fs::write(path, std::process::id().to_string())?;
Ok(listener)
}
#[cfg(windows)]
pub fn ipc_connect_reply(
path: &std::path::Path,
) -> std::io::Result<interprocess::local_socket::Stream> {
use interprocess::local_socket::{prelude::*, GenericNamespaced, Stream as LocalSocketStream};
let name = format!("{}-reply", path.to_string_lossy());
let ns_name = name.to_ns_name::<GenericNamespaced>()?;
LocalSocketStream::connect(ns_name)
}
#[cfg(windows)]
pub fn ipc_bind_reply(
path: &std::path::Path,
) -> std::io::Result<interprocess::local_socket::Listener> {
use interprocess::local_socket::{prelude::*, GenericNamespaced, ListenerOptions};
let name = format!("{}-reply", path.to_string_lossy());
let ns_name = name.to_ns_name::<GenericNamespaced>()?;
ListenerOptions::new().name(ns_name).create_sync()
}
#[cfg(unix)]
pub use unix_only::*;
#[cfg(unix)]
mod unix_only {
use super::*;
use crate::envs;
pub use crate::shared::set_permissions;
use lazy_static::lazy_static;
use nix::unistd::Uid;
use std::env::temp_dir;
#[cfg(target_os = "macos")]
pub const ZELLIJ_SOCK_MAX_LENGTH: usize = 104;
#[cfg(not(target_os = "macos"))]
pub const ZELLIJ_SOCK_MAX_LENGTH: usize = 108;
lazy_static! {
static ref UID: Uid = Uid::current();
pub static ref ZELLIJ_TMP_DIR: PathBuf = temp_dir().join(format!("zellij-{}", *UID));
pub static ref ZELLIJ_TMP_LOG_DIR: PathBuf = ZELLIJ_TMP_DIR.join("zellij-log");
pub static ref ZELLIJ_TMP_LOG_FILE: PathBuf = ZELLIJ_TMP_LOG_DIR.join("zellij.log");
pub static ref ZELLIJ_SOCK_DIR: PathBuf = {
let mut ipc_dir = envs::get_socket_dir().map_or_else(
|_| {
ZELLIJ_PROJ_DIR
.runtime_dir()
.map_or_else(|| ZELLIJ_TMP_DIR.clone(), |p| p.to_owned())
},
PathBuf::from,
);
ipc_dir.push(CLIENT_SERVER_CONTRACT_DIR.clone());
ipc_dir
};
pub static ref WEBSERVER_SOCKET_PATH: PathBuf = ZELLIJ_SOCK_DIR.join("web_server_bus");
}
}
#[cfg(not(unix))]
pub use not_unix::*;
#[cfg(not(unix))]
mod not_unix {
use super::*;
use crate::envs;
pub use crate::shared::set_permissions;
#[cfg(windows)]
use dunce;
use lazy_static::lazy_static;
use std::env::temp_dir;
#[cfg(windows)]
fn canonicalize_path(path: PathBuf) -> PathBuf {
dunce::canonicalize(&path).unwrap_or(path)
}
#[cfg(not(windows))]
fn canonicalize_path(path: PathBuf) -> PathBuf {
path
}
pub const ZELLIJ_SOCK_MAX_LENGTH: usize = 256;
lazy_static! {
pub static ref ZELLIJ_TMP_DIR: PathBuf = {
let tmp_dir = canonicalize_path(temp_dir());
tmp_dir.join("zellij")
};
pub static ref ZELLIJ_TMP_LOG_DIR: PathBuf = ZELLIJ_TMP_DIR.join("zellij-log");
pub static ref ZELLIJ_TMP_LOG_FILE: PathBuf = ZELLIJ_TMP_LOG_DIR.join("zellij.log");
pub static ref ZELLIJ_SOCK_DIR: PathBuf = {
let mut ipc_dir = canonicalize_path(envs::get_socket_dir().map_or_else(
|_| {
ZELLIJ_PROJ_DIR
.runtime_dir()
.map_or_else(|| ZELLIJ_TMP_DIR.clone(), |p| p.to_owned())
},
PathBuf::from,
));
ipc_dir.push(CLIENT_SERVER_CONTRACT_DIR.clone());
ipc_dir
};
pub static ref WEBSERVER_SOCKET_PATH: PathBuf = ZELLIJ_SOCK_DIR.join("web_server_bus");
}
}