Callback-Cleanup

 view release on metacpan or  search on metacpan

lib/Callback/Cleanup.pm  view on Meta::CPAN

#!/usr/bin/perl

package Callback::Cleanup;

use strict;
use warnings;

use Sub::Clone qw(clone_if_immortal);

use Hash::Util::FieldHash::Compat qw(idhash);

use Sub::Exporter -setup => {
	exports => [qw(cleanup callback)],
	groups  => { default => [":all"] },
};

our $VERSION = "0.03";

sub cleanup (&;$) {
	my ( $cleanup, $sub ) = @_;
	$sub ? __PACKAGE__->new( $sub, $cleanup ) : $cleanup;
}

sub callback (&;$) {
	my ( $sub, $cleanup ) = @_;
	$cleanup ? __PACKAGE__->new( $sub, $cleanup ) : $sub;
}

idhash my %cleanups;

sub new {
	my ( $class, $body, $cleanup ) = @_;

	my $refcounted = clone_if_immortal($body);

	$cleanups{$refcounted} = $cleanup;

	bless $refcounted, $class;
}

sub DESTROY { delete($cleanups{$_[0]})->() }

__PACKAGE__;

__END__

=pod

=head1 NAME

Callback::Cleanup - Declare callbacks that clean themselves up

=head1 SYNOPSIS

	use Callback::Cleanup;

	my $anon_sub = callback {
		# this is the sub body
	} cleanup {
		# this is called on DESTROY
	}

	# or

	Callback::Cleanup->new(
		sub { }, # callback
		sub { }, # cleanup
	);

=head1 DESCRIPTION

This is a very simple module that provides syntactic sugar for callbacks that
need to finalize somehow.

Callbacks are very convenient APIs when they have no definite end of life. If
an end of life behavior is required this helps keep the cleanup code and
callback code together.

=head1 EXPORTS

=over 4

=item callback BLOCK $cleanup

=item cleanup BLOCK $callback

Both of these exports act as the identity function when given only one
parameter.

When given enough arguments they will create a Callback::Cleanup object.

This means that you can declare a callback with a cleanup like this:

	my $cleans_up = callback {

	} cleanup {
	
	}

Or a derived sub that cleans up an existing subref:

	my $cleans_up = cleanup {

	} \&needs_cleanup;

As well as a few other useless forms.

=back

=head1 CLOSURES AND GARBAGE COLLECTION

In perl code references that are not closures aren't garbage collected (they
are shared).

This module uses L<Sub::Clone/clone_if_immortal> to make sure timely
destruction of these callbacks happens.

=head1 AUTHOR

Yuval Kogman <nothingmuch@woobling.org>

=head1 COPYRIGHT & LICENSE

	Copyright (c) 2006, 2008 the aforementioned authors. All rights
	reserved. This program is free software; you can redistribute
	it and/or modify it under the same terms as Perl itself.

=cut




( run in 1.475 second using v1.01-cache-2.11-cpan-140bd7fdf52 )