#panic #defer #unwind #linker

no-std scopeguard

A RAII scope guard that will run a given closure when it goes out of scope, even if the code between panics (assuming unwinding panic). Defines the macros defer!, defer_on_unwind!, defer_on_success! as shorthands for guards with one of the implemented strategies

10 releases (3 stable)

Uses old Rust 2015

1.2.0 Jul 17, 2023
1.1.0 Feb 16, 2020
1.0.0 Dec 19, 2018
0.3.3 Oct 13, 2017
0.1.0 Apr 30, 2015

#36 in Rust patterns

Download history 2835588/week @ 2024-09-24 3490454/week @ 2024-10-01 3717825/week @ 2024-10-08 3525718/week @ 2024-10-15 2710275/week @ 2024-10-22 2572667/week @ 2024-10-29 2651848/week @ 2024-11-05 2838073/week @ 2024-11-12 2862089/week @ 2024-11-19 2283977/week @ 2024-11-26 2948739/week @ 2024-12-03 3041544/week @ 2024-12-10 2722619/week @ 2024-12-17 1282096/week @ 2024-12-24 1690110/week @ 2024-12-31 2328295/week @ 2025-01-07

8,494,566 downloads per month
Used in 35,939 crates (272 directly)

MIT/Apache

22KB
263 lines

scopeguard

Rust crate for a convenient RAII scope guard that will run a given closure when it goes out of scope, even if the code between panics (assuming unwinding panic).

The defer! macro and guard are no_std compatible (require only core), but the on unwinding / not on unwinding strategies require linking to std. By default, the use_std crate feature is enabled. Disable the default features for no_std support.

Please read the API documentation here.

Minimum supported Rust version: 1.20

build_status crates

How to use

#[macro_use(defer)]
extern crate scopeguard;

use scopeguard::guard;

fn f() {
    defer! {
        println!("Called at return or panic");
    }
    panic!();
}

use std::fs::File;
use std::io::Write;

fn g() {
    let f = File::create("newfile.txt").unwrap();
    let mut file = guard(f, |f| {
        // write file at return or panic
        let _ = f.sync_all();
    });
    // access the file through the scope guard itself
    file.write_all(b"test me\n").unwrap();
}

Recent Changes

  • 1.2.0

    • Use ManuallyDrop instead of mem::forget in into_inner. (by @willtunnels)
    • Warn if the guard is not assigned to a variable and is dropped immediately instead of at the scope's end. (by @sergey-v-galtsev)
  • 1.1.0

    • Change macros (defer!, defer_on_success! and defer_on_unwind!) to accept statements. (by @konsumlamm)
  • 1.0.0

    • Change the closure type from FnMut(&mut T) to FnOnce(T): Passing the inner value by value instead of a mutable reference is a breaking change, but allows the guard closure to consume it. (by @tormol)

    • Add defer_on_success!, guard_on_success() and OnSuccess strategy, which triggers when scope is exited without panic. It's the opposite to defer_on_unwind! / guard_on_unwind() / OnUnwind.

    • Add ScopeGuard::into_inner(), which "defuses" the guard and returns the guarded value. (by @tormol)

    • Implement Sync for guards with non-Sync closures.

    • Require Rust 1.20

  • 0.3.3

    • Use #[inline] on a few more functions by @stjepang (#14)
    • Add examples to crate documentation
  • 0.3.2

    • Add crate categories
  • 0.3.1

    • Add defer_on_unwind!, Strategy trait
    • Rename GuardScopeGuard
    • Add ScopeGuard::with_strategy.
    • ScopeGuard now implements Debug.
    • Require Rust 1.11
  • 0.2.0

    • Require Rust 1.6
    • Use no_std unconditionally
    • No other changes
  • 0.1.2

    • Add macro defer!

No runtime deps

Features