2012-07-04 21:53:12 +00:00
|
|
|
/*!
|
|
|
|
* Operations on the ubiquitous `option` type.
|
|
|
|
*
|
|
|
|
* Type `option` represents an optional value.
|
|
|
|
*
|
2012-08-20 19:23:37 +00:00
|
|
|
* Every `Option<T>` value can either be `Some(T)` or `none`. Where in other
|
2012-07-04 21:53:12 +00:00
|
|
|
* languages you might use a nullable type, in Rust you would use an option
|
|
|
|
* type.
|
|
|
|
*/
|
|
|
|
|
2012-09-22 02:37:57 +00:00
|
|
|
// NB: transitionary, de-mode-ing.
|
2012-09-26 00:02:22 +00:00
|
|
|
#[warn(deprecated_mode)];
|
2012-09-22 02:37:57 +00:00
|
|
|
#[forbid(deprecated_pattern)];
|
|
|
|
|
2012-08-28 00:24:15 +00:00
|
|
|
use cmp::Eq;
|
|
|
|
|
2012-07-04 21:53:12 +00:00
|
|
|
/// The option type
|
2012-08-20 19:23:37 +00:00
|
|
|
enum Option<T> {
|
|
|
|
None,
|
|
|
|
Some(T),
|
2011-12-14 00:25:51 +00:00
|
|
|
}
|
|
|
|
|
2012-09-22 02:37:57 +00:00
|
|
|
pure fn get<T: Copy>(opt: &Option<T>) -> T {
|
2012-07-04 21:53:12 +00:00
|
|
|
/*!
|
|
|
|
* Gets the value out of an option
|
|
|
|
*
|
|
|
|
* # Failure
|
|
|
|
*
|
|
|
|
* Fails if the value equals `none`
|
|
|
|
*/
|
2011-12-14 00:25:51 +00:00
|
|
|
|
2012-09-22 02:37:57 +00:00
|
|
|
match *opt {
|
2012-09-28 20:00:07 +00:00
|
|
|
Some(copy x) => return x,
|
2012-08-20 19:23:37 +00:00
|
|
|
None => fail ~"option::get none"
|
2012-08-04 02:59:04 +00:00
|
|
|
}
|
2011-12-14 00:25:51 +00:00
|
|
|
}
|
|
|
|
|
2012-08-20 19:23:37 +00:00
|
|
|
pure fn get_ref<T>(opt: &r/Option<T>) -> &r/T {
|
2012-08-17 01:06:05 +00:00
|
|
|
/*!
|
|
|
|
* Gets an immutable reference to the value inside an option.
|
|
|
|
*
|
|
|
|
* # Failure
|
|
|
|
*
|
|
|
|
* Fails if the value equals `none`
|
|
|
|
*/
|
|
|
|
match *opt {
|
2012-08-20 19:23:37 +00:00
|
|
|
Some(ref x) => x,
|
|
|
|
None => fail ~"option::get_ref none"
|
2012-08-17 01:06:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-22 02:37:57 +00:00
|
|
|
pure fn expect<T: Copy>(opt: &Option<T>, +reason: ~str) -> T {
|
2012-08-24 23:02:47 +00:00
|
|
|
/*!
|
|
|
|
* Gets the value out of an option, printing a specified message on
|
|
|
|
* failure
|
|
|
|
*
|
|
|
|
* # Failure
|
|
|
|
*
|
|
|
|
* Fails if the value equals `none`
|
|
|
|
*/
|
2012-09-28 20:00:07 +00:00
|
|
|
match *opt { Some(copy x) => x, None => fail reason }
|
2012-07-04 00:28:44 +00:00
|
|
|
}
|
|
|
|
|
2012-09-26 23:27:12 +00:00
|
|
|
pure fn map<T, U>(opt: &Option<T>, f: fn(x: &T) -> U) -> Option<U> {
|
2012-08-14 23:17:59 +00:00
|
|
|
//! Maps a `some` value by reference from one type to another
|
|
|
|
|
2012-08-20 19:23:37 +00:00
|
|
|
match *opt { Some(ref x) => Some(f(x)), None => None }
|
2012-08-14 23:17:59 +00:00
|
|
|
}
|
|
|
|
|
2012-09-23 11:39:27 +00:00
|
|
|
pure fn map_consume<T, U>(+opt: Option<T>, f: fn(+v: T) -> U) -> Option<U> {
|
2012-07-17 22:24:51 +00:00
|
|
|
/*!
|
|
|
|
* As `map`, but consumes the option and gives `f` ownership to avoid
|
|
|
|
* copying.
|
|
|
|
*/
|
2012-09-12 00:17:54 +00:00
|
|
|
if opt.is_some() { Some(f(option::unwrap(move opt))) } else { None }
|
2012-07-17 22:24:51 +00:00
|
|
|
}
|
|
|
|
|
2012-09-28 00:01:28 +00:00
|
|
|
pure fn chain<T, U>(+opt: Option<T>, f: fn(+t: T) -> Option<U>) -> Option<U> {
|
2012-07-04 21:53:12 +00:00
|
|
|
/*!
|
|
|
|
* Update an optional value by optionally running its content through a
|
|
|
|
* function that returns an option.
|
|
|
|
*/
|
2012-03-07 03:09:32 +00:00
|
|
|
|
2012-09-28 00:01:28 +00:00
|
|
|
// XXX write with move match
|
|
|
|
if opt.is_some() {
|
|
|
|
f(unwrap(opt))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2012-02-22 12:18:15 +00:00
|
|
|
}
|
|
|
|
|
2012-08-20 19:23:37 +00:00
|
|
|
pure fn chain_ref<T, U>(opt: &Option<T>,
|
|
|
|
f: fn(x: &T) -> Option<U>) -> Option<U> {
|
2012-08-14 23:17:59 +00:00
|
|
|
/*!
|
|
|
|
* Update an optional value by optionally running its content by reference
|
|
|
|
* through a function that returns an option.
|
|
|
|
*/
|
|
|
|
|
2012-08-20 19:23:37 +00:00
|
|
|
match *opt { Some(ref x) => f(x), None => None }
|
2012-08-14 23:17:59 +00:00
|
|
|
}
|
|
|
|
|
2012-08-20 19:23:37 +00:00
|
|
|
pure fn or<T>(+opta: Option<T>, +optb: Option<T>) -> Option<T> {
|
2012-08-21 00:01:06 +00:00
|
|
|
/*!
|
|
|
|
* Returns the leftmost some() value, or none if both are none.
|
|
|
|
*/
|
|
|
|
match opta {
|
2012-09-10 19:14:14 +00:00
|
|
|
Some(_) => move opta,
|
|
|
|
_ => move optb
|
2012-08-21 00:01:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-17 17:54:24 +00:00
|
|
|
#[inline(always)]
|
2012-09-23 11:39:27 +00:00
|
|
|
pure fn while_some<T>(+x: Option<T>, blk: fn(+v: T) -> Option<T>) {
|
2012-07-17 17:54:24 +00:00
|
|
|
//! Applies a function zero or more times until the result is none.
|
|
|
|
|
|
|
|
let mut opt <- x;
|
|
|
|
while opt.is_some() {
|
2012-09-12 00:17:54 +00:00
|
|
|
opt = blk(unwrap(move opt));
|
2012-07-17 17:54:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-22 02:37:57 +00:00
|
|
|
pure fn is_none<T>(opt: &Option<T>) -> bool {
|
2012-07-04 21:53:12 +00:00
|
|
|
//! Returns true if the option equals `none`
|
2012-03-07 03:09:32 +00:00
|
|
|
|
2012-09-22 02:37:57 +00:00
|
|
|
match *opt { None => true, Some(_) => false }
|
2011-12-14 00:25:51 +00:00
|
|
|
}
|
|
|
|
|
2012-09-22 02:37:57 +00:00
|
|
|
pure fn is_some<T>(opt: &Option<T>) -> bool {
|
2012-07-04 21:53:12 +00:00
|
|
|
//! Returns true if the option contains some value
|
2011-12-14 00:25:51 +00:00
|
|
|
|
2012-03-07 03:09:32 +00:00
|
|
|
!is_none(opt)
|
|
|
|
}
|
2011-12-14 00:25:51 +00:00
|
|
|
|
2012-09-22 02:37:57 +00:00
|
|
|
pure fn get_default<T: Copy>(opt: &Option<T>, +def: T) -> T {
|
2012-07-04 21:53:12 +00:00
|
|
|
//! Returns the contained value or a default
|
2012-03-07 03:09:32 +00:00
|
|
|
|
2012-09-28 20:00:07 +00:00
|
|
|
match *opt { Some(copy x) => x, None => def }
|
2011-12-14 00:25:51 +00:00
|
|
|
}
|
|
|
|
|
2012-09-26 23:27:12 +00:00
|
|
|
pure fn map_default<T, U>(opt: &Option<T>, +def: U,
|
2012-08-14 23:17:59 +00:00
|
|
|
f: fn(x: &T) -> U) -> U {
|
|
|
|
//! Applies a function to the contained value or returns a default
|
|
|
|
|
2012-09-10 19:14:14 +00:00
|
|
|
match *opt { None => move def, Some(ref t) => f(t) }
|
2012-08-14 23:17:59 +00:00
|
|
|
}
|
|
|
|
|
2012-09-28 00:01:28 +00:00
|
|
|
pure fn iter<T>(opt: &Option<T>, f: fn(x: &T)) {
|
2012-08-14 23:17:59 +00:00
|
|
|
//! Performs an operation on the contained value by reference
|
2012-08-20 19:23:37 +00:00
|
|
|
match *opt { None => (), Some(ref t) => f(t) }
|
2012-08-14 23:17:59 +00:00
|
|
|
}
|
|
|
|
|
2012-09-10 20:16:34 +00:00
|
|
|
// tjc: shouldn't this be - instead of +?
|
|
|
|
// then could get rid of some superfluous moves
|
2012-07-11 22:00:40 +00:00
|
|
|
#[inline(always)]
|
2012-09-22 02:37:57 +00:00
|
|
|
pure fn unwrap<T>(+opt: Option<T>) -> T {
|
2012-07-04 21:53:12 +00:00
|
|
|
/*!
|
|
|
|
* Moves a value out of an option type and returns it.
|
|
|
|
*
|
|
|
|
* Useful primarily for getting strings, vectors and unique pointers out
|
|
|
|
* of option types without copying them.
|
|
|
|
*/
|
2012-08-24 23:02:47 +00:00
|
|
|
match move opt {
|
2012-09-10 19:14:14 +00:00
|
|
|
Some(move x) => move x,
|
2012-08-20 19:23:37 +00:00
|
|
|
None => fail ~"option::unwrap none"
|
2012-06-25 03:18:18 +00:00
|
|
|
}
|
2012-02-21 07:06:47 +00:00
|
|
|
}
|
|
|
|
|
2012-08-02 18:40:42 +00:00
|
|
|
/// The ubiquitous option dance.
|
|
|
|
#[inline(always)]
|
2012-08-20 19:23:37 +00:00
|
|
|
fn swap_unwrap<T>(opt: &mut Option<T>) -> T {
|
2012-08-02 18:40:42 +00:00
|
|
|
if opt.is_none() { fail ~"option::swap_unwrap none" }
|
2012-08-20 19:23:37 +00:00
|
|
|
unwrap(util::replace(opt, None))
|
2012-08-02 18:40:42 +00:00
|
|
|
}
|
|
|
|
|
2012-08-20 19:23:37 +00:00
|
|
|
pure fn unwrap_expect<T>(+opt: Option<T>, reason: &str) -> T {
|
2012-07-24 19:46:40 +00:00
|
|
|
//! As unwrap, but with a specified failure message.
|
2012-08-03 18:43:10 +00:00
|
|
|
if opt.is_none() { fail reason.to_unique(); }
|
2012-09-12 00:17:54 +00:00
|
|
|
unwrap(move opt)
|
2012-07-24 19:46:40 +00:00
|
|
|
}
|
|
|
|
|
2012-08-20 19:23:37 +00:00
|
|
|
// Some of these should change to be &Option<T>, some should not. See below.
|
|
|
|
impl<T> Option<T> {
|
2012-07-04 21:53:12 +00:00
|
|
|
/// Returns true if the option equals `none`
|
2012-09-22 02:37:57 +00:00
|
|
|
pure fn is_none() -> bool { is_none(&self) }
|
2012-07-04 21:53:12 +00:00
|
|
|
/// Returns true if the option contains some value
|
2012-09-22 02:37:57 +00:00
|
|
|
pure fn is_some() -> bool { is_some(&self) }
|
2012-06-18 19:57:06 +00:00
|
|
|
}
|
2012-04-11 23:27:11 +00:00
|
|
|
|
2012-08-20 19:23:37 +00:00
|
|
|
impl<T> &Option<T> {
|
2012-08-14 23:17:59 +00:00
|
|
|
/**
|
|
|
|
* Update an optional value by optionally running its content by reference
|
|
|
|
* through a function that returns an option.
|
|
|
|
*/
|
2012-08-20 19:23:37 +00:00
|
|
|
pure fn chain_ref<U>(f: fn(x: &T) -> Option<U>) -> Option<U> {
|
2012-08-14 23:17:59 +00:00
|
|
|
chain_ref(self, f)
|
|
|
|
}
|
|
|
|
/// Applies a function to the contained value or returns a default
|
2012-09-26 23:27:12 +00:00
|
|
|
pure fn map_default<U>(+def: U, f: fn(x: &T) -> U) -> U
|
|
|
|
{ map_default(self, move def, f) }
|
2012-08-14 23:17:59 +00:00
|
|
|
/// Performs an operation on the contained value by reference
|
2012-09-28 00:01:28 +00:00
|
|
|
pure fn iter(f: fn(x: &T)) { iter(self, f) }
|
2012-08-14 23:17:59 +00:00
|
|
|
/// Maps a `some` value from one type to another by reference
|
2012-09-26 23:27:12 +00:00
|
|
|
pure fn map<U>(f: fn(x: &T) -> U) -> Option<U> { map(self, f) }
|
2012-08-17 01:06:05 +00:00
|
|
|
/// Gets an immutable reference to the value inside a `some`.
|
|
|
|
pure fn get_ref() -> &self/T { get_ref(self) }
|
2012-08-14 23:17:59 +00:00
|
|
|
}
|
|
|
|
|
2012-09-07 21:52:28 +00:00
|
|
|
impl<T: Copy> Option<T> {
|
2012-07-04 21:53:12 +00:00
|
|
|
/**
|
|
|
|
* Gets the value out of an option
|
|
|
|
*
|
|
|
|
* # Failure
|
|
|
|
*
|
|
|
|
* Fails if the value equals `none`
|
|
|
|
*/
|
2012-09-22 02:37:57 +00:00
|
|
|
pure fn get() -> T { get(&self) }
|
|
|
|
pure fn get_default(+def: T) -> T { get_default(&self, def) }
|
2012-07-04 21:53:12 +00:00
|
|
|
/**
|
|
|
|
* Gets the value out of an option, printing a specified message on
|
|
|
|
* failure
|
|
|
|
*
|
|
|
|
* # Failure
|
|
|
|
*
|
|
|
|
* Fails if the value equals `none`
|
|
|
|
*/
|
2012-09-22 02:37:57 +00:00
|
|
|
pure fn expect(+reason: ~str) -> T { expect(&self, reason) }
|
2012-07-17 17:54:24 +00:00
|
|
|
/// Applies a function zero or more times until the result is none.
|
2012-09-23 11:39:27 +00:00
|
|
|
pure fn while_some(blk: fn(+v: T) -> Option<T>) { while_some(self, blk) }
|
2012-03-17 00:49:58 +00:00
|
|
|
}
|
|
|
|
|
2012-09-20 01:00:26 +00:00
|
|
|
impl<T: Eq> Option<T> : Eq {
|
|
|
|
pure fn eq(other: &Option<T>) -> bool {
|
|
|
|
match self {
|
|
|
|
None => {
|
|
|
|
match (*other) {
|
|
|
|
None => true,
|
|
|
|
Some(_) => false
|
|
|
|
}
|
|
|
|
}
|
2012-09-28 20:00:07 +00:00
|
|
|
Some(ref self_contents) => {
|
2012-09-20 01:00:26 +00:00
|
|
|
match (*other) {
|
|
|
|
None => false,
|
|
|
|
Some(ref other_contents) =>
|
2012-09-28 20:00:07 +00:00
|
|
|
(*self_contents).eq(other_contents)
|
2012-09-20 01:00:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pure fn ne(other: &Option<T>) -> bool { !self.eq(other) }
|
|
|
|
}
|
2012-08-28 00:24:15 +00:00
|
|
|
|
2012-02-21 07:06:47 +00:00
|
|
|
#[test]
|
|
|
|
fn test_unwrap_ptr() {
|
|
|
|
let x = ~0;
|
2012-09-29 04:51:14 +00:00
|
|
|
let addr_x = ptr::p2::addr_of(&(*x));
|
2012-08-20 19:23:37 +00:00
|
|
|
let opt = Some(x);
|
2012-02-21 07:06:47 +00:00
|
|
|
let y = unwrap(opt);
|
2012-09-29 04:51:14 +00:00
|
|
|
let addr_y = ptr::p2::addr_of(&(*y));
|
2012-02-21 07:06:47 +00:00
|
|
|
assert addr_x == addr_y;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_unwrap_str() {
|
2012-07-14 05:57:48 +00:00
|
|
|
let x = ~"test";
|
2012-09-29 04:51:14 +00:00
|
|
|
let addr_x = str::as_buf(x, |buf, _len| ptr::p2::addr_of(&buf));
|
2012-08-20 19:23:37 +00:00
|
|
|
let opt = Some(x);
|
2012-02-21 07:06:47 +00:00
|
|
|
let y = unwrap(opt);
|
2012-09-29 04:51:14 +00:00
|
|
|
let addr_y = str::as_buf(y, |buf, _len| ptr::p2::addr_of(&buf));
|
2012-02-21 07:06:47 +00:00
|
|
|
assert addr_x == addr_y;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_unwrap_resource() {
|
2012-09-02 22:39:37 +00:00
|
|
|
struct R {
|
2012-09-07 02:40:15 +00:00
|
|
|
i: @mut int,
|
2012-06-22 04:30:16 +00:00
|
|
|
drop { *(self.i) += 1; }
|
2012-02-21 07:06:47 +00:00
|
|
|
}
|
2012-09-04 22:23:28 +00:00
|
|
|
|
|
|
|
fn R(i: @mut int) -> R {
|
|
|
|
R {
|
|
|
|
i: i
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-27 01:35:18 +00:00
|
|
|
let i = @mut 0;
|
2012-02-21 07:06:47 +00:00
|
|
|
{
|
2012-09-02 22:39:37 +00:00
|
|
|
let x = R(i);
|
2012-08-20 19:23:37 +00:00
|
|
|
let opt = Some(x);
|
2012-02-21 22:25:51 +00:00
|
|
|
let _y = unwrap(opt);
|
2012-02-21 07:06:47 +00:00
|
|
|
}
|
|
|
|
assert *i == 1;
|
|
|
|
}
|
|
|
|
|
2012-08-02 18:40:42 +00:00
|
|
|
#[test]
|
|
|
|
fn test_option_dance() {
|
2012-08-20 19:23:37 +00:00
|
|
|
let x = Some(());
|
|
|
|
let mut y = Some(5);
|
2012-08-02 18:40:42 +00:00
|
|
|
let mut y2 = 0;
|
|
|
|
do x.iter |_x| {
|
|
|
|
y2 = swap_unwrap(&mut y);
|
|
|
|
}
|
|
|
|
assert y2 == 5;
|
|
|
|
assert y.is_none();
|
|
|
|
}
|
|
|
|
#[test] #[should_fail] #[ignore(cfg(windows))]
|
|
|
|
fn test_option_too_much_dance() {
|
2012-08-20 19:23:37 +00:00
|
|
|
let mut y = Some(util::NonCopyable());
|
2012-08-02 18:40:42 +00:00
|
|
|
let _y2 = swap_unwrap(&mut y);
|
|
|
|
let _y3 = swap_unwrap(&mut y);
|
|
|
|
}
|
|
|
|
|
2012-07-17 17:54:24 +00:00
|
|
|
#[test]
|
|
|
|
fn test_option_while_some() {
|
|
|
|
let mut i = 0;
|
2012-08-20 19:23:37 +00:00
|
|
|
do Some(10).while_some |j| {
|
2012-07-17 17:54:24 +00:00
|
|
|
i += 1;
|
|
|
|
if (j > 0) {
|
2012-08-20 19:23:37 +00:00
|
|
|
Some(j-1)
|
2012-07-17 17:54:24 +00:00
|
|
|
} else {
|
2012-08-20 19:23:37 +00:00
|
|
|
None
|
2012-07-17 17:54:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
assert i == 11;
|
|
|
|
}
|
|
|
|
|
2011-12-14 00:25:51 +00:00
|
|
|
// Local Variables:
|
|
|
|
// mode: rust;
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
|
|
|
// End:
|