package perl5i::2::Meta;
use strict;
use warnings;
use 5.010_000;
# Be very careful not to import anything.
require Carp::Fix::1_25;
require mro;
require perl5i::2::Meta::Instance;
require perl5i::2::Meta::Class;
sub UNIVERSAL::mo {
# Be careful to pass through an alias, not a copy
return perl5i::2::Meta::Instance->new($_[0]);
}
sub UNIVERSAL::mc {
return perl5i::2::Meta::Class->new($_[0]);
}
sub new {
my $class = shift;
# Be careful to take a reference to an alias, not a copy
return bless \\$_[0], $class;
}
sub ISA {
my $class = $_[0]->class;
no strict 'refs';
return wantarray ? @{$class.'::ISA'} : \@{$class.'::ISA'};
}
sub linear_isa {
my $self = shift;
my $class = $self->class;
# get_linear_isa() does not return UNIVERSAL
my @extra;
@extra = qw(UNIVERSAL) unless $class eq 'UNIVERSAL';
my $isa = [@{mro::get_linear_isa($class)}, @extra];
return wantarray ? @$isa : $isa;
}
sub methods {
my $self = shift;
my $opts = shift // {};
my $top = $self->class;
my %exclude;
state $defaults = {
with_UNIVERSAL => 0,
just_mine => 0,
};
$opts = { %$defaults, %$opts };
$exclude{UNIVERSAL} = !$opts->{with_UNIVERSAL};
my @classes = $opts->{just_mine} ? $self->class : $self->linear_isa;
my %all_methods;
for my $class (@classes) {
next if $exclude{$class} && $class ne $top;
my $sym_table = $class->mc->symbol_table;
for my $name (keys %$sym_table) {
my $glob = $sym_table->{$name};
next unless ref \$glob eq "GLOB";
next unless *{$glob}{CODE};
$all_methods{$name} = $class;
}
}
return wantarray ? keys %all_methods : [keys %all_methods];
}
sub symbol_table {
my $self = shift;
my $class = $self->class;
no strict 'refs';
return \%{$class.'::'};
}
# A single place to put the "method not found" error.
my $method_not_found = sub {
my $class = shift;
my $method = shift;
Carp::Fix::1_25::croak(
sprintf q[Can't locate object method "%s" via package "%s"],
$method, $class
);
};
# caller() will return if its inside an eval, need to skip over those.
my $find_method = sub {
my $method;
my $height = 2;
do {
$method = (caller($height))[3];
$height++;
} until( !defined $method or $method ne '(eval)' );
return $method;
};
sub super {
my $self = shift;
my $class = $self->class;
my $fq_method = $find_method->();
Carp::Fix::1_25::croak("super() called outside a method") unless $fq_method;
my($parent, $method) = $fq_method =~ /^(.*)::(\w+)$/;
Carp::Fix::1_25::croak(
sprintf qq["%s" is not a parent class of "%s"],
$parent, $class
) unless $class->isa($parent);
my @isa = $self->linear_isa();
while(@isa) {
my $class = shift @isa;
last if $class eq $parent;
}
for (@isa) {
my $code = $_->can($method);
@_ = ($$$self, @_);
goto &$code if $code;
}
$class->$method_not_found($method);
}
1;