summaryrefslogtreecommitdiffstats
path: root/third_party/rust/intl_pluralrules/src/lib.rs
diff options
context:
space:
mode:
Diffstat (limited to 'third_party/rust/intl_pluralrules/src/lib.rs')
-rw-r--r--third_party/rust/intl_pluralrules/src/lib.rs225
1 files changed, 225 insertions, 0 deletions
diff --git a/third_party/rust/intl_pluralrules/src/lib.rs b/third_party/rust/intl_pluralrules/src/lib.rs
new file mode 100644
index 0000000000..84a2fd6353
--- /dev/null
+++ b/third_party/rust/intl_pluralrules/src/lib.rs
@@ -0,0 +1,225 @@
+//! A crate for generating plural rule operands from numberical input.
+//!
+//! This crate generates plural operands according to the specifications outlined at [Unicode's website](https://unicode.org/reports/tr35/tr35-numbers.html#Operands).
+//!
+//! Input is supported for int, float, and &str.
+//!
+//! # Examples
+//!
+//! Plural rules example for Polish
+//!
+//! ```
+//! use intl_pluralrules::{PluralRules, PluralRuleType, PluralCategory};
+//! use unic_langid::LanguageIdentifier;
+//!
+//! let langid: LanguageIdentifier = "pl".parse().expect("Parsing failed.");
+//!
+//! assert!(PluralRules::get_locales(PluralRuleType::CARDINAL).contains(&langid));
+//!
+//! let pr = PluralRules::create(langid.clone(), PluralRuleType::CARDINAL).unwrap();
+//! assert_eq!(pr.select(1), Ok(PluralCategory::ONE));
+//! assert_eq!(pr.select("3"), Ok(PluralCategory::FEW));
+//! assert_eq!(pr.select(12), Ok(PluralCategory::MANY));
+//! assert_eq!(pr.select("5.0"), Ok(PluralCategory::OTHER));
+//!
+//! assert_eq!(pr.get_locale(), &langid);
+//! ```
+
+/// A public AST module for plural rule representations.
+pub mod operands;
+#[cfg(not(tarpaulin_include))]
+mod rules;
+
+use std::convert::TryInto;
+
+use unic_langid::LanguageIdentifier;
+
+use crate::operands::PluralOperands;
+use crate::rules::*;
+
+/// A public enum for handling the plural category.
+/// Each plural category will vary, depending on the language that is being used and whether that language has that plural category.
+#[derive(Debug, Eq, PartialEq)]
+pub enum PluralCategory {
+ ZERO,
+ ONE,
+ TWO,
+ FEW,
+ MANY,
+ OTHER,
+}
+
+/// A public enum for handling plural type.
+#[derive(Copy, Clone, Hash, PartialEq, Eq)]
+pub enum PluralRuleType {
+ /// Ordinal numbers express position or rank in a sequence. [More about oridinal numbers](https://en.wikipedia.org/wiki/Ordinal_number_(linguistics))
+ ORDINAL,
+ /// Cardinal numbers are natural numbers. [More about cardinal numbers](https://en.wikipedia.org/wiki/Cardinal_number)
+ CARDINAL,
+}
+
+// pub use rules::PluralRuleType;
+/// CLDR_VERSION is the version of CLDR extracted from the file used to generate rules.rs.
+pub use crate::rules::CLDR_VERSION;
+
+/// The main structure for selecting plural rules.
+///
+/// # Examples
+///
+/// ```
+/// use intl_pluralrules::{PluralRules, PluralRuleType, PluralCategory};
+/// use unic_langid::LanguageIdentifier;
+///
+/// let langid: LanguageIdentifier = "naq".parse().expect("Parsing failed.");
+/// let pr_naq = PluralRules::create(langid, PluralRuleType::CARDINAL).unwrap();
+/// assert_eq!(pr_naq.select(1), Ok(PluralCategory::ONE));
+/// assert_eq!(pr_naq.select("2"), Ok(PluralCategory::TWO));
+/// assert_eq!(pr_naq.select(5.0), Ok(PluralCategory::OTHER));
+/// ```
+#[derive(Clone)]
+pub struct PluralRules {
+ locale: LanguageIdentifier,
+ function: PluralRule,
+}
+
+impl PluralRules {
+ /// Returns an instance of PluralRules.
+ ///
+ /// # Examples
+ /// ```
+ /// use intl_pluralrules::{PluralRules, PluralRuleType, PluralCategory};
+ /// use unic_langid::LanguageIdentifier;
+ ///
+ /// let langid: LanguageIdentifier = "naq".parse().expect("Parsing failed.");
+ /// let pr_naq = PluralRules::create(langid, PluralRuleType::CARDINAL);
+ /// assert_eq!(pr_naq.is_ok(), !pr_naq.is_err());
+ ///
+ /// let langid: LanguageIdentifier = "xx".parse().expect("Parsing failed.");
+ /// let pr_broken = PluralRules::create(langid, PluralRuleType::CARDINAL);
+ /// assert_eq!(pr_broken.is_err(), !pr_broken.is_ok());
+ /// ```
+ pub fn create<L: Into<LanguageIdentifier>>(
+ langid: L,
+ prt: PluralRuleType,
+ ) -> Result<Self, &'static str> {
+ let langid = langid.into();
+ let returned_rule = match prt {
+ PluralRuleType::CARDINAL => {
+ let idx = rules::PRS_CARDINAL.binary_search_by_key(&&langid, |(l, _)| l);
+ idx.map(|idx| rules::PRS_CARDINAL[idx].1)
+ }
+ PluralRuleType::ORDINAL => {
+ let idx = rules::PRS_ORDINAL.binary_search_by_key(&&langid, |(l, _)| l);
+ idx.map(|idx| rules::PRS_ORDINAL[idx].1)
+ }
+ };
+ match returned_rule {
+ Ok(returned_rule) => Ok(Self {
+ locale: langid,
+ function: returned_rule,
+ }),
+ Err(_) => Err("unknown locale"),
+ }
+ }
+
+ /// Returns a result of the plural category for the given input.
+ ///
+ /// If the input is not numeric.
+ ///
+ /// # Examples
+ /// ```
+ /// use intl_pluralrules::{PluralRules, PluralRuleType, PluralCategory};
+ /// use unic_langid::LanguageIdentifier;
+ ///
+ /// let langid: LanguageIdentifier = "naq".parse().expect("Parsing failed.");
+ /// let pr_naq = PluralRules::create(langid, PluralRuleType::CARDINAL).unwrap();
+ /// assert_eq!(pr_naq.select(1), Ok(PluralCategory::ONE));
+ /// assert_eq!(pr_naq.select(2), Ok(PluralCategory::TWO));
+ /// assert_eq!(pr_naq.select(5), Ok(PluralCategory::OTHER));
+ /// ```
+ pub fn select<N: TryInto<PluralOperands>>(
+ &self,
+ number: N,
+ ) -> Result<PluralCategory, &'static str> {
+ let ops = number.try_into();
+ let pr = self.function;
+ match ops {
+ Ok(ops) => Ok(pr(&ops)),
+ Err(_) => Err("Argument can not be parsed to operands."),
+ }
+ }
+
+ /// Returns a list of the available locales.
+ ///
+ /// # Examples
+ /// ```
+ /// use intl_pluralrules::{PluralRules, PluralRuleType};
+ ///
+ /// assert_eq!(
+ /// PluralRules::get_locales(PluralRuleType::CARDINAL).is_empty(),
+ /// false
+ /// );
+ /// ```
+ pub fn get_locales(prt: PluralRuleType) -> Vec<LanguageIdentifier> {
+ let prs = match prt {
+ PluralRuleType::CARDINAL => rules::PRS_CARDINAL,
+ PluralRuleType::ORDINAL => rules::PRS_ORDINAL,
+ };
+ prs.iter().map(|(l, _)| l.clone()).collect()
+ }
+
+ /// Returns the locale name for this PluralRule instance.
+ ///
+ /// # Examples
+ /// ```
+ /// use intl_pluralrules::{PluralRules, PluralRuleType};
+ /// use unic_langid::LanguageIdentifier;
+ ///
+ /// let langid: LanguageIdentifier = "naq".parse().expect("Parsing failed.");
+ /// let pr_naq = PluralRules::create(langid.clone(), PluralRuleType::CARDINAL).unwrap();
+ /// assert_eq!(pr_naq.get_locale(), &langid);
+ /// ```
+ pub fn get_locale(&self) -> &LanguageIdentifier {
+ &self.locale
+ }
+}
+
+#[cfg(test)]
+mod tests {
+ use super::{PluralCategory, PluralRuleType, PluralRules, CLDR_VERSION};
+ use unic_langid::LanguageIdentifier;
+
+ #[test]
+ fn cardinals_test() {
+ let langid: LanguageIdentifier = "naq".parse().expect("Parsing failed.");
+ let pr_naq = PluralRules::create(langid, PluralRuleType::CARDINAL).unwrap();
+ assert_eq!(pr_naq.select(1), Ok(PluralCategory::ONE));
+ assert_eq!(pr_naq.select(2), Ok(PluralCategory::TWO));
+ assert_eq!(pr_naq.select(5), Ok(PluralCategory::OTHER));
+
+ let langid: LanguageIdentifier = "xx".parse().expect("Parsing failed.");
+ let pr_broken = PluralRules::create(langid, PluralRuleType::CARDINAL);
+ assert_eq!(pr_broken.is_err(), !pr_broken.is_ok());
+ }
+
+ #[test]
+ fn ordinals_rules() {
+ let langid: LanguageIdentifier = "uk".parse().expect("Parsing failed.");
+ let pr_naq = PluralRules::create(langid, PluralRuleType::ORDINAL).unwrap();
+ assert_eq!(pr_naq.select(33), Ok(PluralCategory::FEW));
+ assert_eq!(pr_naq.select(113), Ok(PluralCategory::OTHER));
+ }
+
+ #[test]
+ fn version_test() {
+ assert_eq!(CLDR_VERSION, 37);
+ }
+
+ #[test]
+ fn locale_test() {
+ assert_eq!(
+ PluralRules::get_locales(PluralRuleType::CARDINAL).is_empty(),
+ false
+ );
+ }
+}