7 use Data::Dump qw/dump/;
8 use Carp qw/confess cluck/;
11 #use Devel::LeakTrace::Fast;
15 CWMP::Request - parse SOAP request metods
19 All methods described below call triggers with same name
23 our $state; # FIXME check this!
26 #_default => 'content trim',
28 my ($tag_name, $tag_hash, $context, $parent_data) = @_;
29 warn dump( $tag_name, $tag_hash, $context );
32 my ($tag_name, $tag_hash, $context, $parent_data) = @_;
33 $state->{ID} = $tag_hash->{_content};
34 chomp( $state->{ID} );
38 my ($tag_name, $tag_hash, $context, $parent_data) = @_;
39 foreach my $name ( keys %$tag_hash ) {
40 next if $name eq '_content';
42 $key =~ s/^\w+://; # stip namespace
43 $state->{DeviceId}->{ $key } = _tag( $tag_hash, $name, '_content' );
46 'EventStruct' => sub {
47 my ($tag_name, $tag_hash, $context, $parent_data) = @_;
48 push @{ $state->{EventStruct} }, $tag_hash->{EventCode}->{_content};
50 qr/(MaxEnvelopes|CurrentTime|RetryCount)/ => sub {
51 my ($tag_name, $tag_hash, $context, $parent_data) = @_;
52 $state->{$tag_name} = $tag_hash->{_content};
54 'ParameterValueStruct' => sub {
55 my ($tag_name, $tag_hash, $context, $parent_data) = @_;
56 # Name/Value tags must be case insnesitive
57 my $value = (grep( /value/i, keys %$tag_hash ))[0];
58 $state->{Parameter}->{ _tag($tag_hash, 'Name', '_content') } = _tag($tag_hash, 'Value', '_content' );
59 $state->{_trigger} = 'ParameterValue';
66 Generate InformResponse to CPE
72 $state->{_dispatch} = 'InformResponse'; # what reponse to call
73 $state->{_trigger} = 'Inform';
76 =head2 GetRPCMethodsResponse
81 qr/^(?:^\w+:)*string$/ => 'content array',
83 my ($tag_name, $tag_hash, $context, $parent_data) = @_;
84 $state->{MethodList} = _tag( $tag_hash, 'string' );
85 $state->{_trigger} = 'GetRPCMethodsResponse';
88 =head2 GetParameterNamesResponse
93 'ParameterInfoStruct' => sub {
94 my ($tag_name, $tag_hash, $context, $parent_data) = @_;
95 my $name = _tag($tag_hash, 'Name', '_content');
96 my $writable = _tag($tag_hash, 'Writable', '_content' );
98 confess "need state" unless ( $state ); # don't remove!
100 $state->{ParameterInfo}->{$name} = $writable;
102 #warn "## state = dump( $state ), "\n";
104 $state->{_trigger} = 'GetParameterNamesResponse';
107 =head2 GetParameterAttributesResponse
112 'ParameterAttributeStruct' => sub {
113 my ($tag_name, $tag_hash, $context, $parent_data) = @_;
114 warn dump( $tag_name, $tag_hash, $context );
116 confess "need state" unless ( $state ); # don't remove!
118 my $name = _tag($tag_hash, 'Name', '_content');
121 $state->{ParameterAttribute}->{$name} = {
122 Notification => _tag($tag_hash, 'Notification', '_content' ),
123 AccessList => _tag($tag_hash, 'AccessList', 'string' ),
126 $state->{_trigger} = 'GetParameterAttributesResponse';
135 my ($tag_name, $tag_hash, $context, $parent_data) = @_;
137 FaultCode => _tag( $tag_hash, 'FaultCode', '_content' ),
138 FaultString => _tag( $tag_hash, 'FaultString', '_content' ),
140 warn "FAULT: ", $state->{Fault}->{FaultCode}, " ", $state->{Fault}->{FaultString}, "\n";
141 $state->{_trigger} = 'Fault';
148 my $state = CWMP::Request->parse( "<soap>request</soap>" );
155 my $xml = shift || confess "no xml?";
159 my $parser = XML::Rules->new(
161 # '^division_name,fax' => 'skip',
164 'http://schemas.xmlsoap.org/soap/envelope/' => 'soapenv',
165 'http://schemas.xmlsoap.org/soap/encoding/' => 'soap',
166 'http://www.w3.org/2001/XMLSchema' => 'xsd',
167 'http://www.w3.org/2001/XMLSchema-instance' => 'xsi',
168 'urn:dslforum-org:cwmp-1-0' => '',
173 # warn "## created $parser\n";
175 $parser->parsestring( $xml );
179 if ( my $trigger = $state->{_trigger} ) {
180 warn "### call_trigger( $trigger )\n";
181 $self->call_trigger( $trigger, $state );
183 # XXX propagate _trigger (useful for symlinks)
190 Get value of tag. Tag name is case insensitive (don't ask why),
191 we ignore namespaces and can take optional C<sub_key>
192 (usually C<_content>).
194 _tag( $tag_hash, $name, $sub_key )
199 my ( $tag_hash, $name, $sub_key ) = @_;
200 confess "need hash as first argument" unless ( ref $tag_hash eq 'HASH' );
201 $name = (grep { m/^(?:\w+:)*$name$/i } keys %$tag_hash )[0];
202 # $name =~ s/^\w+://;
203 if ( defined $tag_hash->{$name} ) {
204 if ( ! defined $sub_key ) {
205 return $tag_hash->{$name};
206 } elsif ( defined $tag_hash->{$name}->{$sub_key} ) {
207 return $tag_hash->{$name}->{$sub_key};
209 return if ( $name =~ m/^value$/i );
210 warn "can't find '$name/$sub_key' in ", dump( $tag_hash );
214 warn "can't find '$name' in ", dump( $tag_hash );