Raw content of Bio::EnsEMBL::Analysis::RunnableDB::TranscriptCoalescer # # Ensembl module for Bio::EnsEMBL::Analysis::RunnableDB::TranscriptCoalescer # # Copyright (c) 2006 Ensembl # # written by Jan-Hinnerk Vogel # # =head1 NAME Bio::EnsEMBL::Analysis::RunnableDB::TranscriptCoalescer =head1 SYNOPSIS my $cond = Bio::EnsEMBL::Analysis::RunnableDB::TranscriptCoalescer->new ( -analysis => $analysis, -db => $db, -input_id => 'chromosome:JGI2:chr_04q:70000:80000:1' ); $cond->fetch_input; $cond->run; $cond->write_output; =head1 DESCRIPTION This module acts as an intermediate between the runnable and the core database. It reads configuration and uses information from the analysis object to setup the runnable and then write the results back to the database specified in the config file. =head1 CONTACT Post questions to the Ensembl development list: ensembl-dev@ebi.ac.uk =cut package Bio::EnsEMBL::Analysis::RunnableDB::TranscriptCoalescer; use strict; use warnings; use Bio::EnsEMBL::Analysis::RunnableDB::BaseGeneBuild; use Bio::EnsEMBL::Analysis::Config::Exonerate2Genes; use Bio::EnsEMBL::Analysis::Config::Databases; use Bio::EnsEMBL::Analysis::Config::GeneBuild::TranscriptCoalescer; use Bio::EnsEMBL::Analysis::Runnable::TranscriptCoalescer; use Bio::EnsEMBL::Analysis::Tools::GeneBuildUtils::ExonExtended ; use Bio::EnsEMBL::Analysis::Tools::GeneBuildUtils::TranscriptExtended ; use Bio::EnsEMBL::Analysis::Tools::GeneBuildUtils::TranscriptUtils; use Bio::EnsEMBL::Analysis::Tools::Utilities qw ( merge_config_details ) ; use Bio::EnsEMBL::Utils::Exception qw(verbose throw warning info stack_trace_dump ); use vars qw(@ISA); @ISA = qw(Bio::EnsEMBL::Analysis::RunnableDB::BaseGeneBuild); sub new { my ($class,@args) = @_; my $self = $class->SUPER::new(@args); my @est_biotypes = @{ $EST_SETS } ; my @simgw_biotypes = @{ $SIMGW_SETS } ; my @abinitio_logicnames = @{ $ABINITIO_SETS } ; # # dont' change the key-names of the $tmp hash - they are # used in the runnable to distinguish between the differnt sets ! # $self->{evidence_sets}= { 'est' => \@est_biotypes , 'simgw' => \@simgw_biotypes , 'abinitio' => \@abinitio_logicnames , } ; return $self; print STDERR " TC-db : new runnable created\n" ; } =head2 fetch_input Arg [1] : Bio::EnsEMBL::Analysis::RunnableDB::TranscriptCoalescer Function : fetch input (gene-objects) out of different databases specified in the TranscriptCoalescer config. TranscriptCoalescer is not processing single-exon genes. Returntype: 1 Exceptions: none Example : =cut sub fetch_input{ my ($self) = @_; $self->throw("No input id") unless defined($self->input_id); my $slice = $self->fetch_sequence($self->input_id, $self->db ); my (%est_genes, %simgw_genes, %pred_transcripts ) ; $self->query($slice); # the config for TranscriptCoalescer.pm is distributed over 3 files # # - Databases.pm # - Exonerate2Genes.pm # - TranscriptCoalescer.pm # # we merge the configs such that we have all in one hash $database # my %databases = %{$self->merge_config_details( $DATABASES, $EXONERATE_CONFIG_BY_LOGIC, $TRANSCRIPT_COALESCER_DB_CONFIG)} ; # check if every biotype/logic_name belongs to an EVIDENCE set _check_config(\%databases, $TRANSCRIPT_COALESCER_DB_CONFIG, $self->{evidence_sets} ) ; # now looping through all keys of %databases definend in the (merged) configs my %biotypes_to_genes ; for my $database_class (keys %databases ) { # skip all databases out of Exonerate2Genes.pm & Databases.pm which # have no configuration in TranscriptCoalescer.pm next unless (exists $$TRANSCRIPT_COALESCER_DB_CONFIG{$database_class}) ; #my $dba = new Bio::EnsEMBL::DBSQL::DBAdaptor( %{ ${$databases{$database_class}}{db}} ) ; # attache refdb as dnadb #$dba->dnadb($self->db) ; my $dba = $self->get_dbadaptor($database_class); my $slice = $self->fetch_sequence($self->input_id, $dba ); # # organise genes into "EVIDENCE_SETS" depending on their underlying source # these sets are specified in the config file TranscriptCoalescer.pm # # for my $biotype ( @{$databases{$database_class}{BIOTYPES} }) { my $genes = $slice->get_all_Genes_by_type($biotype, undef, 1) ; my ( $set ) = $self->_get_evidence_set ( $biotype ) ; my $db_used = "${$databases{$database_class}}{db}{-dbname}\@". "${$databases{$database_class}}{db}{-host}::". "${$databases{$database_class}}{db}{-port}"; info( "$db_used :" ); info(scalar (@{$genes}) . "\t$biotype-genes fetched \n") ; if (scalar( @{$genes} ) > 0 ) { # store genes of specific biotype in hash which holds them in an array # add specific information to exons / re-bless them my @multi_exon_genes ; for my $g (@$genes){ my $single_exon_gene ; for my $t (@{$g->get_all_Transcripts}){ throw ("gene has more than one transcript - only processing 1-gene-1-transcript-genes") if (@{$g->get_all_Transcripts}>1) ; bless $t, "Bio::EnsEMBL::Analysis::Tools::GeneBuildUtils::TranscriptExtended" ; my @all_exons = @{ $t->get_all_Exons } ; $single_exon_gene = 1 if (@all_exons == 1 ) ; map { bless $_,"Bio::EnsEMBL::Analysis::Tools::GeneBuildUtils::ExonExtended" } @all_exons ; for (my $int=0; $int < @all_exons ; $int++) { my $e = $all_exons[$int] ; $e->biotype($g->biotype) ; $e->ev_set($set) ; $e->transcript($t) ; $e->analysis($g->analysis) ; # setting pointer to previous exon $e->next_exon($all_exons[$int+1]); # setting pointer to previous exon if ($int == 0 ) { $e->prev_exon(0); } else { $e->prev_exon($all_exons[$int-1]); } } } unless ($single_exon_gene) { push @multi_exon_genes, $g ; }else { info("gene " . $g->dbID ." ( " . $g->biotype . " ) is single_exon_gene - SKIPPING \n") ; } } push @{$biotypes_to_genes{$biotype} } , @multi_exon_genes ; } } # # PREDICTION TRANSCRIPT PROCESSING # # get all PredictionTranscripts by their logic_name # these transcripts are converted to Genes with biotype eq logicname of analysis # for my $logic_name_becomes_biotype ( @{$databases{$database_class}{AB_INITIO_LOGICNAMES} }) { # get all PredictionTranscripts and convert them to Genes, set biotype to logic_name my $pt = $slice->get_all_PredictionTranscripts( $logic_name_becomes_biotype, 1 ) ; # get ev-set my $result_set_name = $self->_get_evidence_set( $logic_name_becomes_biotype ) ; my $ab_initio_genes = convert_prediction_transcripts_to_genes( $pt,$logic_name_becomes_biotype,$result_set_name ) ; info ( scalar (@{$ab_initio_genes}) . "\t$logic_name_becomes_biotype-genes\n") ; if ( scalar(@$ab_initio_genes)>0){ push @{$biotypes_to_genes{$logic_name_becomes_biotype} } , @{$ab_initio_genes} ; } } } # # remove redundant transcripts which have all same exons to speed computation up # print "trying to remove redundant genes\n" ; my $non_redundant_genes = _remove_redundant_genes(\%biotypes_to_genes) ; my $runnable = Bio::EnsEMBL::Analysis::Runnable::TranscriptCoalescer->new ( -query => $self->query, -analysis => $self->analysis, -all_genes => $non_redundant_genes , # ref to $hash{biotype_of_gene} = @all_genes_of_this_biotype -evidence_sets => $self->{evidence_sets} , -dnadb => $self->db , -utils_verbosity => $self->{utils_verbosity}, ); $self->runnable($runnable); return 1; } =head2 write_output Arg [1] : $self Function : get appropriate adaptors and write output to database after validating and attaching sequence and analysis objects Returntype: undef Exceptions: throws if the store fails Example : =cut sub write_output{ my ($self) = @_; #my $out_dba = new Bio::EnsEMBL::DBSQL::DBAdaptor( %{$$DATABASES{COALESCER_DB}}) ; my $out_dba = $self->get_dbadaptor($OUTPUT_DATABASE); my $gene_a = $out_dba->get_GeneAdaptor() ; info ("trying to write output") ; foreach my $gene (@{$self->output}){ info("storing $gene" ); eval{ $gene_a->store($gene) ; }; if($@){ warning("Failed to store gene ".$@); } } return ; } =head2 convert_prediction_transcripts_to_genes Arg [0] : reference to an array of Bio::EnsEMBL::PredictionTranscript-objects Arg [1] : String describing the logic_name Arg [2] : String describing name of the evidence-set Function : Creates Bio::EnsEMBL::Analysis::Tools::GeneBuildUtils::TranscriptExtended-Objects from a set of Bio::EnsEMBL::PredictionTranscript-Objects and adds further information to these Transcripts. The PredictionExons are re-blessed to Bio::EnsEMBL::Analysis::Tools::GeneBuildUtils::ExonExtended-objects Returntype: Ref. to arrray of Bio::EnsEMBL::Analysis::Tools::GeneBuildUtils::TranscriptExtended-objects Example : =cut sub convert_prediction_transcripts_to_genes { my ($pt,$logic_name_becomes_biotype,$ev_set_name ) = @_ ; my @new_genes ; for my $pt (@$pt) { # conversion my $gene_from_pt = Bio::EnsEMBL::Gene->new( -start => $pt->start , -end => $pt->end , -strand => $pt->strand , -slice =>$pt->slice , -biotype => $logic_name_becomes_biotype, -analysis=>$pt->analysis, ) ; my $new_tr = Bio::EnsEMBL::Analysis::Tools::GeneBuildUtils::TranscriptExtended->new( -BIOTYPE => $logic_name_becomes_biotype , -ANALYSIS => $pt->analysis , ) ; my @pt_exons = @{$pt->get_all_Exons} ; for (my $i=0 ; $i<scalar(@pt_exons) ; $i++) { # converting Bio::EnsEMBL::PredictionExon into ExonExtened (ISA Bio::EnsEMBL::Exon) my $pte =$pt_exons[$i] ; bless $pte,"Bio::EnsEMBL::Analysis::Tools::GeneBuildUtils::ExonExtended" ; $pte->biotype($logic_name_becomes_biotype) ; $pte->ev_set($ev_set_name) ; $pte->end_phase(0); $pte->phase(0); $pte->next_exon($pt_exons[$i+1]) ; $pte->prev_exon($pt_exons[$i-1]) ; $pte->transcript($new_tr) ; $pte->analysis($pt->analysis) ; } ; # # Extending the Bio::EnsEMBL::Transcript object by ev_set methods # for (@pt_exons) { $new_tr->add_Exon($_); } $gene_from_pt->add_Transcript($new_tr) ; push @new_genes , $gene_from_pt ; } return \@new_genes ; } =head2 _get_evidence_set ($logic_name_or_biotype) Name : get_evidence_set( $logic_name_or_biotype ) Arg : String Func : returns the name of the evidence_set of a genee / PredictionTranscript Returnval: String describing evidence_set_name =cut sub _get_evidence_set { my ($self, $logic_name_or_biotype) = @_ ; my %ev_sets = %{ $self->{evidence_sets} } ; my $result_set_name ; for my $set_name (keys %ev_sets){ my @logic_names = @{$ev_sets{$set_name}} ; for my $ln (@logic_names ) { if ($ln eq $logic_name_or_biotype) { $result_set_name = $set_name ; } } } return $result_set_name ; } sub _remove_redundant_genes { my ($biotype2genes ) = @_ ; info ("removing redundant Transcripts........\n\n" ); my %biotype2genes = %{$biotype2genes} ; my %result ; my %tmp ; for my $biotype ( keys %biotype2genes) { print "biotype $biotype\n" ; my ( %non_redundant_genes) ; my @genes = @{$biotype2genes{$biotype}} ; push @{$tmp{$biotype}}, scalar(@genes) ; info ( "RunnableDB: $biotype: " . scalar( @genes ) . "\n" ); next if ( scalar(@genes) == 0 ) ; my $red = 0 ; for my $g (@genes ) { $red++; my @all_tr = @{ $g->get_all_Transcripts } ; throw ("Only processing genes with 1gene:1transcript relation") if (@all_tr > 1 ) ; my $tr_hashkey = "" ; for my $t ( @all_tr ) { for my $e ( @{ $t->get_all_Exons} ) { $e->end_phase(0) unless ($e->end_phase); $tr_hashkey .=$e->hashkey; } $non_redundant_genes{$tr_hashkey}=$g; } } for my $key (keys %non_redundant_genes) { push @{ $result{$biotype}}, $non_redundant_genes{$key} ; } push @{$tmp{$biotype}}, scalar(@{$result{$biotype}}) ; } for (keys %tmp) { if (exists ${$tmp{$_}}[1]){ info("having ${$tmp{$_}}[1] non-redundant genes ( out of ${ $tmp{$_} }[0] of biotype $_)\n") ; } } return \%result ; } =head2 _check_config Arg[0] : Hash-reference to $TRANSCRIPT_COALESCER_DB_CONFIG-Hash out of TranscriptCoalescer.pm Arg[1] : href with evidence sets Func : Checks if every logic_name / biotype of gene belongs to an Evidence set Return : 1 if all is ok =cut sub _check_config{ my ( $databases, $TRANSCRIPT_COALESCER_DB_CONFIG, $ev_sets) = @_ ; # check TranscriptCoalescer.pm: for my $db_class (keys %{$TRANSCRIPT_COALESCER_DB_CONFIG}) { # check that each db_class (like REFERENCE_DB ...) in TranscriptCoalescer.pm has # also a database in Databases.pm / Exonerate2Genes.pm throw ( "\n\tConfig-Error: There is configuration for \"$db_class\" in Analysis/Config/GeneBuild/TranscriptCoalescer.pm " . "but no Database defined in Exonerate2Genes.pm or Databases.pm") unless exists $$databases{$db_class} ; } # check that every biotype / logic_name in TranscriptCoalescer.pm # belongs to an ev-set my %database_definition ; for my $db_class (keys %{$TRANSCRIPT_COALESCER_DB_CONFIG}) { for my $key ( keys %{ $$TRANSCRIPT_COALESCER_DB_CONFIG{$db_class} } ) { map $database_definition{$_}=(),@{ ${$$TRANSCRIPT_COALESCER_DB_CONFIG{$db_class}}{$key}}; } } my %ev_set_definitions; for my $set_name (keys %{ $ev_sets} ) { map $ev_set_definitions{$_}=(), @{$$ev_sets{$set_name}} ; } # check that every biotype / logicname mentioned in the evidence_sets # has a database where it's fetched from for my $ln ( keys %ev_set_definitions ) { throw ("\n\tError in config-file Analysis/Config/GeneBuild/TranscriptCoalescer.pm\n ". "\tType $ln has an evidence set but there's NO database for this type defined in Config". " TranscriptCoalescer.pm : $ln\n") unless ( exists $database_definition{$ln} ) ; } # check if every ev-set has an entry in the db-section as well # for my $ln ( keys %database_definition) { throw ("\n\tError in config-file Analysis/Config/GeneBuild/TranscriptCoalescer.pm\n". "\tType $ln has an entry in a database-section but there's NO evidence-set defined in Config TranscriptCoalescer.pm for this type : $ln \n" ) unless ( exists $ev_set_definitions{$ln} ) ; } return 1 ; } 1;