// 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 clone::Clone; use container::Container; use cmp::Eq; use iterator::Iterator; 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) } } /// Converts a `Either` to a `Result` /// /// Converts an `Either` type to a `Result` type, making the "right" choice /// an `Ok` result, and the "left" choice a `Err` #[inline] pub fn to_result(self) -> Result { match self { Right(r) => result::Ok(r), Left(l) => result::Err(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") } } // FIXME: #8228 Replaceable by an external iterator? /// Extracts from a vector of either all the left values pub fn lefts(eithers: &[Either]) -> ~[L] { do vec::build_sized(eithers.len()) |push| { for elt in eithers.iter() { match *elt { Left(ref l) => { push((*l).clone()); } _ => { /* fallthrough */ } } } } } // FIXME: #8228 Replaceable by an external iterator? /// Extracts from a vector of either all the right values pub fn rights(eithers: &[Either]) -> ~[R] { do vec::build_sized(eithers.len()) |push| { for elt in eithers.iter() { match *elt { Right(ref r) => { push((*r).clone()); } _ => { /* fallthrough */ } } } } } // 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 mut lefts: ~[L] = ~[]; let mut rights: ~[R] = ~[]; 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::*; #[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); assert_eq!(result, ~[10, 12, 14]); } #[test] fn test_lefts_none() { let input: ~[Either] = ~[Right(10), Right(10)]; let result = lefts(input); assert_eq!(result.len(), 0u); } #[test] fn test_lefts_empty() { let input: ~[Either] = ~[]; let result = lefts(input); 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); assert_eq!(result, ~[11, 13]); } #[test] fn test_rights_none() { let input: ~[Either] = ~[Left(10), Left(10)]; let result = rights(input); assert_eq!(result.len(), 0u); } #[test] fn test_rights_empty() { let input: ~[Either] = ~[]; let result = rights(input); 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); } }