use core::panic;
use std::{
convert::TryFrom,
fmt::{Display, Write},
ops::Deref,
};
use serde::{de, Deserialize, Serialize};
use static_assertions::assert_impl_all;
use zvariant::Structure;
use crate::{
names::{BusName, InterfaceName, MemberName, UniqueName},
zvariant::{ObjectPath, Str, Type},
Error, MatchRuleBuilder, MessageType, Result,
};
#[derive(Clone, Debug, PartialEq, Eq, Hash, Type)]
#[zvariant(signature = "s")]
pub struct MatchRule<'m> {
pub(crate) msg_type: Option<MessageType>,
pub(crate) sender: Option<BusName<'m>>,
pub(crate) interface: Option<InterfaceName<'m>>,
pub(crate) member: Option<MemberName<'m>>,
pub(crate) path_spec: Option<MatchRulePathSpec<'m>>,
pub(crate) destination: Option<UniqueName<'m>>,
pub(crate) args: Vec<(u8, Str<'m>)>,
pub(crate) arg_paths: Vec<(u8, ObjectPath<'m>)>,
pub(crate) arg0namespace: Option<InterfaceName<'m>>,
pub(crate) arg0ns: Option<Str<'m>>,
}
assert_impl_all!(MatchRule<'_>: Send, Sync, Unpin);
impl<'m> MatchRule<'m> {
pub fn builder() -> MatchRuleBuilder<'m> {
MatchRuleBuilder::new()
}
pub fn sender(&self) -> Option<&BusName<'_>> {
self.sender.as_ref()
}
pub fn msg_type(&self) -> Option<MessageType> {
self.msg_type
}
pub fn interface(&self) -> Option<&InterfaceName<'_>> {
self.interface.as_ref()
}
pub fn member(&self) -> Option<&MemberName<'_>> {
self.member.as_ref()
}
pub fn path_spec(&self) -> Option<&MatchRulePathSpec<'_>> {
self.path_spec.as_ref()
}
pub fn destination(&self) -> Option<&UniqueName<'_>> {
self.destination.as_ref()
}
pub fn args(&self) -> &[(u8, Str<'_>)] {
self.args.as_ref()
}
pub fn arg_paths(&self) -> &[(u8, ObjectPath<'_>)] {
self.arg_paths.as_ref()
}
#[deprecated = "use arg0ns instead"]
pub fn arg0namespace(&self) -> Option<&InterfaceName<'_>> {
self.arg0namespace.as_ref()
}
pub fn arg0ns(&self) -> Option<&Str<'m>> {
self.arg0ns.as_ref()
}
pub fn to_owned(&self) -> MatchRule<'static> {
MatchRule {
msg_type: self.msg_type,
sender: self.sender.as_ref().map(|s| s.to_owned()),
interface: self.interface.as_ref().map(|i| i.to_owned()),
member: self.member.as_ref().map(|m| m.to_owned()),
path_spec: self.path_spec.as_ref().map(|p| p.to_owned()),
destination: self.destination.as_ref().map(|d| d.to_owned()),
args: self.args.iter().map(|(i, s)| (*i, s.to_owned())).collect(),
arg_paths: self
.arg_paths
.iter()
.map(|(i, p)| (*i, p.to_owned()))
.collect(),
arg0namespace: self.arg0namespace.as_ref().map(|a| a.to_owned()),
arg0ns: self.arg0ns.as_ref().map(|a| a.to_owned()),
}
}
pub fn into_owned(self) -> MatchRule<'static> {
MatchRule {
msg_type: self.msg_type,
sender: self.sender.map(|s| s.into_owned()),
interface: self.interface.map(|i| i.into_owned()),
member: self.member.map(|m| m.into_owned()),
path_spec: self.path_spec.map(|p| p.into_owned()),
destination: self.destination.map(|d| d.into_owned()),
args: self
.args
.into_iter()
.map(|(i, s)| (i, s.into_owned()))
.collect(),
arg_paths: self
.arg_paths
.into_iter()
.map(|(i, p)| (i, p.into_owned()))
.collect(),
arg0namespace: self.arg0namespace.map(|a| a.into_owned()),
arg0ns: self.arg0ns.map(|a| a.into_owned()),
}
}
pub fn matches(&self, msg: &zbus::Message) -> Result<bool> {
let hdr = msg.header()?;
if let Some(msg_type) = self.msg_type() {
if msg_type != msg.message_type() {
return Ok(false);
}
}
if let Some(sender) = self.sender() {
match sender {
BusName::Unique(name) if Some(name) != hdr.sender()? => {
return Ok(false);
}
BusName::Unique(_) => (),
BusName::WellKnown(_) => (),
}
}
if let Some(interface) = self.interface() {
match msg.interface().as_ref() {
Some(msg_interface) if interface != msg_interface => return Ok(false),
Some(_) => (),
None => return Ok(false),
}
}
if let Some(member) = self.member() {
match msg.member().as_ref() {
Some(msg_member) if member != msg_member => return Ok(false),
Some(_) => (),
None => return Ok(false),
}
}
if let Some(destination) = self.destination() {
match hdr.destination()? {
Some(BusName::Unique(name)) if destination != name => {
return Ok(false);
}
Some(BusName::Unique(_)) | None => (),
Some(BusName::WellKnown(_)) => (),
};
}
if let Some(path_spec) = self.path_spec() {
let msg_path = match msg.path() {
Some(p) => p,
None => return Ok(false),
};
match path_spec {
MatchRulePathSpec::Path(path) if path != &msg_path => return Ok(false),
MatchRulePathSpec::PathNamespace(path_ns)
if !msg_path.starts_with(path_ns.as_str()) =>
{
return Ok(false);
}
MatchRulePathSpec::Path(_) | MatchRulePathSpec::PathNamespace(_) => (),
}
}
if let Some(arg0_ns) = self.arg0ns() {
if let Ok(arg0) = msg.body_unchecked::<BusName<'_>>() {
match arg0.strip_prefix(arg0_ns.as_str()) {
None => return Ok(false),
Some(s) if !s.is_empty() && !s.starts_with('.') => return Ok(false),
_ => (),
}
} else {
return Ok(false);
}
}
if self.args().is_empty() && self.arg_paths().is_empty() {
return Ok(true);
}
let structure = match msg.body::<Structure<'_>>() {
Ok(s) => s,
Err(_) => return Ok(false),
};
let args = structure.fields();
for (i, arg) in self.args() {
match args.get(*i as usize) {
Some(msg_arg) => match <&str>::try_from(msg_arg) {
Ok(msg_arg) if arg != msg_arg => return Ok(false),
Ok(_) => (),
Err(_) => return Ok(false),
},
None => return Ok(false),
}
}
for (i, path) in self.arg_paths() {
match args.get(*i as usize) {
Some(msg_arg) => match <ObjectPath<'_>>::try_from(msg_arg) {
Ok(msg_arg) if *path != msg_arg => return Ok(false),
Ok(_) => (),
Err(_) => return Ok(false),
},
None => return Ok(false),
}
}
Ok(true)
}
}
impl Display for MatchRule<'_> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let mut first_component = true;
if let Some(msg_type) = self.msg_type() {
let type_str = match msg_type {
MessageType::Error => "error",
MessageType::Invalid => panic!("invalid message type"),
MessageType::MethodCall => "method_call",
MessageType::MethodReturn => "method_return",
MessageType::Signal => "signal",
};
write_match_rule_string_component(f, "type", type_str, &mut first_component)?;
}
if let Some(sender) = self.sender() {
write_match_rule_string_component(f, "sender", sender, &mut first_component)?;
}
if let Some(interface) = self.interface() {
write_match_rule_string_component(f, "interface", interface, &mut first_component)?;
}
if let Some(member) = self.member() {
write_match_rule_string_component(f, "member", member, &mut first_component)?;
}
if let Some(destination) = self.destination() {
write_match_rule_string_component(f, "destination", destination, &mut first_component)?;
}
if let Some(path_spec) = self.path_spec() {
let (key, value) = match path_spec {
MatchRulePathSpec::Path(path) => ("path", path),
MatchRulePathSpec::PathNamespace(ns) => ("path_namespace", ns),
};
write_match_rule_string_component(f, key, value, &mut first_component)?;
}
for (i, arg) in self.args() {
write_comma(f, &mut first_component)?;
write!(f, "arg{i}='{arg}'")?;
}
for (i, arg_path) in self.arg_paths() {
write_comma(f, &mut first_component)?;
write!(f, "arg{i}path='{arg_path}'")?;
}
if let Some(arg0namespace) = self.arg0ns() {
write_comma(f, &mut first_component)?;
write!(f, "arg0namespace='{arg0namespace}'")?;
}
Ok(())
}
}
fn write_match_rule_string_component(
f: &mut std::fmt::Formatter<'_>,
key: &str,
value: &str,
first_component: &mut bool,
) -> std::fmt::Result {
write_comma(f, first_component)?;
f.write_str(key)?;
f.write_str("='")?;
f.write_str(value)?;
f.write_char('\'')?;
Ok(())
}
fn write_comma(f: &mut std::fmt::Formatter<'_>, first_component: &mut bool) -> std::fmt::Result {
if *first_component {
*first_component = false;
} else {
f.write_char(',')?;
}
Ok(())
}
impl<'m> TryFrom<&'m str> for MatchRule<'m> {
type Error = Error;
fn try_from(s: &'m str) -> Result<Self> {
let components = s.split(',');
if components.clone().peekable().peek().is_none() {
return Err(Error::InvalidMatchRule);
}
let mut builder = MatchRule::builder();
for component in components {
let (key, value) = component.split_once('=').ok_or(Error::InvalidMatchRule)?;
if key.is_empty()
|| value.len() < 2
|| !value.starts_with('\'')
|| !value.ends_with('\'')
{
return Err(Error::InvalidMatchRule);
}
let value = &value[1..value.len() - 1];
builder = match key {
"type" => {
let msg_type = match value {
"error" => MessageType::Error,
"method_call" => MessageType::MethodCall,
"method_return" => MessageType::MethodReturn,
"signal" => MessageType::Signal,
_ => return Err(Error::InvalidMatchRule),
};
builder.msg_type(msg_type)
}
"sender" => builder.sender(value)?,
"interface" => builder.interface(value)?,
"member" => builder.member(value)?,
"path" => builder.path(value)?,
"path_namespace" => builder.path_namespace(value)?,
"destination" => builder.destination(value)?,
"arg0namespace" => builder.arg0ns(value)?,
key if key.starts_with("arg") => {
if let Some(trailing_idx) = key.find("path") {
let idx = key[3..trailing_idx]
.parse::<u8>()
.map_err(|_| Error::InvalidMatchRule)?;
builder.arg_path(idx, value)?
} else {
let idx = key[3..]
.parse::<u8>()
.map_err(|_| Error::InvalidMatchRule)?;
builder.arg(idx, value)?
}
}
_ => return Err(Error::InvalidMatchRule),
};
}
Ok(builder.build())
}
}
impl<'de: 'm, 'm> Deserialize<'de> for MatchRule<'m> {
fn deserialize<D>(deserializer: D) -> core::result::Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
let name = <&str>::deserialize(deserializer)?;
Self::try_from(name).map_err(|e| de::Error::custom(e.to_string()))
}
}
impl Serialize for MatchRule<'_> {
fn serialize<S>(&self, serializer: S) -> core::result::Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
serializer.serialize_str(&self.to_string())
}
}
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
pub enum MatchRulePathSpec<'m> {
Path(ObjectPath<'m>),
PathNamespace(ObjectPath<'m>),
}
assert_impl_all!(MatchRulePathSpec<'_>: Send, Sync, Unpin);
impl<'m> MatchRulePathSpec<'m> {
fn to_owned(&self) -> MatchRulePathSpec<'static> {
match self {
MatchRulePathSpec::Path(path) => MatchRulePathSpec::Path(path.to_owned()),
MatchRulePathSpec::PathNamespace(ns) => MatchRulePathSpec::PathNamespace(ns.to_owned()),
}
}
pub fn into_owned(self) -> MatchRulePathSpec<'static> {
match self {
MatchRulePathSpec::Path(path) => MatchRulePathSpec::Path(path.into_owned()),
MatchRulePathSpec::PathNamespace(ns) => {
MatchRulePathSpec::PathNamespace(ns.into_owned())
}
}
}
}
#[derive(Clone, Debug, Hash, PartialEq, Eq, Serialize, Type)]
pub struct OwnedMatchRule(#[serde(borrow)] MatchRule<'static>);
assert_impl_all!(OwnedMatchRule: Send, Sync, Unpin);
impl OwnedMatchRule {
pub fn into_inner(self) -> MatchRule<'static> {
self.0
}
pub fn inner(&self) -> &MatchRule<'static> {
&self.0
}
}
impl Deref for OwnedMatchRule {
type Target = MatchRule<'static>;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl From<OwnedMatchRule> for MatchRule<'static> {
fn from(o: OwnedMatchRule) -> Self {
o.into_inner()
}
}
impl<'unowned, 'owned: 'unowned> From<&'owned OwnedMatchRule> for MatchRule<'unowned> {
fn from(rule: &'owned OwnedMatchRule) -> Self {
rule.inner().clone()
}
}
impl From<MatchRule<'_>> for OwnedMatchRule {
fn from(rule: MatchRule<'_>) -> Self {
OwnedMatchRule(rule.into_owned())
}
}
impl TryFrom<&'_ str> for OwnedMatchRule {
type Error = Error;
fn try_from(value: &str) -> Result<Self> {
Ok(Self::from(MatchRule::try_from(value)?))
}
}
impl<'de> Deserialize<'de> for OwnedMatchRule {
fn deserialize<D>(deserializer: D) -> std::result::Result<Self, D::Error>
where
D: de::Deserializer<'de>,
{
String::deserialize(deserializer)
.and_then(|r| {
MatchRule::try_from(r.as_str())
.map(|r| r.to_owned())
.map_err(|e| de::Error::custom(e.to_string()))
})
.map(Self)
}
}
impl PartialEq<MatchRule<'_>> for OwnedMatchRule {
fn eq(&self, other: &MatchRule<'_>) -> bool {
self.0 == *other
}
}