2024-05-03 22:29:29 +02:00
|
|
|
use futures::{stream::StreamExt, Stream};
|
2024-08-03 16:27:11 +02:00
|
|
|
use mpvipc_async::{parse_property, Event, Mpv, MpvError, MpvExt, Property};
|
2024-05-03 22:29:29 +02:00
|
|
|
use thiserror::Error;
|
|
|
|
use tokio::time::sleep;
|
|
|
|
use tokio::time::{timeout, Duration};
|
|
|
|
|
|
|
|
use test_log::test;
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
2024-12-14 12:55:14 +01:00
|
|
|
const MPV_CHANNEL_ID: u64 = 1337;
|
2024-05-03 22:29:29 +02:00
|
|
|
|
|
|
|
#[derive(Error, Debug)]
|
|
|
|
enum PropertyCheckingThreadError {
|
|
|
|
#[error("Unexpected property: {0:?}")]
|
2024-08-03 16:27:11 +02:00
|
|
|
UnexpectedPropertyError(Property),
|
2024-05-03 22:29:29 +02:00
|
|
|
|
|
|
|
#[error(transparent)]
|
|
|
|
MpvError(#[from] MpvError),
|
|
|
|
}
|
|
|
|
|
2024-08-03 15:36:53 +02:00
|
|
|
/// This function will create an ongoing tokio task that collects [`Event::PropertyChange`] events,
|
2024-08-03 16:27:11 +02:00
|
|
|
/// and parses them into [`Property`]s. It will then run the property through the provided
|
2024-08-03 15:36:53 +02:00
|
|
|
/// closure, and return an error if the closure returns false.
|
|
|
|
///
|
|
|
|
/// The returned cancellation token can be used to stop the task.
|
2024-05-03 22:29:29 +02:00
|
|
|
fn create_interruptable_event_property_checking_thread<T>(
|
|
|
|
mut events: impl Stream<Item = Result<Event, MpvError>> + Unpin + Send + 'static,
|
|
|
|
on_property: T,
|
|
|
|
) -> (
|
|
|
|
tokio::task::JoinHandle<Result<(), PropertyCheckingThreadError>>,
|
|
|
|
tokio_util::sync::CancellationToken,
|
|
|
|
)
|
|
|
|
where
|
2024-08-03 16:27:11 +02:00
|
|
|
T: Fn(Property) -> bool + Send + 'static,
|
2024-05-03 22:29:29 +02:00
|
|
|
{
|
|
|
|
let cancellation_token = tokio_util::sync::CancellationToken::new();
|
|
|
|
let cancellation_token_clone = cancellation_token.clone();
|
|
|
|
let handle = tokio::spawn(async move {
|
|
|
|
loop {
|
|
|
|
tokio::select! {
|
|
|
|
event = events.next() => {
|
|
|
|
match event {
|
|
|
|
Some(Ok(event)) => {
|
|
|
|
match event {
|
2024-12-15 15:31:01 +01:00
|
|
|
Event::PropertyChange { id: Some(MPV_CHANNEL_ID), name, data } => {
|
2024-05-04 23:01:17 +02:00
|
|
|
let property = parse_property(&name, data).unwrap();
|
2024-05-03 22:29:29 +02:00
|
|
|
if !on_property(property.clone()) {
|
|
|
|
return Err(PropertyCheckingThreadError::UnexpectedPropertyError(property))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
log::trace!("Received unrelated event, ignoring: {:?}", event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Some(Err(err)) => return Err(err.into()),
|
|
|
|
None => return Ok(()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ = cancellation_token_clone.cancelled() => return Ok(()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
(handle, cancellation_token)
|
|
|
|
}
|
|
|
|
|
2024-08-03 15:36:53 +02:00
|
|
|
/// This helper function will gracefully shut down both the event checking thread and the mpv process.
|
|
|
|
/// It will also return an error if the event checking thread happened to panic, or if it times out
|
|
|
|
/// The timeout is hardcoded to 500ms.
|
2024-05-03 22:29:29 +02:00
|
|
|
async fn graceful_shutdown(
|
|
|
|
cancellation_token: tokio_util::sync::CancellationToken,
|
|
|
|
handle: tokio::task::JoinHandle<Result<(), PropertyCheckingThreadError>>,
|
|
|
|
mpv: Mpv,
|
|
|
|
mut proc: tokio::process::Child,
|
|
|
|
) -> Result<(), MpvError> {
|
|
|
|
cancellation_token.cancel();
|
|
|
|
|
|
|
|
match timeout(Duration::from_millis(500), handle).await {
|
|
|
|
Ok(Ok(Ok(()))) => {}
|
|
|
|
Ok(Ok(Err(err))) => match err {
|
|
|
|
PropertyCheckingThreadError::UnexpectedPropertyError(property) => {
|
|
|
|
return Err(MpvError::Other(format!(
|
|
|
|
"Unexpected property: {:?}",
|
|
|
|
property
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
PropertyCheckingThreadError::MpvError(err) => return Err(err),
|
|
|
|
},
|
|
|
|
Ok(Err(_)) => {
|
|
|
|
return Err(MpvError::InternalConnectionError(
|
|
|
|
"Event checking thread timed out".to_owned(),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
Err(_) => {
|
|
|
|
return Err(MpvError::InternalConnectionError(
|
|
|
|
"Event checking thread panicked".to_owned(),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mpv.kill().await?;
|
|
|
|
proc.wait().await.map_err(|err| {
|
|
|
|
MpvError::InternalConnectionError(format!(
|
|
|
|
"Failed to wait for mpv process to exit: {}",
|
|
|
|
err
|
|
|
|
))
|
|
|
|
})?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2024-08-03 15:36:53 +02:00
|
|
|
/// Test correct parsing of different values of the "pause" property
|
2024-05-03 22:29:29 +02:00
|
|
|
#[test(tokio::test)]
|
|
|
|
#[cfg(target_family = "unix")]
|
|
|
|
async fn test_highlevel_event_pause() -> Result<(), MpvError> {
|
|
|
|
let (proc, mpv) = spawn_headless_mpv().await?;
|
|
|
|
|
|
|
|
mpv.observe_property(MPV_CHANNEL_ID, "pause").await?;
|
|
|
|
|
|
|
|
let events = mpv.get_event_stream().await;
|
|
|
|
let (handle, cancellation_token) =
|
|
|
|
create_interruptable_event_property_checking_thread(events, |property| match property {
|
2024-08-03 16:27:11 +02:00
|
|
|
Property::Pause(_) => {
|
2024-05-03 22:29:29 +02:00
|
|
|
log::debug!("{:?}", property);
|
|
|
|
true
|
|
|
|
}
|
|
|
|
_ => false,
|
|
|
|
});
|
|
|
|
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
mpv.set_property("pause", false).await?;
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
mpv.set_property("pause", true).await?;
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
|
|
|
|
graceful_shutdown(cancellation_token, handle, mpv, proc).await?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2024-08-03 15:36:53 +02:00
|
|
|
/// Test correct parsing of different values of the "volume" property
|
2024-05-03 22:29:29 +02:00
|
|
|
#[test(tokio::test)]
|
|
|
|
#[cfg(target_family = "unix")]
|
|
|
|
async fn test_highlevel_event_volume() -> Result<(), MpvError> {
|
|
|
|
let (proc, mpv) = spawn_headless_mpv().await?;
|
|
|
|
|
|
|
|
mpv.observe_property(1337, "volume").await?;
|
|
|
|
let events = mpv.get_event_stream().await;
|
|
|
|
let (handle, cancellation_token) =
|
|
|
|
create_interruptable_event_property_checking_thread(events, |property| match property {
|
2024-08-03 16:27:11 +02:00
|
|
|
Property::Volume(_) => {
|
2024-05-03 22:29:29 +02:00
|
|
|
log::trace!("{:?}", property);
|
|
|
|
true
|
|
|
|
}
|
|
|
|
_ => false,
|
|
|
|
});
|
|
|
|
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
mpv.set_property("volume", 100.0).await?;
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
mpv.set_property("volume", 40).await?;
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
mpv.set_property("volume", 0.0).await?;
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
|
|
|
|
graceful_shutdown(cancellation_token, handle, mpv, proc).await?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2024-08-03 15:36:53 +02:00
|
|
|
/// Test correct parsing of different values of the "mute" property
|
2024-05-03 22:29:29 +02:00
|
|
|
#[test(tokio::test)]
|
|
|
|
#[cfg(target_family = "unix")]
|
|
|
|
async fn test_highlevel_event_mute() -> Result<(), MpvError> {
|
|
|
|
let (proc, mpv) = spawn_headless_mpv().await?;
|
|
|
|
|
|
|
|
mpv.observe_property(1337, "mute").await?;
|
|
|
|
let events = mpv.get_event_stream().await;
|
|
|
|
let (handle, cancellation_token) =
|
|
|
|
create_interruptable_event_property_checking_thread(events, |property| match property {
|
2024-08-03 16:27:11 +02:00
|
|
|
Property::Mute(_) => {
|
2024-05-03 22:29:29 +02:00
|
|
|
log::trace!("{:?}", property);
|
|
|
|
true
|
|
|
|
}
|
|
|
|
_ => false,
|
|
|
|
});
|
|
|
|
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
mpv.set_property("mute", true).await?;
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
mpv.set_property("mute", false).await?;
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
|
|
|
|
graceful_shutdown(cancellation_token, handle, mpv, proc).await?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2024-08-03 15:36:53 +02:00
|
|
|
/// Test correct parsing of different values of the "duration" property
|
2024-05-03 22:29:29 +02:00
|
|
|
#[test(tokio::test)]
|
|
|
|
#[cfg(target_family = "unix")]
|
|
|
|
async fn test_highlevel_event_duration() -> Result<(), MpvError> {
|
|
|
|
let (proc, mpv) = spawn_headless_mpv().await?;
|
|
|
|
|
|
|
|
mpv.observe_property(1337, "duration").await?;
|
|
|
|
|
|
|
|
let events = mpv.get_event_stream().await;
|
|
|
|
let (handle, cancellation_token) =
|
|
|
|
create_interruptable_event_property_checking_thread(events, |property| match property {
|
2024-08-03 16:27:11 +02:00
|
|
|
Property::Duration(_) => {
|
2024-05-03 22:29:29 +02:00
|
|
|
log::trace!("{:?}", property);
|
|
|
|
true
|
|
|
|
}
|
|
|
|
_ => false,
|
|
|
|
});
|
|
|
|
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
mpv.set_property("pause", true).await?;
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
mpv.set_property("pause", false).await?;
|
|
|
|
sleep(Duration::from_millis(5)).await;
|
|
|
|
|
|
|
|
graceful_shutdown(cancellation_token, handle, mpv, proc).await?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|