use strictures 1;
use base qw(Exporter);
use Moo::_Utils;
+use Scalar::Util qw(weaken);
+
+our $VERSION = '1.003001';
+$VERSION = eval $VERSION;
our @EXPORT = qw(defer_sub undefer_sub);
my ($target, $maker, $undeferred_ref) = @{
$DEFERRED{$deferred}||return $deferred
};
+ return ${$undeferred_ref}
+ if ${$undeferred_ref};
${$undeferred_ref} = my $made = $maker->();
- if (defined($target)) {
+
+ # make sure the method slot has not changed since deferral time
+ if (defined($target) && $deferred eq *{_getglob($target)}{CODE}||'') {
no warnings 'redefine';
+
+ # I believe $maker already evals with the right package/name, so that
+ # _install_coderef calls are not necessary --ribasushi
*{_getglob($target)} = $made;
}
+ weaken($DEFERRED{$made} = $DEFERRED{$deferred});
+
return $made;
}
+sub defer_info {
+ my ($deferred) = @_;
+ $DEFERRED{$deferred||''};
+}
+
sub defer_sub {
my ($target, $maker) = @_;
my $undeferred;
- my $deferred_string;
+ my $deferred_info;
my $deferred = sub {
- goto &{$undeferred ||= undefer_sub($deferred_string)};
+ $undeferred ||= undefer_sub($deferred_info->[3]);
+ goto &$undeferred;
};
- $deferred_string = "$deferred";
- $DEFERRED{$deferred} = [ $target, $maker, \$undeferred ];
- *{_getglob $target} = $deferred if defined($target);
+ $deferred_info = [ $target, $maker, \$undeferred, $deferred ];
+ weaken($DEFERRED{$deferred} = $deferred_info);
+ _install_coderef($target => $deferred) if defined $target;
return $deferred;
}
-1;
+sub CLONE {
+ %DEFERRED = map { defined $_ ? ($_->[3] => $_) : () } values %DEFERRED;
+ weaken($_) for values %DEFERRED;
+}
-=pod
+1;
+__END__
=head1 NAME
If the passed coderef has not been deferred, this will just return it.
If this is confusing, take a look at the example in the L</SYNOPSIS>.
+
+=head1 SUPPORT
+
+See L<Moo> for support and contact information.
+
+=head1 AUTHORS
+
+See L<Moo> for authors.
+
+=head1 COPYRIGHT AND LICENSE
+
+See L<Moo> for the copyright and license.
+
+=cut