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

com.github.jessemull.microflexbigdecimal.io.PlateReader Maven / Gradle / Ivy

There is a newer version: 1.0.1
Show newest version
/**
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */

/*--------------------------- Package Declaration ----------------------------*/

package com.github.jessemull.microflexbigdecimal.io;
 
/*------------------------------- Dependencies -------------------------------*/

import java.io.BufferedReader;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
import java.io.StringReader;
import java.io.UnsupportedEncodingException;
import java.math.BigDecimal;
import java.nio.charset.Charset;
import java.nio.charset.CharsetDecoder;
import java.util.ArrayList;
import java.util.InputMismatchException;
import java.util.List;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.Unmarshaller;

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;

import com.github.fge.jackson.JsonLoader;
import com.github.fge.jsonschema.core.exceptions.ProcessingException;
import com.github.fge.jsonschema.main.JsonSchema;
import com.github.fge.jsonschema.main.JsonSchemaFactory;
import com.google.common.io.CharStreams;

import com.github.jessemull.microflexbigdecimal.plate.Plate;
import com.github.jessemull.microflexbigdecimal.plate.Stack; 
import com.github.jessemull.microflexbigdecimal.plate.Well;
import com.github.jessemull.microflexbigdecimal.plate.WellSet;

/**
 * Parses an input stream containing BigDecimal stacks, plates, well sets, 
 * and/or wells. Supports the following inputs:
 * 
 * 
 *    
 *    
 *       
 *    
 *    
 *       
 *    
 *    
 *       
 *    
 *    
 *       
 *    
 * 
Formats
Plate Map
Result Table
JSON
XML
* * Result tables and maps must be separated using a user defined delimiter or the * default tab delimiter. Input must be formatted as follows: * *

* *
*
*

Example Result Plate Map:

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
Result
123456
A9.1142.6674.2145.1541.2937.882
B8.9534.8275.1499.7922.5146.448
C6.3284.8965.1309.6555.1201.485
D2.6615.2904.0572.3741.2002.724
*
*
*

Example Result Table:

* * * * * * * * * * * * * * * * * * * * * * * * * *
Result
IndexValue
A148.77
A250.32
A367.23
B149.81
*
*
*

Example Result JSON:

*
 *  {
 *   "results" : [ {
 *     "type" : "BigDecimal",
 *     "label" : "Example Result",
 *     "size" : 10,
 *     "wells" : {
 *       "A11" : 56.455852093078924,
 *       "A2" : 61.49253731775972,
 *       "A5" : 58.462812164254956,
 *       "D10" : 58.957595115303654,
 *       "D2" : 54.5241886265687,
 *       "E4" : 56.821464250881434,
 *       "E8" : 61.16876436088828,
 *       "E9" : 49.45000301303015,
 *       "G6" : 58.74507294128508,
 *       "H1" : 49.115877056175314
 *     }
 *   } ]
 *  }
 *       
*
*
*

Example Result XML:

*
 *  <?xml version="1.0" encoding="UTF-8" standalone="yes"?>
 *  <results>
 *     <result>
 *         <type>BigDecimal</type>
 *         <label>Example Result</label>
 *         <size>3</size>
 *         <wells>
 *             <well>
 *                 <index>A2</index>
 *                 <value>55.873951592640616</value>
 *             </well>
 *             <well>
 *                 <index>A3</index>
 *                 <value>53.03191840271489</value>
 *             </well>
 *             <well>
 *                 <index>A4</index>
 *                 <value>49.02962476222037</value>
 *             </well>
 *         </wells>
 *     </result>
 *  </results>
 *       
*
*
*
*
*

Example Well JSON:

*
 *  {
 *    "wells" : [ {
 *      "type" : "BigDecimal",
 *      "index" : "F8",
 *      "size" : 24,
 *      "values" : [ 15.11, 28.82, 52.54 ]
 *    } ]
 *  }
 *        
*
*
*

Example Well Set JSON:

*
 *  {
 *    "wellsets" : [ {
 *      "type" : "BigDecimal",
 *      "label" : "Example Well Set",
 *      "size" : 1,
 *      "wells" : [ {
 *        "index" : "A3",
 *        "values" : [ 44.70, 91.16, 52.15 ]
 *      } ]
 *    } ]
 *  }
 *        
*
*
*

Example Plate JSON:

*
 *  {
 *    "plates" : [ {
 *      "type" : "BigDecimal",
 *      "label" : "Example Plate",
 *      "descriptor" : "96-Well",
 *      "rows" : 8,
 *      "columns" : 12,
 *      "size" : 1,
 *      "wellsets" : [ {
 *        "label" : "Example Well Set",
 *        "size" : 3,
 *        "wells" : [ "C5", "D4", "F7" ]
 *      } ],
 *      "wells" : [ {
 *        "index" : "A7",
 *        "values" : [ 44.51, 34.53, 55.92 ]
 *      } ]
 *    } ]
 *  }
 *        
*
*
*

Example Stack JSON:

*
 *  {
 *    "stacks" : [ {
 *      "type" : "BigDecimal",
 *      "label" : "Example Stack",
 *      "rows" : 8,
 *      "columns" : 12,
 *      "size" : 1,
 *      "plates" : [ {
 *        "type" : "BigDecimal",
 *        "label" : "Example Plate",
 *        "descriptor" : "96-Well",
 *        "rows" : 8,
 *        "columns" : 12,
 *        "size" : 1,
 *        "wellsets" : [ {
 *          "label" : "Example Well Set",
 *          "size" : 3,
 *          "wells" : [ "B6", "G1", "G12" ]
 *        } ],
 *        "wells" : [ {
 *          "index" : "A2",
 *          "values" : [ 35.94, 28.17, 21.00 ]
 *        } ]
 *      } ]
 *    } ]
 *  }
 *        
*
*
*
*
*

Example Well XML:

*
 *  <wells>
 *     <well>
 *         <type>BigDecimal</type>
 *         <index>H8</index>
 *         <size>24</size>
 *         <values>
 *             <value>78.70</value>
 *             <value>72.96</value>
 *             <value>38.98</value>
 *         </values>
 *     </well>
 *  </wells>
 *       
*
*
*

Example Well Set XML:

*
 *  <wellsets>
 *     <wellset>
 *         <type>BigDecimal</type>
 *         <label>Example Well Set</label>
 *         <size>1</size>
 *         <wells>
 *             <well>
 *                 <index>A11</index>
 *                 <values>
 *                     <value>17.39</value>
 *                     <value>54.65</value>
 *                     <value>54.12</value>
 *                 </values>
 *             </well>
 *         </wells>
 *     </wellset>
 *  </wellsets>
 *       
*
*
*

Example Plate XML:

*
 *  <plates>
 *     <plate>
 *         <type>BigDecimal</type>
 *         <label>Example Plate</label>
 *         <descriptor>96-Well</descriptor>
 *         <rows>8</rows>
 *         <columns>12</columns>
 *         <size>1</size>
 *         <wellsets>
 *             <wellset>
 *                 <label>Example Well Set</label>
 *                 <size>3</size>
 *                 <wells>
 *                     <well>B12</well>
 *                     <well>E2</well>
 *                     <well>H8</well>
 *                 </wells>
 *             </wellset>
 *         </wellsets>
 *         <wells>
 *             <well>
 *                 <index>A5</index>
 *                 <values>
 *                     <value>13.89</value>
 *                     <value>92.46</value>
 *                     <value>19.48</value>
 *                 </values>
 *             </well>
 *         </wells>
 *     </plate>
 *  </plates>
 *       
*
*
*

Example Stack XML:

*
 *  <stacks>
 *     <stack>
 *         <type>BigDecimal</type>
 *         <label>Example Stack</label>
 *         <rows>8</rows>
 *         <columns>12</columns>
 *         <size>1</size>
 *         <plates>
 *             <plate>
 *                 <type>BigDecimal</type>
 *                 <label>Example Plate</label>
 *                 <descriptor>96-Well</descriptor>
 *                 <rows>8</rows>
 *                 <columns>12</columns>
 *                 <size>1</size>
 *                 <wellsets>
 *                     <wellset>
 *                         <label>Example Well Set</label>
 *                         <size>3</size>
 *                         <wells>
 *                             <well>B7</well>
 *                             <well>H2</well>
 *                             <well>H3</well>
 *                         </wells>
 *                     </wellset>
 *                 </wellsets>
 *                 <wells>
 *                     <well>
 *                         <index>A2</index>
 *                         <values>
 *                             <value>13.03</value>
 *                             <value>53.21</value>
 *                             <value>28.08</value>
 *                         </values>
 *                     </well>
 *                 </wells>
 *             </plate>
 *         </plates>
 *     </stack>
 *  </stacks>
 *       
*
*
* * @author Jesse L. Mull * @update Updated Oct 17, 2016 * @address http://www.jessemull.com * @email [email protected] */ public class PlateReader extends BufferedReader { /*---------------------------- Private Fields ----------------------------*/ private int ALPHA_BASE = 26; // The number of possible characters for row indices private String delimiter = "\t"; // The delimiter private ResultListPOJO resultsJSON; // Result list POJO for JSON input private WellListPOJO wellsJSON; // Well list POJO for JSON input private WellSetListPOJO setsJSON; // Set list POJO for JSON input private PlateListPOJO platesJSON; // Plate list POJO for JSON input private StackListPOJO stacksJSON; // Stack list POJO for JSON input private ResultListXML resultsXML; // Result list for XML input private WellListXML wellsXML; // Well list for XML input private WellSetListXML setsXML; // Set list for XML input private PlateListXML platesXML; // Plate list for XML input private StackListXML stacksXML; // Stack list for XML input private int indexResultsJSON = -1; // Current index into the JSON result list private int indexWellsJSON = -1; // Current index into the JSON well list private int indexSetsJSON = -1; // Current index into the JSON set list private int indexPlatesJSON = -1; // Current index into the JSON plate list private int indexStacksJSON = -1; // Current index into the JSON stack list private int indexResultsXML = -1; // Current index into the XML result list private int indexWellsXML = -1; // Current index into the XML well list private int indexSetsXML = -1; // Current index into the XML set list private int indexPlatesXML = -1; // Current index into the XML plate list private int indexStacksXML = -1; // Current index into the XML stack list private String labelRegex = "^\\s*[^ ]+\\s*$"; // Regex for matching plate map labels private String letters = "^[A-Z]+"; // Regex for matching row index letters /* Regex for matching JSON objects */ private String columnHeadersRegex; // Regex for matching plate map column headers private String tableHeaderRegex; // Regex for matching table headers private Pattern labelPattern = Pattern.compile(labelRegex); // Pattern for matching plate map labels private Pattern lettersPattern = Pattern.compile(letters); // Pattern for matching row index letters private Pattern columnHeadersPattern; // Pattern for matching plate map column headers private Pattern tableHeaderPattern; // Pattern for matching table headers /* Paths to well, well set, plate and stack JSON/XML schemas */ private final String JSON_RESULT_SCHEMA = "json_result_schema.json"; private final String JSON_WELL_SCHEMA = "json_well_schema.json"; private final String JSON_WELLSET_SCHEMA = "json_wellset_schema.json"; private final String JSON_PLATE_SCHEMA = "json_plate_schema.json"; private final String JSON_STACK_SCHEMA = "json_stack_schema.json"; /* Default buffer size for a buffered reader is 8192 bytes/chars */ int readAheadLimit = 8192 * 10000; /*----------------------------- Constructors -----------------------------*/ /** * Constructs a new reader for a buffering character input stream using * the default buffer size. * @param Reader the source */ public PlateReader(Reader source) { super(source); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(source); this.validateJSON(input); this.validateXML(input); } catch (Exception e) { e.printStackTrace(); } } /** * Constructs a new reader for a buffering character input stream using * an input buffer of the specified size. * @param Reader reader source * @param int buffer size * @throws IOException * @throws JAXBException */ public PlateReader(Reader source, int size) { super(source, size); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(source); this.validateJSON(input); this.validateXML(input); } catch (Exception e) { e.printStackTrace(); } } /** * Constructs a new reader for a buffering character input stream using * the default buffer size. * @param InputStream input stream source */ public PlateReader(InputStream source) { super(new InputStreamReader(source)); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(new InputStreamReader(source)); this.validateJSON(input); this.validateXML(input); } catch (Exception e) { e.printStackTrace(); } } /** * Constructs a new reader for a buffering character input stream using * an input buffer of the specified size. * @param InputStream input stream source * @param int buffer size */ public PlateReader(InputStream source, int size) { super(new InputStreamReader(source), size); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(new InputStreamReader(source)); this.validateJSON(input); this.validateXML(input); } catch (Exception e) { e.printStackTrace(); } } /** * Constructs a new reader for a buffering character input stream using * the default buffer size. * @param InputStream input stream source * @param Charset character set */ public PlateReader(InputStream source, Charset set) { super(new BufferedReader(new InputStreamReader(source, set))); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(new InputStreamReader(source, set)); this.validateJSON(input); this.validateXML(input); } catch(Exception e) { e.printStackTrace(); } } /** * Constructs a new reader for a buffering character input stream using * an input buffer of the specified size. * @param InputStream input stream source * @param String character set * @param int buffer size * @throws UnsupportedEncodingException */ public PlateReader(InputStream source, String charsetName, int size) throws UnsupportedEncodingException { super(new BufferedReader(new InputStreamReader(source, charsetName), size)); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(new InputStreamReader(source, charsetName)); this.validateJSON(input); this.validateXML(input); } catch(Exception e) { e.printStackTrace(); } } /** * Constructs a new reader for a buffering character input stream using * the default buffer size. * @param InputStream input stream source * @param CharsetDecoder decoder */ public PlateReader(InputStream source, CharsetDecoder decoder) { super(new InputStreamReader(source, decoder)); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(new InputStreamReader(source, decoder)); this.validateJSON(input); this.validateXML(input); } catch(Exception e) { e.printStackTrace(); } } /** * Constructs a new reader for a buffering character input stream using * an input buffer of the specified size. * @param InputStream input stream source * @param CharsetDecoder decoder * @param int buffer size */ public PlateReader(InputStream source, CharsetDecoder decoder, int size) { super(new InputStreamReader(source, decoder), size); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(new InputStreamReader(source, decoder)); this.validateJSON(input); this.validateXML(input); } catch(Exception e) { e.printStackTrace(); } } /** * Constructs a new reader for a buffering character input stream using * the default buffer size. * @param File source file * @throws IOException * @throws JAXBException */ public PlateReader(File source) throws IOException, JAXBException { super(new BufferedReader(new FileReader(source))); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(new FileReader(source)); this.validateJSON(input); this.validateXML(input); } catch(ProcessingException e) { e.printStackTrace(); } } /** * Constructs a new reader for a buffering character input stream using * an input buffer of the specified size. * @param File source file * @param int buffer size * @throws FileNotFoundException */ public PlateReader(File source, int size) throws FileNotFoundException { super(new FileReader(source), size); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(new FileReader(source)); this.validateJSON(input); this.validateXML(input); } catch(Exception e) { e.printStackTrace(); } } /** * Constructs a new reader for a buffering character input stream using * the default buffer size. * @param File the file * @throws FileNotFoundException */ public PlateReader(String fileName) throws FileNotFoundException { super(new FileReader(new File(fileName))); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(new FileReader(new File(fileName))); this.validateJSON(input); this.validateXML(input); } catch(Exception e) { e.printStackTrace(); } } /** * Constructs a new reader for a buffering character input stream using the * specified file source and charset name. * @param File the file * @param int buffer size * @throws FileNotFoundException * @throws IOException * @throws JAXBException */ public PlateReader(String fileName, int size) throws FileNotFoundException { super(new FileReader(new File(fileName)), size); try { this.setDelimiter(this.delimiter); String input = CharStreams.toString(new FileReader(new File(fileName))); this.validateJSON(input); this.validateXML(input); } catch(Exception e) { e.printStackTrace(); } } /*--------------------- Methods for Plate Map Input ----------------------*/ /** * Returns true if the scanner has another plate map in its input. This * method may block while waiting for input to scan. The scanner does not * advance past any input. * @return true if the scanner has another plate map in its input */ public boolean hasNextMap() { try { /* Check that more lines exist in the input buffer */ if(!this.hasNextLine()) { return false; } this.mark(readAheadLimit); /* Get the label */ String label = this.nextLine(); /* Skip blank lines */ while(label.trim().isEmpty()) { if(!this.hasNextLine()) { return false; } label = this.nextLine(); } /* Check for a valid label */ Matcher labelMatcher = labelPattern.matcher(label); if(!labelMatcher.matches()) { return false; } /* Check for valid column headers */ String headers = this.nextLine(); Matcher headersMatcher = columnHeadersPattern.matcher(headers); int columnNumber = 0; if(!headersMatcher.matches()) { return false; } else { String[] splitHeaders = headers.trim().split(delimiter); for(int i = 0; i < splitHeaders.length; i++) { if(Integer.parseInt(splitHeaders[i]) != i + 1) { throw new IllegalArgumentException("Invalid column index."); } } columnNumber = splitHeaders.length; } /* Validate the rows and row indices */ int currentIndex = 0; String row = null; while((row = this.nextLine().trim()) != null && !row.trim().isEmpty()) { String regex = "^\\s*[A-Za-z]+" + this.delimiter + "(\\s*(\\d+\\.?\\d+|Null)\\s*" + this.delimiter + "|\\s*(\\d+\\.?\\d*|Null))+\\s*\\n?$"; Pattern pattern = Pattern.compile(regex); Matcher rowMatcher = pattern.matcher(row); if(!rowMatcher.lookingAt()) { if(currentIndex == 0) { return false; } break; } else { String[] rowSplit = row.split(delimiter); int numValue = this.parseRow(rowSplit[0].trim()); if(numValue != currentIndex) { return false; } if(rowSplit.length - 1 != columnNumber) { return false; } currentIndex++; } } /* Reset the buffer index and return true */ this.reset(); return true; } catch(Exception e) { /* Reset the buffer index and return true */ try { this.reset(); } catch (IOException ioException) { ioException.printStackTrace(); return false; } e.printStackTrace(); return false; } } /** * Scans the next token of the input as a plate map. If the next token * matches the plate map regular expression defined above then the token is * converted into a plate object. * @return plate containing the data from the map */ public Plate nextMap() { try { /* Check that more lines exist in the input buffer */ if(!this.hasNextLine()) { throw new IOException("No more lines in the input buffer."); } /* Mark the input stream */ this.mark(readAheadLimit); /* Get the first line holding the label */ String label = this.nextLine(); /* Move past any blank lines */ while(label.trim().isEmpty()) { label = this.nextLine().trim(); } /* Check for a valid label */ Matcher labelMatcher = labelPattern.matcher(label); if(!labelMatcher.matches()) { throw new InputMismatchException("Invalid plate map label."); } /* Check for valid column headers */ String headers = this.nextLine(); Matcher headersMatcher = columnHeadersPattern.matcher(headers); int columnNumber = 0; if(!headersMatcher.matches()) { throw new InputMismatchException("Invalid plate map column headers."); } else { String[] splitHeaders = headers.trim().split(delimiter); for(int i = 0; i < splitHeaders.length; i++) { if(Integer.parseInt(splitHeaders[i]) != i + 1) { throw new InputMismatchException("Invalid plate map column headers."); } } columnNumber = splitHeaders.length; } /* Validate the rows and row indices */ int currentIndex = 0; List wells = new ArrayList(); String row = null; while((row = this.nextLine()) != null && !row.trim().isEmpty()) { String regex = "^\\s*[A-Za-z]+" + this.delimiter + "(\\s*(\\d+\\.?\\d+|Null)\\s*" + this.delimiter + "|\\s*(\\d+\\.?\\d*|Null))+\\s*\\n?$"; Pattern pattern = Pattern.compile(regex); Matcher rowMatcher = pattern.matcher(row); if(!rowMatcher.matches()) { break; } else { String[] rowSplit = row.split(delimiter); int numValue = this.parseRow(rowSplit[0].trim()); if(numValue != currentIndex) { throw new InputMismatchException("Invalid plate map row index."); } if(rowSplit.length - 1 != columnNumber) { throw new InputMismatchException("Invalid plate map row length."); } for(int i = 1; i < rowSplit.length; i++) { if(!rowSplit[i].equals("Null")) { Well well = new Well(numValue, i); BigDecimal toAdd = new BigDecimal(rowSplit[i].trim() + ""); well.add(toAdd); wells.add(well); } } currentIndex++; } } Plate input = new Plate(currentIndex, columnNumber, label); for(Well well : wells) { input.addWells(well); } return input; } catch(Exception e) { try { this.reset(); } catch (IOException ioException) { ioException.printStackTrace(); return null; } e.printStackTrace(); return null; } } /*----------------------- Methods for Table Input ------------------------*/ /** * Returns true if the scanner has another table of wells in its input. This * method may block while waiting for input to scan. The scanner does not * advance past any input. * @return true if the scanner has another table of wells in * its input */ public boolean hasNextTable() { try { /* Check that more lines exist in the input buffer */ if(!this.hasNextLine()) { return false; } /* Mark the input stream */ this.mark(readAheadLimit); /* Get the first line holding the label */ String label = this.nextLine(); while(label.trim().isEmpty()) { if(!this.hasNextLine()) { return false; } label = this.nextLine(); } /* Check for a valid label */ Matcher labelMatcher = labelPattern.matcher(label); if(!labelMatcher.matches()) { return false; } /* Get the first line holding the label */ String headers = this.nextLine(); while(headers.trim().isEmpty()) { headers = this.nextLine(); } /* Check for valid table headers */ this.tableHeaderPattern = Pattern.compile(this.tableHeaderRegex); Matcher tableHeaderMatcher = tableHeaderPattern.matcher(headers); if(!tableHeaderMatcher.matches()) { return false; } String row = null; int index = 0; while((row = this.nextLine()) != null && !row.trim().isEmpty()) { String regex = "^\\s*[A-Za-z]+\\d+\\s*" + this.delimiter + "\\s*\\d+\\.?\\d+\\s*$"; Pattern pattern = Pattern.compile(regex); Matcher tableRowMatcher = pattern.matcher(row); if(!tableRowMatcher.matches()) { if(index == 0) { return false; } break; } index++; } this.reset(); return true; } catch(Exception e) { /* Reset the index into the buffer on error */ try { this.reset(); } catch (IOException ioException) { ioException.printStackTrace(); return false; } e.printStackTrace(); return false; } } /** * Scans the next token of the input as a table of wells. If the next token * matches the table regular expression defined above then the token is * converted into a well set. * @return the converted table of wells */ public WellSet nextTable() { try { /* Mark the input stream */ this.mark(readAheadLimit); /* Get the label */ String label = this.nextLine(); while(label.trim().isEmpty()) { label = this.nextLine(); } /* Check for a valid label */ Matcher labelMatcher = labelPattern.matcher(label); if(!labelMatcher.matches()) { throw new IllegalArgumentException("Invalid plate label."); } /* Get the table headers */ String headers = this.nextLine(); while(headers.trim().isEmpty()) { headers = this.nextLine(); } /* Check for valid table headers */ Matcher tableHeaderMatcher = tableHeaderPattern.matcher(headers); if(!tableHeaderMatcher.matches()) { throw new IllegalArgumentException("Invalid table headers."); } List wells = new ArrayList(); String row = null; while((row = this.nextLine()) != null && !row.trim().isEmpty()) { String regex = "^\\s*[A-Za-z]+\\d+\\s*" + this.delimiter + "\\s*\\d+\\.?\\d+\\s*$"; Pattern pattern = Pattern.compile(regex); Matcher tableRowMatcher = pattern.matcher(row); String[] split = row.split(this.delimiter); if(!tableRowMatcher.matches()) { break; } else { Well well = new Well(split[0].trim()); BigDecimal toAdd = new BigDecimal(split[1].trim() + ""); well.add(toAdd); wells.add(well); } } return new WellSet(wells, label); } catch(Exception e) { /* Reset the index into the buffer on error */ try { this.reset(); } catch (IOException ioException) { ioException.printStackTrace(); return null; } e.printStackTrace(); return null; } } /*---------------------- Methods for JSON Result Input ---------------------*/ /** * Returns true if the the next result POJO in the list is a valid result. * @return true if the next result POJO is valid * @throws IOException * @throws ProcessingException */ public boolean hasNextJSONResult() { return this.indexResultsJSON >= 0 && this.indexResultsJSON < this.resultsJSON.size(); } /** * Returns true if the the previous result POJO in the list is a valid result. * @return true if the previous result POJO is valid * @throws IOException * @throws ProcessingException */ public boolean hasPreviousJSONResult() { return this.indexResultsJSON > 0; } /** * Parses the next result in the result list to a well set object and returns the * result. * @return the converted result object */ public WellSet nextJSONResult() { if(this.hasNextJSONResult()) { ResultPOJO result = this.resultsJSON.get(indexResultsJSON++); WellSet set = new WellSet(); set.setLabel(result.getLabel()); Map map = result.getWells(); for(Map.Entry entry : map.entrySet()) { Well well = new Well(entry.getKey()); well.add(entry.getValue()); set.add(well); } return set; } else { return null; } } /** * Parses the previous result in the result list to a well set object and returns the * result. * @return the converted result object */ public WellSet previousJSONResult() { if(this.hasPreviousJSONResult()) { ResultPOJO result = this.resultsJSON.get(--indexResultsJSON); WellSet set = new WellSet(); set.setLabel(result.getLabel()); Map map = result.getWells(); for(Map.Entry entry : map.entrySet()) { Well well = new Well(entry.getKey()); well.add(entry.getValue()); set.add(well); } return set; } else { return null; } } /** * Returns a list containing the result objects with an index greater than the * current index. * @return list of converted result objects */ public List remainingJSONResults() { if(!this.hasNextJSONResult()) { return null; } List remaining = new ArrayList(); while(this.hasNextJSONResult()) { remaining.add(this.nextJSONResult()); } return remaining; } /** * Returns a list containing the result objects with an index less than the * current index. * @return list of converted result objects */ public List spentJSONResults() { if(!this.hasPreviousJSONResult()) { return null; } List spent = new ArrayList(); while(this.hasPreviousJSONResult()) { spent.add(this.previousJSONResult()); } return spent; } /** * Returns a list containing all the JSON results in the list. * @return list of all result objects */ public List allJSONResults() { List allResults = new ArrayList(); for(ResultPOJO result : this.resultsJSON) { WellSet toAddSet = new WellSet(); toAddSet.setLabel(result.getLabel()); Map map = result.getWells(); for(Map.Entry entry : map.entrySet()) { Well toAddWell = new Well(entry.getKey()); toAddWell.add(entry.getValue()); toAddSet.add(toAddWell); } allResults.add(toAddSet); } return allResults; } /*-------------------- Methods for XML Result Input --------------------*/ /** * Returns true if the the next XML result in the list is a valid result. * @return true if the next XML result is valid * @throws IOException * @throws ProcessingException */ public boolean hasNextXMLResult() { return this.indexResultsXML >= 0 && this.indexResultsXML < this.resultsXML.size(); } /** * Returns true if the the previous XML result in the list is a valid result. * @return true if the previous XML result is valid * @throws IOException * @throws ProcessingException */ public boolean hasPreviousXMLResult() { return this.indexResultsXML > 0; } /** * Parses the next result in the result list to a well set object and * returns the result. * @return the converted result object */ public WellSet nextXMLResult() { return this.hasNextXMLResult() ? this.resultsXML.get(indexResultsXML++).toWellSetObject() : null; } /** * Parses the previous result in the result list to a well set object * and returns the result. * @return the converted result object */ public WellSet previousXMLResult() { return this.hasPreviousXMLResult() ? this.resultsXML.get(--indexResultsXML).toWellSetObject() : null; } /** * Returns a list containing the result objects with an index greater than the * current index. * @return list of converted result objects */ public List remainingXMLResults() { if(!this.hasNextXMLResult()) { return null; } List remaining = new ArrayList(); while(this.hasNextXMLResult()) { remaining.add(this.nextXMLResult()); } return remaining; } /** * Returns a list containing the result objects with an index less than the * current index. * @return list of converted result objects */ public List spentXMLResults() { if(!this.hasPreviousXMLResult()) { return null; } List spent = new ArrayList(); while(this.hasPreviousXMLResult()) { spent.add(this.previousXMLResult()); } return spent; } /** * Returns a list containing all the XML results in the list. * @return list of all converted result objects */ public List allXMLResults() { List allWellResults = new ArrayList(); for(ResultXML result: this.resultsXML) { allWellResults.add(result.toWellSetObject()); } return allWellResults; } /*---------------------- Methods for JSON Well Input ---------------------*/ /** * Returns true if the the next well POJO in the list is a valid well. * @return true if the next well POJO is valid * @throws IOException * @throws ProcessingException */ public boolean hasNextJSONWell() { return this.indexWellsJSON >= 0 && this.indexWellsJSON < this.wellsJSON.size(); } /** * Returns true if the the previous well POJO in the list is a valid well. * @return true if the previous well POJO is valid * @throws IOException * @throws ProcessingException */ public boolean hasPreviousJSONWell() { return this.indexWellsJSON > 0; } /** * Parses the next well in the well list to a well object and returns the * result. * @return the converted well object */ public Well nextJSONWell() { return this.hasNextJSONWell() ? this.wellsJSON.get(indexWellsJSON++).toWellObject() : null; } /** * Parses the previous well in the well list to a well object and returns the * result. * @return the converted well object */ public Well previousJSONWell() { return this.hasPreviousJSONWell() ? this.wellsJSON.get(--indexWellsJSON).toWellObject() : null; } /** * Returns a list containing the well objects with an index greater than the * current index. * @return list of converted well objects */ public List remainingJSONWells() { if(!this.hasNextJSONWell()) { return null; } List remaining = new ArrayList(); while(this.hasNextJSONWell()) { remaining.add(this.nextJSONWell()); } return remaining; } /** * Returns a list containing the well objects with an index less than the * current index. * @return list of converted well objects */ public List spentJSONWells() { if(!this.hasPreviousJSONWell()) { return null; } List spent = new ArrayList(); while(this.hasPreviousJSONWell()) { spent.add(this.previousJSONWell()); } return spent; } /** * Returns a list containing all the JSON wells in the list. * @return list of all well objects */ public List allJSONWells() { List allWells = new ArrayList(); for(WellPOJO well : this.wellsJSON) { allWells.add(well.toWellObject()); } return allWells; } /*---------------------- Methods for XML Well Input ----------------------*/ /** * Returns true if the the next XML well in the list is a valid well. * @return true if the next XML well is valid * @throws IOException * @throws ProcessingException */ public boolean hasNextXMLWell() { return this.indexWellsXML >= 0 && this.indexWellsXML < this.wellsXML.size(); } /** * Returns true if the the previous XML well in the list is a valid well. * @return true if the previous XML well is valid * @throws IOException * @throws ProcessingException */ public boolean hasPreviousXMLWell() { return this.indexWellsXML > 0; } /** * Parses the next well in the well list to a well object and returns the * result. * @return the converted well object */ public Well nextXMLWell() { return this.hasNextXMLWell() ? this.wellsXML.get(indexWellsXML++).toWellObject() : null; } /** * Parses the previous well in the well list to a well object and returns the * result. * @return the converted well object */ public Well previousXMLWell() { return this.hasPreviousXMLWell() ? this.wellsXML.get(--indexWellsXML).toWellObject() : null; } /** * Returns a list containing the well objects with an index greater than the * current index. * @return list of converted well objects */ public List remainingXMLWells() { if(!this.hasNextXMLWell()) { return null; } List remaining = new ArrayList(); while(this.hasNextXMLWell()) { remaining.add(this.nextXMLWell()); } return remaining; } /** * Returns a list containing the well objects with an index less than the * current index. * @return list of converted well objects */ public List spentXMLWells() { if(!this.hasPreviousXMLWell()) { return null; } List spent = new ArrayList(); while(this.hasPreviousXMLWell()) { spent.add(this.previousXMLWell()); } return spent; } /** * Returns a list containing all the XML wells in the list. * @return list of all well objects */ public List allXMLWells() { List allWells = new ArrayList(); for(WellXML well : this.wellsXML) { allWells.add(well.toWellObject()); } return allWells; } /*-------------------- Methods for JSON Well Set Input -------------------*/ /** * Returns true if the the next set POJO in the list is a valid set. * @return true if the next set POJO is valid * @throws IOException * @throws ProcessingException */ public boolean hasNextJSONSet() { return this.indexSetsJSON >= 0 && this.indexSetsJSON < this.setsJSON.size(); } /** * Returns true if the the previous set POJO in the list is a valid set. * @return true if the previous set POJO is valid * @throws IOException * @throws ProcessingException */ public boolean hasPreviousJSONSet() { return this.indexSetsJSON > 0; } /** * Parses the next set in the set list to a well set object and returns the * result. * @return the converted well set object */ public WellSet nextJSONSet() { return this.hasNextJSONSet() ? this.setsJSON.get(indexSetsJSON++).toWellSetObject() : null; } /** * Parses the previous set in the set list to a well set object and returns the * result. * @return the converted well set object */ public WellSet previousJSONSet() { return this.hasPreviousJSONSet() ? this.setsJSON.get(--indexSetsJSON).toWellSetObject() : null; } /** * Returns a list containing the set objects with an index greater than the * current index. * @return list of converted set objects */ public List remainingJSONSets() { if(!this.hasNextJSONSet()) { return null; } List remaining = new ArrayList(); while(this.hasNextJSONSet()) { remaining.add(this.nextJSONSet()); } return remaining; } /** * Returns a list containing the set objects with an index less than the * current index. * @return list of converted set objects */ public List spentJSONSets() { if(!this.hasPreviousJSONSet()) { return null; } List spent = new ArrayList(); while(this.hasPreviousJSONSet()) { spent.add(this.previousJSONSet()); } return spent; } /** * Returns a list containing all the JSON sets in the list. * @return list of all set objects */ public List allJSONSets() { List allSets = new ArrayList(); for(WellSetPOJO set : this.setsJSON) { allSets.add(set.toWellSetObject()); } return allSets; } /*-------------------- Methods for XML Well Set Input --------------------*/ /** * Returns true if the the next XML well set in the list is a valid well set. * @return true if the next XML well set is valid * @throws IOException * @throws ProcessingException */ public boolean hasNextXMLSet() { return this.indexSetsXML >= 0 && this.indexSetsXML < this.setsXML.size(); } /** * Returns true if the the previous XML well set in the list is a valid well set. * @return true if the previous XML well set is valid * @throws IOException * @throws ProcessingException */ public boolean hasPreviousXMLSet() { return this.indexSetsXML > 0; } /** * Parses the next well set in the well set list to a well set object and * returns the result. * @return the converted well set object */ public WellSet nextXMLSet() { return this.hasNextXMLSet() ? this.setsXML.get(indexSetsXML++).toWellSetObject() : null; } /** * Parses the previous well set in the well set list to a well set object * and returns the result. * @return the converted well set object */ public WellSet previousXMLSet() { return this.hasPreviousXMLSet() ? this.setsXML.get(--indexSetsXML).toWellSetObject() : null; } /** * Returns a list containing the well set objects with an index greater than the * current index. * @return list of converted well set objects */ public List remainingXMLSets() { if(!this.hasNextXMLSet()) { return null; } List remaining = new ArrayList(); while(this.hasNextXMLSet()) { remaining.add(this.nextXMLSet()); } return remaining; } /** * Returns a list containing the well set objects with an index less than the * current index. * @return list of converted well set objects */ public List spentXMLSets() { if(!this.hasPreviousXMLSet()) { return null; } List spent = new ArrayList(); while(this.hasPreviousXMLSet()) { spent.add(this.previousXMLSet()); } return spent; } /** * Returns a list containing all the XML well sets in the list. * @return list of all well set objects */ public List allXMLSets() { List allWellSets = new ArrayList(); for(WellSetXML wellset: this.setsXML) { allWellSets.add(wellset.toWellSetObject()); } return allWellSets; } /*---------------------- Methods for JSON Plate Input --------------------*/ /** * Returns true if the the next plate POJO in the list is a valid plate. * @return true if the next plate POJO is valid * @throws IOException * @throws ProcessingException */ public boolean hasNextJSONPlate() { return this.indexPlatesJSON >= 0 && this.indexPlatesJSON < this.platesJSON.size(); } /** * Returns true if the the previous plate POJO in the list is a valid plate. * @return true if the previous plate POJO is valid * @throws IOException * @throws ProcessingException */ public boolean hasPreviousJSONPlate() { return this.indexPlatesJSON > 0; } /** * Parses the next plate in the plate list to a plate object and returns the * result. * @return the converted plate object */ public Plate nextJSONPlate() { return this.hasNextJSONPlate() ? this.platesJSON.get(indexPlatesJSON++).toPlateObject() : null; } /** * Parses the previous plate in the plate list to a plate object and returns the * result. * @return the converted plate object */ public Plate previousJSONPlate() { return this.hasPreviousJSONPlate() ? this.platesJSON.get(--indexPlatesJSON).toPlateObject() : null; } /** * Returns a list containing the plate objects with an index greater than the * current index. * @return list of converted plate objects */ public List remainingJSONPlates() { if(!this.hasNextJSONPlate()) { return null; } List remaining = new ArrayList(); while(this.hasNextJSONPlate()) { remaining.add(this.nextJSONPlate()); } return remaining; } /** * Returns a list containing the plate objects with an index less than the * current index. * @return list of converted plate objects */ public List spentJSONPlates() { if(!this.hasPreviousJSONPlate()) { return null; } List spent = new ArrayList(); while(this.hasPreviousJSONPlate()) { spent.add(this.previousJSONPlate()); } return spent; } /** * Returns a list containing all the JSON plates in the list. * @return list of all plate objects */ public List allJSONPlates() { List allPlates = new ArrayList(); for(PlatePOJO plate : this.platesJSON) { allPlates.add(plate.toPlateObject()); } return allPlates; } /*---------------------- Methods for XML Plate Input ---------------------*/ /** * Returns true if the the next XML plate in the list is a valid plate. * @return true if the next XML plate is valid * @throws IOException * @throws ProcessingException */ public boolean hasNextXMLPlate() { return this.indexPlatesXML >= 0 && this.indexPlatesXML < this.platesXML.size(); } /** * Returns true if the the previous XML plate in the list is a valid plate. * @return true if the previous XML plate is valid * @throws IOException * @throws ProcessingException */ public boolean hasPreviousXMLPlate() { return this.indexPlatesXML > 0; } /** * Parses the next plate in the plate list to a plate object and returns the * result. * @return the converted plate object */ public Plate nextXMLPlate() { return this.hasNextXMLPlate() ? this.platesXML.get(indexPlatesXML++).toPlateObject() : null; } /** * Parses the previous plate in the plate list to a plate object and returns the * result. * @return the converted plate object */ public Plate previousXMLPlate() { return this.hasPreviousXMLPlate() ? this.platesXML.get(--indexPlatesXML).toPlateObject() : null; } /** * Returns a list containing the plate objects with an index greater than the * current index. * @return list of converted plate objects */ public List remainingXMLPlates() { if(!this.hasNextXMLPlate()) { return null; } List remaining = new ArrayList(); while(this.hasNextXMLPlate()) { remaining.add(this.nextXMLPlate()); } return remaining; } /** * Returns a list containing the plate objects with an index less than the * current index. * @return list of converted plate objects */ public List spentXMLPlates() { if(!this.hasPreviousXMLPlate()) { return null; } List spent = new ArrayList(); while(this.hasPreviousXMLPlate()) { spent.add(this.previousXMLPlate()); } return spent; } /** * Returns a list containing all the XML plates in the list. * @return list of all plate objects */ public List allXMLPlates() { List allPlates = new ArrayList(); for(PlateXML plate : this.platesXML) { allPlates.add(plate.toPlateObject()); } return allPlates; } /*------------------ Methods for JSON Plate Stack Input ------------------*/ /** * Returns true if the the next stack POJO in the list is a valid stack. * @return true if the next stack POJO is valid * @throws IOException * @throws ProcessingException */ public boolean hasNextJSONStack() { return this.indexStacksJSON >= 0 && this.indexStacksJSON < this.stacksJSON.size(); } /** * Returns true if the the previous stack POJO in the list is a valid stack. * @return true if the previous stack POJO is valid * @throws IOException * @throws ProcessingException */ public boolean hasPreviousJSONStack() { return this.indexStacksJSON > 0; } /** * Parses the next stack in the stack list to a stack object and returns the * result. * @return the converted stack object */ public Stack nextJSONStack() { return this.hasNextJSONStack() ? this.stacksJSON.get(indexStacksJSON++).toStackObject() : null; } /** * Parses the previous stack in the stack list to a stack object and returns the * result. * @return the converted stack object */ public Stack previousJSONStack() { return this.hasPreviousJSONStack() ? this.stacksJSON.get(--indexStacksJSON).toStackObject() : null; } /** * Returns a list containing the stack objects with an index greater than the * current index. * @return list of converted stack objects */ public List remainingJSONStacks() { if(!this.hasNextJSONStack()) { return null; } List remaining = new ArrayList(); while(this.hasNextJSONStack()) { remaining.add(this.nextJSONStack()); } return remaining; } /** * Returns a list containing the stack objects with an index less than the * current index. * @return list of converted stack objects */ public List spentJSONStacks() { if(!this.hasPreviousJSONStack()) { return null; } List spent = new ArrayList(); while(this.hasPreviousJSONStack()) { spent.add(this.previousJSONStack()); } return spent; } /** * Returns a list containing all the JSON stacks in the list. * @return list of all stack objects */ public List allJSONStacks() { List allStacks = new ArrayList(); for(StackPOJO stack : this.stacksJSON) { allStacks.add(stack.toStackObject()); } return allStacks; } /*------------------ Methods for XML Plate Stack Input -------------------*/ /** * Returns true if the the next XML stack in the list is a valid stack. * @return true if the next XML stack is valid * @throws IOException * @throws ProcessingException */ public boolean hasNextXMLStack() { return this.indexStacksXML >= 0 && this.indexStacksXML < this.stacksXML.size(); } /** * Returns true if the the previous XML stack in the list is a valid stack. * @return true if the previous XML stack is valid * @throws IOException * @throws ProcessingException */ public boolean hasPreviousXMLStack() { return this.indexStacksXML > 0; } /** * Parses the next stack in the stack list to a stack object and returns the * result. * @return the converted stack object */ public Stack nextXMLStack() { return this.hasNextXMLStack() ? this.stacksXML.get(indexStacksXML++).toStackObject() : null; } /** * Parses the previous stack in the stack list to a stack object and returns the * result. * @return the converted stack object */ public Stack previousXMLStack() { return this.hasPreviousXMLStack() ? this.stacksXML.get(--indexStacksXML).toStackObject() : null; } /** * Returns a list containing the stack objects with an index greater than the * current index. * @return list of converted stack objects */ public List remainingXMLStacks() { if(!this.hasNextXMLStack()) { return null; } List remaining = new ArrayList(); while(this.hasNextXMLStack()) { remaining.add(this.nextXMLStack()); } return remaining; } /** * Returns a list containing the stack objects with an index less than the * current index. * @return list of converted stack objects */ public List spentXMLStacks() { if(!this.hasPreviousXMLStack()) { return null; } List spent = new ArrayList(); while(this.hasPreviousXMLStack()) { spent.add(this.previousXMLStack()); } return spent; } /** * Returns a list containing all the XML stacks in the list. * @return list of all stack objects */ public List allXMLStacks() { List allStacks = new ArrayList(); for(StackXML stack : this.stacksXML) { allStacks.add(stack.toStackObject()); } return allStacks; } /*--------------------- Methods for validating input ---------------------*/ /** * Sets the delimiter. * @param String the delimiter */ public void setDelimiter(String delim) { try { this.delimiter = delim; this.columnHeadersRegex = "^\\s*" + this.delimiter + "?\\s*(\\s*\\d+)\\s*(" + this.delimiter + "\\s*\\d+\\s*)*$"; this.tableHeaderRegex = "^\\s*Index\\s*" + this.delimiter + "\\s*Value\\s*$"; this.columnHeadersPattern = Pattern.compile(columnHeadersRegex); this.tableHeaderPattern = Pattern.compile(tableHeaderRegex); } catch(Exception e) { this.delimiter = "\t"; throw new IllegalArgumentException("Invalid delimiter."); } } /** * Returns the delimiter. * @return the delimiter */ public String getDelimiter() { return this.delimiter; } /** * Converts a row ID to an integer value. * @param String the row as a string * @return the row as an integer value */ private int parseRow(String rowString) { int rowInt = 0; int baseIndex = 0; String upper = rowString.toUpperCase().trim(); Matcher lettersMatcher = lettersPattern.matcher(upper); if(lettersMatcher.find()) { String letters = lettersMatcher.group(0); rowInt = letters.charAt(letters.length() - 1) - 65; for(int i = letters.length() - 1; i >= 0; i--) { rowInt += (baseIndex++ * (letters.charAt(i) - 65 + 1) * ALPHA_BASE); } return rowInt; } else { throw new IllegalArgumentException("Invalid row ID: " + rowString); } } /** * Validates a string against a JSON schema. * @param String file path to the JSON schema * @param String string to validate * @return true when string is a valid JSON object */ private boolean validateSchema(String schemaPath, String input) { try { ClassLoader classLoader = getClass().getClassLoader(); File schemaFile = new File(classLoader.getResource(schemaPath).getFile()); /* Convert the string to a JSON node */ JsonNode rootNode = JsonLoader.fromString(input); /* Get the schema */ JsonNode schemaNode = JsonLoader.fromFile(schemaFile); JsonSchemaFactory factory = JsonSchemaFactory.byDefault(); JsonSchema schema = factory.getJsonSchema(schemaNode); return schema.validInstance(rootNode); } catch (Exception e) { e.printStackTrace(); return false; } } /** * Sets the boolean values for JSON schema validation. * @param String string to validate * @throws IOException * @throws JsonProcessingException */ private void validateJSON(String input) throws ProcessingException, IOException { if(this.validateSchema(this.JSON_RESULT_SCHEMA, input)) { ObjectMapper mapper = new ObjectMapper(); this.resultsJSON = mapper.readValue(this, ResultListPOJO.class); this.indexResultsJSON = 0; return; } if(this.validateSchema(this.JSON_WELL_SCHEMA, input)) { ObjectMapper mapper = new ObjectMapper(); this.wellsJSON = mapper.readValue(this, WellListPOJO.class); this.indexWellsJSON = 0; return; } if(this.validateSchema(this.JSON_WELLSET_SCHEMA, input)) { ObjectMapper mapper = new ObjectMapper(); this.setsJSON = mapper.readValue(this, WellSetListPOJO.class); this.indexSetsJSON = 0; return; } if(this.validateSchema(this.JSON_PLATE_SCHEMA, input)) { ObjectMapper mapper = new ObjectMapper(); this.platesJSON = mapper.readValue(this, PlateListPOJO.class); this.indexPlatesJSON = 0; return; } if(this.validateSchema(this.JSON_STACK_SCHEMA, input)) { ObjectMapper mapper = new ObjectMapper(); this.stacksJSON = mapper.readValue(this, StackListPOJO.class); this.indexStacksJSON = 0; return; } } /** * Sets the boolean values for XML schema validation. * @param String string to validate * @throws JAXBException */ private void validateXML(String input) throws JAXBException { try { JAXBContext jaxbContext = JAXBContext.newInstance(ResultListXML.class); Unmarshaller unmarshaller = jaxbContext.createUnmarshaller(); StringReader reader = new StringReader(input); this.resultsXML = (ResultListXML) unmarshaller.unmarshal(reader); this.indexResultsXML = 0; } catch(Exception e) {} try { JAXBContext jaxbContext = JAXBContext.newInstance(WellListXML.class); Unmarshaller unmarshaller = jaxbContext.createUnmarshaller(); StringReader reader = new StringReader(input); this.wellsXML = (WellListXML) unmarshaller.unmarshal(reader); this.indexWellsXML = 0; } catch(Exception e) {} try { JAXBContext jaxbContext = JAXBContext.newInstance(WellSetListXML.class); Unmarshaller unmarshaller = jaxbContext.createUnmarshaller(); StringReader reader = new StringReader(input); this.setsXML = (WellSetListXML) unmarshaller.unmarshal(reader); this.indexSetsXML = 0; } catch(Exception e) {} try { JAXBContext jaxbContext = JAXBContext.newInstance(PlateListXML.class); Unmarshaller unmarshaller = jaxbContext.createUnmarshaller(); StringReader reader = new StringReader(input); this.platesXML = (PlateListXML) unmarshaller.unmarshal(reader); this.indexPlatesXML = 0; } catch(Exception e) {} try { JAXBContext jaxbContext = JAXBContext.newInstance(StackListXML.class); Unmarshaller unmarshaller = jaxbContext.createUnmarshaller(); StringReader reader = new StringReader(input); this.stacksXML = (StackListXML) unmarshaller.unmarshal(reader); this.indexStacksXML = 0; } catch(Exception e) {} } /** * Returns the next line in the input buffer. If the next line is null this * method throws a NoSuchElementException. * @return next line in the input buffer * @throws NoSuchElementException */ private String nextLine() { String nextLine = null; try { nextLine = this.readLine(); } catch (IOException e) { e.printStackTrace(); } if(nextLine == null) { throw new NoSuchElementException("There are no more lines in the input buffer."); } else { return nextLine; } } /** * Returns true if there is another line in the input buffer. * @return false if no more lines exist in the buffer */ private boolean hasNextLine() { String nextLine = null; try { this.mark(readAheadLimit); nextLine = this.readLine(); this.reset(); return nextLine != null; } catch (IOException e) { e.printStackTrace(); return false; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy