refactor(headers): Use u16 based newtype for quality value
Using floating point numbers is problematic because comparison is inexact. They also take more space than integral numbers in this case. Add `FromPrimitve`, `ToPrimitive` and `Default` traits to quality newtype. Closes: #330 BREAKING_CHANGE: Replace f32 quality values in quality items with a Quality(u16) newtype. Valid values are from 0 to 1000.
This commit is contained in:
		| @@ -33,7 +33,8 @@ impl_list_header!(Accept, | |||||||
| mod tests { | mod tests { | ||||||
|     use mime::*; |     use mime::*; | ||||||
|  |  | ||||||
|     use header::{Header, QualityItem, qitem}; |     use header::{Header, Quality, QualityItem, qitem}; | ||||||
|  |      | ||||||
|     use super::Accept; |     use super::Accept; | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
| @@ -49,7 +50,7 @@ mod tests { | |||||||
|     fn test_parse_header_with_quality() { |     fn test_parse_header_with_quality() { | ||||||
|         let a: Accept = Header::parse_header([b"text/plain; charset=utf-8; q=0.5".to_vec()].as_slice()).unwrap(); |         let a: Accept = Header::parse_header([b"text/plain; charset=utf-8; q=0.5".to_vec()].as_slice()).unwrap(); | ||||||
|         let b = Accept(vec![ |         let b = Accept(vec![ | ||||||
|             QualityItem::new(Mime(TopLevel::Text, SubLevel::Plain, vec![(Attr::Charset, Value::Utf8)]), 0.5f32), |             QualityItem::new(Mime(TopLevel::Text, SubLevel::Plain, vec![(Attr::Charset, Value::Utf8)]), Quality(500)), | ||||||
|         ]); |         ]); | ||||||
|         assert_eq!(a, b); |         assert_eq!(a, b); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -13,7 +13,7 @@ impl_list_header!(AcceptEncoding, | |||||||
|  |  | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests { | mod tests { | ||||||
|     use header::{Encoding, Header, QualityItem}; |     use header::{Encoding, Header, qitem, Quality, QualityItem}; | ||||||
|  |  | ||||||
|     use super::*; |     use super::*; | ||||||
|  |  | ||||||
| @@ -21,8 +21,8 @@ mod tests { | |||||||
|     fn test_parse_header() { |     fn test_parse_header() { | ||||||
|         let a: AcceptEncoding = Header::parse_header([b"gzip;q=1.0, identity; q=0.5".to_vec()].as_slice()).unwrap(); |         let a: AcceptEncoding = Header::parse_header([b"gzip;q=1.0, identity; q=0.5".to_vec()].as_slice()).unwrap(); | ||||||
|         let b = AcceptEncoding(vec![ |         let b = AcceptEncoding(vec![ | ||||||
|             QualityItem{item: Encoding::Gzip, quality: 1f32}, |             qitem(Encoding::Gzip), | ||||||
|             QualityItem{item: Encoding::Identity, quality: 0.5f32}, |             QualityItem::new(Encoding::Identity, Quality(500)), | ||||||
|         ]); |         ]); | ||||||
|         assert_eq!(a, b); |         assert_eq!(a, b); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -46,41 +46,45 @@ impl_list_header!(AcceptLanguage, | |||||||
|                   "Accept-Language", |                   "Accept-Language", | ||||||
|                   Vec<QualityItem<Language>>); |                   Vec<QualityItem<Language>>); | ||||||
|  |  | ||||||
| bench_header!(bench, AcceptLanguage, | #[cfg(test)] | ||||||
|               { vec![b"en-us;q=1.0, en;q=0.5, fr".to_vec()] }); | mod tests { | ||||||
|  |     use header::{Header, qitem, Quality, QualityItem}; | ||||||
|  |  | ||||||
| #[test] |     use super::*; | ||||||
| fn test_parse_header() { |  | ||||||
|     let a: AcceptLanguage = header::Header::parse_header( |     #[test] | ||||||
|  |     fn test_parse_header() { | ||||||
|  |         let a: AcceptLanguage = Header::parse_header( | ||||||
|             [b"en-us;q=1.0, en;q=0.5, fr".to_vec()].as_slice()).unwrap(); |             [b"en-us;q=1.0, en;q=0.5, fr".to_vec()].as_slice()).unwrap(); | ||||||
|         let b = AcceptLanguage(vec![ |         let b = AcceptLanguage(vec![ | ||||||
|         QualityItem { item: Language{primary: "en".to_string(), |             qitem(Language{primary: "en".to_string(), sub: Some("us".to_string())}), | ||||||
|                                    sub: Some("us".to_string())}, |             QualityItem::new(Language{primary: "en".to_string(), sub: None}, | ||||||
|                       quality: 1f32 }, |                              Quality(500)), | ||||||
|         QualityItem { item: Language{primary: "en".to_string(), sub: None}, |             qitem(Language{primary: "fr".to_string(), sub: None}), | ||||||
|                       quality: 0.5f32 }, |  | ||||||
|         QualityItem { item: Language{primary: "fr".to_string(), sub: None}, |  | ||||||
|                       quality: 1f32 }, |  | ||||||
|         ]); |         ]); | ||||||
|         assert_eq!(format!("{}", a), format!("{}", b)); |         assert_eq!(format!("{}", a), format!("{}", b)); | ||||||
|         assert_eq!(a, b); |         assert_eq!(a, b); | ||||||
| } |     } | ||||||
|  |  | ||||||
| #[test] |     #[test] | ||||||
| fn test_display() { |     fn test_display() { | ||||||
|         assert_eq!("en".to_string(), |         assert_eq!("en".to_string(), | ||||||
|                    format!("{}", Language{primary: "en".to_string(), |                    format!("{}", Language{primary: "en".to_string(), | ||||||
|                                           sub: None})); |                                           sub: None})); | ||||||
|         assert_eq!("en-us".to_string(), |         assert_eq!("en-us".to_string(), | ||||||
|                    format!("{}", Language{primary: "en".to_string(), |                    format!("{}", Language{primary: "en".to_string(), | ||||||
|                                           sub: Some("us".to_string())})); |                                           sub: Some("us".to_string())})); | ||||||
| } |     } | ||||||
|  |  | ||||||
| #[test] |     #[test] | ||||||
| fn test_from_str() { |     fn test_from_str() { | ||||||
|         assert_eq!(Language { primary: "en".to_string(), sub: None }, |         assert_eq!(Language { primary: "en".to_string(), sub: None }, | ||||||
|                    "en".parse().unwrap()); |                    "en".parse().unwrap()); | ||||||
|         assert_eq!(Language { primary: "en".to_string(), |         assert_eq!(Language { primary: "en".to_string(), | ||||||
|                               sub: Some("us".to_string()) }, |                               sub: Some("us".to_string()) }, | ||||||
|                    "en-us".parse().unwrap()); |                    "en-us".parse().unwrap()); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | bench_header!(bench, AcceptLanguage, | ||||||
|  |               { vec![b"en-us;q=1.0, en;q=0.5, fr".to_vec()] }); | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ use unicase::UniCase; | |||||||
| use self::cell::OptCell; | use self::cell::OptCell; | ||||||
| use {http, HttpResult, HttpError}; | use {http, HttpResult, HttpError}; | ||||||
|  |  | ||||||
| pub use self::shared::{Encoding, EntityTag, QualityItem, qitem}; | pub use self::shared::{Encoding, EntityTag, Quality, QualityItem, qitem}; | ||||||
| pub use self::common::*; | pub use self::common::*; | ||||||
|  |  | ||||||
| mod cell; | mod cell; | ||||||
| @@ -540,7 +540,7 @@ mod tests { | |||||||
|     use mime::TopLevel::Text; |     use mime::TopLevel::Text; | ||||||
|     use mime::SubLevel::Plain; |     use mime::SubLevel::Plain; | ||||||
|     use super::{Headers, Header, HeaderFormat, ContentLength, ContentType, |     use super::{Headers, Header, HeaderFormat, ContentLength, ContentType, | ||||||
|                 Accept, Host, QualityItem}; |                 Accept, Host, qitem}; | ||||||
|  |  | ||||||
|     use test::Bencher; |     use test::Bencher; | ||||||
|  |  | ||||||
| @@ -562,7 +562,7 @@ mod tests { | |||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
|     fn test_accept() { |     fn test_accept() { | ||||||
|         let text_plain = QualityItem{item: Mime(Text, Plain, vec![]), quality: 1f32}; |         let text_plain = qitem(Mime(Text, Plain, vec![])); | ||||||
|         let application_vendor = "application/vnd.github.v3.full+json; q=0.5".parse().unwrap(); |         let application_vendor = "application/vnd.github.v3.full+json; q=0.5".parse().unwrap(); | ||||||
|  |  | ||||||
|         let accept = Header::parse_header([b"text/plain".to_vec()].as_slice()); |         let accept = Header::parse_header([b"text/plain".to_vec()].as_slice()); | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| pub use self::encoding::Encoding; | pub use self::encoding::Encoding; | ||||||
| pub use self::entity::EntityTag; | pub use self::entity::EntityTag; | ||||||
| pub use self::quality_item::{QualityItem, qitem}; | pub use self::quality_item::{Quality, QualityItem, qitem}; | ||||||
|  |  | ||||||
| mod encoding; | mod encoding; | ||||||
| mod entity; | mod entity; | ||||||
|   | |||||||
| @@ -3,10 +3,79 @@ | |||||||
| //! [RFC7231 Section 5.3.1](https://tools.ietf.org/html/rfc7231#section-5.3.1) | //! [RFC7231 Section 5.3.1](https://tools.ietf.org/html/rfc7231#section-5.3.1) | ||||||
| //! gives more information on quality values in HTTP header fields. | //! gives more information on quality values in HTTP header fields. | ||||||
|  |  | ||||||
| use std::fmt; |  | ||||||
| use std::str; |  | ||||||
| use std::cmp; | use std::cmp; | ||||||
| #[cfg(test)] use super::encoding::*; | use std::default::Default; | ||||||
|  | use std::fmt; | ||||||
|  | use std::num::{FromPrimitive, ToPrimitive}; | ||||||
|  | use std::str; | ||||||
|  |  | ||||||
|  | /// Represents a quality used in quality values. | ||||||
|  | /// | ||||||
|  | /// `Quality` should only be created using the `FromPrimitve` trait methods `from_f32` and | ||||||
|  | /// `from_f64`, they take a value between 0.0 and 1.0. To create a quality with the value 1.0, the | ||||||
|  | /// default you can use `let q: Quality = Default::default()`. | ||||||
|  | /// | ||||||
|  | /// # Implementation notes | ||||||
|  | /// The quality value is defined as a number between 0 and 1 with three decimal places. This means | ||||||
|  | /// there are 1000 possible values. Since floating point numbers are not exact and the smallest | ||||||
|  | /// floating point data type (`f32`) consumes four bytes, hyper uses an `u16` value to store the | ||||||
|  | /// quality internally. For performance reasons you may set quality directly to a value between | ||||||
|  | /// 0 and 1000 e.g. `Quality(532)` matches the quality `q=0.532`. | ||||||
|  | #[derive(Clone, Debug, Eq, Ord, PartialEq, PartialOrd)] | ||||||
|  | pub struct Quality(pub u16); | ||||||
|  |  | ||||||
|  | impl fmt::Display for Quality { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|  |         if self.0 == 1000 { | ||||||
|  |             write!(f, "") | ||||||
|  |         } else { | ||||||
|  |             write!(f, "; q=0.{}", format!("{:03}", self.0).trim_right_matches('0')) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl FromPrimitive for Quality { | ||||||
|  |     fn from_i64(n: i64) -> Option<Quality> { | ||||||
|  |         match n >= 0 { | ||||||
|  |             true => FromPrimitive::from_u64(n as u64), | ||||||
|  |             false => None, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn from_u64(n: u64) -> Option<Quality> { | ||||||
|  |         match n <= 1000 { | ||||||
|  |             true => Some(Quality(n as u16)), | ||||||
|  |             false => None, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn from_f64(n: f64) -> Option<Quality> { | ||||||
|  |         match n >= 0f64 && n <= 1f64 { | ||||||
|  |             true => Some(Quality((n * 1000f64) as u16)), | ||||||
|  |             false => None, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl ToPrimitive for Quality { | ||||||
|  |     fn to_i64(&self) -> Option<i64> { | ||||||
|  |         Some(self.0 as i64) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn to_u64(&self) -> Option<u64> { | ||||||
|  |         Some(self.0 as u64) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn to_f64(&self) -> Option<f64> { | ||||||
|  |         Some((self.0 as f64) / 1000f64) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl Default for Quality { | ||||||
|  |     fn default() -> Quality { | ||||||
|  |         Quality(1000) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| /// Represents an item with a quality value as defined in | /// Represents an item with a quality value as defined in | ||||||
| /// [RFC7231](https://tools.ietf.org/html/rfc7231#section-5.3.1). | /// [RFC7231](https://tools.ietf.org/html/rfc7231#section-5.3.1). | ||||||
| @@ -15,14 +84,14 @@ pub struct QualityItem<T> { | |||||||
|     /// The actual contents of the field. |     /// The actual contents of the field. | ||||||
|     pub item: T, |     pub item: T, | ||||||
|     /// The quality (client or server preference) for the value. |     /// The quality (client or server preference) for the value. | ||||||
|     pub quality: f32, |     pub quality: Quality, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<T> QualityItem<T> { | impl<T> QualityItem<T> { | ||||||
|     /// Creates a new `QualityItem` from an item and a quality. |     /// Creates a new `QualityItem` from an item and a quality. | ||||||
|     /// The item can be of any type. |     /// The item can be of any type. | ||||||
|     /// The quality should be a value in the range [0, 1]. |     /// The quality should be a value in the range [0, 1]. | ||||||
|     pub fn new(item: T, quality: f32) -> QualityItem<T> { |     pub fn new(item: T, quality: Quality) -> QualityItem<T> { | ||||||
|         QualityItem{item: item, quality: quality} |         QualityItem{item: item, quality: quality} | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -35,12 +104,7 @@ impl<T: PartialEq> cmp::PartialOrd for QualityItem<T> { | |||||||
|  |  | ||||||
| impl<T: fmt::Display> fmt::Display for QualityItem<T> { | impl<T: fmt::Display> fmt::Display for QualityItem<T> { | ||||||
|     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|         if self.quality == 1.0 { |         write!(f, "{}{}", self.item, format!("{}", self.quality)) | ||||||
|             write!(f, "{}", self.item) |  | ||||||
|         } else { |  | ||||||
|             write!(f, "{}; q={}", self.item, |  | ||||||
|                    format!("{:.3}", self.quality).trim_right_matches(&['0', '.'][..])) |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -59,8 +123,7 @@ impl<T: str::FromStr> str::FromStr for QualityItem<T> { | |||||||
|                 if q_part.len() > 5 { |                 if q_part.len() > 5 { | ||||||
|                     return Err(()); |                     return Err(()); | ||||||
|                 } |                 } | ||||||
|                 let x: Result<f32, _> = q_part.parse(); |                 match q_part.parse::<f32>() { | ||||||
|                 match x { |  | ||||||
|                     Ok(q_value) => { |                     Ok(q_value) => { | ||||||
|                         if 0f32 <= q_value && q_value <= 1f32 { |                         if 0f32 <= q_value && q_value <= 1f32 { | ||||||
|                             quality = q_value; |                             quality = q_value; | ||||||
| @@ -73,11 +136,8 @@ impl<T: str::FromStr> str::FromStr for QualityItem<T> { | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         let x: Result<T, _> = raw_item.parse(); |         match raw_item.parse::<T>() { | ||||||
|         match x { |             Ok(item) => Ok(QualityItem::new(item, FromPrimitive::from_f32(quality).unwrap())), | ||||||
|             Ok(item) => { |  | ||||||
|                 Ok(QualityItem{ item: item, quality: quality, }) |  | ||||||
|             }, |  | ||||||
|             Err(_) => return Err(()), |             Err(_) => return Err(()), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -86,58 +146,81 @@ impl<T: str::FromStr> str::FromStr for QualityItem<T> { | |||||||
| /// Convinience function to wrap a value in a `QualityItem` | /// Convinience function to wrap a value in a `QualityItem` | ||||||
| /// Sets `q` to the default 1.0 | /// Sets `q` to the default 1.0 | ||||||
| pub fn qitem<T>(item: T) -> QualityItem<T> { | pub fn qitem<T>(item: T) -> QualityItem<T> { | ||||||
|     QualityItem::new(item, 1.0) |     QualityItem::new(item, Default::default()) | ||||||
| } | } | ||||||
|  |  | ||||||
| #[test] | #[cfg(test)] | ||||||
| fn test_quality_item_show1() { | mod tests { | ||||||
|  |     use std::num::FromPrimitive; | ||||||
|  |  | ||||||
|  |     use super::*; | ||||||
|  |     use super::super::encoding::*; | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_quality_item_show1() { | ||||||
|         let x = qitem(Chunked); |         let x = qitem(Chunked); | ||||||
|         assert_eq!(format!("{}", x), "chunked"); |         assert_eq!(format!("{}", x), "chunked"); | ||||||
| } |     } | ||||||
| #[test] |     #[test] | ||||||
| fn test_quality_item_show2() { |     fn test_quality_item_show2() { | ||||||
|     let x = QualityItem::new(Chunked, 0.001); |         let x = QualityItem::new(Chunked, Quality(1)); | ||||||
|         assert_eq!(format!("{}", x), "chunked; q=0.001"); |         assert_eq!(format!("{}", x), "chunked; q=0.001"); | ||||||
| } |     } | ||||||
| #[test] |     #[test] | ||||||
| fn test_quality_item_show3() { |     fn test_quality_item_show3() { | ||||||
|         // Custom value |         // Custom value | ||||||
|         let x = QualityItem{ |         let x = QualityItem{ | ||||||
|             item: EncodingExt("identity".to_string()), |             item: EncodingExt("identity".to_string()), | ||||||
|         quality: 0.5f32, |             quality: Quality(500), | ||||||
|         }; |         }; | ||||||
|         assert_eq!(format!("{}", x), "identity; q=0.5"); |         assert_eq!(format!("{}", x), "identity; q=0.5"); | ||||||
| } |     } | ||||||
|  |  | ||||||
| #[test] |     #[test] | ||||||
| fn test_quality_item_from_str1() { |     fn test_quality_item_from_str1() { | ||||||
|         let x: Result<QualityItem<Encoding>, ()> = "chunked".parse(); |         let x: Result<QualityItem<Encoding>, ()> = "chunked".parse(); | ||||||
|     assert_eq!(x.unwrap(), QualityItem{ item: Chunked, quality: 1f32, }); |         assert_eq!(x.unwrap(), QualityItem{ item: Chunked, quality: Quality(1000), }); | ||||||
| } |     } | ||||||
| #[test] |     #[test] | ||||||
| fn test_quality_item_from_str2() { |     fn test_quality_item_from_str2() { | ||||||
|         let x: Result<QualityItem<Encoding>, ()> = "chunked; q=1".parse(); |         let x: Result<QualityItem<Encoding>, ()> = "chunked; q=1".parse(); | ||||||
|     assert_eq!(x.unwrap(), QualityItem{ item: Chunked, quality: 1f32, }); |         assert_eq!(x.unwrap(), QualityItem{ item: Chunked, quality: Quality(1000), }); | ||||||
| } |     } | ||||||
| #[test] |     #[test] | ||||||
| fn test_quality_item_from_str3() { |     fn test_quality_item_from_str3() { | ||||||
|         let x: Result<QualityItem<Encoding>, ()> = "gzip; q=0.5".parse(); |         let x: Result<QualityItem<Encoding>, ()> = "gzip; q=0.5".parse(); | ||||||
|     assert_eq!(x.unwrap(), QualityItem{ item: Gzip, quality: 0.5f32, }); |         assert_eq!(x.unwrap(), QualityItem{ item: Gzip, quality: Quality(500), }); | ||||||
| } |     } | ||||||
| #[test] |     #[test] | ||||||
| fn test_quality_item_from_str4() { |     fn test_quality_item_from_str4() { | ||||||
|         let x: Result<QualityItem<Encoding>, ()> = "gzip; q=0.273".parse(); |         let x: Result<QualityItem<Encoding>, ()> = "gzip; q=0.273".parse(); | ||||||
|     assert_eq!(x.unwrap(), QualityItem{ item: Gzip, quality: 0.273f32, }); |         assert_eq!(x.unwrap(), QualityItem{ item: Gzip, quality: Quality(273), }); | ||||||
| } |     } | ||||||
| #[test] |     #[test] | ||||||
| fn test_quality_item_from_str5() { |     fn test_quality_item_from_str5() { | ||||||
|         let x: Result<QualityItem<Encoding>, ()> = "gzip; q=0.2739999".parse(); |         let x: Result<QualityItem<Encoding>, ()> = "gzip; q=0.2739999".parse(); | ||||||
|         assert_eq!(x, Err(())); |         assert_eq!(x, Err(())); | ||||||
| } |     } | ||||||
| #[test] |     #[test] | ||||||
| fn test_quality_item_ordering() { |     fn test_quality_item_ordering() { | ||||||
|         let x: QualityItem<Encoding> = "gzip; q=0.5".parse().ok().unwrap(); |         let x: QualityItem<Encoding> = "gzip; q=0.5".parse().ok().unwrap(); | ||||||
|         let y: QualityItem<Encoding> = "gzip; q=0.273".parse().ok().unwrap(); |         let y: QualityItem<Encoding> = "gzip; q=0.273".parse().ok().unwrap(); | ||||||
|         let comparision_result: bool = x.gt(&y); |         let comparision_result: bool = x.gt(&y); | ||||||
|         assert_eq!(comparision_result, true) |         assert_eq!(comparision_result, true) | ||||||
|  |     } | ||||||
|  |     #[test] | ||||||
|  |     fn test_quality() { | ||||||
|  |         assert_eq!(Some(Quality(421)), FromPrimitive::from_f64(0.421f64)); | ||||||
|  |         assert_eq!(Some(Quality(421)), FromPrimitive::from_f32(0.421f32)); | ||||||
|  |         assert_eq!(Some(Quality(421)), FromPrimitive::from_i16(421i16)); | ||||||
|  |         assert_eq!(Some(Quality(421)), FromPrimitive::from_i32(421i32)); | ||||||
|  |         assert_eq!(Some(Quality(421)), FromPrimitive::from_i64(421i64)); | ||||||
|  |         assert_eq!(Some(Quality(421)), FromPrimitive::from_u16(421u16)); | ||||||
|  |         assert_eq!(Some(Quality(421)), FromPrimitive::from_u32(421u32)); | ||||||
|  |         assert_eq!(Some(Quality(421)), FromPrimitive::from_u64(421u64)); | ||||||
|  |  | ||||||
|  |         assert_eq!(None::<Quality>, FromPrimitive::from_i16(-5i16)); | ||||||
|  |         assert_eq!(None::<Quality>, FromPrimitive::from_i32(5000i32)); | ||||||
|  |         assert_eq!(None::<Quality>, FromPrimitive::from_f32(2.5f32)); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user