The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
# Copyright 2009, 2010 Kevin Ryde

# This file is part of Time-Duration-Locale.
#
# Time-Duration-Locale is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License as published
# by the Free Software Foundation; either version 3, or (at your option) any
# later version.
#
# Time-Duration-Locale is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General
# Public License for more details.
#
# You should have received a copy of the GNU General Public License along
# with Time-Duration-Locale.  If not, see <http://www.gnu.org/licenses/>.

package Module::DynamicDelegate;
use strict;
use warnings;
use Carp;
use vars qw(@ISA @EXPORT $AUTOLOAD @CARP_NOT);

use Exporter;
@ISA = ('Exporter');
@EXPORT = qw(can AUTOLOAD);

# uncomment this to run the ### lines
#use Smart::Comments;


# stub generation to hit right method when subclassing ?



# subclassing Exporter
sub export {
  my ($class, $caller_package) = @_;

  # when code here croaks, don't report it against the package using
  # Module::DynamicDelegate but against whatever was calling to there
  push @CARP_NOT, $caller_package;

  return shift->SUPER::export (@_);
}

sub can {
  my ($origin_module, $funcname) = @_;
  return ($origin_module->SUPER::can ($funcname)
          || sub { goto (_lookup($origin_module,$funcname)) });
}

# regexp per AutoLoader::find_filename()
#
sub AUTOLOAD {
  my ($origin_module, $funcname) = ($AUTOLOAD =~ /(.*)::([^:]+)$/);
  ### DynamicDelegate: $AUTOLOAD
  ### dispatch to: $origin_module->_module_dynamicdelegate

  goto (_lookup($origin_module,$funcname));
}

sub _lookup {
  my ($origin_module, $funcname) = @_;
  my $target_module = $origin_module->_module_dynamicdelegate;
  my $func = $target_module->can($funcname)
    || croak "No such function $funcname in $target_module";
}

1;
__END__

=head1 NAME

Module::DynamicDelegate - delegate to a dynamically chosen module

=head1 SYNOPSIS

 use Module::DynamicDelegate;
 sub _module_dynamicdelegate { return 'Some::Module::Name' }

=head1 DESCRIPTION

B<This is experimental, don't use it!>

=head1 SEE ALSO

L<Module::Alias>

=cut