Compare commits
No commits in common. "f5ca2ebde95b409b9a28d97dd11be49efc188b76" and "03bcf9976df7d9c0cdf42e062ec758959f564bf0" have entirely different histories.
f5ca2ebde9
...
03bcf9976d
|
@ -22,8 +22,7 @@ tokio-stream = { version = "0.1.15", features = ["sync"] }
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
env_logger = "0.10.0"
|
env_logger = "0.10.0"
|
||||||
test-log = "0.2.15"
|
test-log = "0.2.15"
|
||||||
tokio = { version = "1.37.0", features = ["rt-multi-thread", "time", "process"] }
|
tokio = { version = "1.37.0", features = ["rt-multi-thread", "time"] }
|
||||||
uuid = { version = "1.8.0", features = ["v4"] }
|
|
||||||
|
|
||||||
[lib]
|
[lib]
|
||||||
doctest = false
|
doctest = false
|
||||||
|
|
|
@ -26,10 +26,9 @@
|
||||||
devShell = forAllSystems (system: pkgs: toolchain: pkgs.mkShell {
|
devShell = forAllSystems (system: pkgs: toolchain: pkgs.mkShell {
|
||||||
packages = [
|
packages = [
|
||||||
(toolchain.withComponents [
|
(toolchain.withComponents [
|
||||||
"cargo" "rustc" "rustfmt" "clippy" "llvm-tools"
|
"cargo" "rustc" "rustfmt" "clippy"
|
||||||
])
|
])
|
||||||
pkgs.mpv
|
pkgs.mpv
|
||||||
pkgs.grcov
|
|
||||||
];
|
];
|
||||||
RUST_SRC_PATH = "${toolchain.rust-src}/lib/rustlib/src/rust/";
|
RUST_SRC_PATH = "${toolchain.rust-src}/lib/rustlib/src/rust/";
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,23 +0,0 @@
|
||||||
#!/usr/bin/env bash
|
|
||||||
rm -rf target/coverage || true
|
|
||||||
mkdir -p target/coverage
|
|
||||||
|
|
||||||
echo "Running tests"
|
|
||||||
RUST_LOG=mpvipc=trace RUSTFLAGS="-Cinstrument-coverage" LLVM_PROFILE_FILE="target/coverage/%p-%m.profraw" cargo test --all-features --release --no-fail-fast
|
|
||||||
|
|
||||||
echo "Generating coverage report"
|
|
||||||
grcov \
|
|
||||||
--source-dir . \
|
|
||||||
--binary-path ./target/release/deps/ \
|
|
||||||
--excl-start 'mod test* \{' \
|
|
||||||
--ignore 'tests/*' \
|
|
||||||
--ignore "*test.rs" \
|
|
||||||
--ignore "*tests.rs" \
|
|
||||||
--ignore "*github.com*" \
|
|
||||||
--ignore "*libcore*" \
|
|
||||||
--ignore "*rustc*" \
|
|
||||||
--ignore "*liballoc*" \
|
|
||||||
--ignore "*cargo*" \
|
|
||||||
-t html \
|
|
||||||
-o ./target/coverage/html \
|
|
||||||
target/coverage/
|
|
|
@ -68,7 +68,7 @@ pub(crate) trait IntoRawCommandPart {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Generic data type representing all possible data types that mpv can return.
|
/// Generic data type representing all possible data types that mpv can return.
|
||||||
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||||
pub enum MpvDataType {
|
pub enum MpvDataType {
|
||||||
Array(Vec<MpvDataType>),
|
Array(Vec<MpvDataType>),
|
||||||
Bool(bool),
|
Bool(bool),
|
||||||
|
|
|
@ -16,7 +16,7 @@ use crate::{ipc::MpvIpcEvent, Error, ErrorCode, MpvDataType};
|
||||||
///
|
///
|
||||||
/// See <https://mpv.io/manual/master/#properties> for
|
/// See <https://mpv.io/manual/master/#properties> for
|
||||||
/// the upstream list of properties.
|
/// the upstream list of properties.
|
||||||
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||||
pub enum Property {
|
pub enum Property {
|
||||||
Path(Option<String>),
|
Path(Option<String>),
|
||||||
Pause(bool),
|
Pause(bool),
|
||||||
|
@ -35,7 +35,7 @@ pub enum Property {
|
||||||
///
|
///
|
||||||
/// See <https://mpv.io/manual/master/#list-of-events> for
|
/// See <https://mpv.io/manual/master/#list-of-events> for
|
||||||
/// the upstream list of events.
|
/// the upstream list of events.
|
||||||
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||||
pub enum Event {
|
pub enum Event {
|
||||||
Shutdown,
|
Shutdown,
|
||||||
StartFile,
|
StartFile,
|
||||||
|
|
59
src/ipc.rs
59
src/ipc.rs
|
@ -2,11 +2,12 @@
|
||||||
|
|
||||||
use futures::{SinkExt, StreamExt};
|
use futures::{SinkExt, StreamExt};
|
||||||
use serde_json::{json, Value};
|
use serde_json::{json, Value};
|
||||||
|
use std::mem;
|
||||||
use tokio::{
|
use tokio::{
|
||||||
net::UnixStream,
|
net::UnixStream,
|
||||||
sync::{broadcast, mpsc, oneshot},
|
sync::{broadcast, mpsc, oneshot, Mutex},
|
||||||
};
|
};
|
||||||
use tokio_util::codec::{Framed, LinesCodec};
|
use tokio_util::codec::{Framed, LinesCodec, LinesCodecError};
|
||||||
|
|
||||||
use crate::{Error, ErrorCode};
|
use crate::{Error, ErrorCode};
|
||||||
|
|
||||||
|
@ -14,6 +15,9 @@ use crate::{Error, ErrorCode};
|
||||||
/// and message passing with [`Mpv`](crate::Mpv) controllers.
|
/// and message passing with [`Mpv`](crate::Mpv) controllers.
|
||||||
pub(crate) struct MpvIpc {
|
pub(crate) struct MpvIpc {
|
||||||
socket: Framed<UnixStream, LinesCodec>,
|
socket: Framed<UnixStream, LinesCodec>,
|
||||||
|
// I had trouble with reading and writing to the socket when it was wrapped
|
||||||
|
// in a MutexGuard, so I'm using a separate Mutex to lock the socket when needed.
|
||||||
|
socket_lock: Mutex<()>,
|
||||||
command_channel: mpsc::Receiver<(MpvIpcCommand, oneshot::Sender<MpvIpcResponse>)>,
|
command_channel: mpsc::Receiver<(MpvIpcCommand, oneshot::Sender<MpvIpcResponse>)>,
|
||||||
event_channel: broadcast::Sender<MpvIpcEvent>,
|
event_channel: broadcast::Sender<MpvIpcEvent>,
|
||||||
}
|
}
|
||||||
|
@ -46,14 +50,14 @@ impl MpvIpc {
|
||||||
MpvIpc {
|
MpvIpc {
|
||||||
socket: Framed::new(socket, LinesCodec::new()),
|
socket: Framed::new(socket, LinesCodec::new()),
|
||||||
command_channel,
|
command_channel,
|
||||||
|
socket_lock: Mutex::new(()),
|
||||||
event_channel,
|
event_channel,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) async fn send_command(&mut self, command: &[Value]) -> Result<Option<Value>, Error> {
|
pub(crate) async fn send_command(&mut self, command: &[Value]) -> Result<Option<Value>, Error> {
|
||||||
// let lock = self.socket_lock.lock().await;
|
let lock = self.socket_lock.lock().await;
|
||||||
// START CRITICAL SECTION
|
// START CRITICAL SECTION
|
||||||
|
|
||||||
let ipc_command = json!({ "command": command });
|
let ipc_command = json!({ "command": command });
|
||||||
let ipc_command_str = serde_json::to_string(&ipc_command)
|
let ipc_command_str = serde_json::to_string(&ipc_command)
|
||||||
.map_err(|why| Error(ErrorCode::JsonParseError(why.to_string())))?;
|
.map_err(|why| Error(ErrorCode::JsonParseError(why.to_string())))?;
|
||||||
|
@ -65,7 +69,6 @@ impl MpvIpc {
|
||||||
.await
|
.await
|
||||||
.map_err(|why| Error(ErrorCode::ConnectError(why.to_string())))?;
|
.map_err(|why| Error(ErrorCode::ConnectError(why.to_string())))?;
|
||||||
|
|
||||||
let response = loop {
|
|
||||||
let response = self
|
let response = self
|
||||||
.socket
|
.socket
|
||||||
.next()
|
.next()
|
||||||
|
@ -73,26 +76,14 @@ impl MpvIpc {
|
||||||
.ok_or(Error(ErrorCode::MissingValue))?
|
.ok_or(Error(ErrorCode::MissingValue))?
|
||||||
.map_err(|why| Error(ErrorCode::ConnectError(why.to_string())))?;
|
.map_err(|why| Error(ErrorCode::ConnectError(why.to_string())))?;
|
||||||
|
|
||||||
let parsed_response = serde_json::from_str::<Value>(&response)
|
|
||||||
.map_err(|why| Error(ErrorCode::JsonParseError(why.to_string())));
|
|
||||||
|
|
||||||
if parsed_response
|
|
||||||
.as_ref()
|
|
||||||
.ok()
|
|
||||||
.and_then(|v| v.as_object().map(|o| o.contains_key("event")))
|
|
||||||
.unwrap_or(false)
|
|
||||||
{
|
|
||||||
self.handle_event(parsed_response).await;
|
|
||||||
} else {
|
|
||||||
break parsed_response;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
// END CRITICAL SECTION
|
// END CRITICAL SECTION
|
||||||
// mem::drop(lock);
|
mem::drop(lock);
|
||||||
|
|
||||||
log::trace!("Received response: {:?}", response);
|
log::trace!("Received response: {}", response);
|
||||||
|
|
||||||
parse_mpv_response_data(response?)
|
serde_json::from_str::<Value>(&response)
|
||||||
|
.map_err(|why| Error(ErrorCode::JsonParseError(why.to_string())))
|
||||||
|
.and_then(parse_mpv_response_data)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) async fn get_mpv_property(
|
pub(crate) async fn get_mpv_property(
|
||||||
|
@ -126,8 +117,16 @@ impl MpvIpc {
|
||||||
.await
|
.await
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn handle_event(&mut self, event: Result<Value, Error>) {
|
async fn handle_event(&mut self, event: Result<String, LinesCodecError>) {
|
||||||
match &event {
|
let parsed_event = event
|
||||||
|
.as_ref()
|
||||||
|
.map_err(|why| Error(ErrorCode::ConnectError(why.to_string())))
|
||||||
|
.and_then(|event| {
|
||||||
|
serde_json::from_str::<Value>(event)
|
||||||
|
.map_err(|why| Error(ErrorCode::JsonParseError(why.to_string())))
|
||||||
|
});
|
||||||
|
|
||||||
|
match parsed_event {
|
||||||
Ok(event) => {
|
Ok(event) => {
|
||||||
log::trace!("Parsed event: {:?}", event);
|
log::trace!("Parsed event: {:?}", event);
|
||||||
if let Err(broadcast::error::SendError(_)) =
|
if let Err(broadcast::error::SendError(_)) =
|
||||||
|
@ -137,7 +136,7 @@ impl MpvIpc {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
log::trace!("Error parsing event, ignoring:\n {:?}\n {:?}", &event, e);
|
log::trace!("Error parsing event, ignoring:\n {:?}\n {:?}", event, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -147,14 +146,8 @@ impl MpvIpc {
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
Some(event) = self.socket.next() => {
|
Some(event) = self.socket.next() => {
|
||||||
log::trace!("Got event: {:?}", event);
|
log::trace!("Got event: {:?}", event);
|
||||||
|
// TODO: error handling
|
||||||
let parsed_event = event
|
self.handle_event(event).await;
|
||||||
.map_err(|why| Error(ErrorCode::ConnectError(why.to_string())))
|
|
||||||
.and_then(|event|
|
|
||||||
serde_json::from_str::<Value>(&event)
|
|
||||||
.map_err(|why| Error(ErrorCode::JsonParseError(why.to_string()))));
|
|
||||||
|
|
||||||
self.handle_event(parsed_event).await;
|
|
||||||
}
|
}
|
||||||
Some((cmd, tx)) = self.command_channel.recv() => {
|
Some((cmd, tx)) = self.command_channel.recv() => {
|
||||||
log::trace!("Handling command: {:?}", cmd);
|
log::trace!("Handling command: {:?}", cmd);
|
||||||
|
|
|
@ -136,27 +136,61 @@ pub(crate) fn json_map_to_hashmap(
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn json_array_to_vec(array: &[Value]) -> Vec<MpvDataType> {
|
pub(crate) fn json_array_to_vec(array: &[Value]) -> Vec<MpvDataType> {
|
||||||
array
|
let mut output: Vec<MpvDataType> = Vec::new();
|
||||||
.iter()
|
if !array.is_empty() {
|
||||||
.map(|entry| match entry {
|
match array[0] {
|
||||||
Value::Array(a) => MpvDataType::Array(json_array_to_vec(a)),
|
Value::Array(_) => {
|
||||||
Value::Bool(b) => MpvDataType::Bool(*b),
|
for entry in array {
|
||||||
Value::Number(n) => {
|
if let Value::Array(ref a) = *entry {
|
||||||
|
output.push(MpvDataType::Array(json_array_to_vec(a)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Value::Bool(_) => {
|
||||||
|
for entry in array {
|
||||||
|
if let Value::Bool(ref b) = *entry {
|
||||||
|
output.push(MpvDataType::Bool(*b));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Value::Number(_) => {
|
||||||
|
for entry in array {
|
||||||
|
if let Value::Number(ref n) = *entry {
|
||||||
if n.is_u64() {
|
if n.is_u64() {
|
||||||
MpvDataType::Usize(n.as_u64().unwrap() as usize)
|
output.push(MpvDataType::Usize(n.as_u64().unwrap() as usize));
|
||||||
} else if n.is_f64() {
|
} else if n.is_f64() {
|
||||||
MpvDataType::Double(n.as_f64().unwrap())
|
output.push(MpvDataType::Double(n.as_f64().unwrap()));
|
||||||
} else {
|
} else {
|
||||||
panic!("unimplemented number");
|
panic!("unimplemented number");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Value::Object(ref o) => MpvDataType::HashMap(json_map_to_hashmap(o)),
|
}
|
||||||
Value::String(s) => MpvDataType::String(s.to_owned()),
|
}
|
||||||
|
|
||||||
|
Value::Object(_) => {
|
||||||
|
for entry in array {
|
||||||
|
if let Value::Object(ref map) = *entry {
|
||||||
|
output.push(MpvDataType::HashMap(json_map_to_hashmap(map)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Value::String(_) => {
|
||||||
|
for entry in array {
|
||||||
|
if let Value::String(ref s) = *entry {
|
||||||
|
output.push(MpvDataType::String(s.to_string()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Value::Null => {
|
Value::Null => {
|
||||||
unimplemented!();
|
unimplemented!();
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
.collect()
|
}
|
||||||
|
output
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn json_array_to_playlist(array: &[Value]) -> Vec<PlaylistEntry> {
|
pub(crate) fn json_array_to_playlist(array: &[Value]) -> Vec<PlaylistEntry> {
|
||||||
|
@ -183,137 +217,3 @@ pub(crate) fn json_array_to_playlist(array: &[Value]) -> Vec<PlaylistEntry> {
|
||||||
}
|
}
|
||||||
output
|
output
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod test {
|
|
||||||
use super::*;
|
|
||||||
use crate::MpvDataType;
|
|
||||||
use serde_json::json;
|
|
||||||
use std::collections::HashMap;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_json_map_to_hashmap() {
|
|
||||||
let json = json!({
|
|
||||||
"array": [1, 2, 3],
|
|
||||||
"bool": true,
|
|
||||||
"double": 1.0,
|
|
||||||
"usize": 1,
|
|
||||||
"string": "string",
|
|
||||||
"object": {
|
|
||||||
"key": "value"
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
let mut expected = HashMap::new();
|
|
||||||
expected.insert(
|
|
||||||
"array".to_string(),
|
|
||||||
MpvDataType::Array(vec![
|
|
||||||
MpvDataType::Usize(1),
|
|
||||||
MpvDataType::Usize(2),
|
|
||||||
MpvDataType::Usize(3),
|
|
||||||
]),
|
|
||||||
);
|
|
||||||
expected.insert("bool".to_string(), MpvDataType::Bool(true));
|
|
||||||
expected.insert("double".to_string(), MpvDataType::Double(1.0));
|
|
||||||
expected.insert("usize".to_string(), MpvDataType::Usize(1));
|
|
||||||
expected.insert(
|
|
||||||
"string".to_string(),
|
|
||||||
MpvDataType::String("string".to_string()),
|
|
||||||
);
|
|
||||||
expected.insert(
|
|
||||||
"object".to_string(),
|
|
||||||
MpvDataType::HashMap(HashMap::from([(
|
|
||||||
"key".to_string(),
|
|
||||||
MpvDataType::String("value".to_string()),
|
|
||||||
)])),
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_eq!(json_map_to_hashmap(json.as_object().unwrap()), expected);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn test_json_map_to_hashmap_fail_on_null() {
|
|
||||||
json_map_to_hashmap(
|
|
||||||
json!({
|
|
||||||
"null": null
|
|
||||||
})
|
|
||||||
.as_object()
|
|
||||||
.unwrap(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_json_array_to_vec() {
|
|
||||||
let json = json!([
|
|
||||||
[1, 2, 3],
|
|
||||||
true,
|
|
||||||
1.0,
|
|
||||||
1,
|
|
||||||
"string",
|
|
||||||
{
|
|
||||||
"key": "value"
|
|
||||||
}
|
|
||||||
]);
|
|
||||||
|
|
||||||
println!("{:?}", json.as_array().unwrap());
|
|
||||||
println!("{:?}", json_array_to_vec(json.as_array().unwrap()));
|
|
||||||
|
|
||||||
let expected = vec![
|
|
||||||
MpvDataType::Array(vec![
|
|
||||||
MpvDataType::Usize(1),
|
|
||||||
MpvDataType::Usize(2),
|
|
||||||
MpvDataType::Usize(3),
|
|
||||||
]),
|
|
||||||
MpvDataType::Bool(true),
|
|
||||||
MpvDataType::Double(1.0),
|
|
||||||
MpvDataType::Usize(1),
|
|
||||||
MpvDataType::String("string".to_string()),
|
|
||||||
MpvDataType::HashMap(HashMap::from([(
|
|
||||||
"key".to_string(),
|
|
||||||
MpvDataType::String("value".to_string()),
|
|
||||||
)])),
|
|
||||||
];
|
|
||||||
|
|
||||||
assert_eq!(json_array_to_vec(json.as_array().unwrap()), expected);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn test_json_array_to_vec_fail_on_null() {
|
|
||||||
json_array_to_vec(json!([null]).as_array().unwrap().as_slice());
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_json_array_to_playlist() {
|
|
||||||
let json = json!([
|
|
||||||
{
|
|
||||||
"filename": "file1",
|
|
||||||
"title": "title1",
|
|
||||||
"current": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"filename": "file2",
|
|
||||||
"title": "title2",
|
|
||||||
"current": false
|
|
||||||
}
|
|
||||||
]);
|
|
||||||
|
|
||||||
let expected = vec![
|
|
||||||
PlaylistEntry {
|
|
||||||
id: 0,
|
|
||||||
filename: "file1".to_string(),
|
|
||||||
title: "title1".to_string(),
|
|
||||||
current: true,
|
|
||||||
},
|
|
||||||
PlaylistEntry {
|
|
||||||
id: 1,
|
|
||||||
filename: "file2".to_string(),
|
|
||||||
title: "title2".to_string(),
|
|
||||||
current: false,
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
assert_eq!(json_array_to_playlist(json.as_array().unwrap()), expected);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,100 +0,0 @@
|
||||||
use mpvipc::{Error, Mpv, MpvExt};
|
|
||||||
use std::path::Path;
|
|
||||||
use tokio::{
|
|
||||||
process::{Child, Command},
|
|
||||||
time::{sleep, timeout, Duration},
|
|
||||||
};
|
|
||||||
|
|
||||||
#[cfg(target_family = "unix")]
|
|
||||||
async fn spawn_headless_mpv() -> Result<(Child, Mpv), Error> {
|
|
||||||
let socket_path_str = format!("/tmp/mpv-ipc-{}", uuid::Uuid::new_v4());
|
|
||||||
let socket_path = Path::new(&socket_path_str);
|
|
||||||
|
|
||||||
let process_handle = Command::new("mpv")
|
|
||||||
.arg("--no-config")
|
|
||||||
.arg("--idle")
|
|
||||||
.arg("--no-video")
|
|
||||||
.arg("--no-audio")
|
|
||||||
.arg(format!(
|
|
||||||
"--input-ipc-server={}",
|
|
||||||
&socket_path.to_str().unwrap()
|
|
||||||
))
|
|
||||||
.spawn()
|
|
||||||
.expect("Failed to start mpv");
|
|
||||||
|
|
||||||
if timeout(Duration::from_millis(500), async {
|
|
||||||
while !&socket_path.exists() {
|
|
||||||
sleep(Duration::from_millis(10)).await;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.await
|
|
||||||
.is_err()
|
|
||||||
{
|
|
||||||
panic!("Failed to create mpv socket at {:?}", &socket_path);
|
|
||||||
}
|
|
||||||
|
|
||||||
let mpv = Mpv::connect(socket_path.to_str().unwrap()).await.unwrap();
|
|
||||||
Ok((process_handle, mpv))
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tokio::test]
|
|
||||||
#[cfg(target_family = "unix")]
|
|
||||||
async fn test_get_mpv_version() {
|
|
||||||
let (mut proc, mpv) = spawn_headless_mpv().await.unwrap();
|
|
||||||
let version: String = mpv.get_property("mpv-version").await.unwrap();
|
|
||||||
assert!(version.starts_with("mpv"));
|
|
||||||
|
|
||||||
mpv.kill().await.unwrap();
|
|
||||||
proc.kill().await.unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tokio::test]
|
|
||||||
#[cfg(target_family = "unix")]
|
|
||||||
async fn test_set_property() {
|
|
||||||
let (mut proc, mpv) = spawn_headless_mpv().await.unwrap();
|
|
||||||
mpv.set_property("pause", true).await.unwrap();
|
|
||||||
let paused: bool = mpv.get_property("pause").await.unwrap();
|
|
||||||
assert!(paused);
|
|
||||||
|
|
||||||
mpv.kill().await.unwrap();
|
|
||||||
proc.kill().await.unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tokio::test]
|
|
||||||
#[cfg(target_family = "unix")]
|
|
||||||
async fn test_events() {
|
|
||||||
use futures::stream::StreamExt;
|
|
||||||
|
|
||||||
let (mut proc, mpv) = spawn_headless_mpv().await.unwrap();
|
|
||||||
|
|
||||||
mpv.observe_property(1337, "pause").await.unwrap();
|
|
||||||
|
|
||||||
let mut events = mpv.get_event_stream().await;
|
|
||||||
let event_checking_thread = tokio::spawn(async move {
|
|
||||||
loop {
|
|
||||||
let event = events.next().await.unwrap().unwrap();
|
|
||||||
if let mpvipc::Event::PropertyChange { id, property } = event {
|
|
||||||
if id == 1337 {
|
|
||||||
assert_eq!(property, mpvipc::Property::Pause(true));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
tokio::time::sleep(Duration::from_millis(10)).await;
|
|
||||||
|
|
||||||
mpv.set_property("pause", true).await.unwrap();
|
|
||||||
|
|
||||||
if let Err(_) = tokio::time::timeout(
|
|
||||||
tokio::time::Duration::from_millis(500),
|
|
||||||
event_checking_thread,
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
{
|
|
||||||
panic!("Event checking thread timed out");
|
|
||||||
}
|
|
||||||
|
|
||||||
mpv.kill().await.unwrap();
|
|
||||||
proc.kill().await.unwrap();
|
|
||||||
}
|
|
Loading…
Reference in New Issue