// Copyright 2013 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. /*! The Finally trait provides a method, `finally` on stack closures that emulates Java-style try/finally blocks. # Example ~~~ do || { ... }.finally { alway_run_this(); } ~~~ */ use ops::Drop; use task::{spawn, failing}; #[cfg(stage0)] pub trait Finally { fn finally(&self, +dtor: &fn()) -> T; } #[cfg(stage1)] #[cfg(stage2)] #[cfg(stage3)] pub trait Finally { fn finally(&self, dtor: &fn()) -> T; } #[cfg(stage0)] impl Finally for &fn() -> T { // FIXME #4518: Should not require a mode here fn finally(&self, +dtor: &fn()) -> T { let _d = Finallyalizer { dtor: dtor }; (*self)() } } #[cfg(stage1)] #[cfg(stage2)] #[cfg(stage3)] impl Finally for &fn() -> T { fn finally(&self, dtor: &fn()) -> T { let _d = Finallyalizer { dtor: dtor }; (*self)() } } struct Finallyalizer { dtor: &fn() } impl Drop for Finallyalizer { fn finalize(&self) { (self.dtor)(); } } #[test] fn test_success() { let mut i = 0; do (|| { i = 10; }).finally { assert !failing(); assert i == 10; i = 20; } assert i == 20; } #[test] #[ignore(cfg(windows))] #[should_fail] fn test_fail() { let mut i = 0; do (|| { i = 10; fail!(); }).finally { assert failing(); assert i == 10; } } #[test] fn test_retval() { let i = do (fn&() -> int { 10 }).finally { }; assert i == 10; } #[test] fn test_compact() { // FIXME #4727: Should be able to use a fn item instead // of a closure for do_some_fallible_work, // but it's a type error. let do_some_fallible_work: &fn() = || { }; fn but_always_run_this_function() { } do_some_fallible_work.finally( but_always_run_this_function); }