Mercurial > pub > Impl
changeset 209:a8db61d0ed33
IMPL::Class::Meta refactoring
author | cin |
---|---|
date | Mon, 28 May 2012 19:58:56 +0400 (2012-05-28) |
parents | 3d433a977e3b |
children | 6adaeb86945d |
files | Lib/IMPL/Class/Meta.pm Lib/IMPL/Class/PropertyInfo.pm Lib/IMPL/Code/Loader.pm Lib/IMPL/DOM/Node.pm Lib/IMPL/declare.pm _test/Test/Class/Meta.pm _test/temp.pl |
diffstat | 7 files changed, 91 insertions(+), 222 deletions(-) [+] |
line wrap: on
line diff
--- a/Lib/IMPL/Class/Meta.pm Fri May 18 18:43:00 2012 +0400 +++ b/Lib/IMPL/Class/Meta.pm Mon May 28 19:58:56 2012 +0400 @@ -1,12 +1,13 @@ package IMPL::Class::Meta; use strict; +use Carp qw(carp); use IMPL::clone qw(clone); my %class_meta; my %class_data; -sub set_meta { +sub SetMeta { my ($class,$meta_data) = @_; $class = ref $class if ref $class; @@ -17,7 +18,11 @@ push @{$class_meta{$class}{ref $meta_data}},$meta_data; } -sub get_meta { +sub set_meta { + goto &SetMeta; +} + +sub GetMeta { my ($class,$meta_class,$predicate,$deep) = @_; $class = ref $class if ref $class; no strict 'refs'; @@ -35,10 +40,16 @@ wantarray ? @result : \@result; } +sub get_meta { + goto &GetMeta; +} + sub class_data { my $class = shift; $class = ref $class || $class; + carp 'The method is obsolete, use static_accessor($name,$value,\'clone\') instead'; + if (@_ > 1) { my ($name,$value) = @_; return $class_data{$class}{$name} = $value; @@ -58,23 +69,26 @@ } sub static_accessor { - my ($class,$name,$value) = @_; + my ($class,$name,$value,$clone) = @_; $class = ref $class || $class; no strict 'refs'; *{"${class}::${name}"} = sub { - if (@_ > 1) { - my $self = shift; - $self = ref $self || $self; + my $self = shift; + + if (@_ > 0) { + $self = ref $self || $self; if ($class ne $self) { - $self->static_accessor( $name => $_[0]); # define own class data + $self->static_accessor( $name => $_[0] ); # define own class data } else { $value = $_[0]; } } else { - $value; + ($clone and $class ne $self) + ? $self->static_accessor($name => clone($value),$clone) + : $value and $value ; } }; $value @@ -152,13 +166,11 @@ =head1 MEMBERS -=over - -=item C<set_meta($meta_data)> +=head2 C<set_meta($meta_data)> Добавляет метаданные C<$meta_data> к классу. -=item C<get_meta($meta_class,$predicate,$deep)> +=head2 C<get_meta($meta_class,$predicate,$deep)> Выбирает метаданные типа C<$meta_class> и его наследников, с возможностью фильтрации и получения метаданных базовых классов. @@ -201,50 +213,13 @@ =back -=item C<class_data($name,$new_value)> - -В отличии от метаданных, C<class_data> не накапливает информацию, -а хранит только один экземпляр для одного ключа C<$name>. - -Если новое значение не задано, то осуществляется выборка сохраненного, -если текущий класс не имеет сохраненного значения, то оно ищется в базовых -классах, затем копия найденного значения сохраняется в текущем классе и -возвращается наружу. Это позволяет базовым классам задавать значение по-умолчанию, -которые могут быть изменены или заменены субклассами. - -=begin code - -package Foo; -use parent qw(IMPL::Class::Meta); - -__PACKAGE__->class_data( info => { version => 1 } ); # will be default for all subclasses - -sub say_version { - my ($self) = @_; - - print $self->class_data('info')->{version}; -} - -package Bar; -use parent qw(Foo); - -__PACKAGE__->class_data('info')->{ language } = 'English'; - -package main; - -Foo->class_data('info')->{version} = 2; -Bar->say_version; # will print '1'; -Foo->say_version; # will print '2'; - -=end code - -=item C<static_accessor($name[,$value])> +=head2 C<static_accessor($name[,$value[,$clone]])> Создает статическое свойство с именем C<$name> и начальным значением C<$value>. -Использование данного свойство аналогично использованию C<class_data>, за исключением -того, что C<class_data> гарантирует, что наследник обладает собственной копией данных, -изменение которых не коснется ни базового класса, ни соседей. +Параметр C<$clone> контролирует то, как наследуются значения, если требуется каждому классу обеспечить +свое уникальное значение, то при первом обращении оно будет клонировано, по умолчанию клонирование не +происходит. =begin code @@ -260,8 +235,5 @@ __PACKAGE__->info({language => 'English'}); # will define own 'info' but will loose original data. =end code - - -=back =cut
--- a/Lib/IMPL/Class/PropertyInfo.pm Fri May 18 18:43:00 2012 +0400 +++ b/Lib/IMPL/Class/PropertyInfo.pm Mon May 28 19:58:56 2012 +0400 @@ -2,10 +2,13 @@ use strict; use IMPL::_core::version; -use parent qw(IMPL::Class::MemberInfo); +use IMPL::declare { + base =>{ + 'IMPL::Class::MemberInfo' => '@_' + } +}; __PACKAGE__->mk_accessors(qw(Type Mutators canGet canSet ownerSet)); -__PACKAGE__->PassThroughArgs; my %LoadedModules; @@ -49,4 +52,15 @@ =pod +=head1 NAME + +C<IMPL::Class::PropertyInfo> - метаданные о свойствах объектов. Используются для отражения и +проверки данных объектов. + +=head1 DESCRIPTION + +В зависимости от типа каждый объект предоставляет способ хранения данных, например хеши позволяют +хранить состояние в виде ассоциативного массива и т.д. Информация о свойстве предоставляет определенный +уровень абстракции. + =cut
--- a/Lib/IMPL/Code/Loader.pm Fri May 18 18:43:00 2012 +0400 +++ b/Lib/IMPL/Code/Loader.pm Mon May 28 19:58:56 2012 +0400 @@ -2,32 +2,47 @@ use strict; use warnings; -my %packages; +use IMPL::lang qw(:declare :constants); + +use IMPL::declare { + require => { + Exception => 'IMPL::Exception', + ArgumentException => '-IMPL::InvalidArgumentException' + }, + base => { + 'IMPL::Object' => undef, + 'IMPL::Object::Autofill' => '@_' + } +}; -sub Provide { - my ($self,$package) = @_; - - my ($declaringPackage,$file) = caller(); - $packages{$package} = { declaringPackage => $declaringPackage, file => $file, evidence => 'provide' }; +my $default; +sub default { + $default ||= new IMPL::Code::Loader; } +my $safe; +sub safe { + $safe ||= new IMPL::Code::Loader(verifyNames => 1); +} + +BEGIN { + public property verifyNames => PROP_GET | PROP_OWNERSET; + public property prefix => PROP_GET | PROP_OWNERSET; +} + + sub Require { - my ($self,$package) = @_; + my ($this,$package) = @_; - return 1 if $packages{$package}; + if ($this->verifyNames) { + $package =~ m/^([a-zA-Z_0-9]+(?:::[a-zA-Z_0-9]+)*)$/ or die ArgumentException->new("package") ; + } - if (my $file = $INC{$package}) { - $packages{$package} = { file => $file, evidence => 'inc' }; - return 1; - } + $package = $this->prefix . $package if $this->prefix; - undef $@; + my $file = join('/', split(/::/,$package)) . ".pm"; - if ( eval "require $package; 1;" and not $packages{$package}) { - $packages{$package} = { file => $INC{$package}, evidence => 'inc' }; - }; - - die $@ if $@ and not $!; + require $file; } 1;
--- a/Lib/IMPL/DOM/Node.pm Fri May 18 18:43:00 2012 +0400 +++ b/Lib/IMPL/DOM/Node.pm Mon May 28 19:58:56 2012 +0400 @@ -22,7 +22,6 @@ public _direct property schemaSource => prop_get | owner_set; private _direct property _propertyMap => prop_all ; - __PACKAGE__->class_data(property_bind => {}); } our %Axes = (
--- a/Lib/IMPL/declare.pm Fri May 18 18:43:00 2012 +0400 +++ b/Lib/IMPL/declare.pm Mon May 28 19:58:56 2012 +0400 @@ -2,6 +2,7 @@ use strict; use Scalar::Util qw(set_prototype); +use Carp qw(carp); sub import { my ($self,$args) = @_; @@ -30,6 +31,7 @@ my @isa; if (ref $base eq 'ARRAY') { + carp("will be changed in next version"); @isa = map _require($_), @$base if @$base; } elsif (ref $base eq 'HASH' ) { while ( my ($class,$mapper) = each %$base ) {
--- a/_test/Test/Class/Meta.pm Fri May 18 18:43:00 2012 +0400 +++ b/_test/Test/Class/Meta.pm Mon May 28 19:58:56 2012 +0400 @@ -5,42 +5,43 @@ __PACKAGE__->PassThroughArgs; -use IMPL::Test qw(test failed); +use IMPL::Test qw(test failed assert); test defineFooClassData => sub { - Foo->class_data(info => {}); + Foo->static_accessor(info => {},'clone'); }; test updateFooClassData => sub { - Foo->class_data('info')->{data} = 'Foo' ; + Foo->info->{data} = 'Foo' ; }; test getFooClassData => sub { - failed "Wrong class data", "Expected: Foo", "Got: ".Foo->class_data('info')->{data} unless Foo->class_data('info')->{data} eq 'Foo'; + assert( Foo->info->{data} eq 'Foo' ); }; test getBazClassData => sub { - failed "Wrong class data", "Expected: Foo", "Got: ".Baz->class_data('info')->{data} unless Baz->class_data('info')->{data} eq 'Foo'; + assert( Baz->info->{data} eq 'Foo' ); + assert( Bar->info->{data} eq 'Foo' ); }; test updateBarClassData => sub { - Bar->class_data('info')->{data} = 'Bar'; + Bar->info->{data} = 'Bar'; }; test getBarClassData => sub { - failed "Wrong class data", "Expected: Bar", "Got: ".Bar->class_data('info')->{data} unless Bar->class_data('info')->{data} eq 'Bar'; + assert( Bar->info->{data} eq 'Bar'); }; test validatetFooClassData => sub { - failed "Wrong class data", "Expected: Foo", "Got: ".Foo->class_data('info')->{data} unless Foo->class_data('info')->{data} eq 'Foo'; + assert( Foo->info->{data} eq 'Foo'); }; test validateBazClassData => sub { - failed "Wrong class data", "Expected: Foo", "Got: ".Baz->class_data('info')->{data} unless Baz->class_data('info')->{data} eq 'Foo'; + assert( Baz->info->{data} eq 'Foo'); }; test getwrongBazClassData => sub { - failed "Wrong class data", "Expected: undef", "Got: ".Foo->class_data( 'info2' ) if Foo->class_data( 'info2' ); + assert( not eval { Foo->info2; 1; } ); };
--- a/_test/temp.pl Fri May 18 18:43:00 2012 +0400 +++ b/_test/temp.pl Mon May 28 19:58:56 2012 +0400 @@ -1,135 +1,1 @@ -#!/usr/bin/perl -use strict; - -use Data::Dumper(); - -=pod - -{ - bar => { - next => { - foo => { - data => 'teo' - }, - baz => { - data => 'ioh' - } - }, - data => 'duo' - }, - wee => { - data => 'iwy' - } -} - -=cut - -my $tree = {}; - -foreach my $selector( - { path => [qw( foo bar )], data => 'teo' }, - { path => [qw( {x:.*} zoo bar )], data => 'view/{x}'}, - { path => [qw( foo >zoo >bar )], data => 'ilo' }, - { path => [qw( bar )], data => 'duo' }, - { path => [qw( wee )], data => 'iwy'}, - { path => [qw( foo wee )], data => 'fwy'}, - { path => [qw( {x:\w+} )], data => 'x:{x}'}, - { path => [qw( boo {x:\w+} )], data => 'boo/{x}'}, -) { - my $t = $tree; - my @path = reverse @{$selector->{path}}; - my $last = pop @path; - my $level = 1; - foreach my $prim (@path ) { - $t = ($t->{$prim}->{next} ||= {}); - $level ++; - } - $t->{$last}->{level} = $level; - $t->{$last}->{data} = $selector->{data}; -} - -my @target = qw( foo zoo bar ); -my @results; -my $alternatives = [ { selector => $tree, immediate => 1 } ]; - -$alternatives = MatchAlternatives($_,$alternatives,\@results) foreach reverse @target; - - -sub MatchAlternatives { - my ($segment,$alternatives,$results) = @_; - - warn "alternatives: ", scalar @$alternatives,", segment: $segment"; - - my @next; - - foreach my $alt (@$alternatives) { - while (my ($selector,$match) = each %{$alt->{selector}} ) { - warn $selector; - - warn "\timmediate" if $alt->{immediate}; - warn "\thas children" if $match->{next}; - - my $context = { - vars => \%{ $alt->{vars} || {} }, - selector => $match->{next} - }; - - if ($selector =~ s/^>//) { - $context->{immediate} = 1; - } - - if (my ($name,$rx) = ($selector =~ m/^\{(?:(\w+)\:)?(.*)\}$/) ) { - #this is a regexp - warn "\tregexp: [$name] $rx"; - - if ( my @captures = ($segment =~ m/($rx)/) ) { - $context->{success} = 1; - - warn "\t",join(',',@captures); - - if ($name) { - $context->{vars}->{$name} = \@captures; - } - } - } else { - #this is a segment name - if ($segment eq $selector) { - $context->{success} = 1; - } - } - - # test if there were a match - if (delete $context->{success}) { - warn "\tmatch"; - if (my $data = $match->{data}) { - # interpolate data - $data =~ s/{(\w+)(?:\:(\d+))?}/ - my ($name,$index) = ($1,$2 || 0); - - if ($context->{vars}{$name}) { - $context->{vars}{$name}[$index]; - } else { - ""; - } - /gex; - - push @$results, { level => $match->{level}, result => $data }; - } - warn "\tnext" if $context->{selector}; - push @next, $context if $context->{selector}; - } else { - #repeat current alternative if it's not required to be immediate - push @next, { - selector => { $selector, $match }, - vars => $alt->{vars} - } unless $alt->{immediate}; - } - } - } - - warn "end, next trip: ",scalar @next, " alternatives"; - - return \@next; -} - -print Data::Dumper->Dump([$tree,\@results],[qw(tree results)]); \ No newline at end of file +print "asd::asd" =~ /^[a-zA-Z]+(?:::[a-zA-Z]+)*$/; \ No newline at end of file