Lines
97.56 %
Functions
21 %
Branches
30.78 %
//! The `Parse` trait for CSS properties, and utilities for parsers.
use cssparser::{Parser, ParserInput, Token};
use markup5ever::QualName;
use std::str;
use crate::error::*;
/// Trait to parse values using `cssparser::Parser`.
pub trait Parse: Sized {
/// Parses a value out of the `parser`.
///
/// All value types should implement this for composability.
fn parse<'i>(parser: &mut Parser<'i, '_>) -> Result<Self, ParseError<'i>>;
/// Convenience function to parse a value out of a `&str`.
/// This is useful mostly for tests which want to avoid creating a
/// `cssparser::Parser` by hand. Property types do not need to reimplement this.
fn parse_str(s: &str) -> Result<Self, ParseError<'_>> {
let mut input = ParserInput::new(s);
let mut parser = Parser::new(&mut input);
let res = Self::parse(&mut parser)?;
parser.expect_exhausted()?;
Ok(res)
}
/// Consumes a comma if it exists, or does nothing.
pub fn optional_comma(parser: &mut Parser<'_, '_>) {
let _ = parser.try_parse(|p| p.expect_comma());
/// Parses an `f32` and ensures that it is not an infinity or NaN.
pub fn finite_f32(n: f32) -> Result<f32, ValueErrorKind> {
if n.is_finite() {
Ok(n)
} else {
Err(ValueErrorKind::Value("expected finite number".to_string()))
pub trait ParseValue<T: Parse> {
/// Parses a `value` string into a type `T`.
fn parse(&self, value: &str) -> Result<T, ElementError>;
impl<T: Parse> ParseValue<T> for QualName {
fn parse(&self, value: &str) -> Result<T, ElementError> {
let mut input = ParserInput::new(value);
T::parse(&mut parser).attribute(self.clone())
impl<T: Parse> Parse for Option<T> {
fn parse<'i>(parser: &mut Parser<'i, '_>) -> Result<Self, ParseError<'i>> {
T::parse(parser).map(Some)
impl Parse for f64 {
let loc = parser.current_source_location();
let n = parser.expect_number()?;
Ok(f64::from(n))
Err(loc.new_custom_error(ValueErrorKind::value_error("expected finite number")))
/// Non-Negative number
#[derive(Debug, Copy, Clone, PartialEq)]
pub struct NonNegative(pub f64);
impl Parse for NonNegative {
let n = Parse::parse(parser)?;
if n >= 0.0 {
Ok(NonNegative(n))
Err(loc.new_custom_error(ValueErrorKind::value_error("expected non negative number")))
/// CSS number-optional-number
/// SVG1.1: <https://www.w3.org/TR/SVG11/types.html#DataTypeNumberOptionalNumber>
pub struct NumberOptionalNumber<T: Parse>(pub T, pub T);
impl<T: Parse + Copy> Parse for NumberOptionalNumber<T> {
let x = Parse::parse(parser)?;
if !parser.is_exhausted() {
optional_comma(parser);
let y = Parse::parse(parser)?;
Ok(NumberOptionalNumber(x, y))
Ok(NumberOptionalNumber(x, x))
/// CSS number-percentage
/// CSS Values and Units 3: <https://www.w3.org/TR/css3-values/#typedef-number-percentage>
pub struct NumberOrPercentage {
pub value: f64,
impl Parse for NumberOrPercentage {
let value = match parser.next()? {
Token::Number { value, .. } => Ok(*value),
Token::Percentage { unit_value, .. } => Ok(*unit_value),
tok => Err(loc.new_unexpected_token_error(tok.clone())),
}?;
let v = finite_f32(value).map_err(|e| parser.new_custom_error(e))?;
Ok(NumberOrPercentage {
value: f64::from(v),
})
impl Parse for i32 {
/// CSS integer
/// SVG1.1: <https://www.w3.org/TR/SVG11/types.html#DataTypeInteger>
Ok(parser.expect_integer()?)
impl Parse for u32 {
let n = parser.expect_integer()?;
if n >= 0 {
Ok(n as u32)
Err(loc.new_custom_error(ValueErrorKind::value_error("expected unsigned number")))
/// Number lists with bounds for the required and maximum number of items.
#[derive(Clone, Debug, PartialEq)]
pub struct NumberList<const REQUIRED: usize, const MAX: usize>(pub Vec<f64>);
impl<const REQUIRED: usize, const MAX: usize> Parse for NumberList<REQUIRED, MAX> {
let mut v = Vec::<f64>::with_capacity(MAX);
for i in 0..MAX {
if i != 0 {
v.push(f64::parse(parser)?);
if parser.is_exhausted() {
break;
if REQUIRED > 0 && v.len() < REQUIRED {
Err(loc.new_custom_error(ValueErrorKind::value_error("expected more numbers")))
Ok(NumberList(v))
/// Parses a list of identifiers from a `cssparser::Parser`
/// # Example
/// ```
/// # use cssparser::{ParserInput, Parser};
/// # use rsvg::parse_identifiers;
/// # fn main() -> Result<(), cssparser::BasicParseError<'static>> {
/// # let mut input = ParserInput::new("true");
/// # let mut parser = Parser::new(&mut input);
/// let my_boolean = parse_identifiers!(
/// parser,
/// "true" => true,
/// "false" => false,
/// )?;
/// # Ok(())
/// # }
#[doc(hidden)]
#[macro_export]
macro_rules! parse_identifiers {
($parser:expr,
$($str:expr => $val:expr,)+) => {
{
let loc = $parser.current_source_location();
let token = $parser.next()?;
match token {
$(cssparser::Token::Ident(ref cow) if cow.eq_ignore_ascii_case($str) => Ok($val),)+
_ => Err(loc.new_basic_unexpected_token_error(token.clone()))
};
/// CSS Custom identifier.
/// CSS Values and Units 4: <https://www.w3.org/TR/css-values-4/#custom-idents>
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct CustomIdent(pub String);
impl Parse for CustomIdent {
let token = parser.next()?;
// CSS-wide keywords and "default" are errors here
// https://www.w3.org/TR/css-values-4/#css-wide-keywords
Token::Ident(ref cow) => {
for s in &["initial", "inherit", "unset", "default"] {
if cow.eq_ignore_ascii_case(s) {
return Err(loc.new_basic_unexpected_token_error(token.clone()).into());
Ok(CustomIdent(cow.as_ref().to_string()))
_ => Err(loc.new_basic_unexpected_token_error(token.clone()).into()),
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn parses_number_optional_number() {
assert_eq!(
NumberOptionalNumber::parse_str("1, 2").unwrap(),
NumberOptionalNumber(1.0, 2.0)
);
NumberOptionalNumber::parse_str("1 2").unwrap(),
NumberOptionalNumber::parse_str("1").unwrap(),
NumberOptionalNumber(1.0, 1.0)
NumberOptionalNumber::parse_str("-1, -2").unwrap(),
NumberOptionalNumber(-1.0, -2.0)
NumberOptionalNumber::parse_str("-1 -2").unwrap(),
NumberOptionalNumber::parse_str("-1").unwrap(),
NumberOptionalNumber(-1.0, -1.0)
fn invalid_number_optional_number() {
assert!(NumberOptionalNumber::<f64>::parse_str("").is_err());
assert!(NumberOptionalNumber::<f64>::parse_str("1x").is_err());
assert!(NumberOptionalNumber::<f64>::parse_str("x1").is_err());
assert!(NumberOptionalNumber::<f64>::parse_str("1 x").is_err());
assert!(NumberOptionalNumber::<f64>::parse_str("1 , x").is_err());
assert!(NumberOptionalNumber::<f64>::parse_str("1 , 2x").is_err());
assert!(NumberOptionalNumber::<f64>::parse_str("1 2 x").is_err());
fn parses_integer() {
assert_eq!(i32::parse_str("0").unwrap(), 0);
assert_eq!(i32::parse_str("1").unwrap(), 1);
assert_eq!(i32::parse_str("-1").unwrap(), -1);
assert_eq!(u32::parse_str("0").unwrap(), 0);
assert_eq!(u32::parse_str("1").unwrap(), 1);
fn invalid_integer() {
assert!(i32::parse_str("").is_err());
assert!(i32::parse_str("1x").is_err());
assert!(i32::parse_str("1.5").is_err());
assert!(u32::parse_str("").is_err());
assert!(u32::parse_str("1x").is_err());
assert!(u32::parse_str("1.5").is_err());
assert!(u32::parse_str("-1").is_err());
fn parses_integer_optional_integer() {
NumberOptionalNumber(1, 2)
NumberOptionalNumber(1, 1)
NumberOptionalNumber(-1, -2)
NumberOptionalNumber(-1, -1)
fn invalid_integer_optional_integer() {
assert!(NumberOptionalNumber::<i32>::parse_str("").is_err());
assert!(NumberOptionalNumber::<i32>::parse_str("1x").is_err());
assert!(NumberOptionalNumber::<i32>::parse_str("x1").is_err());
assert!(NumberOptionalNumber::<i32>::parse_str("1 x").is_err());
assert!(NumberOptionalNumber::<i32>::parse_str("1 , x").is_err());
assert!(NumberOptionalNumber::<i32>::parse_str("1 , 2x").is_err());
assert!(NumberOptionalNumber::<i32>::parse_str("1 2 x").is_err());
assert!(NumberOptionalNumber::<i32>::parse_str("1.5").is_err());
assert!(NumberOptionalNumber::<i32>::parse_str("1 2.5").is_err());
assert!(NumberOptionalNumber::<i32>::parse_str("1, 2.5").is_err());
fn parses_number_list() {
NumberList::<1, 1>::parse_str("5").unwrap(),
NumberList(vec![5.0])
NumberList::<4, 4>::parse_str("1 2 3 4").unwrap(),
NumberList(vec![1.0, 2.0, 3.0, 4.0])
NumberList::<0, 5>::parse_str("1 2 3 4 5").unwrap(),
NumberList(vec![1.0, 2.0, 3.0, 4.0, 5.0])
NumberList::<0, 5>::parse_str("1 2 3").unwrap(),
NumberList(vec![1.0, 2.0, 3.0])
fn errors_on_invalid_number_list() {
// empty
assert!(NumberList::<1, 1>::parse_str("").is_err());
assert!(NumberList::<0, 1>::parse_str("").is_err());
// garbage
assert!(NumberList::<1, 1>::parse_str("foo").is_err());
assert!(NumberList::<2, 2>::parse_str("1foo").is_err());
assert!(NumberList::<2, 2>::parse_str("1 foo").is_err());
assert!(NumberList::<2, 2>::parse_str("1 foo 2").is_err());
assert!(NumberList::<2, 2>::parse_str("1,foo").is_err());
// too many
assert!(NumberList::<1, 1>::parse_str("1 2").is_err());
// extra token
assert!(NumberList::<1, 1>::parse_str("1,").is_err());
assert!(NumberList::<0, 1>::parse_str("1,").is_err());
// too few
assert!(NumberList::<2, 2>::parse_str("1").is_err());
assert!(NumberList::<3, 3>::parse_str("1 2").is_err());
fn parses_custom_ident() {
CustomIdent::parse_str("hello").unwrap(),
CustomIdent("hello".to_string())
fn invalid_custom_ident_yields_error() {
assert!(CustomIdent::parse_str("initial").is_err());
assert!(CustomIdent::parse_str("inherit").is_err());
assert!(CustomIdent::parse_str("unset").is_err());
assert!(CustomIdent::parse_str("default").is_err());
assert!(CustomIdent::parse_str("").is_err());