File Coverage

blib/lib/Bio/Tools/Run/BEDTools.pm
Criterion Covered Total %
statement 26 28 92.8
branch n/a
condition n/a
subroutine 10 10 100.0
pod n/a
total 36 38 94.7


line stmt bran cond sub pod time code
1             #
2             # BioPerl module for Bio::Tools::Run::BEDTools
3             #
4             # Please direct questions and support issues to
5             #
6             # Cared for by Dan Kortschak
7             #
8             # Copyright Dan Kortschak
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::Tools::Run::BEDTools - Run wrapper for the BEDTools suite of programs *BETA*
17              
18             =head1 SYNOPSIS
19              
20             # use a BEDTools program
21             $bedtools_fac = Bio::Tools::Run::BEDTools->new( -command => 'subtract' );
22             $result_file = $bedtools_fac->run( -bed1 => 'genes.bed', -bed2 => 'mask.bed' );
23            
24             # if IO::Uncompress::Gunzip is available...
25             $result_file = $bedtools_fac->run( -bed1 => 'genes.bed.gz', -bed2 => 'mask.bed.gz' );
26            
27             # be more strict
28             $bedtools_fac->set_parameters( -strandedness => 1 );
29            
30             # and even more...
31             $bedtools_fac->set_parameters( -minimum_overlap => 1e-6 );
32            
33             # create a Bio::SeqFeature::Collection object
34             $features = $bedtools_fac->result( -want => 'Bio::SeqFeature::Collection' );
35            
36             =head1 DEPRECATION WARNING
37              
38             Most executables from BEDTools v>=2.10.1 can read GFF and VCF formats
39             in addition to BED format. This requires the use of a new input file param,
40             shown in the following documentation, '-bgv', in place of '-bed' for the
41             executables that can do this.
42              
43             This behaviour breaks existing scripts.
44              
45             =head1 DESCRIPTION
46              
47             This module provides a wrapper interface for Aaron R. Quinlan and Ira M. Hall's
48             utilities C that allow for (among other things):
49              
50             =over
51              
52             =item * Intersecting two BED files in search of overlapping features.
53              
54             =item * Merging overlapping features.
55              
56             =item * Screening for paired-end (PE) overlaps between PE sequences and existing genomic features.
57              
58             =item * Calculating the depth and breadth of sequence coverage across defined "windows" in a genome.
59              
60             =back
61              
62             (see L for manuals and downloads).
63              
64              
65             =head1 OPTIONS
66              
67             C is a suite of 17 commandline executable. This module attempts to
68             provide and options comprehensively. You can browse the choices like so:
69              
70             $bedtools_fac = Bio::Tools::Run::BEDTools->new;
71              
72             # all bowtie commands
73             @all_commands = $bedtools_fac->available_parameters('commands');
74             @all_commands = $bedtools_fac->available_commands; # alias
75              
76             # just for default command ('bam_to_bed')
77             @btb_params = $bedtools_fac->available_parameters('params');
78             @btb_switches = $bedtools_fac->available_parameters('switches');
79             @btb_all_options = $bedtools_fac->available_parameters();
80              
81             Reasonably mnemonic names have been assigned to the single-letter
82             command line options. These are the names returned by
83             C, and can be used in the factory constructor
84             like typical BioPerl named parameters.
85              
86             As a number of options are mutually exclusive, and the interpretation of
87             intent is based on last-pass option reaching bowtie with potentially unpredicted
88             results. This module will prevent inconsistent switches and parameters
89             from being passed.
90              
91             See L for details of BEDTools options.
92              
93             =head1 FILES
94              
95             When a command requires filenames, these are provided to the C method, not
96             the constructor (C). To see the set of files required by a command, use
97             C or the alias C:
98              
99             $bedtools_fac = Bio::Tools::Run::BEDTools->new( -command => 'pair_to_bed' );
100             @filespec = $bedtools_fac->filespec;
101              
102             This example returns the following array:
103              
104             #bedpe
105             #bam
106             bed
107             #out
108              
109             This indicates that the bed (C BED format) file MUST be
110             specified, and that the out, bedpe (C BEDPE format) and bam
111             (C binary format) file MAY be specified (Note that in this case you
112             MUST provide ONE of bedpe OR bam, the module at this stage does not allow
113             this information to be queried). Use these in the C call like so:
114              
115             $bedtools_fac->run( -bedpe => 'paired.bedpe',
116             -bgv => 'genes.bed',
117             -out => 'overlap' );
118              
119             The object will store the programs STDERR output for you in the C
120             attribute:
121              
122             handle_bed_warning($bedtools_fac) if ($bedtools_fac->stderr =~ /Usage:/);
123              
124             For the commands 'fasta_from_bed' and 'mask_fasta_from_bed' STDOUT will also
125             be captured in the C attribute by default and all other commands
126             can be forced to capture program output in STDOUT by setting the -out
127             filespec parameter to '-'.
128              
129             =head1 FEEDBACK
130              
131             =head2 Mailing Lists
132              
133             User feedback is an integral part of the evolution of this and other
134             Bioperl modules. Send your comments and suggestions preferably to
135             the Bioperl mailing list. Your participation is much appreciated.
136              
137             bioperl-l@bioperl.org - General discussion
138             http://bioperl.org/wiki/Mailing_lists - About the mailing lists
139              
140             =head2 Support
141              
142             Please direct usage questions or support issues to the mailing list:
143              
144             L
145              
146             Rather than to the module maintainer directly. Many experienced and
147             reponsive experts will be able look at the problem and quickly
148             address it. Please include a thorough description of the problem
149             with code and data examples if at all possible.
150              
151             =head2 Reporting Bugs
152              
153             Report bugs to the Bioperl bug tracking system to help us keep track
154             of the bugs and their resolution. Bug reports can be submitted via
155             the web:
156              
157             http://redmine.open-bio.org/projects/bioperl/
158              
159             =head1 AUTHOR - Dan Kortschak
160              
161             Email dan.kortschak adelaide.edu.au
162              
163             =head1 CONTRIBUTORS
164              
165             Additional contributors names and emails here
166              
167             =head1 APPENDIX
168              
169             The rest of the documentation details each of the object methods.
170             Internal methods are usually preceded with a _
171              
172             =cut
173              
174             # Let the code begin...
175              
176              
177             package Bio::Tools::Run::BEDTools;
178 1     1   127673 use strict;
  1         2  
  1         35  
179             our $HAVE_IO_UNCOMPRESS;
180              
181             BEGIN {
182 1     1   54 eval 'require IO::Uncompress::Gunzip; $HAVE_IO_UNCOMPRESS = 1';
183             }
184              
185 1     1   8 use IPC::Run;
  1         3  
  1         37  
186              
187             # Object preamble - inherits from Bio::Root::Root
188              
189 1     1   337 use lib '../../..';
  1         5078  
  1         10  
190 1     1   831 use Bio::Tools::Run::BEDTools::Config;
  1         6  
  1         239  
191 1     1   557 use Bio::Tools::Run::WrapperBase;
  1         4  
  1         45  
192 1     1   635 use Bio::Tools::Run::WrapperBase::CommandExts;
  1         6  
  1         75  
193 1     1   637 use Bio::Tools::GuessSeqFormat;
  1         5013  
  1         65  
194 1     1   594 use Bio::SeqFeature::Generic;
  1         73874  
  1         37  
195 1     1   383 use Bio::SeqFeature::Collection;
  0            
  0            
196             use Bio::SeqIO;
197             use File::Sort qw( sort_file );
198              
199             use base qw( Bio::Tools::Run::WrapperBase );
200              
201             ## BEDTools
202             our $program_name = '*bedtools';
203             our $default_cmd = 'bam_to_bed';
204              
205             # Note: Other globals imported from Bio::Tools::Run::BEDTools::Config
206             our $qual_param = undef;
207             our $use_dash = 'single';
208             our $join = ' ';
209              
210             our %strand_translate = (
211             '+' => 1,
212             '-' => -1,
213             '.' => 0
214             );
215              
216             =head2 new()
217              
218             Title : new
219             Usage : my $obj = new Bio::Tools::Run::BEDTools();
220             Function: Builds a new Bio::Tools::Run::BEDTools object
221             Returns : an instance of Bio::Tools::Run::BEDTools
222             Args :
223              
224             =cut
225              
226             sub new {
227             my ($class,@args) = @_;
228             unless (grep /command/, @args) {
229             push @args, '-command', $default_cmd;
230             }
231             my $self = $class->SUPER::new(@args);
232             foreach (keys %command_executables) {
233             $self->executables($_, $self->_find_executable($command_executables{$_}));
234             }
235             $self->want($self->_rearrange([qw(WANT)],@args));
236             $self->parameters_changed(1); # set on instantiation, per Bio::ParameterBaseI
237             return $self;
238             }
239              
240             =head2 run()
241              
242             Title : run
243             Usage : $result = $bedtools_fac->run(%params);
244             Function: Run a BEDTools command.
245             Returns : Command results (file, IO object or Bio object)
246             Args : Dependent on filespec for command.
247             See $bedtools_fac->filespec and BEDTools Manual.
248             Also accepts -want => '(raw|format|)' - see want().
249             Note : gzipped inputs are allowed if IO::Uncompress::Gunzip
250             is available
251              
252             =cut
253              
254             sub run {
255             my $self = shift;
256              
257             my ($ann, $bed, $bg, $bgv, $bgv1, $bgv2, $bam, $bedpe, $bedpe1, $bedpe2, $seq, $genome, $out);
258            
259             if (!(@_ % 2)) {
260             my %args = @_;
261             if ((grep /^-\w+/, keys %args) == keys %args) {
262             ($ann, $bed, $bg, $bgv, $bgv1, $bgv2, $bam, $bedpe, $bedpe1, $bedpe2, $seq, $genome, $out) =
263             $self->_rearrange([qw( ANN BED BG BGV BGV1 BGV2 BAM
264             BEDPE BEDPE1 BEDPE2
265             SEQ GENOME OUT )], @_);
266             } else {
267             $self->throw("Badly formed named args: ".join(' ',@_));
268             }
269             } else {
270             if (grep /^-\w+/, @_) {
271             $self->throw("Badly formed named args: ".join(' ',@_));
272             } else {
273             $self->throw("Require named args.");
274             }
275             }
276              
277             # Sanity checks
278             $self->executable || $self->throw("No executable!");
279             my $cmd = $self->command if $self->can('command');
280              
281             for ($cmd) {
282              
283             =pod
284              
285             Command
286              
287             annotate bgv ann(s) #out
288              
289             =cut
290             m/^annotate$/ && do {
291             $bgv = $self->_uncompress($bgv);
292             $self->_validate_file_input(-bgv => $bgv) || $self->throw("File '$bgv' not BED/GFF/VCF format.");
293             @$ann = map {
294             my $a = $_;
295             $a = $self->_uncompress($a);
296             $self->_validate_file_input(-ann => $a) || $self->throw("File '$a' not BED/GFF/VCF format.");
297             $a;
298             } @$ann;
299             last;
300             };
301            
302             =pod
303              
304             graph_union bg_files #out
305              
306             =cut
307             m/^graph_union$/ && do {
308             @$bg = map {
309             my $g = $_;
310             $g = $self->_uncompress($g);
311             $self->_validate_file_input(-bg => $g) || $self->throw("File '$a' not BedGraph format.");
312             $g;
313             } @$bg;
314             last;
315             };
316            
317             =pod
318              
319             fasta_from_bed seq bgv #out
320              
321             mask_fasta_from_bed seq bgv #out
322              
323             =cut
324             m/fasta_from_bed$/ && do {
325             ($out // 0) eq '-' &&
326             $self->throw("Cannot capture results in STDOUT with sequence commands.");
327             $seq = $self->_uncompress($seq);
328             $self->_validate_file_input(-seq => $seq) || $self->throw("File '$seq' not fasta format.");
329             $bgv = $self->_uncompress($bgv);
330             $self->_validate_file_input(-bgv => $bgv) || $self->throw("File '$bgv' not BED/GFF/VCF format.");
331             last;
332             };
333            
334             =pod
335              
336             bam_to_bed bam #out
337              
338             =cut
339              
340             m/^bam_to_bed$/ && do {
341             $bam = $self->_uncompress($bam);
342             $self->_validate_file_input(-bam => $bam) || $self->throw("File '$bam' not BAM format.");
343             last;
344             };
345              
346              
347             =pod
348              
349             bed_to_IGV bgv #out
350              
351             merge bgv #out
352              
353             sort bgv #out
354              
355             links bgv #out
356              
357             =cut
358              
359             m/^(?:bed_to_IGV|merge|sort|links)$/ && do {
360             $bgv = $self->_uncompress($bgv);
361             $self->_validate_file_input(-bgv => $bgv) || $self->throw("File '$bgv' not BED/GFF/VCF format.");
362             };
363              
364             =pod
365              
366             b12_to_b6 bed #out
367              
368             overlap bed #out
369              
370             group_by bed #out
371              
372             =cut
373              
374             m/^(?:b12_to_b6|overlap|group_by)$/ && do {
375             $bed = $self->_uncompress($bed);
376             $self->_validate_file_input(-bed => $bed) || $self->throw("File '$bgv' not BED format.");
377             if ($cmd eq 'group_by') {
378             my $c =(my @c)= split(",",$self->columns);
379             my $o =(my @o)= split(",",$self->operations);
380             unless ($c > 0 && $o == $c) {
381             $self->throw("The command 'group_by' requires "."paired "x($o == $c)."'-columns' and '-operations' parameters");
382             }
383             }
384             last;
385             };
386              
387             =pod
388              
389             bed_to_bam bgv #out
390              
391             shuffle bgv genome #out
392              
393             slop bgv genome #out
394              
395             complement bgv genome #out
396              
397             =cut
398              
399             m/^(?:bed_to_bam|shuffle|slop|complement)$/ && do {
400             $bgv = $self->_uncompress($bgv);
401             $self->_validate_file_input(-bgv => $bgv) || $self->throw("File '$bgv' not BED/GFF/VCF format.");
402             $genome = $self->_uncompress($genome);
403             $self->_validate_file_input(-genome => $genome) || $self->throw("File '$genome' not genome format.");
404             if ($cmd eq 'slop') {
405             my $l = defined $self->add_to_left;
406             my $r = defined $self->add_to_right;
407             my $b = defined $self->add_bidirectional;
408             # I think I have a lisp
409             unless (($l && $r) || ($b xor ($l || $r))) {
410             $self->throw("The command 'slop' requires an unambiguous description of the slop you want");
411             }
412             }
413             last;
414             };
415              
416             =pod
417              
418             genome_coverage bed genome #out
419              
420             =cut
421              
422             m/^genome_coverage$/ && do {
423             $bed = $self->_uncompress($bed);
424             $self->_validate_file_input(-bed => $bed) || $self->throw("File '$bed' not BED format.");
425             $genome = $self->_uncompress($genome);
426             $self->_validate_file_input(-genome => $genome) || $self->throw("File '$genome' not genome format.");
427             my ($th, $tf) = $self->io->tempfile( -dir => $self->tempdir(), -suffix => '.bed' );
428             $th->close;
429             sort_file({k => 1, I => $bed, o => $tf});
430             $bed = $tf;
431             last;
432             };
433              
434             =pod
435              
436             window bgv1 bgv2 #out
437              
438             closest bgv1 bgv2 #out
439              
440             coverage bgv1 bgv2 #out
441              
442             subtract bgv1 bgv2 #out
443              
444             =cut
445              
446             m/^(?:window|closest|coverage|subtract)$/ && do {
447             $bgv1 = $self->_uncompress($bgv1);
448             $self->_validate_file_input(-bgv1 => $bgv1) || $self->throw("File '$bgv1' not BED/GFF/VCF format.");
449             $bgv2 = $self->_uncompress($bgv2);
450             $self->_validate_file_input(-bgv2 => $bgv2) || $self->throw("File '$bgv2' not BED/GFF/VCF format.");
451             };
452              
453             =pod
454              
455             pair_to_pair bedpe1 bedpe2 #out
456              
457             =cut
458             m/^pair_to_pair$/ && do {
459             $bedpe1 = $self->_uncompress($bedpe1);
460             $self->_validate_file_input(-bedpe1 => $bedpe1) || $self->throw("File '$bedpe1' not BEDPE format.");
461             $bedpe2 = $self->_uncompress($bedpe2);
462             $self->_validate_file_input(-bedpe2 => $bedpe2) || $self->throw("File '$bedpe2' not BEDPE format.");
463             last;
464             };
465              
466             =pod
467              
468             intersect bgv1|bam bgv2 #out
469              
470             =cut
471             m/^intersect$/ && do {
472             $bgv1 = $self->_uncompress($bgv1);
473             $bam = $self->_uncompress($bam);
474             ($bam && $self->_validate_file_input(-bam => $bam)) || ($bgv1 && $self->_validate_file_input(-bgv1 => $bgv1)) ||
475             $self->throw("File in position 1. not correct format.");
476             $bgv2 = $self->_uncompress($bgv2);
477             $self->_validate_file_input(-bgv2 => $bgv2) || $self->throw("File '$bgv2' not BED/GFF/VCF format.");
478             last;
479             };
480              
481             =pod
482              
483             pair_to_bed bedpe|bam bgv #out
484              
485             bgv* signifies any of BED, GFF or VCF. ann is a bgv.
486            
487             NOTE: Replace 'bgv' with 'bed' unless $use_bgv is set.
488              
489              
490             =cut
491              
492             m/^pair_to_bed$/ && do {
493             $bedpe = $self->_uncompress($bedpe);
494             $bam = $self->_uncompress($bam);
495             ($bam && $self->_validate_file_input(-bam => $bam)) || ($bedpe && $self->_validate_file_input(-bedpe => $bedpe)) ||
496             $self->throw("File in position 1. not correct format.");
497             $bgv = $self->_uncompress($bgv);
498             $self->_validate_file_input(-bgv => $bgv) || $self->throw("File '$bed' not BED/GFF/VCF format.");
499             last;
500             }
501             }
502            
503             my %params = (
504             '-ann' => $ann,
505             '-bam' => $bam,
506             '-bed' => $bed,
507             '-bgv' => $bgv,
508             '-bg' => $bg,
509             '-bgv1' => $bgv1,
510             '-bgv2' => $bgv2,
511             '-bedpe' => $bedpe,
512             '-bedpe1' => $bedpe1,
513             '-bedpe2' => $bedpe2,
514             '-seq' => $seq,
515             '-genome' => $genome
516             );
517             map {
518             delete $params{$_} unless defined $params{$_}
519             } keys %params;
520              
521             my $format = $self->_determine_format(\%params);
522             my $suffix = '.'.$format;
523            
524             if (!defined $out) {
525             my ($outh, $outf) = $self->io->tempfile( -dir => $self->tempdir(), -suffix => $suffix );
526             $outh->close;
527             $out = $outf;
528             } elsif ($out ne '-') {
529             $out .= $suffix;
530             } else {
531             undef $out;
532             }
533             $params{'-out'} = $out if defined $out;
534              
535             $self->_run(%params);
536              
537             $self->{'_result'}->{'file_name'} = $out // '-';
538             $self->{'_result'}->{'format'} = $format;
539             $self->{'_result'}->{'file'} = defined $out ? Bio::Root::IO->new( -file => $out ) : undef;
540            
541             return $self->result;
542             }
543              
544             sub _uncompress {
545             my ($self, $file) = @_;
546              
547             return if !defined $file;
548              
549             return $file unless ($file =~ m/\.gz[^.]*$/);
550              
551             return $file unless (-e $file && -r _); # other people will deal with this
552              
553             unless ($HAVE_IO_UNCOMPRESS) {
554             croak( "IO::Uncompress::Gunzip not available, can't expand '$file'" );
555             }
556             my ($tfh, $tf) = $self->io->tempfile( -dir => $self->tempdir() );
557             my $z = IO::Uncompress::Gunzip->new($file);
558             while (my $block = $z->getline) { print $tfh $block }
559             close $tfh;
560              
561             return $tf
562             }
563              
564             =head2 want()
565              
566             Title : want
567             Usage : $bowtiefac->want( $class )
568             Function: make factory return $class, or 'raw' results in file
569             or 'format' for result format
570             All commands can return Bio::Root::IO
571             commands returning: can return object:
572             - BED or BEDPE - Bio::SeqFeature::Collection
573             - sequence - Bio::SeqIO
574             Returns : return wanted type
575             Args : [optional] string indicating class or raw of wanted result
576              
577             =cut
578              
579             sub want {
580             my $self = shift;
581             return $self->{'_want'} = shift if @_;
582             return $self->{'_want'};
583             }
584              
585             =head2 result()
586              
587             Title : result
588             Usage : $bedtoolsfac->result( [-want => $type|$format] )
589             Function: return result in wanted format
590             Returns : results
591             Args : [optional] hashref of wanted type
592             Note : -want arg does not persist between result() call when
593             specified in result(), for persistence, use want()
594              
595             =cut
596              
597             sub result {
598             my ($self, @args) = @_;
599            
600             my $want = $self->_rearrange([qw(WANT)],@args);
601             $want ||= $self->want;
602             my $cmd = $self->command if $self->can('command');
603             my $format = $self->{'_result'}->{'format'};
604             my $file_name = $self->{'_result'}->{'file_name'};
605              
606             return $self->{'_result'}->{'format'} if (defined $want && $want eq 'format');
607             return $self->{'_result'}->{'file_name'} if (!$want || $want eq 'raw');
608             return $self->{'_result'}->{'file'} if ($want =~ m/^Bio::Root::IO/); # this will be undef if -out eq '-'
609            
610             for ($format) { # these are dissected more finely than seems resonable to allow easy extension
611             m/bed/ && do {
612             for ($want) {
613             m/Bio::SeqFeature::Collection/ && do {
614             unless (defined $self->{'_result'}->{'object'} &&
615             ref($self->{'_result'}->{'object'}) =~ m/^Bio::SeqFeature::Collection/) {
616             $self->{'_result'}->{'object'} = $self->_read_bed;
617             }
618             return $self->{'_result'}->{'object'};
619             };
620             $self->warn("Cannot make '$_' for $format.");
621             return;
622             }
623             last;
624             };
625             m/bedpe/ && do {
626             for ($want) {
627             m/Bio::SeqFeature::Collection/ && do {
628             unless (defined $self->{'_result'}->{'object'} &&
629             ref($self->{'_result'}->{'object'}) =~ m/^Bio::SeqFeature::Collection/) {
630             $self->{'_result'}->{'object'} = $self->_read_bedpe;
631             }
632             return $self->{'_result'}->{'object'};
633             };
634             $self->warn("Cannot make '$_' for $format.");
635             return;
636             }
637             last;
638             };
639             m/bam/ && do {
640             $self->warn("Cannot make '$_' for $format.");
641             return;
642             };
643             m/^(?:fasta|raw)$/ && do {
644             for ($want) {
645             m/Bio::SeqIO/ && do {
646             $file_name eq '-' && $self->throw("Cannot make a SeqIO object from STDOUT.");
647             unless (defined $self->{'_result'}->{'object'} &&
648             ref($self->{'_result'}->{'object'}) =~ m/^Bio::SeqIO/) {
649             $self->{'_result'}->{'object'} =
650             Bio::SeqIO->new(-file => $file_name,
651             -format => $format);
652             }
653             return $self->{'_result'}->{'object'};
654             };
655             $self->warn("Cannot make '$_' for $format.");
656             return;
657             }
658             last;
659             };
660             m/tab/ && do {
661             $self->warn("Cannot make '$_' for $format.");
662             return;
663             };
664             m/igv/ && do {
665             $self->warn("Cannot make '$_' for $format.");
666             return;
667             };
668             m/html/ && do {
669             $self->warn("Cannot make '$_' for $format.");
670             return;
671             };
672             do {
673             $self->warn("Result format '$_' not recognised - have you called run() yet?");
674             }
675             }
676             }
677              
678             =head2 _determine_format()
679              
680             Title : _determine_format( $has_run )
681             Usage : $bedtools-fac->_determine_format
682             Function: determine the format of output for current options
683             Returns : format of bowtie output
684             Args : [optional] boolean to indicate result exists
685              
686             =cut
687              
688             sub _determine_format {
689             my ($self, $params) = @_;
690              
691             my $cmd = $self->command if $self->can('command');
692             my $format = $format_lookup{$cmd};
693            
694             #special cases - dependent on switches and files
695             for ($cmd) {
696             m/^intersect$/ && do {
697             return 'bed' if !defined $$params{'-bam'} || $self->write_bed;
698             return 'bam';
699             };
700             m/^pair_to_bed$/ && do {
701             return 'bedpe' if !defined $$params{'-bam'} || $self->write_bedpe;
702             return 'bam';
703             };
704             m/^fasta_from_bed$/ && do {
705             return $self->output_tab_format ? 'tab' : 'fasta';
706             }
707             }
708              
709             return $format;
710             }
711              
712             =head2 _read_bed()
713              
714             Title : _read_bed()
715             Usage : $bedtools_fac->_read_bed
716             Function: return a Bio::SeqFeature::Collection object from a BED file
717             Returns : Bio::SeqFeature::Collection
718             Args :
719              
720             =cut
721              
722             sub _read_bed {
723             my ($self) = shift;
724            
725             my @features;
726            
727             if ($self->{'_result'}->{'file_name'} ne '-') {
728             my $in = $self->{'_result'}->{'file'};
729             while (my $feature = $in->_readline) {
730             chomp $feature;
731             push @features, _read_bed_line($feature);
732             }
733             } else {
734             for my $feature (split("\cJ", $self->stdout)) {
735             push @features, _read_bed_line($feature);
736             }
737             }
738            
739             my $collection = Bio::SeqFeature::Collection->new;
740             $collection->add_features(\@features);
741            
742             return $collection;
743             }
744              
745             sub _read_bed_line {
746             my $feature = shift;
747              
748             my ($chr, $start, $end, $name, $score, $strand,
749             $thick_start, $thick_end, $item_RGB, $block_count, $block_size, $block_start) =
750             split("\cI",$feature);
751             $strand ||= '.'; # BED3 doesn't have strand data - for 'merge' and 'complement'
752            
753             return Bio::SeqFeature::Generic->new( -seq_id => $chr,
754             -primary => $name,
755             -start => $start,
756             -end => $end,
757             -strand => $strand_translate{$strand},
758             -score => $score,
759             -tag => { thick_start => $thick_start,
760             thick_end => $thick_end,
761             item_RGB => $item_RGB,
762             block_count => $block_count,
763             block_size => $block_size,
764             block_start => $block_size }
765             );
766             }
767              
768             =head2 _read_bedpe()
769              
770             Title : _read_bedpe()
771             Usage : $bedtools_fac->_read_bedpe
772             Function: return a Bio::SeqFeature::Collection object from a BEDPE file
773             Returns : Bio::SeqFeature::Collection
774             Args :
775              
776             =cut
777              
778             sub _read_bedpe {
779             my ($self) = shift;
780            
781             my @features;
782            
783             if ($self->{'_result'}->{'file_name'} ne '-') {
784             my $in = $self->{'_result'}->{'file'};
785             while (my $feature = $in->_readline) {
786             chomp $feature;
787             push @features, _read_bedpe_line($feature);
788             }
789             } else {
790             for my $feature (split("\cJ", $self->stdout)) {
791             push @features, _read_bedpe_line($feature);
792             }
793             }
794            
795             my $collection = Bio::SeqFeature::Collection->new;
796             $collection->add_features(\@features);
797            
798             return $collection;
799             }
800              
801             sub _read_bedpe_line {
802             my $feature = shift;
803            
804             my ($chr1, $start1, $end1, $chr2, $start2, $end2, $name, $score, $strand1, $strand2, @add) =
805             split("\cI",$feature);
806             $strand1 ||= '.';
807             $strand2 ||= '.';
808            
809             return Bio::SeqFeature::FeaturePair->new( -primary => $name,
810             -seq_id => $chr1,
811             -start => $start1,
812             -end => $end1,
813             -strand => $strand_translate{$strand1},
814              
815             -hprimary_tag => $name,
816             -hseqname => $chr2,
817             -hstart => $start2,
818             -hend => $end2,
819             -hstrand => $strand_translate{$strand2},
820            
821             -score => $score
822             );
823             }
824              
825             =head2 _validate_file_input()
826              
827             Title : _validate_file_input
828             Usage : $bedtools_fac->_validate_file_input( -type => $file )
829             Function: validate file type for file spec
830             Returns : file type if valid type for file spec
831             Args : hash of filespec => file_name
832              
833             =cut
834              
835             sub _validate_file_input {
836             my ($self, @args) = @_;
837             my (%args);
838             if (grep (/^-/, @args) && (@args > 1)) { # named parms
839             $self->throw("Wrong number of args - requires one named arg") if (@args > 2);
840             s/^-// for @args;
841             %args = @args;
842             } else {
843             $self->throw("Must provide named filespec");
844             }
845            
846             for (keys %args) {
847             m/bam/ && do {
848             return 'bam';
849             };
850             do {
851             return unless ( -e $args{$_} && -r _ );
852             my $guesser = Bio::Tools::GuessSeqFormat->new(-file=>$args{$_});
853             return $guesser->guess if grep {$guesser->guess =~ m/$_/} @{$accepted_types{$_}};
854             }
855             }
856             return;
857             }
858              
859             =head2 version()
860              
861             Title : version
862             Usage : $version = $bedtools_fac->version()
863             Function: Returns the program version (if available)
864             Returns : string representing location and version of the program
865              
866             =cut
867              
868             sub version{
869             my ($self) = @_;
870              
871             my $cmd = $self->command if $self->can('command');
872              
873             defined $cmd or $self->throw("No command defined - cannot determine program executable");
874              
875             # new bahaviour for some BEDTools executables - breaks previous approach to getting version
876             # $dummy can be any non-recognised parameter - '-version' works for most
877             my $dummy = '-version';
878             $dummy = '-examples' if $cmd =~ /graph_union/;
879              
880             my ($in, $out, $err);
881             my $dum;
882             $in = \$dum;
883             $out = \$self->{'stdout'};
884             $err = \$self->{'stderr'};
885              
886             # Get program executable
887             my $exe = $self->executable;
888              
889             my @ipc_args = ( $exe, $dummy );
890            
891             eval {
892             IPC::Run::run(\@ipc_args, $in, $out, $err) or
893             die ("There was a problem running $exe : $!");
894             };
895             # We don't bother trying to catch this: version is returned as an illegal file seek
896              
897             my @details = split("\n",$self->stderr);
898             (my $version) = grep /^Program: .*$/, @details;
899             $version =~ s/^Program: //;
900              
901             return $version;
902             }
903              
904             sub available_commands { shift->available_parameters('commands') };
905              
906             sub filespec { shift->available_parameters('filespec') };
907              
908             1;