2019-06-19 00:51:11 +02:00
|
|
|
use super::*;
|
2024-04-17 00:07:11 +02:00
|
|
|
use futures::{SinkExt, StreamExt};
|
|
|
|
use serde_json::{json, Value};
|
|
|
|
use std::mem;
|
|
|
|
use tokio::net::UnixStream;
|
2024-04-19 01:36:11 +02:00
|
|
|
use tokio::sync::mpsc;
|
|
|
|
use tokio::sync::{broadcast, oneshot, Mutex};
|
|
|
|
use tokio_util::codec::{Framed, LinesCodec, LinesCodecError};
|
2024-04-17 00:07:11 +02:00
|
|
|
|
|
|
|
pub(crate) struct MpvIpc {
|
|
|
|
socket: Framed<UnixStream, LinesCodec>,
|
2024-04-19 01:36:11 +02:00
|
|
|
command_channel: mpsc::Receiver<(MpvIpcCommand, oneshot::Sender<MpvIpcResponse>)>,
|
2024-04-17 00:07:11 +02:00
|
|
|
socket_lock: Mutex<()>,
|
2024-04-19 01:36:11 +02:00
|
|
|
event_channel: broadcast::Sender<MpvIpcEvent>,
|
2017-05-22 18:31:20 +02:00
|
|
|
}
|
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
|
|
pub(crate) enum MpvIpcCommand {
|
|
|
|
Command(Vec<String>),
|
|
|
|
GetProperty(String),
|
|
|
|
SetProperty(String, Value),
|
|
|
|
ObserveProperty(isize, String),
|
|
|
|
UnobserveProperty(isize),
|
|
|
|
Exit,
|
2017-05-22 18:31:20 +02:00
|
|
|
}
|
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub(crate) struct MpvIpcResponse(pub(crate) Result<Option<Value>, Error>);
|
|
|
|
|
2024-04-19 01:36:11 +02:00
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub(crate) struct MpvIpcEvent(pub(crate) Value);
|
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
impl MpvIpc {
|
|
|
|
pub(crate) fn new(
|
|
|
|
socket: UnixStream,
|
2024-04-19 01:36:11 +02:00
|
|
|
command_channel: mpsc::Receiver<(MpvIpcCommand, oneshot::Sender<MpvIpcResponse>)>,
|
|
|
|
event_channel: broadcast::Sender<MpvIpcEvent>,
|
2024-04-17 00:07:11 +02:00
|
|
|
) -> Self {
|
|
|
|
MpvIpc {
|
|
|
|
socket: Framed::new(socket, LinesCodec::new()),
|
|
|
|
command_channel,
|
|
|
|
socket_lock: Mutex::new(()),
|
2024-04-19 01:36:11 +02:00
|
|
|
event_channel,
|
2024-04-17 00:07:11 +02:00
|
|
|
}
|
|
|
|
}
|
2024-04-16 22:48:27 +02:00
|
|
|
|
2024-04-19 01:36:11 +02:00
|
|
|
pub(crate) async fn send_command(&mut self, command: &[Value]) -> Result<Option<Value>, Error> {
|
2024-04-17 00:07:11 +02:00
|
|
|
let lock = self.socket_lock.lock().await;
|
|
|
|
// START CRITICAL SECTION
|
|
|
|
let ipc_command = json!({ "command": command });
|
|
|
|
let ipc_command_str = serde_json::to_string(&ipc_command)
|
|
|
|
.map_err(|why| Error(ErrorCode::JsonParseError(why.to_string())))?;
|
2023-08-02 11:15:37 +02:00
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
log::trace!("Sending command: {}", ipc_command_str);
|
2017-05-22 18:31:20 +02:00
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
self.socket
|
|
|
|
.send(ipc_command_str)
|
|
|
|
.await
|
|
|
|
.map_err(|why| Error(ErrorCode::ConnectError(why.to_string())))?;
|
2024-04-16 22:48:27 +02:00
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
let response = self
|
|
|
|
.socket
|
|
|
|
.next()
|
|
|
|
.await
|
|
|
|
.ok_or(Error(ErrorCode::MissingValue))?
|
|
|
|
.map_err(|why| Error(ErrorCode::ConnectError(why.to_string())))?;
|
2017-05-22 18:31:20 +02:00
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
// END CRITICAL SECTION
|
|
|
|
mem::drop(lock);
|
2024-04-16 22:48:27 +02:00
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
log::trace!("Received response: {}", response);
|
2017-05-22 18:31:20 +02:00
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
serde_json::from_str::<Value>(&response)
|
|
|
|
.map_err(|why| Error(ErrorCode::JsonParseError(why.to_string())))
|
|
|
|
.and_then(parse_mpv_response_data)
|
|
|
|
}
|
2022-07-10 15:16:11 +02:00
|
|
|
|
2024-04-19 01:36:11 +02:00
|
|
|
pub(crate) async fn get_mpv_property(
|
|
|
|
&mut self,
|
|
|
|
property: &str,
|
|
|
|
) -> Result<Option<Value>, Error> {
|
|
|
|
self.send_command(&[json!("get_property"), json!(property)])
|
|
|
|
.await
|
2024-04-17 00:07:11 +02:00
|
|
|
}
|
2019-06-19 00:30:16 +02:00
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
pub(crate) async fn set_mpv_property(
|
|
|
|
&mut self,
|
|
|
|
property: &str,
|
|
|
|
value: Value,
|
|
|
|
) -> Result<Option<Value>, Error> {
|
2024-04-19 01:36:11 +02:00
|
|
|
// let str_value = match &value {
|
|
|
|
// Value::String(s) => s,
|
|
|
|
// v => &serde_json::to_string(&v).unwrap(),
|
|
|
|
// };
|
|
|
|
self.send_command(&[json!("set_property"), json!(property), value])
|
2024-04-17 00:07:11 +02:00
|
|
|
.await
|
|
|
|
}
|
2022-07-19 21:27:02 +02:00
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
pub(crate) async fn observe_property(
|
|
|
|
&mut self,
|
|
|
|
id: isize,
|
|
|
|
property: &str,
|
|
|
|
) -> Result<Option<Value>, Error> {
|
2024-04-19 01:36:11 +02:00
|
|
|
self.send_command(&[json!("observe_property"), json!(id), json!(property)])
|
2024-04-17 00:07:11 +02:00
|
|
|
.await
|
|
|
|
}
|
2017-05-22 18:31:20 +02:00
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
pub(crate) async fn unobserve_property(&mut self, id: isize) -> Result<Option<Value>, Error> {
|
2024-04-19 01:36:11 +02:00
|
|
|
self.send_command(&[json!("unobserve_property"), json!(id)])
|
2024-04-17 00:07:11 +02:00
|
|
|
.await
|
|
|
|
}
|
2017-05-31 19:32:46 +02:00
|
|
|
|
2024-04-19 01:36:11 +02:00
|
|
|
async fn handle_event(&mut self, event: Result<String, LinesCodecError>) {
|
|
|
|
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) => {
|
|
|
|
log::trace!("Parsed event: {:?}", event);
|
|
|
|
if let Err(broadcast::error::SendError(_)) =
|
|
|
|
self.event_channel.send(MpvIpcEvent(event.to_owned()))
|
|
|
|
{
|
|
|
|
log::trace!("Failed to send event to channel, ignoring");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
log::trace!("Error parsing event, ignoring:\n {:?}\n {:?}", event, e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-17 00:07:11 +02:00
|
|
|
pub(crate) async fn run(mut self) -> Result<(), Error> {
|
|
|
|
loop {
|
|
|
|
tokio::select! {
|
|
|
|
Some(event) = self.socket.next() => {
|
2024-04-19 01:36:11 +02:00
|
|
|
log::trace!("Got event: {:?}", event);
|
|
|
|
// TODO: error handling
|
|
|
|
self.handle_event(event).await;
|
2024-04-17 00:07:11 +02:00
|
|
|
}
|
|
|
|
Some((cmd, tx)) = self.command_channel.recv() => {
|
|
|
|
log::trace!("Handling command: {:?}", cmd);
|
|
|
|
match cmd {
|
|
|
|
MpvIpcCommand::Command(command) => {
|
2024-04-19 01:36:11 +02:00
|
|
|
let refs = command.iter().map(|s| json!(s)).collect::<Vec<Value>>();
|
2024-04-17 00:07:11 +02:00
|
|
|
let response = self.send_command(refs.as_slice()).await;
|
|
|
|
tx.send(MpvIpcResponse(response)).unwrap()
|
|
|
|
}
|
|
|
|
MpvIpcCommand::GetProperty(property) => {
|
|
|
|
let response = self.get_mpv_property(&property).await;
|
|
|
|
tx.send(MpvIpcResponse(response)).unwrap()
|
|
|
|
}
|
|
|
|
MpvIpcCommand::SetProperty(property, value) => {
|
|
|
|
let response = self.set_mpv_property(&property, value).await;
|
|
|
|
tx.send(MpvIpcResponse(response)).unwrap()
|
|
|
|
}
|
|
|
|
MpvIpcCommand::ObserveProperty(id, property) => {
|
|
|
|
let response = self.observe_property(id, &property).await;
|
|
|
|
tx.send(MpvIpcResponse(response)).unwrap()
|
|
|
|
}
|
|
|
|
MpvIpcCommand::UnobserveProperty(id) => {
|
|
|
|
let response = self.unobserve_property(id).await;
|
|
|
|
tx.send(MpvIpcResponse(response)).unwrap()
|
|
|
|
}
|
|
|
|
MpvIpcCommand::Exit => {
|
|
|
|
tx.send(MpvIpcResponse(Ok(None))).unwrap();
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-05-22 18:31:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-04-17 00:07:11 +02:00
|
|
|
|
|
|
|
fn parse_mpv_response_data(value: Value) -> Result<Option<Value>, Error> {
|
|
|
|
log::trace!("Parsing mpv response data: {:?}", value);
|
|
|
|
let result = value
|
|
|
|
.as_object()
|
|
|
|
.map(|o| (o.get("error").and_then(|e| e.as_str()), o.get("data")))
|
|
|
|
.ok_or(Error(ErrorCode::UnexpectedValue))
|
|
|
|
.and_then(|(error, data)| match error {
|
|
|
|
Some("success") => Ok(data),
|
|
|
|
Some(e) => Err(Error(ErrorCode::MpvError(e.to_string()))),
|
|
|
|
None => Err(Error(ErrorCode::UnexpectedValue)),
|
|
|
|
});
|
|
|
|
match &result {
|
|
|
|
Ok(v) => log::trace!("Successfully parsed mpv response data: {:?}", v),
|
|
|
|
Err(e) => log::trace!("Error parsing mpv response data: {:?}", e),
|
|
|
|
}
|
|
|
|
result.map(|opt| opt.map(|val| val.clone()))
|
2024-04-19 01:36:11 +02:00
|
|
|
}
|