1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
use crate::value::MAXIMUM_NAME_LENGTH;
use std::cmp::{Eq, PartialEq};
use std::convert::{From, TryFrom};
use std::fmt::{Display, Formatter, Result as FmtResult};
use thiserror::Error;

enum Input {
    /// [A-Z][a-z]_
    AlphabeticAndUnderscore,
    /// [0-9]
    Digit,
}

impl TryFrom<u8> for Input {
    type Error = MemberError;

    fn try_from(c: u8) -> Result<Self, Self::Error> {
        if c.is_ascii_alphabetic() || c == b'_' {
            Ok(Input::AlphabeticAndUnderscore)
        } else if c.is_ascii_digit() {
            Ok(Input::Digit)
        } else {
            Err(MemberError::InvalidChar(c))
        }
    }
}

/// Check if the given bytes is a valid [member name].
///
/// [member name]: https://dbus.freedesktop.org/doc/dbus-specification.html#message-protocol-names-member
fn check(member: &[u8]) -> Result<(), MemberError> {
    let member_len = member.len();
    if MAXIMUM_NAME_LENGTH < member_len {
        return Err(MemberError::ExceedMaximum(member_len));
    }

    let mut member_iter = member.iter();
    match member_iter.next() {
        Some(c) => {
            if let Input::Digit = Input::try_from(*c)? {
                return Err(MemberError::BeginDigit);
            }
        }
        None => return Err(MemberError::Empty),
    }

    for c in member_iter {
        Input::try_from(*c)?;
    }
    Ok(())
}

/// This represents a [member name].
///
/// [member name]: https://dbus.freedesktop.org/doc/dbus-specification.html#message-protocol-names-member
#[derive(Debug, Clone, PartialOrd, PartialEq, Ord, Eq, Hash)]
pub struct Member(String);

/// An enum representing all errors, which can occur during the handling of a [`Member`].
#[derive(Debug, PartialEq, Eq, Error)]
pub enum MemberError {
    #[error("Member must not be empty")]
    Empty,
    #[error("Member must not begin with a digit")]
    BeginDigit,
    #[error("Member must not exceed the maximum length: {MAXIMUM_NAME_LENGTH} < {0}")]
    ExceedMaximum(usize),
    #[error("Member contians an invalid char: {0}")]
    InvalidChar(u8),
}

impl From<Member> for String {
    fn from(member: Member) -> Self {
        member.0
    }
}

impl TryFrom<String> for Member {
    type Error = MemberError;

    fn try_from(member: String) -> Result<Self, Self::Error> {
        check(member.as_bytes())?;
        Ok(Member(member))
    }
}

impl TryFrom<&str> for Member {
    type Error = MemberError;

    fn try_from(member: &str) -> Result<Self, Self::Error> {
        check(member.as_bytes())?;
        Ok(Member(member.to_owned()))
    }
}

impl TryFrom<&[u8]> for Member {
    type Error = MemberError;

    fn try_from(member: &[u8]) -> Result<Self, Self::Error> {
        check(member)?;
        let member = member.to_vec();
        //  The vector only contains valid UTF-8 (ASCII) characters because it was already
        //  checked by the `check` function above
        unsafe { Ok(Member(String::from_utf8_unchecked(member))) }
    }
}

impl Display for Member {
    fn fmt(&self, f: &mut Formatter<'_>) -> FmtResult {
        write!(f, "{}", self.0)
    }
}

impl AsRef<str> for Member {
    fn as_ref(&self) -> &str {
        &self.0
    }
}

impl PartialEq<str> for Member {
    fn eq(&self, other: &str) -> bool {
        self.0 == other
    }
}