// 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 clone::Clone; use container::Container; use cmp::Eq; use iterator::IteratorUtil; use result::Result; use result; use str::StrSlice; use vec; use vec::{OwnedVector, ImmutableVector}; /// The either type #[deriving(Clone, Eq)] pub enum Either { Left(T), Right(U) } /// Applies a function based on the given either value /// /// If `value` is left(T) then `f_left` is applied to its contents, if /// `value` is right(U) then `f_right` is applied to its contents, and the /// result is returned. #[inline] pub fn either(f_left: &fn(&T) -> V, f_right: &fn(&U) -> V, value: &Either) -> V { match *value { Left(ref l) => f_left(l), Right(ref r) => f_right(r) } } /// Extracts from a vector of either all the left values pub fn lefts(eithers: &[Either]) -> ~[T] { do vec::build_sized(eithers.len()) |push| { for eithers.iter().advance |elt| { match *elt { Left(ref l) => { push((*l).clone()); } _ => { /* fallthrough */ } } } } } /// Extracts from a vector of either all the right values pub fn rights(eithers: &[Either]) -> ~[U] { do vec::build_sized(eithers.len()) |push| { for eithers.iter().advance |elt| { match *elt { Right(ref r) => { push((*r).clone()); } _ => { /* fallthrough */ } } } } } /// 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]) -> (~[T], ~[U]) { let mut lefts: ~[T] = ~[]; let mut rights: ~[U] = ~[]; for eithers.consume_iter().advance |elt| { match elt { Left(l) => lefts.push(l), Right(r) => rights.push(r) } } return (lefts, rights); } /// Flips between left and right of a given either #[inline] pub fn flip(eith: Either) -> Either { match eith { Right(r) => Left(r), Left(l) => Right(l) } } /// Converts either::t to a result::t /// /// Converts an `either` type to a `result` type, making the "right" choice /// an ok result, and the "left" choice a fail #[inline] pub fn to_result(eith: Either) -> Result { match eith { Right(r) => result::Ok(r), Left(l) => result::Err(l) } } /// Checks whether the given value is a left #[inline] pub fn is_left(eith: &Either) -> bool { match *eith { Left(_) => true, _ => false } } /// Checks whether the given value is a right #[inline] pub fn is_right(eith: &Either) -> bool { match *eith { Right(_) => true, _ => false } } /// Retrieves the value in the left branch. /// Fails with a specified reason if the either is Right. #[inline] pub fn expect_left(eith: Either, reason: &str) -> T { match eith { Left(x) => x, Right(_) => fail!(reason.to_owned()) } } /// Retrieves the value in the left branch. Fails if the either is Right. #[inline] pub fn unwrap_left(eith: Either) -> T { expect_left(eith, "either::unwrap_left Right") } /// Retrieves the value in the right branch. /// Fails with a specified reason if the either is Left. #[inline] pub fn expect_right(eith: Either, reason: &str) -> U { match eith { Right(x) => x, Left(_) => fail!(reason.to_owned()) } } /// Retrieves the value in the right branch. Fails if the either is Left. pub fn unwrap_right(eith: Either) -> U { expect_right(eith, "either::unwrap_right Left") } impl Either { #[inline] pub fn either(&self, f_left: &fn(&T) -> V, f_right: &fn(&U) -> V) -> V { either(f_left, f_right, self) } #[inline] pub fn flip(self) -> Either { flip(self) } #[inline] pub fn to_result(self) -> Result { to_result(self) } #[inline] pub fn is_left(&self) -> bool { is_left(self) } #[inline] pub fn is_right(&self) -> bool { is_right(self) } #[inline] pub fn expect_left(self, reason: &str) -> T { expect_left(self, reason) } #[inline] pub fn unwrap_left(self) -> T { unwrap_left(self) } #[inline] pub fn expect_right(self, reason: &str) -> U { expect_right(self, reason) } #[inline] pub fn unwrap_right(self) -> U { unwrap_right(self) } } #[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!((either(f_left, f_right, &val))); } #[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!((either(f_left, f_right, &val))); } #[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); }