# HG changeset patch # User Sergey # Date 1253625458 -14400 # Node ID 267460284fb38d7e1f28d9ed44b9db6eb0cd26d3 # Parent 1ca530e5c9c5e2fb652ffba8a7f9f2648be1f3bf DOM Schema diff -r 1ca530e5c9c5 -r 267460284fb3 Lib/IMPL/DOM/Schema.pm --- a/Lib/IMPL/DOM/Schema.pm Fri Sep 11 16:30:39 2009 +0400 +++ b/Lib/IMPL/DOM/Schema.pm Tue Sep 22 17:17:38 2009 +0400 @@ -34,7 +34,11 @@ sub Validate { my ($this,$node) = @_; - #return IMPL::DOM::Schema::NodeSet->new()->appendRange(@{$this->childNodes})->Validate($node); + if ( my ($schemaNode) = $this->selectNodes(sub { $_[0]->name eq $node->nodeName })) { + $schemaNode->Validate($node); + } else { + return IMPL::DOM::Schema::VaidationError(Message=> "A specified document doesn't match the schema"); + } } my $schema; @@ -46,42 +50,42 @@ $schema = new IMPL::DOM::Schema(nodeName => 'schema'); $schema->appendRange( - IMPL::DOM::Schema::ComplexNode->new(nodeName => 'schema')->appendRange( + IMPL::DOM::Schema::ComplexNode->new(name => 'schema')->appendRange( IMPL::DOM::Schema::NodeSet->new()->appendRange( - IMPL::DOM::Schema::Node->new(nodeName => 'ComplexNode', type => 'ComplexNode', minOccur => 0, maxOccur=>'unbounded'), - IMPL::DOM::Schema::Node->new(nodeName => 'ComplexType', type => 'ComplexType', minOccur => 0, maxOccur=>'unbounded'), - IMPL::DOM::Schema::Node->new(nodeName => 'SimpleNode', type => 'SimpleNode', minOccur => 0, maxOccur=>'unbounded'), - IMPL::DOM::Schema::Node->new(nodeName => 'SimpleType', type => 'SimpleType', minOccur => 0, maxOccur=>'unbounded'), - IMPL::DOM::Schema::SimpleNode->new(nodeName => 'Node', minOccur => 0, maxOccur=>'unbounded'), - IMPL::DOM::Schema::SimpleNode->new(nodeName => 'Include', minOccur => 0, maxOccur=>'unbounded') + IMPL::DOM::Schema::Node->new(name => 'ComplexNode', type => 'ComplexNode', minOccur => 0, maxOccur=>'unbounded'), + IMPL::DOM::Schema::Node->new(name => 'ComplexType', type => 'ComplexType', minOccur => 0, maxOccur=>'unbounded'), + IMPL::DOM::Schema::Node->new(name => 'SimpleNode', type => 'SimpleNode', minOccur => 0, maxOccur=>'unbounded'), + IMPL::DOM::Schema::Node->new(name => 'SimpleType', type => 'SimpleType', minOccur => 0, maxOccur=>'unbounded'), + IMPL::DOM::Schema::SimpleNode->new(name => 'Node', minOccur => 0, maxOccur=>'unbounded'), + IMPL::DOM::Schema::SimpleNode->new(name => 'Include', minOccur => 0, maxOccur=>'unbounded') ), ), IMPL::DOM::Schema::ComplexType->new(type => 'NodeSet', nativeType => 'IMPL::DOM::Schema::NodeSet')->appendRange( IMPL::DOM::Schema::NodeSet->new()->appendRange( - IMPL::DOM::Schema::Node->new(nodeName => 'ComplexNode', type => 'ComplexNode', minOccur => 0, maxOccur=>'unbounded'), - IMPL::DOM::Schema::Node->new(nodeName => 'SimpleNode', type => 'SimpleNode', minOccur => 0, maxOccur=>'unbounded'), - IMPL::DOM::Schema::SimpleNode->new(nodeName => 'Node', minOccur => 0, maxOccur=>'unbounded'), + IMPL::DOM::Schema::Node->new(name => 'ComplexNode', type => 'ComplexNode', minOccur => 0, maxOccur=>'unbounded'), + IMPL::DOM::Schema::Node->new(name => 'SimpleNode', type => 'SimpleNode', minOccur => 0, maxOccur=>'unbounded'), + IMPL::DOM::Schema::SimpleNode->new(name => 'Node', minOccur => 0, maxOccur=>'unbounded'), ) ), IMPL::DOM::Schema::ComplexType->new(type => 'NodeList', nativeType => 'IMPL::DOM::Schema::NodeList')->appendRange( IMPL::DOM::Schema::NodeSet->new()->appendRange( - IMPL::DOM::Schema::Node->new(nodeName => 'ComplexNode', type => 'ComplexNode', minOccur => 0, maxOccur=>'unbounded'), - IMPL::DOM::Schema::Node->new(nodeName => 'SimpleNode', type => 'SimpleNode', minOccur => 0, maxOccur=>'unbounded'), - IMPL::DOM::Schema::SimpleNode->new(nodeName => 'Node', minOccur => 0, maxOccur=>'unbounded'), + IMPL::DOM::Schema::Node->new(name => 'ComplexNode', type => 'ComplexNode', minOccur => 0, maxOccur=>'unbounded'), + IMPL::DOM::Schema::Node->new(name => 'SimpleNode', type => 'SimpleNode', minOccur => 0, maxOccur=>'unbounded'), + IMPL::DOM::Schema::SimpleNode->new(name => 'Node', minOccur => 0, maxOccur=>'unbounded'), ) ), IMPL::DOM::Schema::ComplexType->new(type => 'ComplexType', nativeType => 'IMPL::DOM::Schema::ComplexType')->appendRange( IMPL::DOM::Schema::NodeList->new()->appendRange( - IMPL::DOM::Schema::Node->new(nodeName => 'NodeSet', minOccur => 0, type => 'NodeSet'), - IMPL::DOM::Schema::Node->new(nodeName => 'NodeList', minOccur => 0, type => 'NodeSet'), - IMPL::DOM::Schema::SimpleNode->new(nodeName => 'Node', minOccur => 0, maxOccur => 'unbounded') + IMPL::DOM::Schema::Node->new(name => 'NodeSet', minOccur => 0, type => 'NodeSet'), + IMPL::DOM::Schema::Node->new(name => 'NodeList', minOccur => 0, type => 'NodeSet'), + IMPL::DOM::Schema::SimpleNode->new(name => 'Node', minOccur => 0, maxOccur => 'unbounded') ) ), IMPL::DOM::Schema::ComplexType->new(type => 'ComplexNode', nativeType => 'IMPL::DOM::Schema::ComplexNode')->appendRange( IMPL::DOM::Schema::NodeList->new()->appendRange( - IMPL::DOM::Schema::Node->new(nodeName => 'NodeSet', minOccur => 0, type => 'NodeSet'), - IMPL::DOM::Schema::Node->new(nodeName => 'NodeList', minOccur => 0, type => 'NodeSet'), - IMPL::DOM::Schema::SimpleNode->new(nodeName => 'Node', minOccur => 0, maxOccur => 'unbounded') + IMPL::DOM::Schema::Node->new(name => 'NodeSet', minOccur => 0, type => 'NodeSet'), + IMPL::DOM::Schema::Node->new(name => 'NodeList', minOccur => 0, type => 'NodeSet'), + IMPL::DOM::Schema::SimpleNode->new(name => 'Node', minOccur => 0, maxOccur => 'unbounded') ) ), IMPL::DOM::Schema::ComplexType->new(type => 'SimpleType', nativeType => 'IMPL::DOM::Schema::SimpleType')->appendRange( @@ -125,43 +129,48 @@ =back +=head1 DESCRIPTION + +DOM схема - это документ, состоящий из определенных узлов, описывающая структуру +других документов. + =head1 META SCHEMA Схема для описания схемы, эта схема используется для постороения других схем - + - - - - - - + + + + + + - - - + + + - - + + - - + + diff -r 1ca530e5c9c5 -r 267460284fb3 Lib/IMPL/DOM/Schema/AnyNode.pm --- a/Lib/IMPL/DOM/Schema/AnyNode.pm Fri Sep 11 16:30:39 2009 +0400 +++ b/Lib/IMPL/DOM/Schema/AnyNode.pm Tue Sep 22 17:17:38 2009 +0400 @@ -5,7 +5,7 @@ use base qw(IMPL::DOM::Schema::Node); our %CTOR = ( - 'IMPL::DOM::Schema::Node' => sub { nodeName => 'AnyNode'} + 'IMPL::DOM::Schema::Node' => sub { nodeName => 'AnyNode', name=> 'AnyNode'} ); 1; diff -r 1ca530e5c9c5 -r 267460284fb3 Lib/IMPL/DOM/Schema/ComplexNode.pm --- a/Lib/IMPL/DOM/Schema/ComplexNode.pm Fri Sep 11 16:30:39 2009 +0400 +++ b/Lib/IMPL/DOM/Schema/ComplexNode.pm Tue Sep 22 17:17:38 2009 +0400 @@ -12,7 +12,9 @@ } } -__PACKAGE__->PassThroughArgs; +our %CTOR = ( + 'IMPL::DOM::Schema::Node' => sub {my %args = @_; $args{nodeName} ||= 'ComplexNode'; %args } +); sub _getContent { $_[0]->firstChild; diff -r 1ca530e5c9c5 -r 267460284fb3 Lib/IMPL/DOM/Schema/ComplexType.pm --- a/Lib/IMPL/DOM/Schema/ComplexType.pm Fri Sep 11 16:30:39 2009 +0400 +++ b/Lib/IMPL/DOM/Schema/ComplexType.pm Tue Sep 22 17:17:38 2009 +0400 @@ -13,9 +13,10 @@ our %CTOR = ( 'IMPL::DOM::Schema::ComplexNode' => sub { my %args = @_; - $args{nodeName} = 'ComplexNode'; + $args{nodeName} ||= 'ComplexType'; $args{minOccur} = 0; $args{maxOccur} = 'unbounded'; + $args{name} ||= 'ComplexType'; %args } ); diff -r 1ca530e5c9c5 -r 267460284fb3 Lib/IMPL/DOM/Schema/Node.pm --- a/Lib/IMPL/DOM/Schema/Node.pm Fri Sep 11 16:30:39 2009 +0400 +++ b/Lib/IMPL/DOM/Schema/Node.pm Tue Sep 22 17:17:38 2009 +0400 @@ -10,17 +10,21 @@ BEGIN { public property minOccur => prop_all; public property maxOccur => prop_all; - public property type => prop_all + public property type => prop_all; + public property name => prop_all; } -__PACKAGE__->PassThroughArgs; +our %CTOR = ( + 'IMPL::DOM::Node' => sub {my %args = @_; $args{nodeName} ||= 'Node'; %args} +); sub CTOR { my ($this,%args) = @_; - $this->minOccur(defined $args{minOcuur} ? $args{minOcuur} : 1); + $this->minOccur(defined $args{minOccur} ? $args{minOccur} : 1); $this->maxOccur(defined $args{maxOccur} ? $args{maxOccur} : 1); $this->type($args{type}); + $this->name($args{name}) or die new IMPL::InvalidArgumentException('Argument is required','name'); } sub Validate { @@ -28,6 +32,8 @@ if (my $schemaType = $this->type ? $this->rootNode->resolveType($this->type) : undef ) { return $schemaType->Validate($node); + } else { + return (); } } @@ -47,6 +53,6 @@ =head1 DESCRIPTION -Базовый класс для элементов схемы. Содержит в себе базовые методы +Базовый класс для элементов схемы. =cut diff -r 1ca530e5c9c5 -r 267460284fb3 Lib/IMPL/DOM/Schema/NodeList.pm --- a/Lib/IMPL/DOM/Schema/NodeList.pm Fri Sep 11 16:30:39 2009 +0400 +++ b/Lib/IMPL/DOM/Schema/NodeList.pm Tue Sep 22 17:17:38 2009 +0400 @@ -26,7 +26,7 @@ my ($this,$node) = @_; my @nodes = map { - {nodeName => $_->nodeName, anyNode => $_->isa('IMPL::DOM::Schema::AnyNode') , Schema => $_, Min => $_->minOccur eq 'unbounded' ? undef : $_->maxOccur, Max => $_->maxOccur, Seen => 0 } + {nodeName => $_->name, anyNode => $_->isa('IMPL::DOM::Schema::AnyNode') , Schema => $_, Min => $_->minOccur eq 'unbounded' ? undef : $_->maxOccur, Max => $_->maxOccur, Seen => 0 } } @{$this->childNodes}; my $info = shift @nodes; diff -r 1ca530e5c9c5 -r 267460284fb3 Lib/IMPL/DOM/Schema/NodeSet.pm --- a/Lib/IMPL/DOM/Schema/NodeSet.pm Fri Sep 11 16:30:39 2009 +0400 +++ b/Lib/IMPL/DOM/Schema/NodeSet.pm Tue Sep 22 17:17:38 2009 +0400 @@ -19,7 +19,7 @@ my ($this,%args) = @_; $this->messageMax( $args{messageMax} || 'Too many %Node.nodeName% nodes'); - $this->messageMin( $args{messageMin} || '%Schema.nodeName% nodes expected'); + $this->messageMin( $args{messageMin} || '%Schema.name% nodes expected'); $this->messageUnexpected( $args{messageUnexpected} || 'A %Node.nodeName% isn\'t allowed here'); } @@ -34,7 +34,7 @@ if ($_->isa('IMPL::DOM::Schema::AnyNode')) { $anyNode = {Schema => $_, Min => $_->minOccur, Max => $_->maxOccur eq 'unbounded' ? undef : $_->maxOccur , Seen => 0 }; } else { - $nodes{$_->nodeName} = {Schema => $_, Min => $_->minOccur, Max => $_->maxOccur eq 'unbounded' ? undef : $_->maxOccur , Seen => 0 }; + $nodes{$_->name} = {Schema => $_, Min => $_->minOccur, Max => $_->maxOccur eq 'unbounded' ? undef : $_->maxOccur , Seen => 0 }; } } diff -r 1ca530e5c9c5 -r 267460284fb3 Lib/IMPL/DOM/Schema/SimpleNode.pm --- a/Lib/IMPL/DOM/Schema/SimpleNode.pm Fri Sep 11 16:30:39 2009 +0400 +++ b/Lib/IMPL/DOM/Schema/SimpleNode.pm Tue Sep 22 17:17:38 2009 +0400 @@ -4,7 +4,9 @@ use base qw(IMPL::DOM::Schema::Node); -__PACKAGE__->PassThroughArgs; +our %CTOR = ( + 'IMPL::DOM::Schema::Node' => sub {my %args = @_; $args{nodeName} ||= 'SimpleNode'; %args} +); sub Validate { my ($this,$node) = @_; diff -r 1ca530e5c9c5 -r 267460284fb3 Lib/IMPL/DOM/Schema/SimpleType.pm --- a/Lib/IMPL/DOM/Schema/SimpleType.pm Fri Sep 11 16:30:39 2009 +0400 +++ b/Lib/IMPL/DOM/Schema/SimpleType.pm Tue Sep 22 17:17:38 2009 +0400 @@ -13,9 +13,10 @@ our %CTOR = ( 'IMPL::DOM::Schema::SimpleNode' => sub { my %args = @_; - $args{nodeName} = 'ComplexNode'; + $args{nodeName} = 'SimpleType'; $args{minOccur} = 0; $args{maxOccur} = 'unbounded'; + $args{name} ||= 'SimpleType'; %args } ); diff -r 1ca530e5c9c5 -r 267460284fb3 impl.kpf --- a/impl.kpf Fri Sep 11 16:30:39 2009 +0400 +++ b/impl.kpf Tue Sep 22 17:17:38 2009 +0400 @@ -170,6 +170,32 @@ default + + + + + + 9011 + + + Lib/IMPL/DOM/Schema/ComplexNode.pm + + Perl + + + + application/x-www-form-urlencoded + GET + 1 + 0 + 0 + + + enabled + + + default + @@ -326,6 +352,32 @@ default + + + + + + 9011 + + + _test/Test/DOM/Schema.pm + + Perl + + + + application/x-www-form-urlencoded + GET + 1 + 0 + 0 + + + enabled + + + default +