@ -16,35 +16,39 @@
// File: cleanup.h
// File: cleanup.h
// -----------------------------------------------------------------------------
// -----------------------------------------------------------------------------
//
//
// `absl::Cleanup` implements the scope guard idiom, invoking `operator()() &&`
// `absl::Cleanup` implements the scope guard idiom, invoking the contained
// on the callback it was constructed with, on scope exit.
// callback's `operator()() &&` on scope exit.
//
//
// Example:
// Example:
//
//
// ```
// ```
// void CopyGoodData(const char* input_path, const char* output_path) {
// absl::Status CopyGoodData(const char* source_path, const char* sink_path) {
// FILE* in_file = fopen(input_path, "r");
// FILE* source_file = fopen(source_path, "r");
// if (in_file == nullptr) return;
// if (source_file == nullptr) {
// return absl::NotFoundError("No source file"); // No cleanups execute
// }
//
//
// // C++17 style using class template argument deduction
// // C++17 style cleanup using class template argument deduction
// absl::Cleanup in_closer = [in_file] { fclose(in _file); };
// absl::Cleanup source_closer = [source_file] { fclose(source _file); };
//
//
// FILE* out_file = fopen(output_path, "w");
// FILE* sink_file = fopen(sink_path, "w");
// if (out_file == nullptr) return; // `in_closer` will run
// if (sink_file == nullptr) {
// return absl::NotFoundError("No sink file"); // First cleanup executes
// }
//
//
// // C++11 style using the factory function
// // C++11 style cleanup using the factory function
// auto out_closer = absl::MakeCleanup([out_file] { fclose(out _file); });
// auto sink_closer = absl::MakeCleanup([sink_file] { fclose(sink _file); });
//
//
// Data data;
// Data data;
// while (ReadData(in _file, &data)) {
// while (ReadData(source _file, &data)) {
// if (data.IsBad()) {
// if (data.IsBad()) {
// LOG(ERROR) << "Found bad data." ;
// absl::Status result = absl::FailedPreconditionError("Read bad data") ;
// return; // `in_closer` and `out_closer` will run
// return result; // Both cleanups execute
// }
// }
// SaveData(out _file, &data);
// SaveData(sink _file, &data);
// }
// }
//
//
// // `in_closer` and `out_closer` will run
// return absl::OkStatus(); // Both cleanups execute
// }
// }
// ```
// ```
//
//
@ -54,6 +58,12 @@
//
//
// `std::move(cleanup).Invoke()` will execute the callback early, before
// `std::move(cleanup).Invoke()` will execute the callback early, before
// destruction, and prevent the callback from executing in the destructor.
// destruction, and prevent the callback from executing in the destructor.
//
// Usage:
//
// `absl::Cleanup` is not an interface type. It is only intended to be used
// within the body of a function. It is not a value type and instead models a
// control flow construct. Check out `defer` in Golang for something similar.
# ifndef ABSL_CLEANUP_CLEANUP_H_
# ifndef ABSL_CLEANUP_CLEANUP_H_
# define ABSL_CLEANUP_CLEANUP_H_
# define ABSL_CLEANUP_CLEANUP_H_
@ -76,9 +86,10 @@ class ABSL_MUST_USE_RESULT Cleanup {
" Callbacks that return values are not supported. " ) ;
" Callbacks that return values are not supported. " ) ;
public :
public :
Cleanup ( Callback callback ) : storage_ ( std : : move ( callback ) ) { } // NOLINT
Cleanup ( Callback callback ) // NOLINT
: storage_ ( std : : move ( callback ) , /*engaged=*/ true ) { }
Cleanup ( Cleanup & & other ) : storage_ ( std : : move ( other . storage_ ) ) { }
Cleanup ( Cleanup & & other ) = default ;
void Cancel ( ) & & {
void Cancel ( ) & & {
ABSL_HARDENING_ASSERT ( storage_ . IsCallbackEngaged ( ) ) ;
ABSL_HARDENING_ASSERT ( storage_ . IsCallbackEngaged ( ) ) ;
@ -103,7 +114,7 @@ class ABSL_MUST_USE_RESULT Cleanup {
// `absl::Cleanup c = /* callback */;`
// `absl::Cleanup c = /* callback */;`
//
//
// C++17 type deduction API for creating an instance of `absl::Cleanup`.
// C++17 type deduction API for creating an instance of `absl::Cleanup`
# if defined(ABSL_HAVE_CLASS_TEMPLATE_ARGUMENT_DEDUCTION)
# if defined(ABSL_HAVE_CLASS_TEMPLATE_ARGUMENT_DEDUCTION)
template < typename Callback >
template < typename Callback >
Cleanup ( Callback callback ) - > Cleanup < cleanup_internal : : Tag , Callback > ;
Cleanup ( Callback callback ) - > Cleanup < cleanup_internal : : Tag , Callback > ;
@ -111,7 +122,7 @@ Cleanup(Callback callback) -> Cleanup<cleanup_internal::Tag, Callback>;
// `auto c = absl::MakeCleanup(/* callback */);`
// `auto c = absl::MakeCleanup(/* callback */);`
//
//
// C++11 type deduction API for creating an instance of `absl::Cleanup`.
// C++11 type deduction API for creating an instance of `absl::Cleanup`
template < typename . . . Args , typename Callback >
template < typename . . . Args , typename Callback >
absl : : Cleanup < cleanup_internal : : Tag , Callback > MakeCleanup ( Callback callback ) {
absl : : Cleanup < cleanup_internal : : Tag , Callback > MakeCleanup ( Callback callback ) {
static_assert ( cleanup_internal : : WasDeduced < cleanup_internal : : Tag , Args . . . > ( ) ,
static_assert ( cleanup_internal : : WasDeduced < cleanup_internal : : Tag , Args . . . > ( ) ,