use std::sync::Arc;
use std::time::{Duration, Instant};
use crate::element::{Element, ElementData, TreeNode};
use crate::error::{Error, Result};
use crate::event_provider::Subscription;
use crate::locator::Locator;
use crate::provider::Provider;
const LOOKUP_POLL_INTERVAL: Duration = Duration::from_millis(100);
fn poll_lookup<F>(timeout: Duration, mut attempt: F) -> Result<App>
where
F: FnMut() -> Result<App>,
{
let start = Instant::now();
loop {
match attempt() {
Ok(app) => return Ok(app),
Err(e @ Error::SelectorNotMatched { .. }) => {
if start.elapsed() >= timeout {
return Err(e);
}
}
Err(e) => return Err(e),
}
std::thread::sleep(LOOKUP_POLL_INTERVAL);
}
}
pub struct App {
pub name: String,
pub pid: Option<u32>,
pub data: ElementData,
provider: Arc<dyn Provider>,
}
impl App {
pub fn by_name_with(
provider: Arc<dyn Provider>,
name: &str,
timeout: Duration,
) -> Result<Self> {
poll_lookup(timeout, || {
let apps = provider.list_apps()?;
let data = apps
.into_iter()
.find(|d| d.name.as_deref() == Some(name))
.ok_or_else(|| Error::SelectorNotMatched {
selector: format!(r#"application[name="{}"]"#, name),
})?;
Ok(Self::from_data(Arc::clone(&provider), data))
})
}
pub fn by_pid_with(provider: Arc<dyn Provider>, pid: u32, timeout: Duration) -> Result<Self> {
poll_lookup(timeout, || {
let apps = provider.list_apps()?;
let data = apps
.into_iter()
.find(|d| d.pid == Some(pid))
.ok_or_else(|| Error::SelectorNotMatched {
selector: format!("application with pid={}", pid),
})?;
Ok(Self::from_data(Arc::clone(&provider), data))
})
}
pub fn list_with(provider: Arc<dyn Provider>) -> Result<Vec<Self>> {
let datas = provider.list_apps()?;
Ok(datas
.into_iter()
.map(|d| Self::from_data(Arc::clone(&provider), d))
.collect())
}
fn from_data(provider: Arc<dyn Provider>, data: ElementData) -> Self {
let name = data.name.clone().unwrap_or_default();
let pid = data.pid;
Self {
name,
pid,
data,
provider,
}
}
pub fn locator(&self, selector: &str) -> Locator {
Locator::new(
Arc::clone(&self.provider),
Some(self.data.clone()),
selector,
)
}
pub fn subscribe(&self) -> Result<Subscription> {
self.provider.subscribe(&self.data)
}
pub fn children(&self) -> Result<Vec<Element>> {
let children = self.provider.get_children(Some(&self.data))?;
Ok(children
.into_iter()
.map(|d| Element::new(d, Arc::clone(&self.provider)))
.collect())
}
pub fn tree(&self, max_depth: Option<usize>) -> Result<TreeNode> {
self.as_element().tree(max_depth)
}
pub fn dump(&self, max_depth: Option<usize>) -> Result<String> {
self.as_element().dump(max_depth)
}
pub fn as_element(&self) -> Element {
Element::new(self.data.clone(), Arc::clone(&self.provider))
}
pub fn provider(&self) -> &Arc<dyn Provider> {
&self.provider
}
}
impl std::fmt::Display for App {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "application \"{}\"", self.name)
}
}
impl std::fmt::Debug for App {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("App")
.field("name", &self.name)
.field("pid", &self.pid)
.finish()
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::mock::build_provider;
use crate::role::Role;
fn mock_app() -> App {
let provider: Arc<dyn Provider> = build_provider();
App::by_name_with(provider, "TestApp", Duration::ZERO)
.expect("TestApp must exist in mock tree")
}
#[test]
fn app_tree_returns_application_root() {
let node = mock_app().tree(None).expect("tree must succeed");
assert_eq!(node.role, "application");
assert_eq!(node.name.as_deref(), Some("TestApp"));
assert!(
!node.children.is_empty(),
"TestApp must have at least one window child"
);
}
#[test]
fn app_tree_max_depth_zero_has_no_children() {
let node = mock_app().tree(Some(0)).expect("tree must succeed");
assert_eq!(node.role, "application");
assert!(node.children.is_empty());
}
#[test]
fn app_tree_max_depth_one_stops_at_direct_children() {
let node = mock_app().tree(Some(1)).expect("tree must succeed");
assert!(!node.children.is_empty());
for child in &node.children {
assert!(
child.children.is_empty(),
"max_depth=1 must stop after direct children"
);
}
}
#[test]
fn app_dump_contains_application_root() {
let s = mock_app().dump(None).expect("dump must succeed");
assert!(
s.contains(r#"application "TestApp""#),
"dump output should include the application root: {s}"
);
}
#[test]
fn app_dump_max_depth_zero_is_one_line() {
let s = mock_app().dump(Some(0)).expect("dump must succeed");
let non_empty: Vec<&str> = s.lines().filter(|l| !l.trim().is_empty()).collect();
assert_eq!(non_empty.len(), 1, "max_depth=0 should be a single line");
assert!(non_empty[0].contains("application"));
}
#[test]
fn app_as_element_is_root() {
let app = mock_app();
let el = app.as_element();
assert_eq!(el.data().role, Role::Application);
assert_eq!(el.data().name.as_deref(), Some("TestApp"));
}
}