49
|
1 package IMPL::DOM::Schema::NodeSet;
|
|
2 use strict;
|
|
3 use warnings;
|
|
4
|
165
|
5 use parent qw(IMPL::DOM::Node);
|
49
|
6 use IMPL::Class::Property;
|
152
|
7 use IMPL::DOM::Property qw(_dom);
|
49
|
8
|
|
9 our %CTOR = (
|
|
10 'IMPL::DOM::Node' => sub { nodeName => 'NodeSet' }
|
|
11 );
|
|
12
|
|
13 BEGIN {
|
152
|
14 public _dom property messageUnexpected => prop_all;
|
|
15 public _dom property messageMax => prop_all;
|
|
16 public _dom property messageMin => prop_all;
|
49
|
17 }
|
|
18
|
|
19 sub CTOR {
|
|
20 my ($this,%args) = @_;
|
|
21
|
238
|
22 $this->messageMax( $args{messageMax} || 'Too many %node.nodeName% nodes');
|
|
23 $this->messageMin( $args{messageMin} || '%schema.name% nodes expected');
|
|
24 $this->messageUnexpected( $args{messageUnexpected} || 'A %node.nodeName% isn\'t allowed in %node.parentNode.path%');
|
49
|
25 }
|
|
26
|
|
27 sub Validate {
|
125
|
28 my ($this,$node,$ctx) = @_;
|
49
|
29
|
|
30 my @errors;
|
|
31
|
|
32 my %nodes;
|
|
33 my $anyNode;
|
125
|
34 my $sourceSchema = $ctx->{Source} || $this->parentNode;
|
|
35
|
49
|
36 foreach (@{$this->childNodes}) {
|
|
37 if ($_->isa('IMPL::DOM::Schema::AnyNode')) {
|
|
38 $anyNode = {Schema => $_, Min => $_->minOccur, Max => $_->maxOccur eq 'unbounded' ? undef : $_->maxOccur , Seen => 0 };
|
|
39 } else {
|
|
40 $nodes{$_->name} = {Schema => $_, Min => $_->minOccur, Max => $_->maxOccur eq 'unbounded' ? undef : $_->maxOccur , Seen => 0 };
|
|
41 }
|
|
42 }
|
|
43
|
|
44 foreach my $child ( @{$node->childNodes} ) {
|
|
45 if (my $info = $nodes{$child->nodeName} || $anyNode) {
|
|
46 $info->{Seen}++;
|
|
47 push @errors,new IMPL::DOM::Schema::ValidationError (
|
236
|
48 source => $sourceSchema,
|
|
49 node => $child,
|
|
50 parent => $node,
|
|
51 schema => $info->{Schema},
|
|
52 message => $this->messageMax
|
49
|
53 ) if ($info->{Max} and $info->{Seen} > $info->{Max});
|
|
54
|
|
55 if (my @localErrors = $info->{Schema}->Validate($child)) {
|
|
56 push @errors,@localErrors;
|
|
57 }
|
|
58 } else {
|
|
59 push @errors, new IMPL::DOM::Schema::ValidationError (
|
236
|
60 source => $sourceSchema,
|
|
61 node => $child,
|
|
62 parent => $node,
|
|
63 message => $this->messageUnexpected
|
49
|
64 )
|
|
65 }
|
|
66 }
|
|
67
|
|
68 foreach my $info (values %nodes) {
|
|
69 push @errors, new IMPL::DOM::Schema::ValidationError (
|
236
|
70 source => $sourceSchema,
|
|
71 schema => $info->{Schema},
|
|
72 parent => $node,
|
|
73 message => $this->messageMin
|
49
|
74 ) if $info->{Min} > $info->{Seen};
|
|
75 }
|
|
76
|
|
77 return @errors;
|
|
78 }
|
|
79
|
|
80 1;
|
|
81
|
|
82 __END__
|
|
83
|
|
84 =pod
|
|
85
|
|
86 =head1 DESCRIPTION
|
|
87
|
180
|
88 Содержимое для сложного узла. Порядок не важен. Дочерними элементами могут быть
|
|
89 только C<IMPL::DOM::Schema::ComplexNode> и C<IMPL::DOM::Schema::SimpleNode>.
|
49
|
90
|
180
|
91 При проверке данного правила, проверяются имеющиеся элементы на соответсие схемы
|
|
92 и количества встречаемости, после чего проверяются количественные ограничения
|
|
93 для несуществующих элементов.
|
49
|
94
|
|
95 =cut
|