fix constructor firing for nonMoo through multiple layers of Moo
[gitmo/Moo.git] / lib / Sub / Defer.pm
CommitLineData
eae82931 1package Sub::Defer;
2
3use strictures 1;
4use base qw(Exporter);
b1eebd55 5use Moo::_Utils;
eae82931 6
a165a07f 7our @EXPORT = qw(defer_sub undefer_sub);
eae82931 8
9our %DEFERRED;
10
a165a07f 11sub undefer_sub {
eae82931 12 my ($deferred) = @_;
13 my ($target, $maker, $undeferred_ref) = @{
14 $DEFERRED{$deferred}||return $deferred
15 };
16 ${$undeferred_ref} = my $made = $maker->();
a165a07f 17 if (defined($target)) {
18 no warnings 'redefine';
19 *{_getglob($target)} = $made;
20 }
c4570291 21 push @{$DEFERRED{$made} = $DEFERRED{$deferred}}, $made;
eae82931 22 return $made;
23}
24
c4570291 25sub defer_info {
26 my ($deferred) = @_;
27 $DEFERRED{$deferred||''};
28}
29
a165a07f 30sub defer_sub {
eae82931 31 my ($target, $maker) = @_;
32 my $undeferred;
33 my $deferred_string;
9187b862 34 my $deferred = sub {
a165a07f 35 goto &{$undeferred ||= undefer_sub($deferred_string)};
9187b862 36 };
eae82931 37 $deferred_string = "$deferred";
38 $DEFERRED{$deferred} = [ $target, $maker, \$undeferred ];
a165a07f 39 *{_getglob $target} = $deferred if defined($target);
eae82931 40 return $deferred;
41}
42
431;
8213644a 44
0b6e5fff 45=head1 NAME
46
47Sub::Defer - defer generation of subroutines until they are first called
48
8213644a 49=head1 SYNOPSIS
50
51 use Sub::Defer;
52
53 my $deferred = defer_sub 'Logger::time_since_first_log' => sub {
54 my $t = time;
55 sub { time - $t };
56 };
57
0b6e5fff 58 Logger->time_since_first_log; # returns 0 and replaces itself
59 Logger->time_since_first_log; # returns time - $t
8213644a 60
61=head1 DESCRIPTION
62
0b6e5fff 63These subroutines provide the user with a convenient way to defer creation of
8213644a 64subroutines and methods until they are first called.
65
66=head1 SUBROUTINES
67
68=head2 defer_sub
69
70 my $coderef = defer_sub $name => sub { ... };
71
0b6e5fff 72This subroutine returns a coderef that encapsulates the provided sub - when
73it is first called, the provided sub is called and is -itself- expected to
74return a subroutine which will be goto'ed to on subsequent calls.
8213644a 75
0b6e5fff 76If a name is provided, this also installs the sub as that name - and when
77the subroutine is undeferred will re-install the final version for speed.
8213644a 78
79=head2 undefer_sub
80
81 my $coderef = undefer_sub \&Foo::name;
82
83If the passed coderef has been L<deferred|/defer_sub> this will "undefer" it.
84If the passed coderef has not been deferred, this will just return it.
85
86If this is confusing, take a look at the example in the L</SYNOPSIS>.