// NB: transitionary, de-mode-ing. #[forbid(deprecated_mode)]; #[forbid(deprecated_pattern)]; //! A type that represents one of two alternatives use cmp::Eq; use result::Result; /// The either type pub enum Either { Left(T), Right(U) } pub fn either(f_left: fn(&T) -> V, f_right: fn(&U) -> V, value: &Either) -> V { /*! * 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. */ match *value { Left(ref l) => f_left(l), Right(ref r) => f_right(r) } } pub fn lefts(eithers: &[Either]) -> ~[T] { //! Extracts from a vector of either all the left values do vec::build_sized(eithers.len()) |push| { for vec::each(eithers) |elt| { match *elt { Left(ref l) => { push(*l); } _ => { /* fallthrough */ } } } } } pub fn rights(eithers: &[Either]) -> ~[U] { //! Extracts from a vector of either all the right values do vec::build_sized(eithers.len()) |push| { for vec::each(eithers) |elt| { match *elt { Right(ref r) => { push(*r); } _ => { /* fallthrough */ } } } } } // XXX bad copies. take arg by val pub fn partition(eithers: &[Either]) -> {lefts: ~[T], rights: ~[U]} { /*! * 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. */ let mut lefts: ~[T] = ~[]; let mut rights: ~[U] = ~[]; for vec::each(eithers) |elt| { match *elt { Left(copy l) => lefts.push(l), Right(copy r) => rights.push(r) } } return {lefts: move lefts, rights: move rights}; } // XXX bad copies pub pure fn flip(eith: &Either) -> Either { //! Flips between left and right of a given either match *eith { Right(copy r) => Left(r), Left(copy l) => Right(l) } } // XXX bad copies pub pure fn to_result(eith: &Either) -> Result { /*! * 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 */ match *eith { Right(copy r) => result::Ok(r), Left(copy l) => result::Err(l) } } pub pure fn is_left(eith: &Either) -> bool { //! Checks whether the given value is a left match *eith { Left(_) => true, _ => false } } pub pure fn is_right(eith: &Either) -> bool { //! Checks whether the given value is a right match *eith { Right(_) => true, _ => false } } // tjc: fix the next two after a snapshot pub pure fn unwrap_left(eith: Either) -> T { //! Retrieves the value in the left branch. Fails if the either is Right. match move eith { Left(move x) => move x, Right(_) => fail ~"either::unwrap_left Right" } } pub pure fn unwrap_right(eith: Either) -> U { //! Retrieves the value in the right branch. Fails if the either is Left. match move eith { Right(move x) => move x, Left(_) => fail ~"either::unwrap_right Left" } } impl Either : Eq { #[cfg(stage0)] pure fn eq(other: &Either) -> bool { match self { Left(ref a) => { match (*other) { Left(ref b) => (*a).eq(b), Right(_) => false } } Right(ref a) => { match (*other) { Left(_) => false, Right(ref b) => (*a).eq(b) } } } } #[cfg(stage1)] #[cfg(stage2)] pure fn eq(&self, other: &Either) -> bool { match (*self) { Left(ref a) => { match (*other) { Left(ref b) => (*a).eq(b), Right(_) => false } } Right(ref a) => { match (*other) { Left(_) => false, Right(ref b) => (*a).eq(b) } } } } #[cfg(stage0)] pure fn ne(other: &Either) -> bool { !self.eq(other) } #[cfg(stage1)] #[cfg(stage2)] pure fn ne(&self, other: &Either) -> bool { !(*self).eq(other) } } #[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 (result == ~[10, 12, 14]); } #[test] fn test_lefts_none() { let input: ~[Either] = ~[Right(10), Right(10)]; let result = lefts(input); assert (vec::len(result) == 0u); } #[test] fn test_lefts_empty() { let input: ~[Either] = ~[]; let result = lefts(input); assert (vec::len(result) == 0u); } #[test] fn test_rights() { let input = ~[Left(10), Right(11), Left(12), Right(13), Left(14)]; let result = rights(input); assert (result == ~[11, 13]); } #[test] fn test_rights_none() { let input: ~[Either] = ~[Left(10), Left(10)]; let result = rights(input); assert (vec::len(result) == 0u); } #[test] fn test_rights_empty() { let input: ~[Either] = ~[]; let result = rights(input); assert (vec::len(result) == 0u); } #[test] fn test_partition() { let input = ~[Left(10), Right(11), Left(12), Right(13), Left(14)]; let result = partition(input); assert (result.lefts[0] == 10); assert (result.lefts[1] == 12); assert (result.lefts[2] == 14); assert (result.rights[0] == 11); assert (result.rights[1] == 13); } #[test] fn test_partition_no_lefts() { let input: ~[Either] = ~[Right(10), Right(11)]; let result = partition(input); assert (vec::len(result.lefts) == 0u); assert (vec::len(result.rights) == 2u); } #[test] fn test_partition_no_rights() { let input: ~[Either] = ~[Left(10), Left(11)]; let result = partition(input); assert (vec::len(result.lefts) == 2u); assert (vec::len(result.rights) == 0u); } #[test] fn test_partition_empty() { let input: ~[Either] = ~[]; let result = partition(input); assert (vec::len(result.lefts) == 0u); assert (vec::len(result.rights) == 0u); } // // Local Variables: // mode: rust // fill-column: 78; // indent-tabs-mode: nil // c-basic-offset: 4 // buffer-file-coding-system: utf-8-unix // End: //