/[Frey]/trunk/lib/Frey/ClassLoader.pm
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Diff of /trunk/lib/Frey/ClassLoader.pm

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 100 by dpavlin, Fri Jul 11 19:19:42 2008 UTC revision 412 by dpavlin, Tue Nov 18 14:07:28 2008 UTC
# Line 1  Line 1 
1  package Frey::ClassLoader;  package Frey::ClassLoader;
2  use Moose;  use Moose;
3    
4    =head1 DESCRIPTION
5    
6    Load L<Frey> classes
7    
8    =cut
9    
10  extends 'Frey';  extends 'Frey';
11    
12  use Data::Dump qw/dump/;  use Data::Dump qw/dump/;
13  use File::Find;  use File::Find;
14    
15  has 'classes' => (  our $class_path;
16          is => 'ro',  our @classes;
17  #       isa => 'HashRef[Str]',  
18          default => sub {  =head2 classes
19                  my $self = shift;  
20                  # FIXME there must be better way to do this in Moose style  Return all local classes by reading from disk
21                  my $classes;  
22                  finddepth({ no_chdir => 1, wanted => sub {    my @classes = $o->classes;
                         return unless s/\.pm$//;  
                         my @a = split(m!/!,$_);  
                         my $package = join('::', @a[ 1 .. $#a ]);  
                         warn ">> $_ ",dump( @a ), " >> $package\n" if $self->debug;  
                         push @$classes, { $package => "$_.pm" };  
                 } }, 'lib');  
                 warn "## classes = ",dump( $classes ) if $self->debug;  
                 $classes;  
         },  
         lazy => 1,  
 );  
   
 =head2 load_package  
   
   my ( $class, $meta, $is_role ) = $o->load_package( 'Some::Package' );  
23    
24  =cut  =cut
25    
26  sub load_package {  sub classes {
27          my ( $self, $package ) = @_;          my $self = shift;
28            return @classes if @classes;
29          #intercept role application so we can accurately generate  
30          #method and attribute information for the parent class.          # FIXME there must be better way to do this in Moose style
31          #this is fragile, but there is not better way that i am aware of          finddepth({ no_chdir => 1, wanted => sub {
32          my $rmeta = Moose::Meta::Role->meta;                  return unless s/\.pm$//;
33          $rmeta->make_mutable if $rmeta->is_immutable;                  my @a = split(m!/!,$_);
34          my $original_apply = $rmeta->get_method("apply")->body;                  my $class = join('::', @a[ 1 .. $#a ]);
35          $rmeta->remove_method("apply");                  warn ">> $_ ",dump( @a ), " >> $class\n" if $self->debug;
36          my @roles_to_apply;                  $class_path->{ $class } = "$_.pm";
37          $rmeta->add_method("apply", sub{push(@roles_to_apply, [@_])});          } }, 'lib');
38          #load the package with the hacked Moose::Meta::Role          warn "## class_path = ",dump( $class_path ) if $self->debug;
39    
40          #eval { Class::MOP::load_class($package); };          @classes = sort keys %$class_path;
41          #confess "Failed to load package ${package} $@" if $@;  }
42          Class::MOP::load_class($package);  
43    =head2 class_path
44    
45    Return any local or loaded class
46    
47      $path = $o->class_path( $class );
48    
49    =cut
50    
51    sub class_path {
52            my ( $self, $class ) = @_;
53            $self->classes unless $class_path;
54            if ( ! defined $class_path->{$class} ) {
55                    my $path = $class;
56                    $path =~ s{::}{/}g;
57                    $path .= '.pm';
58                    $path = $INC{$path};
59                    warn "# $class from INC $path";
60                    $class_path->{$class} = $path || confess "can't find path for $class";
61            }
62            return $class_path->{$class};
63    }
64    
65    =head2 loaded_classes
66    
67      my $available = $o->loaded_classes;
68      $available->{'Frey'} # true
69    
70    =cut
71    
72    our $loaded_class;
73    sub loaded_classes { $loaded_class };
74    
75    =head2 class_meta
76    
77          my $meta = $package->meta;    my ( $meta, $is_role ) = $o->class_meta( 'Some::Class' );
78    
79          my ($class, $is_role);  =cut
80          if($package->meta->isa('Moose::Meta::Role')){  
81    sub class_meta {
82            my ( $self, $class ) = @_;
83    
84            $class ||= $self->class if $self->can('class');
85            warn "# class_meta $class";
86    
87            $self->load_class($class);
88    
89            if ( ! $class->can('meta') ) {
90                    my $instance = Moose::Meta::Class->create_anon_class;
91                    warn "class $class isn't Moose, faking anon class" if $self->debug;
92                    return ( $instance, 0 );
93            }
94    
95            my $meta;
96            my $is_role = 0;
97            my $instance;
98    
99            if($class->meta->isa('Moose::Meta::Role')){
100                  $is_role = 1;                  $is_role = 1;
101                  # we need to apply the role to a class to be able to properly introspect it                  $instance = Moose::Meta::Class->create_anon_class;
102                  $class = Moose::Meta::Class->create_anon_class;                  $class->meta->apply( $instance );
103                  $original_apply->($meta, $class);                  $meta = $class->meta;
104                    die $@ if $@;
105          } else {          } else {
106                  #roles don't have superclasses ...                  $meta = $class->meta;
                 $class = $meta;  
107          }          }
108          return ( $class, $meta, $is_role );          return ( $meta, $is_role );
109    }
110    
111    sub load_class {
112            my ( $self, $class ) = @_;
113            eval {
114                    Class::MOP::load_class($class) if ! $loaded_class->{$class}++;
115            };
116            warn $@ if $@; # && $@ !~ m/role/;
117            warn "# load_class $class" if $self->debug && $loaded_class->{$class} == 1;
118    }
119    
120    sub load_all_classes {
121            my $self = shift;
122            warn "# loaded_class = ",dump( $loaded_class ) if $self->debug;
123            $self->load_class( $_ ) foreach ( $self->classes );
124            $loaded_class;
125  }  }
126    
127    =head2 class_methods
128    
129      my @all_methods = $o->class_methods( $class );
130    
131      my $class_method = $o->class_methods( $class );
132      if ( $class_method->{ $method } ) {
133            # $class has $method
134      }
135    
136    =cut
137    
138    sub class_methods {
139            my ( $self, $class ) = @_;
140    
141            confess "need class" unless $class;
142            if ( ! $class->can('meta') ) {
143                    warn "$class doesn't have meta (isn't Moose class)" if $self->debug;
144                    return;
145            }
146            my $meta = $class->meta;
147    
148            my $attr;
149            my $methods;
150            $attr->{$_}++ foreach $meta->get_attribute_list;
151            my @methods = map { $methods->{$_}++; $_ } grep { ! defined($attr->{$_}) && $_ ne 'meta' } $meta->get_method_list;
152            warn "# methods = ",dump( @methods ) if $self->debug;
153    
154            return @methods if wantarray;
155            return $methods;
156    }
157    
158  1;  1;

Legend:
Removed from v.100  
changed lines
  Added in v.412

  ViewVC Help
Powered by ViewVC 1.1.26