/[Frey]/trunk/lib/Frey/Class/Loader.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/Class/Loader.pm

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

trunk/lib/Frey/ClassLoader.pm revision 137 by dpavlin, Tue Jul 15 18:06:27 2008 UTC trunk/lib/Frey/Class/Loader.pm revision 887 by dpavlin, Wed Dec 24 13:51:59 2008 UTC
# Line 1  Line 1 
1  package Frey::ClassLoader;  package Frey::Class::Loader;
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    with 'Frey::Session';
12    
13  use Data::Dump qw/dump/;  use Data::Dump qw/dump/;
14  use File::Find;  use File::Find;
15    
16  our $package_path;  our $class_path;
17  our @classes;  our @classes;
18    
19    =head2 classes
20    
21    Return all local classes by reading from disk
22    
23      my @classes = $o->classes;
24    
25    =cut
26    
27  sub classes {  sub classes {
28          my $self = shift;          my $self = shift;
29          return @classes if @classes;          return @classes if @classes;
30    
31          # FIXME there must be better way to do this in Moose style          # FIXME there must be better way to do this in Moose style
32          finddepth({ no_chdir => 1, wanted => sub {          finddepth({ no_chdir => 1, wanted => sub {
33                  return unless s/\.pm$//;                  return unless m{\.pm$};
34                  my @a = split(m!/!,$_);                  my $class = $_;
35                  my $package = join('::', @a[ 1 .. $#a ]);                  $class =~ s{^lib/}{};
36                  warn ">> $_ ",dump( @a ), " >> $package\n" if $self->debug;                  $class =~ s{\.pm$}{};
37                  $package_path->{ $package } = "$_.pm";                  $class =~ s{/}{::}g;
38                    if ( 0 && $class =~ m{Mojo} ) { # FIXME remove dead code
39                            $self->TODO( "Mojo support" );
40                            return;
41                    }
42                    $class_path->{ $class } = $_;
43          } }, 'lib');          } }, 'lib');
44          warn "## package_path = ",dump( $package_path ) if $self->debug;          warn "## class_path = ",dump( $class_path ) if $self->debug;
45    
46          @classes = sort keys %$package_path;          @classes = sort keys %$class_path;
47  }  }
48    
49  sub package_path {  =head2 class_path
50          my ( $self, $package ) = @_;  
51          $self->classes unless $package_path;  Return any local or loaded class
52          confess "can't find path for package $package" unless defined $package_path->{$package};  
53          return $package_path->{$package};    $path = $o->class_path( $class );
54    
55    =cut
56    
57    sub class_path {
58            my ( $self, $class ) = @_;
59            $self->classes unless $class_path;
60            if ( ! defined $class_path->{$class} ) {
61                    my $path = $class;
62                    $path =~ s{::}{/}g;
63                    $path .= '.pm';
64                    if ( defined $INC{$path} ) {
65                            $path = $INC{$path};
66                            warn "# $class from INC $path";
67                            $class_path->{$class} = $path;
68                    } elsif ( $path =~ s{\.pm$}{} && -e "lib/${path}.pod" ) {
69                            return "lib/${path}.pod";
70                    } else {
71                            confess "can't find $class at $path";
72                    }
73            }
74            return $class_path->{$class};
75  }  }
76    
77  =head2 load_package  =head2 loaded_classes
78    
79    my ( $class, $meta, $is_role ) = $o->load_package( 'Some::Package' );    my $available = $o->loaded_classes;
80      $available->{'Frey'} # true
81    
82  =cut  =cut
83    
84  sub load_package {  our $loaded_class;
85          my ( $self, $package ) = @_;  sub loaded_classes { $loaded_class };
86    
87          Class::MOP::load_class($package);  =head2 class_meta
88    
89          if ( ! $package->can('meta') ) {    my ( $meta, $is_role, $instance ) = $o->class_meta( 'Some::Class' );
90                  my $class = Moose::Meta::Class->create_anon_class;  
91                  warn "package $package isn't Moose, faking anon class";  =cut
92                  return ( $class, $class->meta, 0 );  
93          }  sub class_meta {
94            my ( $self, $class ) = @_;
95    
96            $class ||= $self->class if $self->can('class');
97            warn "# class_meta $class";
98    
99            $self->load_class($class);
100    
101            my $meta;
102          my $is_role = 0;          my $is_role = 0;
103          my $class;          my $instance;
104    
105          if($package->meta->isa('Moose::Meta::Role')){          if ( ! $class->can('meta') ) {
106                    $instance = Moose::Meta::Class->create_anon_class;
107                    warn "# class $class isn't Moose, faking anon class" if $self->debug;
108                    $meta = $instance->meta;
109            } elsif( $class->meta->isa('Moose::Meta::Role') ) {
110                  $is_role = 1;                  $is_role = 1;
111                  $class = Moose::Meta::Class->create_anon_class;                  $instance = Frey->new;
112                  $package->meta->apply( $class );                  warn "# apply $class on $instance";
113                  die $@ if $@;                  $class->meta->apply( $instance );
114                    $meta = $instance->meta;
115          } else {          } else {
116                  $class = $package->meta;                  $meta = $class->meta;
117          }          }
118          return ( $class, $is_role );          return ( $meta, $is_role, $instance );
119    }
120    
121    sub load_class {
122            my ( $self, $class ) = @_;
123            return if $loaded_class->{$class}++;
124            eval {
125                    Class::MOP::load_class($class)
126            };
127            warn $@ if $@; # && $@ !~ m/role/;
128            warn "# load_class $class" if $self->debug && $loaded_class->{$class} == 1;
129  }  }
130    
131  sub load_all_classes {  sub load_all_classes {
132          my $self = shift;          my $self = shift;
133          my $loaded = 0;          warn "# loaded_class = ",dump( $loaded_class ) if $self->debug;
134          foreach ( $self->classes ) {          $self->load_class( $_ ) foreach ( $self->classes );
135                  Class::MOP::load_class($_);          $loaded_class;
136                  $loaded++;  }
137    
138    =head2 class_methods
139    
140      my @all_methods = $o->class_methods( $class );
141    
142      my $class_method = $o->class_methods( $class );
143      if ( $class_method->{ $method } ) {
144            # $class has $method
145      }
146    
147    =cut
148    
149    sub class_methods {
150            my ( $self, $class ) = @_;
151    
152            confess "need class" unless $class;
153            if ( ! $class->can('meta') ) {
154                    warn "# $class doesn't have meta (isn't Moose class)" if $self->debug;
155                    return;
156          }          }
157          $loaded;          my $meta = $class->meta;
158    
159            my $attr;
160            my $methods;
161            $attr->{$_}++ foreach $meta->get_attribute_list;
162            my @methods = map { $methods->{$_}++; $_ } grep { ! defined($attr->{$_}) && $_ ne 'meta' } $meta->get_method_list;
163            warn "# methods = ",dump( @methods ) if $self->debug;
164    
165            return @methods if wantarray;
166            return $methods;
167    }
168    
169    =head2 class_runnable
170    
171      my @runnable_methods = $o->class_runnable( $class );
172    
173    =cut
174    
175    sub class_runnable_re { m{^as_} || m{_as_} || m{sql} }
176    
177    sub class_runnable {
178            my ( $self, $class ) = @_;
179            my @methods = grep { class_runnable_re } $self->class_methods( $class );
180            return @methods if wantarray;
181            return \@methods;
182    }
183    
184    sub class_inputs {
185            my ( $self, $class ) = @_;
186            my @inputs = grep { m{^(markup/as_data/as_sponge)$} } $self->class_methods( $class );
187            return @inputs if wantarray;
188            return \@inputs;
189    }
190    
191    =head2 new_frey_class
192    
193      my $instance = $o->new_frey_class( $class, $params );
194    
195    This will apply L<Moose::Role> on the fly to provide accessors for
196    C<data> and C<sponge> in form of C<as_*>
197    
198    See L<http://www.perlmonks.org/?node_id=602389>
199    
200    It is used by L<Frey::Run> and L<Frey::Pipe> to create objects
201    
202    =cut
203    
204    {
205            package Frey::Role::as_data;
206            use Moose::Role;
207    
208            sub as_data {
209                    my ($self) = @_;
210                    $self->data;
211            }
212    
213            package Frey::Role::as_sponge;
214            use Moose::Role;
215            sub as_sponge {
216                    my ($self) = @_;
217                    $self->sponge;
218            }
219    }
220    
221    our $syntax_checked_last;
222    
223    sub new_frey_class {
224            my ( $self, $class, $params ) = @_;
225            my $instance;
226    
227            my $path = $self->class_path( $class );
228            if ( $syntax_checked_last->{$class} != -C $path ) {
229                    my $syntax = `perl -Ilib -wc $path 2>&1`;
230                    warn "# syntax: $syntax";
231                    $syntax_checked_last->{$class} = -C $class;
232            }
233    
234            if ( $class->meta->isa('Moose::Meta::Role') ) {
235                    $instance = Frey->new;
236                    Frey::Web->meta->apply( $instance );
237                    warn "new_frey_class $class role with Frey::Web";
238            } else {
239                    if ( $self->can('request_url') ) {
240                            $params->{request_url} = $self->request_url;
241                    } else {
242                            warn "## $self doesn't have request_url";
243                    }
244                    $instance = $class->new( %$params ) or confess "can't $class->new".dump( %$params );
245                    warn "new_frey_class $class";
246            }
247    
248            if ( $instance->can('data') && ! $instance->can('as_data') ) {
249                    Frey::Role::as_data->meta->apply( $instance );
250                    warn "# apply as_data role to $class";
251            }
252            if ( $instance->can('sponge') && ! $instance->can('as_sponge') ) {
253                    Frey::Role::as_sponge->meta->apply( $instance );
254                    warn "# apply as_sponge role to $class";
255            }
256    
257            if ( ! $instance->can('add_status') ) {
258                    Frey::Web->meta->apply( $instance );
259                    warn "# apply Frey::Web role to $class";
260            }
261    
262            $self->add_status({ $class => $params });
263            return $instance;
264  }  }
265    
266  1;  1;

Legend:
Removed from v.137  
changed lines
  Added in v.887

  ViewVC Help
Powered by ViewVC 1.1.26