File Coverage

blib/lib/App/Fasops/Command/join.pm
Criterion Covered Total %
statement 82 87 94.2
branch 17 22 77.2
condition 7 9 77.7
subroutine 12 12 100.0
pod 6 6 100.0
total 124 136 91.1


line stmt bran cond sub pod time code
1             package App::Fasops::Command::join;
2 20     20   13878 use strict;
  20         58  
  20         640  
3 20     20   159 use warnings;
  20         446  
  20         550  
4 20     20   111 use autodie;
  20         46  
  20         109  
5              
6 20     20   105035 use App::Fasops -command;
  20         534  
  20         243  
7 20     20   6726 use App::RL::Common;
  20         48  
  20         520  
8 20     20   109 use App::Fasops::Common;
  20         36  
  20         20656  
9              
10             sub abstract {
11 2     2 1 51 return 'join multiple blocked fasta files by common target';
12             }
13              
14             sub opt_spec {
15             return (
16 4     4 1 22 [ "outfile|o=s", "Output filename. [stdout] for screen" ],
17             [ "name|n=s", "According to this species. Default is the first one" ],
18             { show_defaults => 1, }
19             );
20             }
21              
22             sub usage_desc {
23 4     4 1 43622 return "fasops join [options] [more infiles]";
24             }
25              
26             sub description {
27 1     1 1 827 my $desc;
28 1         4 $desc .= ucfirst(abstract) . ".\n";
29 1         2 $desc .= <<'MARKDOWN';
30              
31             * are paths to axt files, .axt.gz is supported
32             * infile == stdin means reading from STDIN
33              
34             MARKDOWN
35              
36 1         3 return $desc;
37             }
38              
39             sub validate_args {
40 3     3 1 2718 my ( $self, $opt, $args ) = @_;
41              
42 3 100       7 if ( !@{$args} ) {
  3         11  
43 1         3 my $message = "This command need one or more input files.\n\tIt found";
44 1         3 $message .= sprintf " [%s]", $_ for @{$args};
  1         3  
45 1         4 $message .= ".\n";
46 1         9 $self->usage_error($message);
47             }
48 2         5 for ( @{$args} ) {
  2         7  
49 4 50       121 next if lc $_ eq "stdin";
50 4 100       18 if ( !Path::Tiny::path($_)->is_file ) {
51 1         117 $self->usage_error("The input file [$_] doesn't exist.");
52             }
53             }
54              
55 1 50       48 if ( !exists $opt->{outfile} ) {
56             $opt->{outfile}
57 0         0 = Path::Tiny::path( $args->[0] )->absolute . ".join.fas";
58             }
59             }
60              
61             sub execute {
62 1     1 1 8 my ( $self, $opt, $args ) = @_;
63              
64 1         2 my $out_fh;
65 1 50       5 if ( lc( $opt->{outfile} ) eq "stdout" ) {
66 1         5 $out_fh = *STDOUT{IO};
67             }
68             else {
69 0         0 open $out_fh, ">", $opt->{outfile};
70             }
71              
72 1         12 tie my %block_of, "Tie::IxHash";
73 1         19 for my $infile ( @{$args} ) {
  1         3  
74 3         312 my $in_fh;
75 3 50       10 if ( lc $infile eq "stdin" ) {
76 0         0 $in_fh = *STDIN{IO};
77             }
78             else {
79 3         25 $in_fh = IO::Zlib->new( $infile, "rb" );
80             }
81              
82 3         4551 my $content = ''; # content of one block
83 3         7 while (1) {
84 18 100 100     167 last if $in_fh->eof and $content eq '';
85 15         1221 my $line = '';
86 15 100       58 if ( !$in_fh->eof ) {
87 12         527 $line = $in_fh->getline;
88             }
89 15 100 66     2805 if ( ( $line eq '' or $line =~ /^\s+$/ ) and $content ne '' ) {
      66        
90 3         13 my $info_of = App::Fasops::Common::parse_block($content);
91 3         6 $content = '';
92              
93             # set $opt->{name} to the first one of the first block
94 3 50       11 if ( !defined $opt->{name} ) {
95 0         0 ( $opt->{name} ) = keys %{$info_of};
  0         0  
96             }
97              
98             # target name
99 3         14 my $header = App::RL::Common::encode_header( $info_of->{ $opt->{name} } );
100              
101 3 100       456 if ( exists $block_of{$header} ) {
102             my @other_names
103 2         11 = grep { $_ ne $opt->{name} } keys %{$info_of};
  4         38  
  2         7  
104 2         7 for my $name (@other_names) {
105 2         7 $block_of{$header}->{$name} = $info_of->{$name};
106             }
107             }
108             else {
109 1         10 $block_of{$header} = $info_of;
110             }
111             }
112             else {
113 12         80 $content .= $line;
114             }
115             }
116 3         497 $in_fh->close;
117             }
118              
119 1         132 for my $header ( keys %block_of ) {
120 1         18 my $info_of = $block_of{$header};
121              
122 1         8 my @names = keys %{$info_of};
  1         4  
123              
124 1         26 for my $name (@names) {
125 4         119 my $info = $info_of->{$name};
126 4         27 printf {$out_fh} ">%s\n", App::RL::Common::encode_header($info);
  4         15  
127 4         639 printf {$out_fh} "%s\n", $info->{seq};
  4         20  
128             }
129 1         32 print {$out_fh} "\n";
  1         3  
130             }
131              
132 1         19 close $out_fh;
133             }
134              
135             1;