Mercurial > repos > geert-vandeweyer > dc_genotyper
diff DC_Genotyper.pl @ 0:e8a32d824f39 draft
Uploaded
author | geert-vandeweyer |
---|---|
date | Thu, 25 Sep 2014 05:24:35 -0400 |
parents | |
children | 61b6d523acd9 |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/DC_Genotyper.pl Thu Sep 25 05:24:35 2014 -0400 @@ -0,0 +1,530 @@ +#!/usr/bin/perl +############################# +## DEEP COVERAGE GENOTYPER ## +############################# +# version : 0.0.1 +# Principle: +# 1. get allele counts on all positions in specified targets (bed) using igvtools. Only SNPs !! +# 2. remove known dbsnp positions (bcf file) +# 3. Get distribution of background noise (pcr/sequencing errors), by modelling allele fractions as normal distributions. +# 4. Based on these distributions, check each position for significant change from the reference allele (based on allele fraction) +# 5. For abberant positions, check each alternate allele to see if it passes the background signal. +# 6. Generate VCF file. + + +################## +## LOAD MODULES ## +################## +use threads; +use threads::shared; +use Thread::Queue; +use Getopt::Std; + +#################### +## get paramaters ## +#################### +# t: target file +# b: bam file +# R: 2bit version of reference fasta. +# p: number of threads. +# s: dbsnp file +# m: minimal coverage (defaults 400x) +# P: ploidy +# a: outfile for allele distributions +# v: vcf file output. +getopts('t:b:R:p:s:m:P:v:a:', \%opts) ; + +## variables +my $twobit :shared; +my $igvgenome :shared; +if (!defined($opts{'R'})) { + die("Reference Genomes not specified\n"); +} +my @refgenomes = split(",",$opts{'r'}); +if (!-e $refgenomes[0]) { + die("'$refgenomes[0]' is not a valid file path."); +} +else { + $twobit = $refgenomes[0]; +} +if (!-e $refgenomes[1]) { + die("'$refgenomes[1]' is not a valid file path."); +} +else { + $igvgenome = $refgenomes[1]; +} + + +my $mincov :shared; +$mincov = 320; +if (defined($opts{'m'})) { + $mincov = $opts{'m'}; +} + +my $ploidy :shared; +if (defined($opts{'P'}) && $opts{'P'} =~ m/^\d+$/) { + $ploidy = $opts{'P'}; +} +else { + die("Ploidy (-P) was not specified or not an integer\n"); +} + + +if (defined($opts{'v'})) { + $outfile = $opts{'v'}; +} +else { + die("No output vcf-file specified.\n"); +} +if (!defined($opts{'a'})) { + die("No output file specified for distribution details\n"); +} +## create working dir. +my $rand = int(rand(10000)); +while (-d "/tmp/DC_Genotyper_$rand") { + $rand = int(rand(10000)); +} +my $wd :shared; +$wd = "/tmp/DC_Genotyper_$rand"; +system("mkdir '$wd'"); + + +my $snpfile :shared; +my $hassnp :shared; +$hassnp = 'NoDbSNP'; +$snpfile = ''; +if (defined($opts{'s'})) { + $snpfile = $opts{'s'}; + if (!-e $snpfile) { + die("'$snpfile' is not a valid file path."); + } + + my $mime = `file $snpfile`; + if ($mime !~ m/compressed/) { + print "$snpfile is not in compressed format. compressing & indexing the file now.\n"; + #print "... this takes a while\n"; + system("bgzip -c $snpfile > $wd/dbSNP.vcf.bgz"); + system("cd $wd/ && tabix -p vcf dbSNP.vcf.bgz"); + $snpfile = "$wd/dbSNP.vcf.bgz"; + } + elsif (!-e "$snpfile.tbi") { + print "tabix index file is missing for '$snpfile'. creating now.\n"; + ## check if I can write it out for future use + $snpfile =~ m/(.*)/([^\/]+)$/; + my $d = $1; + if (-w $d) { + open OUT, ">$d/lock"; + flock(OUT,2); + system("cd $d && tabix -p vcf $snpfile"); + close OUT; + system("rm $d/lock"); + } + else { + system("cp $snpfile /$wd/dbSNP.vcf.bgz"); + system("cd $wd/ && tabix -p vcf dbSNP.vcf.bgz"); + $snpfile = "$wd/dbSNP.vcf.bgz"; + } + } + $hassnp = 'WithDbSNP'; +} + + +## 1. Get FASTA and prepare output hashes: +my $targets_one = Thread::Queue->new(); +my $targets_two = Thread::Queue->new(); +my $targets_three = Thread::Queue->new(); +open IN, $opts{'t'} or die("Could not open $opts{'t'} file for reading"); +if (-d "$wd/Fasta/") { + system("rm $wd/Fasta/*"); +} +else { + system("mkdir $wd/Fasta"); +} +## create the threads. +for (my $i = 1; $i<= $opts{'p'}; $i++) { + ${'thr'.$i} = threads->create('FetchFasta'); +} + +## enqueue the targets. +my %thash; +while (<IN>) { + chomp; + my ($chr,$start,$stop,$name,$score,$strand) = split(/\t/,$_); + $targets_one->enqueue($_); + $targets_two->enqueue($_); + $targets_three->enqueue($_); + $thash{$chr}{$start} = $stop; +} +close IN; + +## end the threads. +for (my $i = 1; $i <= $opts{'p'}; $i++) { + $targets_one->enqueue(undef); +} + +for (my $i = 1; $i <= $opts{'p'}; $i++) { + ${'thr'.$i}->join(); +} + +## load dbSNP inside target regions into shared structure. +########################################################## +my %dbsnp :shared; +if ($snpfile ne '') { + my $bcf = `which bcftools`; + chomp($bcf); + if ($bcf ne '') { + my $command = "bcftools query -f '\%CHROM\\t\%POS\\t\%REF\\t\%ALT\\t\%ID\\n' -R '".$opts{'t'}."' '$snpfile' > $wd/dbsnp.txt"; + system("$command"); + open IN, "$wd/dbsnp.txt"; + while (<IN>) { + chomp; + my @p = split(/\t/,$_); + $dbsnp{$p[0].'-'.$p[1]} = $p[2].'-'.$p[3].'-'.$p[4]; + } + close IN; + } + else { + print "WARNING: BCFtools is not in the path. Skipping snp handling.\n"; + $snpfile = ''; + system("touch $wd/dbsnp.txt"); + } +} +else { + system("touch $wd/dbsnp.txt"); +} + +## now process the bam file. +mkdir "$wd/WIGS/"; +my $bam :shared; +$bam = $opts{'b'}; +my $bai = $bam.".bai"; +if (!-e $bai) { + print "BAI ($bai) missing for $bam : creating\n"; + system("samtools index $bam"); +} + +for (my $i = 1; $i <= $opts{'p'}; $i++) { + ${'thr'.$i} = threads->create('CountAlleles'); +} +## end the threads. +for (my $i = 1; $i <= $opts{'p'}; $i++) { + $targets_two->enqueue(undef); +} + +for (my $i = 1; $i <= $opts{'p'}; $i++) { + ${'thr'.$i}->join(); +} + +## generate the distributions. +############################## +my $alleles = Thread::Queue->new(); +my %all = ('A' => 1,'C' => 2,'G' => 3, 'T' => 4); +foreach(keys(%all)) { + $alleles->enqueue($_); + my $a = $_; + foreach(keys(%all)) { + if ($_ eq $a) { + next; + } + $alleles->enqueue($a.'-'.$_); + } +} +for (my $i = 1; $i <= $opts{'p'}; $i++) { + ${'thr'.$i} = threads->create('GetDistribution'); +} +## end the threads. +for (my $i = 1; $i <= $opts{'p'}; $i++) { + $alleles->enqueue(undef); +} + +for (my $i = 1; $i <= $opts{'p'}; $i++) { + ${'thr'.$i}->join(); +} + +## group distributions into one file +#################################### +my %map =('A' => 2,'C' => 3,'G' => 4, 'T' => 5); +open OUT, ">".$opts{'a'}; +print OUT "allele\tavg\tsd\n"; +foreach(keys(%map)) { + my $r = $_; + my $f = "$wd/model.$r.$mincov"."x.$hassnp.txt"; + open IN, "$f"; + my $a = <IN>; + chomp($a); + #$dists{$r}{'avg'} = $a; + my $s = <IN>; + chomp($s); + #$dists{$r}{'sd'} = $s; + close IN; + print OUT "$r\t$a\t$s\n"; + foreach(keys(%map)) { + if ($_ eq $r) { + next; + } + my $f = "$wd/model.$r-$_.$mincov"."x.$hassnp.txt"; + open IN, "$f"; + my $a = <IN>; + chomp($a); + my $s = <IN>; + chomp($s); + close IN; + print OUT "$r-$_\t$a\t$s\n"; + } +} +close OUT; + +## CALL SNPs +############ +# create the R script. +open R, ">$wd/CallSNPs.R"; +print R "args <- commandArgs(trailingOnly = TRUE)\n"; +print R "counts <- read.table(file=args[1],header=FALSE, as.is=TRUE)\n"; +print R "ploidy <- as.integer(args[3])\n"; +print R "chr <- args[2]\n"; +print R "snps <- read.table(file=args[5],header=FALSE,as.is=TRUE)\n"; +print R "colnames(snps) <- c('chr','pos','ref','alt','id')\n"; +print R "colnames(counts) <- c('pos','ref','A','C','G','T','TotalDepth')\n"; +print R "dists <- read.table(file='$wd/allelic_distributions.txt',header=TRUE,as.is=TRUE)\n"; +print R 'rownames(dists) = dists$allele'."\n"; +print R 'dists <- dists[,-1]'."\n"; +print R "vcf <- c()\n"; +print R "lower <- c()\n"; +print R "higher <- c()\n"; +print R "for (i in 1:(ploidy)) {\n"; +print R " lower[length(lower)+1] <- (2*i-1)/(2*ploidy)\n"; +print R " higher[length(higher)+1] <- (2*i+1)/(2*ploidy)\n"; +print R "}\n"; +print R "for (i in 1:nrow(counts)) {\n"; +print R " if (counts[i,'TotalDepth'] == 0) next\n"; +print R " # significantly different from reference?\n"; +print R " z <- ((counts[i,counts[i,'ref']]/counts[i,'TotalDepth']) - dists[counts[i,'ref'],'avg']) / dists[counts[i,'ref'],'sd']\n"; +print R " if (abs(z) > 3) {\n"; +print R " # test all alterate alleles to see which one is significant.\n"; +print R " for (j in c('A','C','G','T')) {\n"; +print R " if (j == counts[i,'ref']) next\n"; +print R " z <- ((counts[i,j]/counts[i,'TotalDepth']) - dists[paste(counts[i,'ref'],'-',j,sep=''),'avg']) / dists[paste(counts[i,'ref'],'-',j,sep=''),'sd']\n"; +print R " if (abs(z) > 3){\n"; +print R " filter <- 'PASS'\n"; +print R " phred <- round(-10*log(pnorm(-abs(z))),digits=0)\n"; +print R " if (phred > 9999) phred <- 9999\n"; +print R " frac <- counts[i,j]/counts[i,'TotalDepth']\n"; +print R " for (k in 1:ploidy) {\n"; +print R " if (frac >= lower[k] && frac < higher[k]) {\n"; +print R " sample <- paste(paste(paste(rep(0,(ploidy-k)),sep='',collapse='/'),paste(rep(1,k),sep='',collapse='/'),sep='/',collapse=''),':',counts[i,counts[i,'ref']],',',counts[i,j],sep='',collapse='')\n"; +print R " af <- k/ploidy\n"; +print R " break\n"; +print R " }\n"; +print R " }\n"; +print R " if (frac < lower[1]) {\n"; +print R " sample <- paste(paste(paste(rep(0,(ploidy-1)),sep='',collapse='/'),paste(rep(1,1),sep='',collapse='/'),sep='/',collapse=''),':',counts[i,counts[i,'ref']],',',counts[i,j],sep='',collapse='')\n"; +print R " af <- 1/ploidy\n"; +print R " filter <- 'LowFraction'\n"; +print R " }\n"; +print R " if (counts[i,'TotalDepth'] < $mincov) {\n"; +print R " filter <- 'LowCoverage'\n"; +print R " }\n"; +print R " info <- paste('DP=',counts[i,'TotalDepth'],';AF=',round(af,digits=5),';AR=',round(frac,digits=5),sep='')\n"; +print R " snpids <- which(snps\$chr == chr & snps\$pos == counts[i,'pos'])\n"; +print R " id <- '.'\n"; +print R " if (length(snpids) > 0) id <- snps[snpids[1],'id']\n"; +print R " vcf[length(vcf)+1] <- paste(chr,counts[i,'pos'],id,counts[i,'ref'],j,phred,filter,info,'GT:AD',sample,sep='\\t',collapse='')\n"; +print R " }\n"; +print R " }\n"; +print R " }\n"; +print R "}\n"; +print R "if (length(vcf) > 0) {\n"; +print R " write(file=args[4],paste(vcf,sep='\\n'))\n"; +print R "}\n"; +close R; +system("mkdir $wd/VCF/"); +for (my $i = 1; $i <= $opts{'p'}; $i++) { + ${'thr'.$i} = threads->create('CallSNPs'); +} +## end the threads. +for (my $i = 1; $i <= $opts{'p'}; $i++) { + $targets_three->enqueue(undef); +} + +for (my $i = 1; $i <= $opts{'p'}; $i++) { + ${'thr'.$i}->join(); +} + +## BUILD FINAL VCF +open OUT, ">$outfile"; +print OUT "##fileformat=VCFv4.1\n"; +print OUT "##source=High_Ploidy_Genotyper_v.0.1\n"; +print OUT "##genome_reference=$twobit\n"; +if ($snpfile ne '') { + print OUT "##SNP_file=$snpfile\n"; +} +foreach(keys(%thash)) { + print OUT "##contig=<ID=$_,assembly=hg19,species=\"Homo Sapiens\">\n"; +} +print OUT "##INFO=<ID=DP,Number=1,Type=Integer,Description=\"Total Depth\">\n"; +print OUT "##INFO=<ID=AF,Number=1,Type=Float,Description=\"Allele Frequency\">\n"; +print OUT "##INFO=<ID=AR,Number=1,Type=Float,Description=\"Allelic Ratio\">\n"; +print OUT "##FILTER=<ID=LowFraction,Description=\"Allelic Fraction under 1/2*$ploidy\">\n"; +print OUT "##FILTER=<ID=LowCoverage,Description=\"Total Depth is lower than threshold of $mincov\">\n"; +print OUT "##FORMAT=<ID=GT,Number=1,Type=String,Description=\"Genotype\">\n"; +print OUT "##FORMAT=<ID=AD,Number=2,type=Integer,Description,\"Allelic Depth\">\n"; +print OUT "#CHROM\tPOS\tID\tREF\tALT\tQUAL\tFILTER\tINFO\tFORMAT\tSAMPLE\n"; +close OUT; +@i = ( 1 .. 22,'X','Y','M' ); +foreach(@i) { + my $chr = "chr$_"; + foreach(sort {$a <=> $b} keys(%{$thash{$chr}})) { + my $v = "$wd/VCF/$chr.$_-".$thash{$chr}{$_}.".vcf"; + if (-e $v) { + system("cat '$v' >> '$outfile'"); + } + } +} + +## clean up +system("rm -Rf '$wd'"); + +sub FetchFasta { + while(defined(my $line = $targets_one->dequeue())) { + my ($chr,$start,$stop,$name,$score,$strand) = split(/\t/,$line); + # 2bit is zero based, non-including => decrease start by one + $startposition = $start - 1; + my $command = "twoBitToFa -seq=$chr -start=$startposition -end=$stop -noMask $twobit $wd/Fasta/$chr-$start-$stop.fasta"; + system($command); + } +} + +sub CountAlleles { + # local version of hashes + my $snp = \%dbsnp; + my %counts; + $counts{'A'} = ''; + $counts{'C'} = ''; + $counts{'G'} = ''; + $counts{'T'} = ''; + my %map =('A' => 1,'C' => 2,'G' => 3, 'T' => 4); + my %options; + foreach(keys(%map)) { + my $r = $_; + foreach(keys(%map)) { + if ($_ eq $r) { + next; + } + $options{$r.'-'.$_} = ''; + } + } + while (defined(my $line = $targets_two->dequeue())) { + $out = ''; + my ($chr,$start,$stop,$name,$score,$strand) = split(/\t/,$line); + ## get reference alleles + my %ref_alleles; + open FASTA, "$wd/Fasta/$chr-$start-$stop.fasta"; + my $head = <FASTA>; + my $seq = ''; + while (<FASTA>) { + chomp; + $seq .= $_; + } + close FASTA; + # this generates a hash of the reference alleles once, instead of substr-calls in every bam, on every iteration. + for (my $pos = 0; $pos < length($seq); $pos++) { + $ref_alleles{($pos+$start)} = substr($seq,$pos,1); + } + ## get counts. + my $target = "$chr:$start-$stop"; + my $command = "igvtools count -w 1 --bases --query '$target' '$bam' '$wd/WIGS/$chr-$start-$stop.wig' '$igvgenome' > /dev/null 2>&1"; + system($command); + open WIG, "$wd/WIGS/$chr-$start-$stop.wig"; + my $h = <WIG>; + $h = <WIG>; + $h = <WIG>; + my $target_counts = ''; + while (<WIG>) { + chomp; + #my ($pos, $a, $c, $g, $t , $n) = split(/\t/,$_); + my @p = split(/\t/,$_); + my $s = $p[1] + $p[2] + $p[3] + $p[4]; + $target_counts .= "$p[0]\t$ref_alleles{$p[0]}\t$p[1]\t$p[2]\t$p[3]\t$p[4]\t$s\n"; + ## skip positions with coverage < minimal coverage, and positions in dbsnp if specified (if not specified, snp hash is empty). + if ($s > $mincov && !defined($snp->{$chr.'-'.$p[0]})) { + ## for model of 'non-reference' + my $frac = $p[$map{$ref_alleles{$p[0]}}] / $s; + $counts{$ref_alleles{$p[0]}} .= $frac.','; + $out .= "$target\t$p[0]\t$ref_alleles{$p[0]}\t$p[1]\t$p[2]\t$p[3]\t$p[4]\n"; + ## for each of the options background models + foreach(keys(%map)) { + if ($_ eq $ref_alleles{$p[0]}) { + next; + } + $options{$ref_alleles{$p[0]}.'-'.$_} .= ($p[$map{$_}] / $s) .','; + } + + } + } + close WIG; + open OUT, ">>$wd/allcounts.$mincov"."x.$hassnp.txt"; + flock(OUT, 2); + print OUT $out; + close OUT; + open OUT, ">$wd/WIGS/$chr.$start-$stop.txt"; + print OUT $target_counts; + close OUT; + + } + foreach(keys(%counts)) { + open OUT, ">>$wd/counts_$_.$mincov"."x.$hassnp.txt"; + flock(OUT,2); + print OUT $counts{$_}; + close OUT; + } + foreach(keys(%options)) { + open OUT, ">>$wd/counts_$_.$mincov"."x.$hassnp.txt"; + flock(OUT,2); + print OUT $options{$_}; + close OUT; + } +} + +sub GetDistribution { + while (defined(my $allele = $alleles->dequeue())) { + system("sed -i 's/.\$//' '$wd/counts_$allele.$mincov"."x.$hassnp.txt'"); + open OUT, ">$wd/GetDistribution.$allele.R"; + print OUT "sample <- '$bam'\n"; + print OUT "nt <- '$allele'\n"; + #print OUT "pdf(file='$wd/Distribution.$allele.$mincov"."x.$hassnp.pdf',paper='a4')\n"; + print OUT "data <- scan(file='$wd/counts_$allele.$mincov"."x.$hassnp.txt',sep=',')\n"; + print OUT "nr <- length(data)\n"; + print OUT "avg <- mean(data)\n"; + print OUT "sdd <- sd(data)\n"; + #print OUT "if (avg > 0.5) {\n"; + #print OUT " x <- seq(0.8,1,length=1000)\n"; + #print OUT " y <- dnorm(x,mean=avg,sd=sdd)\n"; + #print OUT " plot(x,y,main='Distribution in sample $bam for nt $allele',xlab='Allelic Ratio',type='l',lwd=1)\n"; + #print OUT " abline(v=(avg-3*sdd),col='red')\n"; + #print OUT " text(0.81,max(y-0.5),paste(c('avg: ',avg,'\\nsd: ',sdd,'\\nnrDataPoints:', nr,'\\n$hassnp\\nMin.Cov:',$mincov),sep=' ',collapse=''),adj=c(0,1))\n"; + #print OUT "} else {\n"; + #print OUT " x <- seq(0,0.3,length=1000)\n"; + #print OUT " y <- dnorm(x,mean=avg,sd=sdd)\n"; + #print OUT " plot(x,y,main='Distribution in sample $bam for nt $allele',xlab='Allelic Ratio',type='l',lwd=1)\n"; + #print OUT " abline(v=(avg+3*sdd),col='red')\n"; + #print OUT " text(0.2,max(y-0.5),paste(c('avg: ',avg,'\\nsd: ',sdd,'\\nnrDataPoints:', nr,'\\n$hassnp\\nMin.Cov:',$mincov),sep=' ',collapse=''),adj=c(0,1))\n"; + #print OUT "}\n"; + #print OUT "dev.off()\n"; + print OUT "write(c(avg,sdd),file='$wd/model.$allele.$mincov"."x.$hassnp.txt',ncolumns=1)\n"; + close OUT; + system("cd $wd && Rscript GetDistribution.$allele.R >/dev/null 2>&1"); + } +} + + +sub CallSNPs { + while (defined(my $line = $targets_three->dequeue())) { + # split. + my ($chr,$start,$stop,$name,$score,$strand) = split(/\t/,$line); + my $file = "$wd/WIGS/$chr.$start-$stop.txt"; + my $ofile = "$wd/VCF/$chr.$start-$stop.vcf"; + system("cd $wd && Rscript CallSNPs.R '$file' '$chr' '$ploidy' '$ofile' '$wd/dbsnp.txt'"); + } + +} +