// Copyright 2012 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. //! A type that represents one of two alternatives #[allow(missing_doc)]; use option::{Some, None}; use option; use clone::Clone; use container::Container; use cmp::Eq; use iter::{Iterator, FilterMap}; use result::Result; use result; use str::StrSlice; use vec; use vec::{OwnedVector, ImmutableVector}; /// `Either` is a type that represents one of two alternatives #[deriving(Clone, Eq, IterBytes)] pub enum Either { Left(L), Right(R) } impl Either { /// Applies a function based on the given either value /// /// If `value` is `Left(L)` then `f_left` is applied to its contents, if /// `value` is `Right(R)` then `f_right` is applied to its contents, and the /// result is returned. #[inline] pub fn either(&self, f_left: &fn(&L) -> T, f_right: &fn(&R) -> T) -> T { match *self { Left(ref l) => f_left(l), Right(ref r) => f_right(r) } } /// Flips between left and right of a given `Either` #[inline] pub fn flip(self) -> Either { match self { Right(r) => Left(r), Left(l) => Right(l) } } /// Checks whether the given value is a `Left` #[inline] pub fn is_left(&self) -> bool { match *self { Left(_) => true, _ => false } } /// Checks whether the given value is a `Right` #[inline] pub fn is_right(&self) -> bool { match *self { Right(_) => true, _ => false } } /// Retrieves the value from a `Left`. /// Fails with a specified reason if the `Either` is `Right`. #[inline] pub fn expect_left(self, reason: &str) -> L { match self { Left(x) => x, Right(_) => fail!(reason.to_owned()) } } /// Retrieves the value from a `Left`. Fails if the `Either` is `Right`. #[inline] pub fn unwrap_left(self) -> L { self.expect_left("called Either::unwrap_left()` on `Right` value") } /// Retrieves the value from a `Right`. /// Fails with a specified reason if the `Either` is `Left`. #[inline] pub fn expect_right(self, reason: &str) -> R { match self { Right(x) => x, Left(_) => fail!(reason.to_owned()) } } /// Retrieves the value from a `Right`. Fails if the `Either` is `Left`. #[inline] pub fn unwrap_right(self) -> R { self.expect_right("called Either::unwrap_right()` on `Left` value") } } /// A generic trait for converting a value to a `Either` pub trait ToEither { /// Convert to the `either` type fn to_either(&self) -> Either; } /// A generic trait for converting a value to a `Either` pub trait IntoEither { /// Convert to the `either` type fn into_either(self) -> Either; } /// A generic trait for converting a value to a `Either` pub trait AsEither { /// Convert to the `either` type fn as_either<'a>(&'a self) -> Either<&'a L, &'a R>; } impl option::ToOption for Either { #[inline] fn to_option(&self)-> option::Option { match *self { Left(_) => None, Right(ref r) => Some(r.clone()), } } } impl option::IntoOption for Either { #[inline] fn into_option(self)-> option::Option { match self { Left(_) => None, Right(r) => Some(r), } } } impl option::AsOption for Either { #[inline] fn as_option<'a>(&'a self) -> option::Option<&'a R> { match *self { Left(_) => None, Right(ref r) => Some(r), } } } impl result::ToResult for Either { #[inline] fn to_result(&self)-> result::Result { match *self { Left(ref l) => result::Err(l.clone()), Right(ref r) => result::Ok(r.clone()), } } } impl result::IntoResult for Either { #[inline] fn into_result(self)-> result::Result { match self { Left(l) => result::Err(l), Right(r) => result::Ok(r), } } } impl result::AsResult for Either { #[inline] fn as_result<'a>(&'a self) -> result::Result<&'a R, &'a L> { match *self { Left(ref l) => result::Err(l), Right(ref r) => result::Ok(r), } } } impl ToEither for Either { fn to_either(&self) -> Either { self.clone() } } impl IntoEither for Either { fn into_either(self) -> Either { self } } impl AsEither for Either { fn as_either<'a>(&'a self) -> Either<&'a L, &'a R> { match *self { Left(ref l) => Left(l), Right(ref r) => Right(r), } } } /// An iterator yielding the `Left` values of its source pub type Lefts = FilterMap<'static, Either, L, Iter>; /// An iterator yielding the `Right` values of its source pub type Rights = FilterMap<'static, Either, R, Iter>; /// Extracts all the left values pub fn lefts>>(eithers: Iter) -> Lefts { do eithers.filter_map |elt| { match elt { Left(x) => Some(x), _ => None, } } } /// Extracts all the right values pub fn rights>>(eithers: Iter) -> Rights { do eithers.filter_map |elt| { match elt { Right(x) => Some(x), _ => None, } } } // FIXME: #8228 Replaceable by an external iterator? /// Extracts from a vector of either all the left values and right values /// /// Returns a structure containing a vector of left values and a vector of /// right values. pub fn partition(eithers: ~[Either]) -> (~[L], ~[R]) { let n_lefts = eithers.iter().count(|elt| elt.is_left()); let mut lefts = vec::with_capacity(n_lefts); let mut rights = vec::with_capacity(eithers.len() - n_lefts); for elt in eithers.move_iter() { match elt { Left(l) => lefts.push(l), Right(r) => rights.push(r) } } return (lefts, rights); } #[cfg(test)] mod tests { use super::*; use option::{IntoOption, ToOption, AsOption}; use option; use result::{IntoResult, ToResult, AsResult}; use result; #[test] fn test_either_left() { let val = Left(10); fn f_left(x: &int) -> bool { *x == 10 } fn f_right(_x: &uint) -> bool { false } assert!(val.either(f_left, f_right)); } #[test] fn test_either_right() { let val = Right(10u); fn f_left(_x: &int) -> bool { false } fn f_right(x: &uint) -> bool { *x == 10u } assert!(val.either(f_left, f_right)); } #[test] fn test_lefts() { let input = ~[Left(10), Right(11), Left(12), Right(13), Left(14)]; let result = lefts(input.move_iter()).to_owned_vec(); assert_eq!(result, ~[10, 12, 14]); } #[test] fn test_lefts_none() { let input: ~[Either] = ~[Right(10), Right(10)]; let result = lefts(input.move_iter()).to_owned_vec(); assert_eq!(result.len(), 0u); } #[test] fn test_lefts_empty() { let input: ~[Either] = ~[]; let result = lefts(input.move_iter()).to_owned_vec(); assert_eq!(result.len(), 0u); } #[test] fn test_rights() { let input = ~[Left(10), Right(11), Left(12), Right(13), Left(14)]; let result = rights(input.move_iter()).to_owned_vec(); assert_eq!(result, ~[11, 13]); } #[test] fn test_rights_none() { let input: ~[Either] = ~[Left(10), Left(10)]; let result = rights(input.move_iter()).to_owned_vec(); assert_eq!(result.len(), 0u); } #[test] fn test_rights_empty() { let input: ~[Either] = ~[]; let result = rights(input.move_iter()).to_owned_vec(); assert_eq!(result.len(), 0u); } #[test] fn test_partition() { let input = ~[Left(10), Right(11), Left(12), Right(13), Left(14)]; let (lefts, rights) = partition(input); assert_eq!(lefts[0], 10); assert_eq!(lefts[1], 12); assert_eq!(lefts[2], 14); assert_eq!(rights[0], 11); assert_eq!(rights[1], 13); } #[test] fn test_partition_no_lefts() { let input: ~[Either] = ~[Right(10), Right(11)]; let (lefts, rights) = partition(input); assert_eq!(lefts.len(), 0u); assert_eq!(rights.len(), 2u); } #[test] fn test_partition_no_rights() { let input: ~[Either] = ~[Left(10), Left(11)]; let (lefts, rights) = partition(input); assert_eq!(lefts.len(), 2u); assert_eq!(rights.len(), 0u); } #[test] fn test_partition_empty() { let input: ~[Either] = ~[]; let (lefts, rights) = partition(input); assert_eq!(lefts.len(), 0u); assert_eq!(rights.len(), 0u); } #[test] pub fn test_to_option() { let right: Either = Right(100); let left: Either = Left(404); assert_eq!(right.to_option(), option::Some(100)); assert_eq!(left.to_option(), option::None); } #[test] pub fn test_into_option() { let right: Either = Right(100); let left: Either = Left(404); assert_eq!(right.into_option(), option::Some(100)); assert_eq!(left.into_option(), option::None); } #[test] pub fn test_as_option() { let right: Either = Right(100); let left: Either = Left(404); assert_eq!(right.as_option().unwrap(), &100); assert_eq!(left.as_option(), option::None); } #[test] pub fn test_to_result() { let right: Either = Right(100); let left: Either = Left(404); assert_eq!(right.to_result(), result::Ok(100)); assert_eq!(left.to_result(), result::Err(404)); } #[test] pub fn test_into_result() { let right: Either = Right(100); let left: Either = Left(404); assert_eq!(right.into_result(), result::Ok(100)); assert_eq!(left.into_result(), result::Err(404)); } #[test] pub fn test_as_result() { let right: Either = Right(100); let left: Either = Left(404); let x = 100; assert_eq!(right.as_result(), result::Ok(&x)); let x = 404; assert_eq!(left.as_result(), result::Err(&x)); } #[test] pub fn test_to_either() { let right: Either = Right(100); let left: Either = Left(404); assert_eq!(right.to_either(), Right(100)); assert_eq!(left.to_either(), Left(404)); } #[test] pub fn test_into_either() { let right: Either = Right(100); let left: Either = Left(404); assert_eq!(right.into_either(), Right(100)); assert_eq!(left.into_either(), Left(404)); } #[test] pub fn test_as_either() { let right: Either = Right(100); let left: Either = Left(404); assert_eq!(right.as_either().unwrap_right(), &100); assert_eq!(left.as_either().unwrap_left(), &404); } }