/************************************************************************* * * * This file is part of the 20n/act project. * * 20n/act enables DNA prediction for synthetic biology/bioengineering. * * Copyright (C) 2017 20n Labs, Inc. * * * * Please direct all queries to act@20n.com. * * * * This program is free software: you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation, either version 3 of the License, or * * (at your option) any later version. * * * * This program is distributed in the hope that it will be useful, * * but WITHOUT ANY WARRANTY; without even the implied warranty of * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * * GNU General Public License for more details. * * * * You should have received a copy of the GNU General Public License * * along with this program. If not, see <http://www.gnu.org/licenses/>. * * * *************************************************************************/ package com.act.lcms.db.analysis; import com.act.biointerpretation.l2expansion.L2PredictionCorpus; import com.act.lcms.MS1; import com.act.lcms.MassCalculator; import com.act.lcms.XZ; import com.act.lcms.db.io.DB; import com.act.lcms.db.io.LoadPlateCompositionIntoDB; import com.act.lcms.db.io.report.IonAnalysisInterchangeModel; import com.act.lcms.db.model.LCMSWell; import com.act.lcms.db.model.Plate; import com.act.lcms.db.model.PlateWell; import com.act.lcms.db.model.ScanFile; import com.act.utils.TSVParser; import org.apache.commons.cli.CommandLine; import org.apache.commons.cli.CommandLineParser; import org.apache.commons.cli.DefaultParser; import org.apache.commons.cli.HelpFormatter; import org.apache.commons.cli.Option; import org.apache.commons.cli.Options; import org.apache.commons.cli.ParseException; import org.apache.commons.lang3.StringUtils; import org.apache.commons.lang3.tuple.Pair; import org.apache.commons.lang3.tuple.Triple; import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; import java.io.BufferedReader; import java.io.File; import java.io.FileReader; import java.io.IOException; import java.sql.SQLException; import java.util.ArrayList; import java.util.Arrays; import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.HashMap; import java.util.Set; import java.util.stream.Collectors; public class IonDetectionAnalysis <T extends PlateWell<T>> { private static final Logger LOGGER = LogManager.getFormatterLogger(IonDetectionAnalysis.class); private static final boolean USE_SNR_FOR_LCMS_ANALYSIS = true; private static final boolean USE_FINE_GRAINED_TOLERANCE = false; private static final String DEFAULT_ION = "M+H"; private static final Double MIN_INTENSITY_THRESHOLD = 10000.0; private static final Double MIN_SNR_THRESHOLD = 1000.0; private static final Double MIN_TIME_THRESHOLD = 15.0; private static final String OPTION_LCMS_FILE_DIRECTORY = "d"; // The OPTION_INPUT_PREDICTION_CORPUS file is a json formatted file that is serialized from the class "L2PredictionCorpus" // or a list of inchis. private static final String OPTION_INPUT_PREDICTION_CORPUS = "s"; private static final String OPTION_OUTPUT_PREFIX = "o"; private static final String OPTION_PLOTTING_DIR = "p"; private static final String OPTION_INCLUDE_IONS = "i"; private static final String OPTION_NON_REPLICATE_ANALYSIS = "n"; private static final String OPTION_LIST_OF_INCHIS_INPUT_FILE = "f"; // This input file is structured as a tsv file with the following schema: // WELL_TYPE PLATE_BARCODE WELL_ROW WELL_COLUMN // eg. POS 12389 0 1 private static final String OPTION_INPUT_POSITIVE_AND_NEGATIVE_CONTROL_WELLS_FILE = "t"; private static final String HEADER_WELL_TYPE = "WELL_TYPE"; private static final String HEADER_WELL_ROW = "WELL_ROW"; private static final String HEADER_WELL_COLUMN = "WELL_COLUMN"; private static final String HEADER_PLATE_BARCODE = "PLATE_BARCODE"; private static final String FAKE_CHEM_PREFIX = "CHEM_%05d"; private static final Set<String> ALL_HEADERS = new HashSet<>(Arrays.asList(HEADER_WELL_TYPE, HEADER_WELL_ROW, HEADER_WELL_COLUMN, HEADER_PLATE_BARCODE)); public static final String HELP_MESSAGE = StringUtils.join(new String[]{ "This class takes as input an experimental setup containing positive wells and negative control well. Along with this ", "the class also takes in a list of chemicals to be validated by the LCMS analysis. It then performs SNR analysis ", "to detect which chemicals are strongly represented in the LCMS analysis and outputs those in a json file per positive sample" }, ""); public static final HelpFormatter HELP_FORMATTER = new HelpFormatter(); static { HELP_FORMATTER.setWidth(100); } public static final List<Option.Builder> OPTION_BUILDERS = new ArrayList<Option.Builder>() {{ add(Option.builder(OPTION_LCMS_FILE_DIRECTORY) .argName("directory") .desc("The directory where LCMS analysis results live") .hasArg().required() .longOpt("data-dir") ); // The OPTION_INPUT_PREDICTION_CORPUS is in the L2PredictionCorpus JSON format. add(Option.builder(OPTION_INPUT_PREDICTION_CORPUS) .argName("input prediction corpus") .desc("The input prediction corpus") .hasArg().required() .longOpt("prediction-corpus") ); add(Option.builder(OPTION_OUTPUT_PREFIX) .argName("output prefix") .desc("A prefix for the output data/pdf files") .hasArg().required() .longOpt("output-prefix") ); add(Option.builder(OPTION_PLOTTING_DIR) .argName("plotting directory") .desc("The absolute path of the plotting directory") .hasArg().required() .longOpt("plotting-dir") ); add(Option.builder(OPTION_INCLUDE_IONS) .argName("ion list") .desc("A comma-separated list of ions to include in the search (ions not in this list will be ignored)") .hasArgs().valueSeparator(',') .longOpt("include-ions") ); add(Option.builder(OPTION_INPUT_POSITIVE_AND_NEGATIVE_CONTROL_WELLS_FILE) .argName("input positive and negative control wells") .desc("A tsv file containing positive and negative wells") .hasArg().required() .longOpt("wells-config") ); add(Option.builder(OPTION_LIST_OF_INCHIS_INPUT_FILE) .argName("file input type") .desc("If this option is specified, the input corpus is a list of inchis") .longOpt("file-input-type") ); add(Option.builder(OPTION_NON_REPLICATE_ANALYSIS) .argName("non replicate analysis") .desc("This option is for lcms analysis on multiple positive wells that are not replicates. The default is " + "replicate analysis. In replicate analysis, we simply add all the peak results from all replicates into one " + "model JSON object while in non-replicate analysis, we do not do this step.") .longOpt("non-replicate-analysis") ); }}; static { // Add DB connection options. OPTION_BUILDERS.addAll(DB.DB_OPTION_BUILDERS); } // Instance variables private File lcmsDir; private String plottingDirPath; private List<T> positiveWells; private List<T> negativeWells; private HashMap<Integer, Plate> plateCache; private Set<Pair<String, Double>> setOfMassCharges; private DB db; private Double progress; public IonDetectionAnalysis(File lcmsDir, List<T> positiveWells, List<T> negativeWells, String plottingDirPath, HashMap<Integer, Plate> plateCache, Set<Pair<String, Double>> setOfMassCharges, DB db) { this.lcmsDir = lcmsDir; this.positiveWells = positiveWells; this.negativeWells = negativeWells; this.plateCache = plateCache; this.setOfMassCharges = setOfMassCharges; this.db = db; this.plottingDirPath = plottingDirPath; this.progress = 0.0; } public static Map<Double, Set<Pair<String, String>>> constructMassChargeToChemicalIonsFromInputFile( File inputPredictionCorpus, Set<String> includeIons, Boolean listOfInchisFormat) throws IOException { List<String> products = new ArrayList<>(); if (listOfInchisFormat) { try (BufferedReader br = new BufferedReader(new FileReader(inputPredictionCorpus))) { // Get the inchis from input file String product; while ((product = br.readLine()) != null) { products.add(product.trim()); } } } else { products.addAll(L2PredictionCorpus.readPredictionsFromJsonFile(inputPredictionCorpus).getUniqueProductInchis()); } Map<Double, Set<Pair<String, String>>> massChargeToChemicalAndIon = new HashMap<>(); for (String inchi : products) { try { // Assume the ion modes are all positive! Map<String, Double> allMasses = MS1.getIonMasses(MassCalculator.calculateMass(inchi), MS1.IonMode.POS); Map<String, Double> metlinMasses = Utils.filterMasses(allMasses, includeIons, null); for (Map.Entry<String, Double> entry : metlinMasses.entrySet()) { Set<Pair<String, String>> res = massChargeToChemicalAndIon.get(entry.getValue()); if (res == null) { res = new HashSet<>(); massChargeToChemicalAndIon.put(entry.getValue(), res); } res.add(Pair.of(inchi, entry.getKey())); } } catch (Exception e) { LOGGER.error("Caught exception when trying to import %s", inchi); } } return massChargeToChemicalAndIon; } /** * This function is used to do two related things: a) create a mapping between a fake chemical id to mass charge * b) create a set of fake chemical id and mass charge pairs. We create these two representations since a) is used * to easily get the mass charge from its fake name (used in plotting) and b) is used to represent a set of mass charge * name and values, which traditionally have been chemical+ion combinations but since we are bucketting multiple * chemical+ion combinations in the same mass charge value, we use the fake chemical id to represent multiple chemical+ion * pairs. * @param massChargeToChemicalAndIon A set of mass charge values * @return A pair of a) A set of all Mass charge name and value pairs b) A mapping of fake chem id to mass charge * mass charge. */ public static Pair<Set<Pair<String, Double>>, Map<String, Double>> constructFakeNameToMassChargeAndSetOfMassChargePairs(Set<Double> massChargeToChemicalAndIon) { Set<Pair<String, Double>> searchMZs = new HashSet<>(); Integer chemicalCounter = 0; Map<String, Double> chemIDToMassCharge = new HashMap<>(); for (Double massCharge : massChargeToChemicalAndIon) { String chemID = String.format(FAKE_CHEM_PREFIX, chemicalCounter); chemIDToMassCharge.put(chemID, massCharge); searchMZs.add(Pair.of(chemID, massCharge)); chemicalCounter++; } return Pair.of(searchMZs, chemIDToMassCharge); } public static Map<ScanData.KIND, List<LCMSWell>> readInputExperimentalSetup(DB db, File inputFile) throws IOException, SQLException, ClassNotFoundException { TSVParser parser = new TSVParser(); parser.parse(inputFile); Set<String> headerSet = new HashSet<>(parser.getHeader()); if (!headerSet.equals(ALL_HEADERS)) { throw new RuntimeException(String.format("Invalid header types! The allowed header types are: %s", StringUtils.join(ALL_HEADERS, ","))); } Map<ScanData.KIND, List<LCMSWell>> result = new HashMap<>(); HashMap<String, Plate> plateCache = new HashMap<>(); for (Map<String, String> row : parser.getResults()) { String wellType = row.get(HEADER_WELL_TYPE); String barcode = row.get(HEADER_PLATE_BARCODE); Integer rowCoordinate = Integer.parseInt(row.get(HEADER_WELL_ROW)); Integer columnCoordinate = Integer.parseInt(row.get(HEADER_WELL_COLUMN)); Plate plate = plateCache.get(barcode); if (plate == null) { plate = Plate.getPlateByBarcode(db, barcode); plateCache.put(barcode, plate); } LCMSWell well = LCMSWell.getInstance().getByPlateIdAndCoordinates(db, plate.getId(), rowCoordinate, columnCoordinate); if (well == null) { throw new RuntimeException(String.format("Well plate id %d, row %d and col %d does not exist", plate.getId(), rowCoordinate, columnCoordinate)); } if (wellType.equals("POS")) { List<LCMSWell> values = result.get(ScanData.KIND.POS_SAMPLE); if (values == null) { values = new ArrayList<>(); result.put(ScanData.KIND.POS_SAMPLE, values); } values.add(well); } else { List<LCMSWell> values = result.get(ScanData.KIND.NEG_CONTROL); if (values == null) { values = new ArrayList<>(); result.put(ScanData.KIND.NEG_CONTROL, values); } values.add(well); } } return result; } /** * Get the intensity-time values for a well * @param well The well to analyze * @param kindOfWell The type of well in the experimental setup (ie pos or neg control) * @return The intensity-time values for a well * @throws Exception */ public ChemicalToMapOfMetlinIonsToIntensityTimeValues getIntensityTimeProfileForMassChargesInWell( T well, ScanData.KIND kindOfWell) throws Exception { Plate plate = plateCache.get(well.getPlateId()); if (plate == null) { plate = Plate.getPlateById(db, well.getPlateId()); plateCache.put(plate.getId(), plate); } ScanFile bestScanFile = AnalysisHelper.pickBestScanFileForWell(db, well); if (bestScanFile == null) { throw new RuntimeException(String.format("Could not find scan file for well id %d", well.getId())); } Map<Pair<String, Double>, ScanData<T>> massChargePairToScanDataResult = AnalysisHelper.getIntensityTimeValuesForEachMassChargeInScanFile(db, lcmsDir, setOfMassCharges, kindOfWell, plateCache, bestScanFile, well, USE_FINE_GRAINED_TOLERANCE, USE_SNR_FOR_LCMS_ANALYSIS); ChemicalToMapOfMetlinIonsToIntensityTimeValues signalProfile = AnalysisHelper.constructChemicalToMapOfMetlinIonsToIntensityTimeValuesFromMassChargeData( massChargePairToScanDataResult, kindOfWell); if (signalProfile == null) { throw new RuntimeException("No signal data available."); } return signalProfile; } /** * This function gets the intensity time values for all positive and negative wells * @return A mapping for well type to list of well and intensity-time value pairs. * @throws Exception */ public Map<ScanData.KIND, List<Pair<T, ChemicalToMapOfMetlinIonsToIntensityTimeValues>>> getIntensityTimeValuesForMassChargesInPositiveAndNegativeWells() throws Exception { Map<ScanData.KIND, List<Pair<T, ChemicalToMapOfMetlinIonsToIntensityTimeValues>>> designUnitToWellIntensityTimeValuePairs = new HashMap<>(); Integer wellCounter = 0; for (T positiveWell : positiveWells) { LOGGER.info("Reading scan data for positive well number: %s", wellCounter.toString()); ChemicalToMapOfMetlinIonsToIntensityTimeValues positiveWellSignalProfiles = getIntensityTimeProfileForMassChargesInWell(positiveWell, ScanData.KIND.POS_SAMPLE); if (positiveWellSignalProfiles == null) { throw new RuntimeException("Peak positive analysis was null"); } List<Pair<T, ChemicalToMapOfMetlinIonsToIntensityTimeValues>> values = designUnitToWellIntensityTimeValuePairs.get(ScanData.KIND.POS_SAMPLE); if (values == null) { values = new ArrayList<>(); designUnitToWellIntensityTimeValuePairs.put(ScanData.KIND.POS_SAMPLE, values); } values.add(Pair.of(positiveWell, positiveWellSignalProfiles)); updateProgress(); wellCounter++; } // Reset well counter for the negative well analysis wellCounter = 0; for (T negativeWell : negativeWells) { LOGGER.info("Reading scan data for negative well number: %s", wellCounter.toString()); ChemicalToMapOfMetlinIonsToIntensityTimeValues negativeWellSignalProfiles = getIntensityTimeProfileForMassChargesInWell(negativeWell, ScanData.KIND.NEG_CONTROL); if (negativeWellSignalProfiles == null) { throw new RuntimeException("Peak negative analysis was null"); } List<Pair<T, ChemicalToMapOfMetlinIonsToIntensityTimeValues>> values = designUnitToWellIntensityTimeValuePairs.get(ScanData.KIND.NEG_CONTROL); if (values == null) { values = new ArrayList<>(); designUnitToWellIntensityTimeValuePairs.put(ScanData.KIND.NEG_CONTROL, values); } values.add(Pair.of(negativeWell, negativeWellSignalProfiles)); updateProgress(); wellCounter++; } return designUnitToWellIntensityTimeValuePairs; } /** * This function does the SNR analysis for each positive well and constructs plots for each mass charge for each * positive well run * @return A map of positive well to map of chem id to plotting file path to pair of max intensity, time and SNR. * @throws Exception */ public Map<T, Map<String, Triple<String, XZ, Double>>> getSnrAndPlotResultsForMassChargesForEachPositiveWell() throws Exception { // Get signal profiles for each positive and negative wells once. This is the rate limiting step of the computation. Map<ScanData.KIND, List<Pair<T, ChemicalToMapOfMetlinIonsToIntensityTimeValues>>> designUnitToListOfWellIntensityTimeValues = getIntensityTimeValuesForMassChargesInPositiveAndNegativeWells(); // This is the object the final results are stored in. Map<T, Map<String, Triple<String, XZ, Double>>> positiveWellToMapOfChemicalToSNRResults = new HashMap<>(); List<Pair<T, ChemicalToMapOfMetlinIonsToIntensityTimeValues>> positiveWellResults = designUnitToListOfWellIntensityTimeValues.get(ScanData.KIND.POS_SAMPLE); List<ChemicalToMapOfMetlinIonsToIntensityTimeValues> negativeWellsSignalProfiles = designUnitToListOfWellIntensityTimeValues.get(ScanData.KIND.NEG_CONTROL).stream().map(pair -> pair.getRight()) .collect(Collectors.toList()); for (Pair<T, ChemicalToMapOfMetlinIonsToIntensityTimeValues> value : positiveWellResults) { T positiveWell = value.getLeft(); ChemicalToMapOfMetlinIonsToIntensityTimeValues positiveWellSignalProfile = value.getRight(); Map<String, Pair<XZ, Double>> snrResults = WaveformAnalysis.performSNRAnalysisAndReturnMetlinIonsRankOrderedBySNRForWells( positiveWellSignalProfile, negativeWellsSignalProfiles, setOfMassCharges); List<T> positiveWellAndNegativeWells = new ArrayList<>(); positiveWellAndNegativeWells.add(positiveWell); positiveWellAndNegativeWells.addAll(negativeWells); // This variable is used as a part of the file path dir to uniquely identify the pos/neg wells for the chemical. StringBuilder indexedPath = new StringBuilder(); for (T well : positiveWellAndNegativeWells) { indexedPath.append(Integer.toString(well.getId()) + "-"); } Map<String, String> plottingFileMappings = ChemicalToMapOfMetlinIonsToIntensityTimeValues.plotPositiveAndNegativeControlsForEachMZ( setOfMassCharges, indexedPath.toString(), positiveWellSignalProfile, negativeWellsSignalProfiles, plottingDirPath); Map<String, Triple<String, XZ, Double>> mzToPlotDirAndSNR = new HashMap<>(); for (Map.Entry<String, Pair<XZ, Double>> entry : snrResults.entrySet()) { String plottingPath = plottingFileMappings.get(entry.getKey()); XZ snr = entry.getValue().getLeft(); if (plottingDirPath == null || snr == null) { throw new RuntimeException("Plotting directory or snr is null"); } mzToPlotDirAndSNR.put(entry.getKey(), Triple.of(plottingPath, entry.getValue().getLeft(), entry.getValue().getRight())); } positiveWellToMapOfChemicalToSNRResults.put(positiveWell, mzToPlotDirAndSNR); } return positiveWellToMapOfChemicalToSNRResults; } /** * This function runs the lcms ion mining analysis based on mass charges that bin the input chemical+ion combinations * and writes the plots the resulting pos+neg control intensity-time values, along with the SNR results, * to the final json file. * @param chemIDToMassCharge Fake chem name (representing each bin) to mass charge. * @param massChargeToChemicalAndIon Mass charge value to pair of Chemical and Ion combination * @param outputPrefix The output prefix to write output results to. * @paam isNonReplicateAnalysis The boolean is true if the analysis is a non replicate analysis. * @throws Exception */ public void runLCMSMiningAnalysisAndPlotResults(Map<String, Double> chemIDToMassCharge, Map<Double, Set<Pair<String, String>>> massChargeToChemicalAndIon, String outputPrefix, Boolean isNonReplicateAnalysis) throws Exception { Map<T, Map<String, Triple<String, XZ, Double>>> lcmsAnalysisResultForEachPositiveWell = getSnrAndPlotResultsForMassChargesForEachPositiveWell(); List<List<IonAnalysisInterchangeModel.ResultForMZ>> allExperimentalResults = new ArrayList<>(); int numberOfMassChargeHits = 0; for (T positiveWell : lcmsAnalysisResultForEachPositiveWell.keySet()) { List<IonAnalysisInterchangeModel.ResultForMZ> experimentalResults = new ArrayList<>(); for (Map.Entry<String, Triple<String, XZ, Double>> mzToPlotAndSnr : lcmsAnalysisResultForEachPositiveWell.get(positiveWell).entrySet()) { Double massCharge = chemIDToMassCharge.get(mzToPlotAndSnr.getKey()); String plot = mzToPlotAndSnr.getValue().getLeft(); Double intensity = mzToPlotAndSnr.getValue().getMiddle().getIntensity(); Double time = mzToPlotAndSnr.getValue().getMiddle().getTime(); Double snr = mzToPlotAndSnr.getValue().getRight(); IonAnalysisInterchangeModel.ResultForMZ resultForMZ = new IonAnalysisInterchangeModel.ResultForMZ(massCharge); if (intensity > MIN_INTENSITY_THRESHOLD && time > MIN_TIME_THRESHOLD && snr > MIN_SNR_THRESHOLD) { resultForMZ.setIsValid(true); numberOfMassChargeHits++; } else { resultForMZ.setIsValid(false); } Set<Pair<String, String>> inchisAndIon = massChargeToChemicalAndIon.get(massCharge); for (Pair<String, String> pair : inchisAndIon) { String inchi = pair.getLeft(); String ion = pair.getRight(); IonAnalysisInterchangeModel.HitOrMiss hitOrMiss = new IonAnalysisInterchangeModel.HitOrMiss(inchi, ion, snr, time, intensity, plot); resultForMZ.addMolecule(hitOrMiss); } experimentalResults.add(resultForMZ); } IonAnalysisInterchangeModel ionAnalysisInterchangeModel = new IonAnalysisInterchangeModel(experimentalResults); // If there is only one positive replicate, the output file path is simply the prefix name. If there are multiple // replicates, we need to do more post processing to combine the results, so generate result files per replicate // and name the combined analysis file with the output prefix. This makes it easier for a workflow program to know // which output file to look at for downstream processing. String outAnalysis = positiveWells.size() == 1 ? outputPrefix : outputPrefix + "_" + positiveWell.getId().toString() + ".json"; ionAnalysisInterchangeModel.writeToJsonFile(new File(outAnalysis)); allExperimentalResults.add(experimentalResults); } LOGGER.info("The number of mass charge hits are %d out of %d", numberOfMassChargeHits, massChargeToChemicalAndIon.keySet().size()); // If the analysis is a replicate analysis, we combine the results from the various replicates into one file by simply // adding all the peak results from the replcates into one model. // If the analysis is a non-replicate analysis, we do not do this step since each of the pos+neg analysis should not // be compared since the positive wells are not replicates. if (!isNonReplicateAnalysis) { LOGGER.info("Conducting post processing since we have more than one positive well and this is a replicate analysis"); // Post process analysis List<IonAnalysisInterchangeModel.ResultForMZ> experimentalResults = new ArrayList<>(); for (int i = 0; i < allExperimentalResults.get(0).size(); i++) { IonAnalysisInterchangeModel.ResultForMZ rep = allExperimentalResults.get(0).get(i); IonAnalysisInterchangeModel.ResultForMZ resultForMZ = new IonAnalysisInterchangeModel.ResultForMZ(rep.getMz()); Boolean areAllValid = true; for (List<IonAnalysisInterchangeModel.ResultForMZ> res : allExperimentalResults) { if (!res.get(i).getIsValid()) { areAllValid = false; } resultForMZ.addMolecules(res.get(i).getMolecules()); } resultForMZ.setIsValid(areAllValid); experimentalResults.add(resultForMZ); } IonAnalysisInterchangeModel model = new IonAnalysisInterchangeModel(experimentalResults); model.writeToJsonFile(new File(outputPrefix)); } } /** * This function updates the progress of the run. Since the runtime is significantly driven by reading scan data * for each well, the function is called after each scan read is finished for each well. Therefore, the total * progress is divided by the total number of wells. */ public void updateProgress() { progress += 100.0 / (positiveWells.size() + negativeWells.size()); LOGGER.info("Progress: %f", progress); } public static void main(String[] args) throws Exception { Options opts = new Options(); for (Option.Builder b : OPTION_BUILDERS) { opts.addOption(b.build()); } CommandLine cl = null; try { CommandLineParser parser = new DefaultParser(); cl = parser.parse(opts, args); } catch (ParseException e) { System.err.format("Argument parsing failed: %s", e.getMessage()); HELP_FORMATTER.printHelp(LoadPlateCompositionIntoDB.class.getCanonicalName(), HELP_MESSAGE, opts, null, true); System.exit(1); } if (cl.hasOption("help")) { HELP_FORMATTER.printHelp(LoadPlateCompositionIntoDB.class.getCanonicalName(), HELP_MESSAGE, opts, null, true); return; } File lcmsDir = new File(cl.getOptionValue(OPTION_LCMS_FILE_DIRECTORY)); if (!lcmsDir.isDirectory()) { System.err.format("File at %s is not a directory", lcmsDir.getAbsolutePath()); HELP_FORMATTER.printHelp(LoadPlateCompositionIntoDB.class.getCanonicalName(), HELP_MESSAGE, opts, null, true); System.exit(1); } String plottingDirectory = cl.getOptionValue(OPTION_PLOTTING_DIR); // Get include and excluse ions from command line Set<String> includeIons; if (cl.hasOption(OPTION_INCLUDE_IONS)) { includeIons = new HashSet<>(Arrays.asList(cl.getOptionValues(OPTION_INCLUDE_IONS))); LOGGER.info("Including ions in search: %s", StringUtils.join(includeIons, ", ")); } else { includeIons = new HashSet<>(); includeIons.add(DEFAULT_ION); } try (DB db = DB.openDBFromCLI(cl)) { ScanFile.insertOrUpdateScanFilesInDirectory(db, lcmsDir); File inputPredictionCorpus = new File(cl.getOptionValue(OPTION_INPUT_PREDICTION_CORPUS)); Map<Double, Set<Pair<String, String>>> massChargeToChemicalAndIon = constructMassChargeToChemicalIonsFromInputFile(inputPredictionCorpus, includeIons, cl.hasOption(OPTION_LIST_OF_INCHIS_INPUT_FILE)); Pair<Set<Pair<String, Double>>, Map<String, Double>> values = constructFakeNameToMassChargeAndSetOfMassChargePairs(massChargeToChemicalAndIon.keySet()); Set<Pair<String, Double>> searchMZs = values.getLeft(); Map<String, Double> chemIDToMassCharge = values.getRight(); LOGGER.info("The number of mass charges are: %d", searchMZs.size()); Map<ScanData.KIND, List<LCMSWell>> wellTypeToLCMSWells = readInputExperimentalSetup(db, new File(cl.getOptionValue(OPTION_INPUT_POSITIVE_AND_NEGATIVE_CONTROL_WELLS_FILE))); // Get experimental setup ie. positive and negative wells from config file List<LCMSWell> positiveWells = wellTypeToLCMSWells.get(ScanData.KIND.POS_SAMPLE); List<LCMSWell> negativeWells = wellTypeToLCMSWells.get(ScanData.KIND.NEG_CONTROL); LOGGER.info("Number of positive wells is: %d", positiveWells.size()); LOGGER.info("Number of negative wells is: %d", negativeWells.size()); HashMap<Integer, Plate> plateCache = new HashMap<>(); String outputPrefix = cl.getOptionValue(OPTION_OUTPUT_PREFIX); IonDetectionAnalysis<LCMSWell> ionDetectionAnalysis = new IonDetectionAnalysis<LCMSWell>(lcmsDir, positiveWells, negativeWells, plottingDirectory, plateCache, searchMZs, db); ionDetectionAnalysis.runLCMSMiningAnalysisAndPlotResults(chemIDToMassCharge, massChargeToChemicalAndIon, outputPrefix, cl.hasOption(OPTION_NON_REPLICATE_ANALYSIS)); } } }