File Coverage

Bio/AlignIO/metafasta.pm
Criterion Covered Total %
statement 74 86 86.0
branch 14 30 46.6
condition 3 9 33.3
subroutine 12 12 100.0
pod 3 3 100.0
total 106 140 75.7


line stmt bran cond sub pod time code
1             #
2             # BioPerl module for Bio::AlignIO::metafasta
3             #
4             # Copyright Heikki Lehvaslaiho
5             #
6             # You may distribute this module under the same terms as perl itself
7             #
8             # POD documentation - main docs before the code
9              
10             =head1 NAME
11              
12             Bio::AlignIO::metafasta - Metafasta MSA Sequence input/output stream
13              
14             =head1 SYNOPSIS
15              
16             Do not use this module directly. Use it via the L class.
17              
18             =head1 DESCRIPTION
19              
20             This object can transform L objects to and from
21             metafasta flat file databases.
22              
23             The format of a metafasta file is
24              
25             >test/1-25
26             ABCDEFHIJKLMNOPQRSTUVWXYZ
27             &charge
28             NBNAANCNJCNNNONNCNNUNNXNZ
29             &chemical
30             LBSAARCLJCLSMOIMCHHULRXRZ
31              
32             where the sequence block is followed by one or several meta blocks.
33             Each meta block starts with the ampersand character '&' in the first
34             column and is immediately followed by the name of the meta data which
35             continues until the new line. The meta data follows it. All
36             characters, except new line, are important in meta data.
37              
38             =head1 SEE ALSO
39              
40             L
41              
42             =head1 FEEDBACK
43              
44             =head2 Support
45              
46             Please direct usage questions or support issues to the mailing list:
47              
48             I
49              
50             rather than to the module maintainer directly. Many experienced and
51             reponsive experts will be able look at the problem and quickly
52             address it. Please include a thorough description of the problem
53             with code and data examples if at all possible.
54              
55             =head2 Reporting Bugs
56              
57             Report bugs to the Bioperl bug tracking system to help us keep track
58             the bugs and their resolution. Bug reports can be submitted via the
59             web:
60              
61             https://github.com/bioperl/bioperl-live/issues
62              
63             =head1 AUTHOR - Heikki Lehvaslaiho
64              
65             Email heikki-at-bioperl-dot-org
66              
67             =head1 APPENDIX
68              
69             The rest of the documentation details each of the object
70             methods. Internal methods are usually preceded with a _
71              
72             =cut
73              
74             # Let the code begin...
75              
76             package Bio::AlignIO::metafasta;
77 3     3   372 use vars qw($WIDTH);
  3         4  
  3         118  
78 3     3   10 use strict;
  3         3  
  3         51  
79              
80 3     3   478 use Bio::SimpleAlign;
  3         5  
  3         70  
81 3     3   348 use Bio::Seq::Meta;
  3         3  
  3         62  
82 3     3   11 use Bio::Seq::SeqFactory;
  3         4  
  3         42  
83 3     3   516 use Bio::Seq::SeqFastaSpeedFactory;
  3         4  
  3         80  
84              
85 3     3   15 use base qw(Bio::AlignIO);
  3         4  
  3         466  
86              
87 3     3   1828 BEGIN { $WIDTH = 60}
88              
89             sub _initialize {
90 5     5   13 my($self,@args) = @_;
91 5         24 $self->SUPER::_initialize(@args);
92 5         19 my ($width) = $self->_rearrange([qw(WIDTH)], @args);
93 5 50       19 $width && $self->width($width);
94             }
95              
96             =head2 next_aln
97              
98             Title : next_aln
99             Usage : $aln = $stream->next_aln()
100             Function: returns the next alignment in the stream.
101             Returns : L object - returns 0 on end of file
102             or on error
103             Args : NONE
104              
105             =cut
106              
107             sub next_aln {
108 3     3 1 10 my( $self ) = @_;
109 3         3 my $seq;
110             my $alphabet;
111 3         12 local $/ = "\n>";
112              
113 3         17 my $aln = Bio::SimpleAlign->new();
114              
115 3         20 while(defined (my $entry = $self->_readline)) {
116 6         9 chomp($entry);
117 6 50       24 if ($entry =~ m/\A\s*\Z/s) { # very first one
118 0 0       0 return unless $entry = $self->_readline;
119 0         0 chomp($entry);
120             }
121 6         13 $entry =~ s/^>//;
122              
123 6         16 my ($top,$sequence) = split(/\n/,$entry,2);
124 6 50       15 defined $sequence && $sequence =~ s/>//g;
125              
126 6         7 my @metas;
127 6         14 ($sequence, @metas) = split /\n&/, $sequence;
128              
129 6         9 my ($id, $start, $end);
130 6 50       27 if ( $top =~ /(\S+)\/(\d+)-(\d+)/ ) {
    0          
131 6         12 $id = $1;
132 6         7 $start = $2;
133 6         8 $end = $3;
134             }
135             elsif ($top =~ /(\S+)/) {
136 0         0 $id = $1;
137 0         0 $start = 1;
138 0         0 $end = length($sequence);
139             }
140              
141 6 50       58 defined $sequence && $sequence =~ s/\s//g; # Remove whitespace
142              
143 6         16 $seq = Bio::Seq::Meta->new('-seq' => $sequence,
144             '-display_id' => $id,
145             '-start' => $start,
146             '-end' => $end,
147             '-alphabet' => $self->alphabet,
148             );
149              
150 6         10 foreach my $meta (@metas) {
151 6         14 my ($name,$string) = split /\n/, $meta;
152 6         8 $string =~ s/\n//g; # Remove newlines, spaces are important
153 6         15 $seq->named_meta($name, $string);
154             }
155              
156 6         20 $aln->add_seq($seq);
157            
158             # alignment needs seqs all the same length, pad with gaps
159 6         13 my $alnlen = $aln->length;
160 6         13 foreach my $seq ( $aln->each_seq ) {
161 9 50       14 if ( $seq->length < $alnlen ) {
162 0         0 my ($diff) = ($alnlen - $seq->length);
163 0         0 $seq->seq( $seq->seq() . "-" x $diff);
164             }
165             }
166             }
167 3 50       13 return $aln if $aln->num_sequences;
168 0         0 return;
169             }
170              
171             =head2 write_aln
172              
173             Title : write_aln
174             Usage : $stream->write_aln(@aln)
175             Function: writes the $aln object into the stream in fasta format
176             Returns : 1 for success and 0 for error
177             Args : L object
178              
179             =cut
180              
181             sub write_aln {
182 2     2 1 4 my ($self,@aln) = @_;
183 2         7 my $width = $self->width;
184              
185 2         6 foreach my $aln (@aln) {
186 2 50 33     21 if( ! $aln || ! $aln->isa('Bio::Align::AlignI') ) {
187 0         0 $self->warn("Must provide a Bio::Align::AlignI object when calling write_aln");
188 0         0 next;
189             }
190 2         7 foreach my $seq ( $aln->each_seq() ) {
191 8         16 my $name = $aln->displayname($seq->get_nse);
192              
193 8         17 my $str = $seq->seq();
194 8 50       14 if(length($str) > 0) {
195 8         106 $str =~ s/(.{1,$width})/$1\n/g;
196             } else {
197 0         0 $str = "\n";
198             }
199 8 50       33 $self->_print (">",$name,"\n",$str) or return;
200 8 100       42 if ($seq->isa('Bio::Seq::MetaI')) {
201 2         5 foreach my $meta ($seq->meta_names) {
202 2         6 my $str = $seq->named_meta($meta);
203 2         25 $str =~ s/(.{1,$width})/$1\n/g;
204 2         6 $self->_print ("&",$meta,"\n",$str);
205             }
206             }
207             }
208             }
209 2 50 33     9 $self->flush if $self->_flush_on_write && defined $self->_fh;
210 2         8 return 1;
211             }
212              
213              
214             =head2 width
215              
216             Title : width
217             Usage : $obj->width($newval)
218             Function: Get/Set the line width for METAFASTA output
219             Returns : value of width
220             Args : newvalue (optional)
221              
222              
223             =cut
224              
225             sub width{
226 2     2 1 3 my ($self,$value) = @_;
227 2 50       7 if( defined $value) {
228 0         0 $self->{'width'} = $value;
229             }
230 2   33     9 return $self->{'width'} || $WIDTH;
231             }
232              
233             1;