All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.maizegenetics.analysis.gbs.v2.SNPQualityProfilerPlugin Maven / Gradle / Ivy

Go to download

TASSEL is a software package to evaluate traits associations, evolutionary patterns, and linkage disequilibrium.

There is a newer version: 5.2.94
Show newest version
package net.maizegenetics.analysis.gbs.v2;


import com.google.common.collect.ImmutableMultimap;
import com.google.common.collect.Multimap;

import net.maizegenetics.dna.map.Position;
import net.maizegenetics.dna.snp.Allele;
import net.maizegenetics.dna.snp.NucleotideAlignmentConstants;
import net.maizegenetics.dna.tag.TagData;
import net.maizegenetics.dna.tag.TagDataSQLite;
import net.maizegenetics.dna.tag.TaxaDistribution;
import net.maizegenetics.plugindef.AbstractPlugin;
import net.maizegenetics.plugindef.DataSet;
import net.maizegenetics.plugindef.PluginParameter;
import net.maizegenetics.taxa.TaxaList;
import net.maizegenetics.taxa.TaxaListIOUtils;

import org.apache.log4j.Logger;

import javax.swing.*;

import java.awt.*;
import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;
import java.util.*;
import java.util.List;
import java.util.concurrent.atomic.LongAdder;
import java.util.stream.Collectors;
import java.util.stream.IntStream;
import java.util.stream.Stream;


/**
 * Scores all discovered SNPs for various coverage, depth, and genotypic statistics for a given set of taxa (samples).
 * For each subset of taxa, there are expectations for segregation that can be used to determine whether the SNP is
 * behaving appropriately.
 *
 * @author Ed Buckler
 */
public class SNPQualityProfilerPlugin extends AbstractPlugin {

    private static final Logger myLogger = Logger.getLogger(SNPQualityProfilerPlugin.class);

    private PluginParameter myTaxaFile = new PluginParameter.Builder("taxa", null, String.class).guiName("Taxa List File").inFile()
            .description("Name of taxa list input file in taxa list format").build();
    private PluginParameter myDBFile = new PluginParameter.Builder("db", null, String.class).guiName("GBS DB File").required(true).outFile()
            .description("Name of output file (e.g. GBSv2.db)").build();
    private PluginParameter myTaxaListName = new PluginParameter.Builder("tname", null, String.class).guiName("Name for taxa set in DB")
            .description("Name of taxa set for database").build();
    private PluginParameter statFileName = new PluginParameter.Builder("statFile",null,String.class).guiName("Name for Stat File Output")
            .description("Name of Stat File for Output").build();
    private PluginParameter myDeleteOldData = new PluginParameter.Builder("deleteOldData",true,Boolean.class).guiName("Delete Old Data")
            .description("Delete existing SNP quality data from db tables").build();
    private TagDataSQLite tagDataWriter;

    public SNPQualityProfilerPlugin() {
        super(null, false);
    }

    public SNPQualityProfilerPlugin(Frame parentFrame, boolean isInteractive) {
        super(parentFrame, isInteractive);
    }

    @Override
    public DataSet processData(DataSet input) {
        tagDataWriter=new TagDataSQLite(dBFile());
        TaxaList taxaList=tagDataWriter.getTaxaList();
        TaxaList subTaxa;
        if (myTaxaFile.isEmpty()) {
            subTaxa=taxaList;
            if (myTaxaListName.isEmpty()) taxaListName("ALL");
        } else {
            subTaxa=TaxaListIOUtils.readTaxaAnnotationFile(taxaFile(), "");
            if (myTaxaListName.isEmpty()) taxaListName(taxaFile());
        }

        if (deleteOldData()) {
            myLogger.info("deleteOldData is TRUE: Clearing existing snpQuality data");
            tagDataWriter.clearSNPQualityData();
        }
        
        subTaxa.stream().filter(t -> taxaList.indexOf(t)<0).forEach(t-> System.err.println("Missing taxon from master:" + t));
        int[] subsetIndices=subTaxa.stream().mapToInt(taxaList::indexOf).filter(i -> i > -1).sorted().toArray();
        // int[] subsetIndices= IntStream.range(0,taxaList.numberOfTaxa()).toArray();  //for testing using all taxa
        System.out.println("sublist");
        System.out.println(Arrays.toString(subsetIndices));
        

        int totalRecords = 2000;
        long startTimeNew = System.currentTimeMillis();
        
        Comparator arrayCompare=Comparator.comparing(depths -> -Arrays.stream(depths).sum());
        
        // Stream> streamOfAlleles = tagDataWriter.getAllAllelesTaxaDistForSNP();
        Stream> streamOfAlleles = tagDataWriter.getAllAllelesTaxaDistForSNPEntries();

        LongAdder adder=new LongAdder();
        
        //These multimaps only have one single entry in them.
        //Iterator> streamIterator = streamOfAlleles.iterator();
        Iterator> streamIterator = streamOfAlleles.iterator();
        if (!streamIterator.hasNext() ) {
            System.out.println("\nERROR: No SNP entries found in the SNP Position table.  \nPlease run DiscoverySNPCallerPluginV2 to call SNPs before running this plugin.\n");
            return null;
        }
        //Set up aggregate objects
        ImmutableMultimap.Builder aggMapBuilder = ImmutableMultimap.builder();
        
        //Grab initial values
        //TODO: Need to error check here for null stream
        
        //ImmutableMultimap currentMap = streamIterator.next();
        Map.Entry currentMap = streamIterator.next();
       
        //Position currentPosition = currentMap.keySet().asList().get(0).position();
        Position currentPosition = currentMap.getKey().position();
        //aggMapBuilder.putAll(currentMap);
        aggMapBuilder.put(currentMap);
        //Set up Statistic OutputFile
        BufferedWriter fileWriter = null;
        if(statFileName.value()!=null) {
            try {
                fileWriter = new BufferedWriter(new FileWriter(statFileName.value()));
                //fileWriter.write("Chromosome,PositionID,avgDepth,minorDepthProp,minor2DepthProp,gapDepthProp,propCovered,propCovered2,taxaCntWithMinorAlleleGE2,genotypeCnt,minorAlleleFreqGE2,hetFreq_DGE2,inbredF_DGE2");
                fileWriter.write("Chromosome\tPositionID\tavgDepth\tminorDepthProp\tminor2DepthProp\tgapDepthProp\tpropCovered\tpropCovered2\ttaxaCntWithMinorAlleleGE2\tgenotypeCnt\tminorAlleleFreqGE2\thetFreq_DGE2\tinbredF_DGE2");
                fileWriter.write("\n");               
            }catch(IOException e) {
                System.out.println(e);
                return null;
            } 
        }
        
        System.out.print("Processing Positions between 0 and 10,000.");
        //Iterate through streamIterator
        while(streamIterator.hasNext()) {
            
            //Grab the object
            currentMap = streamIterator.next();
            
            //If current position is equal to iterator.position then put it in the aggregator
            if(currentPosition.equals(currentMap.getKey().position())) {
                aggMapBuilder.put(currentMap);
            }
            /*
            if(currentPosition.equals(currentMap.keySet().asList().get(0).position())) {
                aggMapBuilder.putAll(currentMap);
            }
            */
            //Else
            else {
                //Build Aggregator object
                Multimap aTDMMap = aggMapBuilder.build();
                //Run set up the previous things
                Map subDepths = convertToSubsetMap(aTDMMap, subsetIndices);
                List depthsInOrder = subDepths.values().stream().sorted(arrayCompare).collect(Collectors.toList());
                Map qualMap = new HashMap<>();
                //depth stats
                int[] alleleDepths = depthsInOrder.stream().mapToInt(depths -> Arrays.stream(depths).sum()).toArray();
                double totalDepth = (double) Arrays.stream(alleleDepths).sum();
                StringBuilder strBuild = new StringBuilder();
                qualMap.put("avgDepth", totalDepth / (double) subsetIndices.length); 
                if (totalDepth > 0) {
                    qualMap.put("minorDepthProp", alleleDepths.length > 1 ? alleleDepths[1] / totalDepth : 0.0);
                    qualMap.put("minor2DepthProp", alleleDepths.length > 2 ? alleleDepths[2] / totalDepth : 0.0);
                    int gapDepth = subDepths.entrySet().stream()
                            .filter(ent -> ent.getKey().allele() == NucleotideAlignmentConstants.GAP_ALLELE)
                            .mapToInt(ent -> Arrays.stream(ent.getValue()).sum()).sum();
                    Arrays.stream(subDepths.getOrDefault(NucleotideAlignmentConstants.GAP_ALLELE, new int[0])).sum();
                    qualMap.put("gapDepthProp", (double) gapDepth / totalDepth);
                    
                    //coverage stats
                    int[] coverage = new int[subsetIndices.length];
                    for (int[] depths : depthsInOrder) {
                        for (int i = 0; i < depths.length; i++) coverage[i] += depths[i];
                    }
                    qualMap.put("propCovered", (double) Arrays.stream(coverage).filter(d -> d > 0).count() / (double) coverage.length);
                    qualMap.put("propCovered2", (double) Arrays.stream(coverage).filter(d -> d > 1).count() / (double) coverage.length);
                    qualMap.put("taxaCntWithMinorAlleleGE2", alleleDepths.length > 1 ? (double) Arrays.stream(depthsInOrder.get(1)).filter(d -> d > 1).count() : 0);
                    
                    //genotypic stats
                    //Only add if there are more than 2 alleles
                    if(depthsInOrder.size()>=2) {
                        GenotypeStats genotypeCnt = callGenotypes(depthsInOrder.get(0), depthsInOrder.get(1));
                        qualMap.put("genotypeCnt", (double) genotypeCnt.totalCnt);
                        qualMap.put("minorAlleleFreqGE2",Double.isNaN(genotypeCnt.minorFreq)?0.0:genotypeCnt.minorFreq);
                        qualMap.put("hetFreq_DGE2", (double) genotypeCnt.hetCnt);
                        qualMap.put("inbredF_DGE2", genotypeCnt.f);
                    }
                    /*
                    GenotypeStats genotypeCnt = callGenotypes(depthsInOrder.get(0), depthsInOrder.get(1));
                    qualMap.put("genotypeCnt", (double) genotypeCnt.totalCnt);
                    qualMap.put("minorAlleleFreqGE2",Double.isNaN(genotypeCnt.minorFreq)?0.0:genotypeCnt.minorFreq);
                    qualMap.put("hetFreq_DGE2", (double) genotypeCnt.hetCnt);
                    qualMap.put("inbredF_DGE2", genotypeCnt.f);
                    */
                    
                    //System.out.println("jsonObject:" + jsonObject.toJSONString());
//                    if ((Double) qualMap.getOrDefault("inbredF_DGE2", 0.0) > 0.9 && qualMap.getOrDefault("genotypeCnt", 0.0) > 10) {
//                        MAF.out.println(adder.intValue() + "\t" + qualMap.get("inbredF_DGE2") + "\t" + qualMap.get("minorDepthProp"));
//                    }
                    
                    Map> resultMap = new HashMap<>();
                    resultMap.put(currentPosition,qualMap);
                    
                    try {
                        tagDataWriter.putSNPQualityProfile(resultMap,myTaxaListName.value(),adder.intValue()); 
                    } catch (Exception sqlE) {
                        // Exit on SQL exception, (quite likely a UNIQUE constraint issue). 
                        // Unique constraint issues happens if the data for this taxa list name already exists.
                        System.out.println("Error processing request.  Quality data may already exist for taxa name "
                                + taxaListName() + "\n " + sqlE.getMessage());
                        try{ // CLose the DB
                            ((TagDataSQLite)tagDataWriter).close();
                        } catch (Exception ex) {ex.printStackTrace();}
                        return null;
                    }
                    
                    adder.increment();
                    if(adder.intValue()%2000==0) {
                        System.out.print(".");
                    }
                    if(adder.intValue()%10000 == 0) {
                        System.out.println("DONE. Time: "+(((double)System.currentTimeMillis()-startTimeNew)/1000));
                        System.out.print("Processing Positions between "+(adder.intValue())+" and "+(adder.intValue()+10000)+".");
                    }

                    // This needs to use currentPosition, NOT currentMap !!
                    strBuild.append(currentPosition.getChromosome().toString());
                    strBuild.append("\t");
                    strBuild.append(currentPosition.getPosition());
                    strBuild.append("\t");
                    strBuild.append(qualMap.get("avgDepth"));
                    strBuild.append("\t");
                    strBuild.append(qualMap.get("minorDepthProp"));
                    strBuild.append("\t");
                    strBuild.append(qualMap.get("minor2DepthProp"));
                    strBuild.append("\t");
                    strBuild.append(qualMap.get("gapDepthProp"));
                    strBuild.append("\t");
                    strBuild.append(qualMap.get("propCovered"));
                    strBuild.append("\t");
                    strBuild.append(qualMap.get("propCovered2"));
                    strBuild.append("\t");
                    strBuild.append(qualMap.get("taxaCntWithMinorAlleleGE2"));
                    
                    if(depthsInOrder.size()>=2) {
                        strBuild.append("\t");
                        strBuild.append(qualMap.get("genotypeCnt"));
                        strBuild.append("\t");
                        strBuild.append(qualMap.get("minorAlleleFreqGE2"));
                        strBuild.append("\t");
                        strBuild.append(qualMap.get("hetFreq_DGE2"));
                        strBuild.append("\t");
                        strBuild.append(qualMap.get("inbredF_DGE2"));   
                    }
                    strBuild.append("\n");
                    
                }
                if(statFileName.value()!=null) {
                    try {               
                        fileWriter.write(strBuild.toString());
                    }
                    catch(IOException e) {
                        System.out.println(e);
                    }
                }
                //Reset currentPosition and aggregatorBuilder
                //currentPosition = currentMap.keySet().asList().get(0).position();
                currentPosition = currentMap.getKey().position();
                aggMapBuilder = ImmutableMultimap.builder();
                //aggMapBuilder.putAll(currentMap);
                aggMapBuilder.put(currentMap);
            }
        }
        System.out.println("DONE");
        long totalTimeNew = System.currentTimeMillis() - startTimeNew;
        //Build final Aggregator object
        Multimap aTDMMap = aggMapBuilder.build();
        //Run setup from old code
        Map subDepths = convertToSubsetMap(aTDMMap, subsetIndices);
        List depthsInOrder = subDepths.values().stream().sorted(arrayCompare).collect(Collectors.toList());
        Map qualMap = new HashMap<>();
        //depth stats
        int[] alleleDepths = depthsInOrder.stream().mapToInt(depths -> Arrays.stream(depths).sum()).toArray();
        StringBuilder strBuild = new StringBuilder();
        double totalDepth = (double) Arrays.stream(alleleDepths).sum();
        qualMap.put("avgDepth", totalDepth / (double) subsetIndices.length);
        if (totalDepth > 0) {
            
            qualMap.put("minorDepthProp", alleleDepths.length > 1 ? alleleDepths[1] / totalDepth : 0.0);
            qualMap.put("minor2DepthProp", alleleDepths.length > 2 ? alleleDepths[2] / totalDepth : 0.0);
            
            int gapDepth = subDepths.entrySet().stream()
                    .filter(ent -> ent.getKey().allele() == NucleotideAlignmentConstants.GAP_ALLELE)
                    .mapToInt(ent -> Arrays.stream(ent.getValue()).sum()).sum();
            Arrays.stream(subDepths.getOrDefault(NucleotideAlignmentConstants.GAP_ALLELE, new int[0])).sum();
            qualMap.put("gapDepthProp", (double) gapDepth / totalDepth);
            
            //coverage stats
            int[] coverage = new int[subsetIndices.length];
            for (int[] depths : depthsInOrder) {
                for (int i = 0; i < depths.length; i++) coverage[i] += depths[i];
            }
            qualMap.put("propCovered", (double) Arrays.stream(coverage).filter(d -> d > 0).count() / (double) coverage.length);
            qualMap.put("propCovered2", (double) Arrays.stream(coverage).filter(d -> d > 1).count() / (double) coverage.length);
            qualMap.put("taxaCntWithMinorAlleleGE2", alleleDepths.length > 1 ? (double) Arrays.stream(depthsInOrder.get(1)).filter(d -> d > 1).count() : 0);
            
            //genotypic stats
            if(depthsInOrder.size()>=2) {
                GenotypeStats genotypeCnt = callGenotypes(depthsInOrder.get(0), depthsInOrder.get(1));
                qualMap.put("genotypeCnt", (double) genotypeCnt.totalCnt);
                qualMap.put("minorAlleleFreqGE2",Double.isNaN(genotypeCnt.minorFreq)?0.0:genotypeCnt.minorFreq);
                qualMap.put("hetFreq_DGE2", (double) genotypeCnt.hetCnt);
                qualMap.put("inbredF_DGE2", genotypeCnt.f);
            }
            
            
            //System.out.println("jsonObject:" + jsonObject.toJSONString());
//            if ((Double) qualMap.getOrDefault("inbredF_DGE2", 0.0) > 0.9 && qualMap.getOrDefault("genotypeCnt", 0.0) > 10) {
//                MAF.out.println(adder.intValue() + "\t" + qualMap.get("inbredF_DGE2") + "\t" + qualMap.get("minorDepthProp"));
//            }
            adder.increment();
                        
            Map> resultMap = new HashMap<>();
            resultMap.put(currentPosition,qualMap);
                        
            try {
                tagDataWriter.putSNPQualityProfile(resultMap,myTaxaListName.value(),-1); 
            } catch (Exception sqlE) {
                // Exit if we hit n SQL error, which is most often a UNIQUE constraint issue. 
                // This happens if the data for this taxa list name already exists.
                System.out.println("Error processing request.  Quality data may already exist for taxa name "
                        + taxaListName() + "\n " + sqlE.getMessage());
                sqlE.printStackTrace();
                try{ // Close/unlock database
                    ((TagDataSQLite)tagDataWriter).close();
                } catch (Exception ex) {ex.printStackTrace();}
                return null;
            }
            
            strBuild.append(currentMap.getKey().position().getChromosome().toString());
            strBuild.append("\t");
            strBuild.append(currentMap.getKey().position().getPosition());
            strBuild.append("\t");
            strBuild.append(qualMap.get("avgDepth"));
            strBuild.append("\t");
            strBuild.append(qualMap.get("minorDepthProp"));
            strBuild.append("\t");
            strBuild.append(qualMap.get("minor2DepthProp"));
            strBuild.append("\t");
            strBuild.append(qualMap.get("gapDepthProp"));
            strBuild.append("\t");
            strBuild.append(qualMap.get("propCovered"));
            strBuild.append("\t");
            strBuild.append(qualMap.get("propCovered2"));
            strBuild.append("\t");
            strBuild.append(qualMap.get("taxaCntWithMinorAlleleGE2"));
            /*
            strBuild.append("\t");
            strBuild.append(qualMap.get("genotypeCnt"));
            strBuild.append("\t");
            strBuild.append(qualMap.get("minorAlleleFreqGE2"));
            strBuild.append("\t");
            strBuild.append(qualMap.get("hetFreq_DGE2"));
            strBuild.append("\t");
            strBuild.append(qualMap.get("inbredF_DGE2"));
            */
            if(depthsInOrder.size()>=2) {
                strBuild.append("\t");
                strBuild.append(qualMap.get("genotypeCnt"));
                strBuild.append("\t");
                strBuild.append(qualMap.get("minorAlleleFreqGE2"));
                strBuild.append("\t");
                strBuild.append(qualMap.get("hetFreq_DGE2"));
                strBuild.append("\t");
                strBuild.append(qualMap.get("inbredF_DGE2"));   
            }
            strBuild.append("\n");
            if(statFileName.value()!=null) { 
                try {               
                    fileWriter.write(strBuild.toString());
                }
                catch(IOException e) {
                    System.out.println(e);
                }
            }
        }
         
        System.out.println("Total Time: " + (double)totalTimeNew/1000+" seconds.\nProcessed "+adder.intValue()+" positions.");
        
        //Close out file
        if(statFileName.value()!=null) {   
            try {
                fileWriter.close();
            }
            catch(IOException e) {
                System.out.println(e);
            }
        }
        try{
            ((TagDataSQLite)tagDataWriter).close();
        } catch (Exception ex) {ex.printStackTrace();}
        //TaxaListIOUtils.exportAnnotatedTaxaListTable();
        return null;
    }

    //Calculate the genotypic classes for all taxa with a depth greater than 2.
    // Returns an object with genotypic stats
    private GenotypeStats callGenotypes(int[] majorAlleleDepth, int[] minorAlleleDepth) {
        int[] genotypes=new int[3];
        for (int i = 0; i < majorAlleleDepth.length; i++) {
            if(majorAlleleDepth[i]+minorAlleleDepth[i]<2) continue;
            int genotype=(majorAlleleDepth[i]>0)?1:0;
            genotype+=(minorAlleleDepth[i]>0)?2:0;
            genotypes[genotype-1]++;
        }
        return new GenotypeStats(genotypes[0],genotypes[2],genotypes[1]);
    }

    private class GenotypeStats {
        double majorFreq, minorFreq;
        int homoMajorCnt, hetCnt, homoMinorCnt;
        int totalCnt;
        double f;

        private GenotypeStats(int homoMajorCnt, int hetCnt, int homoMinorCnt) {
            this.homoMajorCnt = homoMajorCnt;
            this.hetCnt = hetCnt;
            this.homoMinorCnt = homoMinorCnt;
            totalCnt=homoMajorCnt+hetCnt+homoMinorCnt;
            majorFreq=((double)homoMajorCnt+(double)hetCnt*0.5)/(double)totalCnt;
            minorFreq=1-majorFreq;
            double expHets = 2.0 * minorFreq * majorFreq;
            double propHets=(double)hetCnt/(double)totalCnt;
            f = 1.0 - (propHets / expHets);
        }
    }



    /*
    Reduces the TaxaDistributions to depths by allele-> selected subset of taxa
    Some alleles are represented by multiple tags and they are collapsed.
     */
    private Map convertToSubsetMap(Multimap aTDMMap, int[] subsetIndices) {
        Map result=new HashMap<>();
        for (Allele allele : aTDMMap.keySet()) {
            int[] subDepths=new int[subsetIndices.length];
            for (TaxaDistribution taxaDistribution : aTDMMap.get(allele)) {
                int[] depths=taxaDistribution.depths();
                for (int i = 0; i < subDepths.length; i++) {
                    subDepths[i]+=depths[subsetIndices[i]];
                }
            }
            result.put(allele,subDepths);
        }
        return result;
    }


    @Override
    public ImageIcon getIcon() {
        return null;
    }

    @Override
    public String getButtonName() {
        return "SNP Quality Profiler";
    }

    @Override
    public String getToolTipText() {
        return "SNP Quality Profiler";
    }

    // The following getters and setters were auto-generated.
    // Please use this method to re-generate.
    //
    // public static void main(String[] args) {
    //     GeneratePluginCode.generate(SNPQualityProfilerPlugin.class);
    // }

    /**
     * Convenience method to run plugin with one return object.
     */
    public TagData runPlugin(DataSet input) {
        return (TagData) performFunction(input).getData(0).getData();
    }

    /**
     * Name of taxa list input file in taxa list format
     *
     * @return Taxa List File
     */
    public String taxaFile() {
        return myTaxaFile.value();
    }

    /**
     * Set Taxa List File. Name of taxa list input file in
     * taxa list format
     *
     * @param value Taxa List File
     *
     * @return this plugin
     */
    public SNPQualityProfilerPlugin taxaFile(String value) {
        myTaxaFile = new PluginParameter<>(myTaxaFile, value);
        return this;
    }

    /**
     * Name of output file (e.g. GBSv2.db)
     *
     * @return GBS DB File
     */
    public String dBFile() {
        return myDBFile.value();
    }

    /**
     * Set GBS DB File. Name of output file (e.g. GBSv2.db)
     *
     * @param value GBS DB File
     *
     * @return this plugin
     */
    public SNPQualityProfilerPlugin dBFile(String value) {
        myDBFile = new PluginParameter<>(myDBFile, value);
        return this;
    }
    
    /**
     * Name of output file (e.g. GBSv2.db)
     *
     * @return GBS DB File
     */
    public String statFile() {
        return statFileName.value();
    }

    /**
     * Set GBS DB File. Name of output file (e.g. GBSv2.db)
     *
     * @param value GBS DB File
     *
     * @return this plugin
     */
    public SNPQualityProfilerPlugin statFile(String value) {
        statFileName = new PluginParameter<>(statFileName, value);
        return this;
    }

    /**
     * Name of taxa set for database
     *
     * @return Name for taxa set in DB
     */
    public String taxaListName() {
        return myTaxaListName.value();
    }

    /**
     * Set Name for taxa set in DB. Name of taxa set for database
     *
     * @param value Name for taxa set in DB
     *
     * @return this plugin
     */
    public SNPQualityProfilerPlugin taxaListName(String value) {
        myTaxaListName = new PluginParameter<>(myTaxaListName, value);
        return this;
    }
    
    /**
     * Delete exisiting SNP Quality data from DB
     *
     * @return deleteOldData
     */
    public Boolean deleteOldData() {
        return myDeleteOldData.value();
    }

    /**
     * Set Delete old data flag.  True indicates we want the
     * db tables cleared
     *
     * @param value true/false - whether to delete data
     *
     * @return this plugin
     */
    public SNPQualityProfilerPlugin deleteOldData(Boolean value) {
        myDeleteOldData = new PluginParameter<>(myDeleteOldData, value);
        return this;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy