File Coverage

Bio/SearchIO/axt.pm
Criterion Covered Total %
statement 128 152 84.2
branch 27 46 58.7
condition 7 15 46.6
subroutine 18 21 85.7
pod 11 12 91.6
total 191 246 77.6


line stmt bran cond sub pod time code
1             #
2             # BioPerl module for Bio::SearchIO::axt
3             #
4             # Please direct questions and support issues to
5             #
6             # Cared for by Jason Stajich
7             #
8             # Copyright Jason Stajich
9             #
10             # You may distribute this module under the same terms as perl itself
11              
12             # POD documentation - main docs before the code
13              
14             =head1 NAME
15              
16             Bio::SearchIO::axt - a parser for axt format reports
17              
18             =head1 SYNOPSIS
19              
20             use Bio::SearchIO;
21             my $parser = Bio::SearchIO->new(-format => 'axt',
22             -file => 't/data/report.blastz');
23             while( my $result = $parser->next_result ) {
24             while( my $hit = $result->next_hit ) {
25             while( my $hsp = $hit->next_hsp ) {
26             }
27             }
28             }
29              
30             =head1 DESCRIPTION
31              
32             This is a parser and event-generator for AXT format reports. BLASTZ
33             reports (Schwartz et al,(2003) Genome Research, 13:103-107) are normally
34             in LAV format but are commonly post-processed to AXT format; many precomputed
35             BLASTZ reports, such as those found in the UCSC Genome
36             Browser, are in AXT format. This parser will also parse any
37             AXT format produced from any lav report and directly out of BLAT.
38              
39             =head1 FEEDBACK
40              
41             =head2 Mailing Lists
42              
43             User feedback is an integral part of the evolution of this and other
44             Bioperl modules. Send your comments and suggestions preferably to
45             the Bioperl mailing list. Your participation is much appreciated.
46              
47             bioperl-l@bioperl.org - General discussion
48             http://bioperl.org/wiki/Mailing_lists - About the mailing lists
49              
50             =head2 Support
51              
52             Please direct usage questions or support issues to the mailing list:
53              
54             I
55              
56             rather than to the module maintainer directly. Many experienced and
57             reponsive experts will be able look at the problem and quickly
58             address it. Please include a thorough description of the problem
59             with code and data examples if at all possible.
60              
61             =head2 Reporting Bugs
62              
63             Report bugs to the Bioperl bug tracking system to help us keep track
64             of the bugs and their resolution. Bug reports can be submitted via the
65             web:
66              
67             https://github.com/bioperl/bioperl-live/issues
68              
69             =head1 AUTHOR - Jason Stajich
70              
71             Email jason-at-bioperl.org
72              
73             =head1 APPENDIX
74              
75             The rest of the documentation details each of the object methods.
76             Internal methods are usually preceded with a _
77              
78             =cut
79              
80              
81             # Let the code begin...
82              
83              
84             package Bio::SearchIO::axt;
85 1     1   4 use vars qw(%MODEMAP %MAPPING @STATES $GAPCHAR);
  1         1  
  1         57  
86 1     1   3 use strict;
  1         2  
  1         16  
87              
88 1     1   325 use Bio::Search::Result::ResultFactory;
  1         2  
  1         19  
89 1     1   256 use Bio::Search::HSP::HSPFactory;
  1         1  
  1         21  
90 1     1   4 use base qw(Bio::SearchIO);
  1         1  
  1         51  
91              
92 1     1   374 use POSIX;
  1         4365  
  1         6  
93              
94             BEGIN {
95             # mapping of NCBI Blast terms to Bioperl hash keys
96 1     1   1977 %MODEMAP = ('AXTOutput' => 'result',
97             'Hit' => 'hit',
98             'Hsp' => 'hsp'
99             );
100 1         1 $GAPCHAR = '-';
101 1         1218 %MAPPING =
102             (
103             'Hsp_score' => 'HSP-score',
104             'Hsp_query-from' => 'HSP-query_start',
105             'Hsp_query-to' => 'HSP-query_end',
106             'Hsp_hit-from' => 'HSP-hit_start',
107             'Hsp_hit-to' => 'HSP-hit_end',
108             'Hsp_positive' => 'HSP-conserved',
109             'Hsp_identity' => 'HSP-identical',
110             'Hsp_gaps' => 'HSP-hsp_gaps',
111             'Hsp_hitgaps' => 'HSP-hit_gaps',
112             'Hsp_querygaps' => 'HSP-query_gaps',
113             'Hsp_qseq' => 'HSP-query_seq',
114             'Hsp_hseq' => 'HSP-hit_seq',
115             'Hsp_midline' => 'HSP-homology_seq', # ignoring this for now
116             'Hsp_align-len' => 'HSP-hsp_length',
117            
118             'Hit_id' => 'HIT-name',
119             'AXTOutput_query-def'=> 'RESULT-query_name',
120             );
121             }
122              
123             =head2 new
124              
125             Title : new
126             Usage : my $obj = Bio::SearchIO::axt->new();
127             Function: Builds a new Bio::SearchIO::axt object
128             Returns : an instance of Bio::SearchIO::axt
129             Args :
130              
131              
132             =cut
133              
134             sub new {
135 1     1 1 2 my($class,@args) = @_;
136 1         7 my $self = $class->SUPER::new(@args);
137 1         7 return $self;
138             }
139              
140             =head2 next_result
141              
142             Title : next_result
143             Usage : my $hit = $searchio->next_result;
144             Function: Returns the next Result from a search
145             Returns : Bio::Search::Result::ResultI object
146             Args : none
147              
148             =cut
149              
150             sub next_result{
151 1     1 1 6 my ($self) = @_;
152 1         5 local $/ = "\n";
153 1         2 local $_;
154              
155 1         0 my ($curquery,$curhit);
156 1         3 $self->start_document();
157 1         1 my @hit_signifs;
158 1         6 while( defined ($_ = $self->_readline )) {
159 3 100       12 next if (/^\s+$/);
160 2 50       11 if( m/^(\d+)\s+ # alignment number - we'll throw this away anyways
161             (\S+)\s+ # Query name
162             (\d+)\s+(\d+)\s+ # Query start Query end (always + strand, 0 based)
163             (\S+)\s+ # Hit name
164             (\d+)\s+(\d+)\s+ # Hit start Hit end (0 based)
165             ([\-\+])\s+ # Hit strand
166             ([\d\.\-]+)\s+ # Score
167             /ox ) {
168 2         10 my ($alnnum, $qname,$qstart,$qend, $hname,
169             $hstart,$hend,$hstrand, $score) = ($1,$2,$3,$4,$5,
170             $6,$7,$8,$9);
171 2         3 $self->{'_reporttype'} = 'AXT';
172             # Jim's code is 0 based
173             # yes, but axt format is one-based, see bug 3145 - cjfields 10-11-10
174             #$qstart++; $qend++; $hstart++; $hend++;
175 2 50 66     12 if( defined $curquery && $curquery ne $qname ) {
176 0         0 $self->end_element({'Name' => 'Hit'});
177 0         0 $self->_pushback($_);
178 0         0 $self->end_element({'Name' => 'AXTOutput'});
179 0         0 return $self->end_document();
180             }
181            
182 2 50 66     9 if( defined $curhit &&
    100          
183             $curhit ne $hname) {
184             # slight duplication here -- keep these in SYNC
185 0         0 $self->end_element({'Name' => 'Hit'});
186 0         0 $self->start_element({'Name' => 'Hit'});
187 0         0 $self->element({'Name' => 'Hit_id',
188             'Data' => $hname});
189             } elsif ( ! defined $curquery ) {
190 1         5 $self->start_element({'Name' => 'AXTOutput'});
191 1         2 $self->{'_result_count'}++;
192 1         5 $self->element({'Name' => 'AXTOutput_query-def',
193             'Data' => $qname });
194            
195 1         3 $self->start_element({'Name' => 'Hit'});
196 1         4 $self->element({'Name' => 'Hit_id',
197             'Data' => $hname});
198             }
199 2         4 $self->start_element({'Name' => 'Hsp'});
200 2         4 my $queryalign = $self->_readline;
201 2         4 my $hitalign = $self->_readline;
202 2         3 chomp($queryalign);
203 2         3 chomp($hitalign);
204 2         3 my $alnlen = length($queryalign);
205 2         12 my $qgapnum = ( $queryalign =~ s/\Q$GAPCHAR/$GAPCHAR/g);
206 2         6 my $hgapnum = ( $hitalign =~ s/\Q$GAPCHAR/$GAPCHAR/g);
207 2         2 my $totalgaps = ($qgapnum + $hgapnum);
208            
209 2 50       5 if( $hstrand eq '-' ) { # strand gets inferred by start/end
210 2         3 ($hstart,$hend) = ($hend,$hstart);
211             }
212 2         7 $self->element({'Name' => 'Hsp_score',
213             'Data' => $score});
214 2         5 $self->element({'Name' => 'Hsp_query-from',
215             'Data' => $qstart});
216 2         6 $self->element({'Name' => 'Hsp_query-to',
217             'Data' => $qend});
218 2         9 $self->element({'Name' => 'Hsp_hit-from',
219             'Data' => $hstart});
220 2         7 $self->element({'Name' => 'Hsp_hit-to',
221             'Data' => $hend});
222 2         4 $self->element({'Name' => 'Hsp_gaps',
223             'Data' => $qgapnum + $hgapnum});
224 2         6 $self->element({'Name' => 'Hsp_querygaps',
225             'Data' => $qgapnum});
226 2         9 $self->element({'Name' => 'Hsp_hitgaps',
227             'Data' => $hgapnum});
228            
229 2         6 $self->element({'Name' => 'Hsp_identity',
230             'Data' => $alnlen - $totalgaps});
231 2         5 $self->element({'Name' => 'Hsp_positive',
232             'Data' => $alnlen - $totalgaps});
233 2         5 $self->element({'Name' => 'Hsp_qseq',
234             'Data' => $queryalign});
235 2         6 $self->element({'Name' => 'Hsp_hseq',
236             'Data' => $hitalign});
237            
238 2         6 $self->end_element({'Name' => 'Hsp'});
239 2         4 $curquery = $qname;
240 2         6 $curhit = $hname;
241             }
242             }
243             # fence post
244 1 50       3 if( defined $curquery ) {
245 1         3 $self->end_element({'Name' => 'Hit'});
246 1         3 $self->end_element({'Name' => 'AXTOutput'});
247 1         3 return $self->end_document();
248             }
249 0         0 return;
250             }
251              
252             sub _initialize {
253 1     1   2 my ($self,@args) = @_;
254 1         4 $self->SUPER::_initialize(@args);
255 1         6 $self->_eventHandler->register_factory('result', Bio::Search::Result::ResultFactory->new(-type => 'Bio::Search::Result::GenericResult'));
256              
257 1         4 $self->_eventHandler->register_factory('hsp', Bio::Search::HSP::HSPFactory->new(-type => 'Bio::Search::HSP::GenericHSP'));
258             }
259              
260              
261             =head2 start_element
262              
263             Title : start_element
264             Usage : $eventgenerator->start_element
265             Function: Handles a start element event
266             Returns : none
267             Args : hashref with at least 2 keys 'Data' and 'Name'
268              
269              
270             =cut
271              
272             sub start_element{
273 30     30 1 20 my ($self,$data) = @_;
274             # we currently don't care about attributes
275 30         23 my $nm = $data->{'Name'};
276 30 100       81 if( my $type = $MODEMAP{$nm} ) {
277 4         6 $self->_mode($type);
278 4 50       8 if( $self->_eventHandler->will_handle($type) ) {
279 4         11 my $func = sprintf("start_%s",lc $type);
280 4         6 $self->_eventHandler->$func($data->{'Attributes'});
281             }
282 4         5 unshift @{$self->{'_elements'}}, $type;
  4         13  
283             }
284 30 100       43 if($nm eq 'AXTOutput') {
285 1         2 $self->{'_values'} = {};
286 1         2 $self->{'_result'}= undef;
287 1         2 $self->{'_mode'} = '';
288             }
289              
290             }
291              
292             =head2 end_element
293              
294             Title : start_element
295             Usage : $eventgenerator->end_element
296             Function: Handles an end element event
297             Returns : none
298             Args : hashref with at least 2 keys 'Data' and 'Name'
299              
300              
301             =cut
302              
303             sub end_element {
304 30     30 1 21 my ($self,$data) = @_;
305 30         26 my $nm = $data->{'Name'};
306 30         17 my $rc;
307             # Hsp are sort of weird, in that they end when another
308             # object begins so have to detect this in end_element for now
309            
310 30 100       51 if( my $type = $MODEMAP{$nm} ) {
    50          
311 4 50       9 if( $self->_eventHandler->will_handle($type) ) {
312 4         10 my $func = sprintf("end_%s",lc $type);
313             $rc = $self->_eventHandler->$func($self->{'_reporttype'},
314 4         7 $self->{'_values'});
315             }
316 4         5 shift @{$self->{'_elements'}};
  4         5  
317              
318             } elsif( $MAPPING{$nm} ) {
319 26 50       27 if ( ref($MAPPING{$nm}) =~ /hash/i ) {
320 0         0 my $key = (keys %{$MAPPING{$nm}})[0];
  0         0  
321 0         0 $self->{'_values'}->{$key}->{$MAPPING{$nm}->{$key}} = $self->{'_last_data'};
322             } else {
323 26         50 $self->{'_values'}->{$MAPPING{$nm}} = $self->{'_last_data'};
324             }
325             } else {
326 0         0 $self->warn( "unknown nm $nm ignoring\n");
327             }
328 30         28 $self->{'_last_data'} = ''; # remove read data if we are at
329             # end of an element
330 30 100       35 $self->{'_result'} = $rc if( $nm eq 'AXTOutput' );
331 30         27 return $rc;
332              
333             }
334              
335             =head2 element
336              
337             Title : element
338             Usage : $eventhandler->element({'Name' => $name, 'Data' => $str});
339             Function: Convience method that calls start_element, characters, end_element
340             Returns : none
341             Args : Hash ref with the keys 'Name' and 'Data'
342              
343              
344             =cut
345              
346             sub element{
347 26     26 1 22 my ($self,$data) = @_;
348 26         25 $self->start_element($data);
349 26         27 $self->characters($data);
350 26         28 $self->end_element($data);
351             }
352              
353              
354             =head2 characters
355              
356             Title : characters
357             Usage : $eventgenerator->characters($str)
358             Function: Send a character events
359             Returns : none
360             Args : string
361              
362              
363             =cut
364              
365             sub characters{
366 26     26 1 19 my ($self,$data) = @_;
367              
368 26 50       32 return unless ( defined $data->{'Data'} );
369 26 50       45 if( $data->{'Data'} =~ /^\s+$/ ) {
370 0 0       0 return unless $data->{'Name'} =~ /Hsp\_(midline|qseq|hseq)/;
371             }
372              
373 26 100 100     27 if( $self->in_element('hsp') &&
374             $data->{'Name'} =~ /Hsp\_(qseq|hseq|midline)/ ) {
375            
376 4         14 $self->{'_last_hspdata'}->{$data->{'Name'}} .= $data->{'Data'};
377             }
378            
379 26         32 $self->{'_last_data'} = $data->{'Data'};
380             }
381              
382             =head2 _mode
383              
384             Title : _mode
385             Usage : $obj->_mode($newval)
386             Function:
387             Example :
388             Returns : value of _mode
389             Args : newvalue (optional)
390              
391              
392             =cut
393              
394             sub _mode{
395 4     4   4 my ($self,$value) = @_;
396 4 50       6 if( defined $value) {
397 4         5 $self->{'_mode'} = $value;
398             }
399 4         5 return $self->{'_mode'};
400             }
401              
402             =head2 within_element
403              
404             Title : within_element
405             Usage : if( $eventgenerator->within_element($element) ) {}
406             Function: Test if we are within a particular element
407             This is different than 'in' because within can be tested
408             for a whole block.
409             Returns : boolean
410             Args : string element name
411              
412              
413             =cut
414              
415             sub within_element{
416 0     0 1 0 my ($self,$name) = @_;
417             return 0 if ( ! defined $name &&
418             ! defined $self->{'_elements'} ||
419 0 0 0     0 scalar @{$self->{'_elements'}} == 0) ;
  0   0     0  
420 0         0 foreach ( @{$self->{'_elements'}} ) {
  0         0  
421 0 0       0 if( $_ eq $name ) {
422 0         0 return 1;
423             }
424             }
425 0         0 return 0;
426             }
427              
428             =head2 in_element
429              
430             Title : in_element
431             Usage : if( $eventgenerator->in_element($element) ) {}
432             Function: Test if we are in a particular element
433             This is different than 'in' because within can be tested
434             for a whole block.
435             Returns : boolean
436             Args : string element name
437              
438              
439             =cut
440              
441             sub in_element{
442 26     26 1 26 my ($self,$name) = @_;
443 26 50       34 return 0 if ! defined $self->{'_elements'}->[0];
444 26         108 return ( $self->{'_elements'}->[0] eq $name)
445             }
446              
447              
448             =head2 start_document
449              
450             Title : start_document
451             Usage : $eventgenerator->start_document
452             Function: Handles a start document event
453             Returns : none
454             Args : none
455              
456              
457             =cut
458              
459             sub start_document{
460 1     1 1 2 my ($self) = @_;
461 1         2 $self->{'_lasttype'} = '';
462 1         2 $self->{'_values'} = {};
463 1         2 $self->{'_result'}= undef;
464 1         2 $self->{'_mode'} = '';
465 1         2 $self->{'_elements'} = [];
466             }
467              
468              
469             =head2 end_document
470              
471             Title : end_document
472             Usage : $eventgenerator->end_document
473             Function: Handles an end document event
474             Returns : Bio::Search::Result::ResultI object
475             Args : none
476              
477              
478             =cut
479              
480             sub end_document{
481 1     1 1 1 my ($self,@args) = @_;
482 1         5 return $self->{'_result'};
483             }
484              
485             =head2 result_count
486              
487             Title : result_count
488             Usage : my $count = $searchio->result_count
489             Function: Returns the number of results we have processed
490             Returns : integer
491             Args : none
492              
493              
494             =cut
495              
496             sub result_count {
497 0     0 1   my $self = shift;
498 0           return $self->{'_result_count'};
499             }
500              
501 0     0 0   sub report_count { shift->result_count }
502              
503              
504              
505             1;
506