 
                        
        
                        
        org.broadinstitute.hellbender.tools.walkers.rnaseq.GeneExpressionEvaluation Maven / Gradle / Ivy
package org.broadinstitute.hellbender.tools.walkers.rnaseq;
import htsjdk.samtools.AlignmentBlock;
import htsjdk.samtools.Cigar;
import htsjdk.samtools.SAMFileHeader;
import htsjdk.samtools.SAMReadGroupRecord;
import htsjdk.samtools.SAMSequenceDictionary;
import htsjdk.samtools.SAMTag;
import htsjdk.samtools.SAMUtils;
import htsjdk.samtools.TextCigarCodec;
import htsjdk.samtools.util.Interval;
import htsjdk.samtools.util.IntervalList;
import htsjdk.samtools.util.OverlapDetector;
import htsjdk.tribble.annotation.Strand;
import htsjdk.tribble.gff.Gff3BaseData;
import htsjdk.tribble.gff.Gff3Feature;
import org.apache.commons.lang3.tuple.Pair;
import org.broadinstitute.barclay.argparser.Argument;
import org.broadinstitute.barclay.argparser.BetaFeature;
import org.broadinstitute.barclay.argparser.CommandLineProgramProperties;
import org.broadinstitute.barclay.argparser.ExperimentalFeature;
import org.broadinstitute.barclay.help.DocumentedFeature;
import org.broadinstitute.hellbender.cmdline.StandardArgumentDefinitions;
import org.broadinstitute.hellbender.cmdline.programgroups.CoverageAnalysisProgramGroup;
import org.broadinstitute.hellbender.engine.*;
import org.broadinstitute.hellbender.engine.filters.AlignmentAgreesWithHeaderReadFilter;
import org.broadinstitute.hellbender.engine.filters.MappingQualityReadFilter;
import org.broadinstitute.hellbender.engine.filters.ReadFilter;
import org.broadinstitute.hellbender.engine.filters.ReadFilterLibrary;
import org.broadinstitute.hellbender.exceptions.GATKException;
import org.broadinstitute.hellbender.exceptions.UserException;
import org.broadinstitute.hellbender.utils.IntervalUtils;
import org.broadinstitute.hellbender.utils.SimpleInterval;
import org.broadinstitute.hellbender.utils.read.GATKRead;
import org.broadinstitute.hellbender.utils.tsv.DataLine;
import org.broadinstitute.hellbender.utils.tsv.TableColumnCollection;
import org.broadinstitute.hellbender.utils.tsv.TableReader;
import org.broadinstitute.hellbender.utils.tsv.TableWriter;
import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.*;
import java.util.stream.Collectors;
/**
 * Evaluate gene expression from RNA-seq reads aligned to genome.
 *
 * This tool counts fragments to evaluate gene expression from RNA-seq reads aligned to the genome.  Features to evaluate expression over are defined in an input annotation file in gff3 fomat
 * (https://github.com/The-Sequence-Ontology/Specifications/blob/master/gff3.md).  Output is a tsv listing sense and antisense expression for all stranded grouping features,
 * and expression (labeled as sense) for all unstranded grouping features.
 * 
 * 
 * 
 *     
Input
 *     
 *     - BAM file of RNA-seq reads*
- Gff3 file of feature annotations*
* 
 *
 *     
Output
 *     TSV file of gene expression
 * 
 *
 * 
 *     
Usage Examples
 *     
 *         gatk GeneExpressionEvaluation
 *         -I input.bam
 *         -G geneAnnotations.gff3
 *         -O output.tsv
 *     
 * 
 *
 *Reads are assumed to be paired-end.  Reads which are in a "good pair" are counted once together as a single fragment.  Reads which are not in a "good pair" are each counted separately.  Fragment
 * size is not explicitly considered in the "good pair" decision, instead we rely on the aligner to take that into consideration intelligently when deciding whether to set the properly paired flag.
 * A "good pair" is defined as:
 *  
Both reads are mapped 
 *  Properly paired flag is set 
 *  Both reads are on same contig 
 *  Mapping quality of both reads is at least minimum-mapping-quality 
 *
 *
 *
 * Reads can be either from spliced or unspliced RNA.  If from spliced RNA, alignment blocks of reads are taken as their coverage.
 * If from unspliced RNA, the entire region from the most 5' base of either read to most 3' base of either read is taken as the fragment
 * coverage.  Splice status is set through the command line.  By default, splice status is taken to be "spliced." 
 * 
 *     
For spliced RNA
 *     
 *         gatk GeneExpressionEvaluation
 *         -I input.bam
 *         -G geneAnnotations.gff3
 *         -O output.tsv
 *     
 * 
 * 
 *     
For unspliced RNA
 *     
 *         gatk GeneExpressionEvaluation
 *         -I input.bam
 *         -G geneAnnotations.gff3
 *         -O output.tsv
 *         --unspliced
 *     
 * 
 * 
 * Gene expression is aggregated over "groupingType" features, and overlap of fragments with features is determined by "overlapType" features.  By default,
 * "groupingType" is genes, and "overlapType" is exons.  Additional grouping_types and overlap_types can be used as well.
 * 
 *     
Use gene and pseudogene as grouping types
 *     
 *     gatk GeneExpressionEvaluation
 *     -I input.bam
 *     -G geneAnnotations.gff3
 *     -O output.tsv
 *     --grouping-type gene
 *     --grouping-type pseudogene
 *     
 * 
 * 
 * The read orientation can be set using the read-strand argument.  By default, it is assumed that read1 will be on the forward strand,
 *  and read2 on the reverse strand.
 * 
 * 
 *     
Set read strands to read1 reverse, read2 forward
 *     
 *         gatk GeneExpressionEvaluation
 *         -I input.bam
 *         -G geneAnnotations.gff3
 *         -O output.tsv
 *         --read-strands REVERSE-FORWARD
 *     
 *
 *     Set read1 and read2 to both be forward strand
 *     
 *         gatk GeneExpressionEvaluation
 *         -I input.bam
 *         -G geneAnnotations.gff3
 *         -O output.tsv
 *         --read-strands FORWARD-FORWARD
 *     
 *         
 * 
 *
 * Multi-overlapping fragments (fragment alignments which overlap multiple grouping features) can be handled in two ways.  Equal weight can be given to each grouping feature,
 * in which case each is given weight 1/N, for N overlapping grouping features.
 * 
 * 
 *     
Equal weight for multi-overlapping fragments
 *     
 *         gatk GeneExpressionEvaluation
 *         -I input.bam
 *         -G geneAnnotations.gff3
 *         -O output.tsv
 *         --multi-overlap-method EQUAL
 *     
 * 
 * Multi-overlapping fragments can also have weight distributed according to how much of the fragment overlaps each feature.  In this case, each grouping feature is given an unnormalized weight corresponding
 * to the fraction of the fragment that overlaps its overlapping features.  A "non-overlapping" option is also given an unnormalized weight corresponding the the fraction of the fragment that overlaps no features.
 * Final weights for each feature are found by normalizing by the sum of unnormalized weights.  This is the default behavior.
 * 
 * 
 *     
Proportional weight for multi-overlapping fragments
 *     
 *         gatk GeneExpressionEvaluation
 *         -I input.bam
 *         -G geneAnnotations.gff3
 *         -O output.tsv
 *         --multi-overlap-method PROPORTIONAL
 *     
 * 
 *
 * Multi-mapping fragments (fragments whose reads map to multiple locations) can also be handled in two ways.  They can be ignored, in which case only fragments with a single mapping are counted.
 * This is default behavior.
 * 
 * 
 *     
Ignore multi-mapping fragments
 *     
 *         gatk GeneExpressionEvaluation
 *         -I input.bam
 *         -G geneAnnotations.gff3
 *         -O output.tsv
 *         --multi-map-method IGNORE
 *     
 * 
 * Multi-mapping fragments can alternatively have their weight distributed equally between the different alignments.  If run in this setting, minimum-mapping-quality will be set to 0,
 * regardless of its value on the command line.
 * 
 * 
 *     
Equally weight each alignment of multi-mapping fragments
 *     
 *         gatk GeneExpressionEvaluation
 *         -I input.bam
 *         -G geneAnnotations.gff3
 *         -O output.tsv
 *         --multi-map-method EQUAL
 *     
 * 
 * The number of mapping for a particular fragment is extracted via the NH tag.
 * Currently this tool only works on a single sample BAM.  If a readgroup header line indicates multiple samples in the same BAM, the tool will throw an exception.
 */
@CommandLineProgramProperties(
        summary = "This tool evaluates gene expression from RNA-seq reads aligned to genome.  Features to evaluate expression over are defined in an input annotation file in gff3 fomat " +
                "(https://github.com/The-Sequence-Ontology/Specifications/blob/master/gff3.md).  Output is a tsv listing sense and antisense expression for all grouping features. " +
                "For unstranded features, fragments transcribed on the forward strand are counted as sense, and fragments transcribed on the reverse strand are counted as antisense.",
        oneLineSummary = "Evaluate gene expression from RNA-seq reads aligned to genome.",
        programGroup = CoverageAnalysisProgramGroup.class
)
@DocumentedFeature
@BetaFeature
public final class GeneExpressionEvaluation extends ReadWalker {
    @Argument(
            doc = "Output file for gene expression.",
            fullName = StandardArgumentDefinitions.OUTPUT_LONG_NAME,
            shortName = StandardArgumentDefinitions.OUTPUT_SHORT_NAME
    )
    private File outputCountsFile = null;
    @Argument(doc="Gff3 file containing feature annotations", shortName = "G", fullName = "gff-file")
    private FeatureInput gffFile;
    @Argument(doc="Feature types to group by", fullName = "grouping-type")
    private Set groupingType = new HashSet<>(Collections.singleton("gene"));
    @Argument(doc="Feature overlap types", fullName = "overlap-type")
    private Set overlapType = new HashSet<>(Collections.singleton("exon"));
    @Argument(doc="Whether to label features by ID or Name", fullName = "feature-label-key")
    private FeatureLabelType featureLabelKey = FeatureLabelType.NAME;
    @Argument(doc = "Which strands (forward or reverse) each read is expected to be on", fullName = "read-strands")
    private ReadStrands readStrands = ReadStrands.FORWARD_REVERSE;
    @Argument(doc = "How to distribute weight of alignments which overlap multiple features", fullName = "multi-overlap-method")
    private MultiOverlapMethod multiOverlapMethod = MultiOverlapMethod.PROPORTIONAL;
    @Argument(doc = "How to distribute weight of reads with multiple alignments", fullName = "multi-map-method")
    private MultiMapMethod multiMapMethod = MultiMapMethod.IGNORE;
    @Argument(doc = "Whether the rna is unspliced.  If spliced, alignments must be from an aligner run in a splice-aware mode.  If unspliced, alignments must be from an aligner run in a non-splicing mode.")
    private boolean unspliced = false;
    final private Map featureCounts = new LinkedHashMap<>();
    final private OverlapDetector> featureOverlapDetector = new OverlapDetector<>(0,0);
    private String sampleName = null;
    final MappingQualityReadFilter mappingQualityFilter = new MappingQualityReadFilter();
    enum FeatureLabelType {
        NAME("Name") {
            @Override
            String getValue(final Gff3BaseData baseData) {
                return baseData.getName();
            }
        },
        ID("ID") {
            @Override
            String getValue(final Gff3BaseData baseData) {
                return baseData.getId();
            }
        };
        String key;
        FeatureLabelType(final String key) {
            this.key = key;
        }
        String getKey() {
            return key;
        }
        abstract String getValue(final Gff3BaseData baseData);
    }
    enum MultiOverlapMethod {
        EQUAL {
            @Override
            Map getWeights(final List alignmentIntervals, final OverlapDetector> featureOverlapDetector) {
                final Set overlappingFeatures = alignmentIntervals.stream().flatMap(i -> featureOverlapDetector.getOverlaps(i).stream().map(Pair::getLeft)).collect(Collectors.toSet());
                final int nOverlappingFeatures = overlappingFeatures.size();
                final Map weights = new LinkedHashMap<>();
                for (final Gff3BaseData feature : overlappingFeatures) {
                    weights.put(feature, 1.0d/nOverlappingFeatures);
                }
                return weights;
            }
        },
        PROPORTIONAL {
            @Override
            Map getWeights(final List alignmentIntervals, final OverlapDetector> featureOverlapDetector) {
                final List mergedAlignmentIntervals = getMergedIntervals(alignmentIntervals);
                final int basesOnReference = mergedAlignmentIntervals.stream().map(Interval::getLengthOnReference).reduce(0, Integer::sum);
                int totalCoveredBases = 0;
                double summedUnNormalizedWeights = 0;
                final Map weights = new LinkedHashMap<>();
                for (final Interval alignmentInterval : mergedAlignmentIntervals) {
                    final Set> overlaps = featureOverlapDetector.getOverlaps(alignmentInterval);
                    final Map> overlappingIntervalsByFeature = new LinkedHashMap<>();
                    final List allOverlappingIntervals = new ArrayList<>();
                    for (Pair overlap : overlaps) {
                        final List overlappingIntervals = overlappingIntervalsByFeature.computeIfAbsent(overlap.getLeft(), f -> new ArrayList<>());
                        overlappingIntervals.add(overlap.getRight());
                        allOverlappingIntervals.add(overlap.getRight());
                    }
                    final List allOverlappingIntervalsMerged = getMergedIntervals(allOverlappingIntervals);
                    totalCoveredBases += allOverlappingIntervalsMerged.stream().map(alignmentInterval::getIntersectionLength).reduce(0,Integer::sum);
                    for (Map.Entry> overlap : overlappingIntervalsByFeature.entrySet()) {
                        final List mergedOverlapIntervals = getMergedIntervals(overlap.getValue());
                        final double weight = (double)mergedOverlapIntervals.stream().map(alignmentInterval::getIntersectionLength).reduce(0,Integer::sum)/basesOnReference;
                        weights.compute(overlap.getKey(), (k,v) -> v == null? weight : v + weight);
                        summedUnNormalizedWeights += weight;
                    }
                }
                summedUnNormalizedWeights += 1.0 - (double)totalCoveredBases/basesOnReference;
                final double normalizationFactor = 1.0d/summedUnNormalizedWeights;
                for (final Gff3BaseData feature : weights.keySet()) {
                    weights.compute(feature, (k,v) -> v*normalizationFactor);
                }
                return weights;
            }
        };
        abstract Map getWeights(final List alignmentIntervals, final OverlapDetector> featureOverlapDetector);
    }
    enum MultiMapMethod {
        IGNORE {
            @Override
            protected Map getWeightsForMethod(final int nHits, final Map previousWeights) {
                if (nHits == 1) {
                    return previousWeights;
                } else {
                    return Collections.emptyMap();
                }
            }
        },
        EQUAL {
            @Override
            protected Map getWeightsForMethod(final int nHits, final Map previousWeights) {
                if (nHits == 1) {
                    return previousWeights;
                } else {
                    final Map newWeights = new HashMap<>(previousWeights.size());
                    for (final Map.Entry entry : previousWeights.entrySet()) {
                        newWeights.put(entry.getKey(), entry.getValue()/(double)nHits);
                    }
                    return newWeights;
                }
            }
        };
        Map getWeights(final int nHits, final Map previousWeights) {
            if (nHits < 1) {
                throw new GATKException("nHits = " + nHits + ", cannot be less than 1");
            }
            return getWeightsForMethod(nHits, previousWeights);
        }
        abstract protected Map getWeightsForMethod(final int nHits, final Map previousWeights);
    }
    @Override
    public List getDefaultReadFilters() {
        final List readFilters = new ArrayList<>();
        readFilters.add(ReadFilterLibrary.VALID_ALIGNMENT_START);
        readFilters.add(ReadFilterLibrary.VALID_ALIGNMENT_END);
        readFilters.add(new AlignmentAgreesWithHeaderReadFilter(getHeaderForReads()));
        readFilters.add(ReadFilterLibrary.HAS_MATCHING_BASES_AND_QUALS);
        readFilters.add(ReadFilterLibrary.READLENGTH_EQUALS_CIGARLENGTH);
        readFilters.add(ReadFilterLibrary.SEQ_IS_STORED);
        readFilters.add(ReadFilterLibrary.MAPPED);
        readFilters.add(ReadFilterLibrary.NON_ZERO_REFERENCE_LENGTH_ALIGNMENT);
        readFilters.add(ReadFilterLibrary.NOT_DUPLICATE);
        readFilters.add(mappingQualityFilter);
        return readFilters;
    }
    @Override
    public void onTraversalStart() {
        validateOutputFile(outputCountsFile);
        if(multiMapMethod == MultiMapMethod.EQUAL) {
            mappingQualityFilter.minMappingQualityScore = 0;
        }
        final SAMSequenceDictionary dict = getBestAvailableSequenceDictionary();
        final SAMFileHeader header = getHeaderForReads();
        for (final SAMReadGroupRecord readGroupRecord : header.getReadGroups()) {
            if (sampleName == null) {
                sampleName = readGroupRecord.getSample();
            } else {
                if (!sampleName.equals(readGroupRecord.getSample())) {
                    throw new GATKException("Cannot run GeneExpressionEvaluation on multi-sample bam.");
                }
            }
        }
        if (dict == null) {
            throw new GATKException("sequence dictionary must be specified (" + StandardArgumentDefinitions.SEQUENCE_DICTIONARY_NAME + ").");
        }
        logger.info("collecting list of features");
        final List allIntervals = hasUserSuppliedIntervals()? getTraversalIntervals() : IntervalUtils.getAllIntervalsForReference(dict);
        for (final SimpleInterval interval : allIntervals) {
            final List contigFeatures = features.getFeatures(gffFile, interval);
            logger.info("collecting features in " + interval.getContig() + ":" + interval.getStart() + "-" + interval.getEnd());
            for (final Gff3Feature feature : contigFeatures) {
                if (groupingType.contains(feature.getType())) {
                    final List overlappingFeatures = feature.getDescendents().stream().filter(f -> overlapType.contains(f.getType())).map(f -> new Interval(f.getContig(), f.getStart(), f.getEnd())).collect(Collectors.toList());
                    final Gff3BaseData shrunkGroupingBaseData = shrinkBaseData(feature.getBaseData());
                    addGroupingFeature(shrunkGroupingBaseData, overlappingFeatures);
                }
            }
        }
        logger.info("Collecting read counts...");
    }
    private void validateOutputFile(final File file) {
        if (file.exists()) {
            if (!Files.isWritable(file.toPath())) {
                throw new UserException.CouldNotCreateOutputFile(file, " is not writable");
            }
        } else {
            if (!Files.isWritable(file.getAbsoluteFile().getParentFile().toPath())) {
                throw new UserException.CouldNotCreateOutputFile(file, " is not writable");
            }
        }
    }
    private Gff3BaseData shrinkBaseData(final Gff3BaseData baseData) {
        //remove all but featureLabelKey attributes
        final Map> shrunkAttributes = baseData.getAttributes().entrySet().stream().filter(e -> e.getKey().equals(featureLabelKey.getKey())).collect(Collectors.toMap(Map.Entry::getKey,Map.Entry::getValue));
        return new Gff3BaseData(baseData.getContig(), baseData.getSource(), baseData.getType(), baseData.getStart(), baseData.getEnd(), baseData.getScore(), baseData.getStrand(), baseData.getPhase(), shrunkAttributes);
    }
    private void addGroupingFeature(final Gff3BaseData groupingBaseData, final List overlappingFeatures) {
        final String geneLabel = featureLabelKey.getValue(groupingBaseData);
        if (geneLabel == null) {
            throw new UserException("no geneid field " + featureLabelKey + " found in feature at " + groupingBaseData.getContig() + ":" + groupingBaseData.getStart() + "-" + groupingBaseData.getEnd());
        }
        featureCounts.put(groupingBaseData, new Coverage(0, 0));
        for (final Interval overlappingFeature : overlappingFeatures) {
            featureOverlapDetector.addLhs(Pair.of(groupingBaseData, overlappingFeature), overlappingFeature);
        }
    }
    static boolean inGoodPair(final GATKRead read, int minimumMappingQuality, final ReadStrands readStrands) {
        boolean ret = !read.mateIsUnmapped() && read.isProperlyPaired() && read.getContig().equals(read.getMateContig());
        if (ret) {
            final boolean readsOnSameStrand = read.isReverseStrand() == read.mateIsReverseStrand();
            ret = readStrands.expectReadsOnSameStrand == readsOnSameStrand;
        }
        if (ret) {
            if (!read.hasAttribute(SAMTag.MQ.toString())) {
                throw new GATKException("Mate quality must be included.  Consider running FixMateInformation.");
            }
            ret = read.getAttributeAsInteger(SAMTag.MQ.toString()) >= minimumMappingQuality;
        }
        return ret;
    }
    /**
     * Get the alignment intervals associated with the particular read.  If the read is part of a "good pair" then the alignment intervals will be calculated based on the read and its
     * mate, otherwise based on only the read.  For spliced data, the alignment intervals are based on the alignment blocks of the read (and its mate if a "good pair").  For unspliced data,
     * only a single interval is returned, which is the interval from the 5' most base covered by the read or its mate, and the 3' most base covered by the read or its mate (or by the read only
     * if not a "good pair").  Note that for good pairs this method is only called for one of the reads in the pair so as to not double count.
     *
     * @param read                  the read
     * @param unspliced             whether data is unspliced
     * @param minimumMappingQuality the minimum mapping quality required of both the read and its mate in order to be considered a "good pair"
     * @return alignment intervals to search for overlapping genes
     */
    static List getAlignmentIntervals(final GATKRead read, final boolean unspliced, final int minimumMappingQuality, final ReadStrands readStrands) {
        if (!unspliced) {
            final List alignmentIntervals = new ArrayList<>();
            final List readAlignmentBlocks = SAMUtils.getAlignmentBlocks(read.getCigar(), read.getStart(), "read cigar");
            for( final AlignmentBlock block : readAlignmentBlocks) {
                alignmentIntervals.add(new Interval(read.getContig(), block.getReferenceStart(), block.getReferenceStart()+block.getLength() - 1));
            }
            if (inGoodPair(read, minimumMappingQuality, readStrands)) {
                final String mateCigarString = read.getAttributeAsString(SAMTag.MC.toString());
                if(mateCigarString == null) {
                    throw new GATKException("Mate cigar must be present if using spliced reads");
                }
                final List mateAlignmentBlocks = SAMUtils.getAlignmentBlocks(TextCigarCodec.decode(mateCigarString), read.getMateStart(), "mate cigar");
                for( final AlignmentBlock block : mateAlignmentBlocks) {
                    final Interval alignmentBlockInterval = new Interval(read.getMateContig(), block.getReferenceStart(), block.getReferenceStart()+block.getLength() - 1);
                    alignmentIntervals.add(alignmentBlockInterval);
                }
            }
            return getMergedIntervals(alignmentIntervals);
        } else {
            if (read.isUnmapped()) {
                return Collections.emptyList();
            }
            final boolean inGoodPair = inGoodPair(read, minimumMappingQuality, readStrands);
            final int start = inGoodPair? Math.min(read.getStart(), read.getMateStart()) : read.getStart();
            final int end = inGoodPair? start + Math.abs(read.getFragmentLength()) - 1 : read.getEnd();
            return Collections.singletonList(new Interval(read.getContig(), start, end));
        }
    }
    @Override
    public void apply(GATKRead read, ReferenceContext referenceContext, FeatureContext featureContext) {
        if ((read.isFirstOfPair() || !inGoodPair(read, mappingQualityFilter.minMappingQualityScore, readStrands))) {
            final List alignmentIntervals = getAlignmentIntervals(read, unspliced, mappingQualityFilter.minMappingQualityScore, readStrands);
            final Map initialWeights = multiOverlapMethod.getWeights(alignmentIntervals, featureOverlapDetector);
            final Map finalWeights = multiMapMethod.getWeights(read.hasAttribute(SAMTag.NH.toString()) ? read.getAttributeAsInteger(SAMTag.NH.toString()) : 1, initialWeights);
            for (final Map.Entry weight : finalWeights.entrySet()) {
                final Gff3BaseData feature = weight.getKey();
                final boolean isSense = readStrands.isSense(read, feature);
                if (isSense) {
                    featureCounts.get(feature).addSenseCount(weight.getValue());
                } else {
                    featureCounts.get(feature).addAntiSenseCount(weight.getValue());
                }
            }
        }
    }
    @Override
    public Object onTraversalSuccess() {
        logger.info(String.format("Writing read counts to %s...", outputCountsFile.getAbsolutePath()));
        try (final FragmentCountWriter writer = new FragmentCountWriter(outputCountsFile.toPath(), sampleName, featureLabelKey)) {
            int i=0;
            for (final GATKPath input_bam: readArguments.getReadPathSpecifiers()) {
                writer.writeMetadata("input_bam_"+i, input_bam.toString());
                i++;
            }
            writer.writeMetadata("annotation_file", gffFile.toString());
            for (final Map.Entry featureCoverage : featureCounts.entrySet()) {
                final Gff3BaseData feature = featureCoverage.getKey();
                final Coverage coverage = featureCoverage.getValue();
                writer.writeRecord(new SingleStrandFeatureCoverage(feature, coverage.sense_count, true));
                if (feature.getStrand() != Strand.NONE) {
                    writer.writeRecord(new SingleStrandFeatureCoverage(feature, coverage.antisense_count, false));
                }
            }
        } catch (IOException ex) {
            throw new UserException(ex.getMessage());
        }
        logger.info(String.format("%s complete.", getClass().getSimpleName()));
        return null;
    }
    public static class FragmentCountWriter extends TableWriter {
        final String name;
        final FeatureLabelType gene_label_key;
        public FragmentCountWriter(final Path file , final String name, final FeatureLabelType gene_label_key) throws IOException {
            super(file, new TableColumnCollection("gene_label", "contig", "start", "stop", "strand", "sense_antisense", name+"_counts"));
            this.name = name;
            this.gene_label_key = gene_label_key;
        }
        protected void composeLine(final SingleStrandFeatureCoverage fragmentCount, final DataLine dataLine) {
            final String gene_label = gene_label_key.getValue(fragmentCount.baseData);
            dataLine.set("contig", fragmentCount.baseData.getContig())
                    .set("start", fragmentCount.baseData.getStart())
                    .set("stop", fragmentCount.baseData.getEnd())
                    .set("strand", fragmentCount.baseData.getStrand().encode())
                    .set("sense_antisense", fragmentCount.sense? "sense" : "antisense")
                    .set(name != null? name+"_counts" : "counts", fragmentCount.count, 2)
                    .set("gene_label", gene_label == null? "" : gene_label);
        }
    }
    public static class FragmentCountReader extends TableReader {
        public FragmentCountReader(final Path file) throws IOException {
            super(file);
        }
        @Override
        protected GeneExpressionEvaluation.SingleStrandFeatureCoverage createRecord(final DataLine dataLine) {
            return new GeneExpressionEvaluation.SingleStrandFeatureCoverage(new Gff3BaseData(dataLine.get("contig"),".", ".",
                    dataLine.getInt("start"), dataLine.getInt("stop"), -1d, Strand.decode(dataLine.get("strand")), -1,
                    Collections.singletonMap("ID", Collections.singletonList(dataLine.get("gene_label")))), dataLine.getDouble(6), dataLine.get("sense_antisense").equals("sense"));
        }
    }
    public static class Coverage {
        private double sense_count;
        private double antisense_count;
        public Coverage(final double sense_count, final double antisense_count) {
            this.sense_count = sense_count;
            this.antisense_count = antisense_count;
        }
        public void addSenseCount(double count) {
            sense_count += count;
        }
        public void addAntiSenseCount(double count) {
            antisense_count += count;
        }
        public double getSenseCount() {
            return sense_count;
        }
        public double getAntisenseCount() {
            return antisense_count;
        }
    }
    public static class SingleStrandFeatureCoverage {
        public final Gff3BaseData baseData;
        public final double count;
        public final boolean sense;
        SingleStrandFeatureCoverage(final Gff3BaseData baseData, final double count, final boolean sense) {
            this.baseData = baseData;
            this.count = count;
            this.sense = sense;
        }
    }
    enum ReadStrands {
        FORWARD_FORWARD(true, true),
        FORWARD_REVERSE(true, false),
        REVERSE_FORWARD(false, true),
        REVERSE_REVERSE(false, false);
        ReadStrands(final boolean r1TranscriptionStrand, final boolean r2TranscriptionStrand) {
            this.r1TranscriptionStrand = r1TranscriptionStrand;
            this.r2TranscriptionStrand = r2TranscriptionStrand;
            this.expectReadsOnSameStrand = r1TranscriptionStrand == r2TranscriptionStrand;
        }
        final boolean r1TranscriptionStrand;
        final boolean r2TranscriptionStrand;
        final boolean expectReadsOnSameStrand;
        boolean isSense(final GATKRead read, final Gff3BaseData feature) {
            final Strand senseStrand = feature.getStrand() == Strand.NONE ? Strand.POSITIVE : feature.getStrand(); // call forward strand sense for unstranded features
            final boolean isTranscriptionStrand = read.isFirstOfPair() ? r1TranscriptionStrand : r2TranscriptionStrand;
            final boolean trancriptionStrandIsReverseStrand = isTranscriptionStrand == read.isReverseStrand();
            final Strand transcriptionStrand = trancriptionStrandIsReverseStrand? Strand.NEGATIVE : Strand.POSITIVE;
            return transcriptionStrand == senseStrand;
        }
    }
    private static List getMergedIntervals(final List intervals) {
        final List intervalsCopy = new ArrayList<>(intervals);
        Collections.sort(intervalsCopy);
        final IntervalList.IntervalMergerIterator mergeringIterator = new IntervalList.IntervalMergerIterator(intervalsCopy.iterator(), true, true, false);
        final List merged = new ArrayList<>();
        while (mergeringIterator.hasNext()) {
            merged.add(mergeringIterator.next());
        }
        return merged;
    }
}