Mercurial > pub > Impl
diff lib/IMPL/DOM/Schema/NodeSet.pm @ 407:c6e90e02dd17 ref20150831
renamed Lib->lib
author | cin |
---|---|
date | Fri, 04 Sep 2015 19:40:23 +0300 |
parents | |
children |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/lib/IMPL/DOM/Schema/NodeSet.pm Fri Sep 04 19:40:23 2015 +0300 @@ -0,0 +1,104 @@ +package IMPL::DOM::Schema::NodeSet; +use strict; +use warnings; + +use IMPL::Const qw(:prop); +use IMPL::declare { + require => { + Label => 'IMPL::DOM::Schema::Label', + ValidationError => 'IMPL::DOM::Schema::ValidationError', + AnyNode => '-IMPL::DOM::Schema::AnyNode' + }, + base => [ + 'IMPL::DOM::Node' => sub { nodeName => 'NodeSet' } + ], + props => [ + messageUnexpected => { get => 1, set => 1, dom => 1}, + messageMax => { get => 1, set => 1, dom => 1}, + messageMin => { get => 1, set => 1, dom => 1} + ] +}; + +sub CTOR { + my ($this,%args) = @_; + + $this->messageMax( $args{messageMax} || 'Too many %node.nodeName% nodes'); + $this->messageMin( $args{messageMin} || '%schemaNode.name% nodes expected'); + $this->messageUnexpected( $args{messageUnexpected} || 'A %node.nodeName% isn\'t allowed in %node.parentNode.path%'); +} + +sub Validate { + my ($this,$node,$ctx) = @_; + + my @errors; + + my %nodes; + my $anyNode; + + foreach (@{$this->childNodes}) { + if ($_->isa(AnyNode)) { + $anyNode = {schemaNode => $_, min => $_->minOccur, max => $_->maxOccur eq 'unbounded' ? undef : $_->maxOccur , seen => 0 }; + } else { + $nodes{$_->name} = {schemaNode => $_, min => $_->minOccur, max => $_->maxOccur eq 'unbounded' ? undef : $_->maxOccur , seen => 0 }; + } + } + + foreach my $child ( @{$node->childNodes} ) { + if (my $info = $nodes{$child->nodeName} || $anyNode) { + $info->{seen}++; + push @errors,ValidationError->new( + schemaNode => $info->{schemaNode}, + node => $child, + parent => $node, + message => $this->_MakeLabel($this->messageMax) + ) if ($info->{max} and $info->{seen} > $info->{max}); + + if (my @localErrors = $info->{schemaNode}->Validate($child)) { + push @errors,@localErrors; + } + } else { + push @errors, ValidationError->new( + node => $child, + parent => $node, + message => $this->_MakeLabel($this->messageUnexpected) + ) + } + } + + foreach my $info (values %nodes) { + push @errors, ValidationError->new( + schemaNode => $info->{schemaNode}, + parent => $node, + message => $this->_MakeLabel($this->messageMin) + ) if $info->{min} > $info->{seen}; + } + + return @errors; +} + +sub _MakeLabel { + my ($this,$label) = @_; + + if ($label =~ /^ID:(\w+)$/) { + return Label->new($this->document->stringMap, $1); + } else { + return $label; + } +} + +1; + +__END__ + +=pod + +=head1 DESCRIPTION + +Содержимое для сложного узла. Порядок не важен. Дочерними элементами могут быть +только C<IMPL::DOM::Schema::ComplexNode> и C<IMPL::DOM::Schema::SimpleNode>. + +При проверке данного правила, проверяются имеющиеся элементы на соответсие схемы +и количества встречаемости, после чего проверяются количественные ограничения +для несуществующих элементов. + +=cut