001/* 002 * @(#)SequenceUtil.java 1.0 September 2009 003 * 004 * Copyright (c) 2009 Peter Troshin 005 * 006 * BioJava development code 007 * 008 * This code may be freely distributed and modified under the 009 * terms of the GNU Lesser General Public Licence. This should 010 * be distributed with the code. If you do not have a copy, 011 * see: 012 * 013 * http://www.gnu.org/copyleft/lesser.html 014 * 015 * Copyright for this code is held jointly by the individual 016 * authors. These should be listed in @author doc comments. 017 * 018 * For more information on the BioJava project and its aims, 019 * or to join the biojava-l mailing list, visit the home page 020 * at: 021 * 022 * http://www.biojava.org/ 023 * 024 */ 025 026package org.biojava.nbio.data.sequence; 027 028import org.slf4j.Logger; 029import org.slf4j.LoggerFactory; 030 031import java.io.*; 032import java.util.ArrayList; 033import java.util.List; 034import java.util.regex.Matcher; 035import java.util.regex.Pattern; 036 037/** 038 * Utility class for operations on sequences 039 * 040 * @author Peter Troshin 041 * @version 1.0 042 * @since 3.0.2 043 */ 044public final class SequenceUtil { 045 046 private static final Logger logger = LoggerFactory.getLogger(SequenceUtil.class); 047 048 /** 049 * A whitespace character: [\t\n\x0B\f\r] 050 */ 051 public static final Pattern WHITE_SPACE = Pattern.compile("\\s"); 052 053 /** 054 * A digit 055 */ 056 public static final Pattern DIGIT = Pattern.compile("\\d"); 057 058 /** 059 * Non word 060 */ 061 public static final Pattern NONWORD = Pattern.compile("\\W"); 062 063 /** 064 * Valid Amino acids 065 */ 066 public static final Pattern AA = Pattern.compile("[ARNDCQEGHILKMFPSTWYVUO]+", 067 Pattern.CASE_INSENSITIVE); 068 069 /** 070 * inversion of AA pattern 071 */ 072 public static final Pattern NON_AA = Pattern.compile( 073 "[^ARNDCQEGHILKMFPSTWYVXUO]+", Pattern.CASE_INSENSITIVE); 074 075 /** 076 * Same as AA pattern but with one additional letters - X 077 */ 078 public static final Pattern AMBIGUOUS_AA = Pattern.compile( 079 "[ARNDCQEGHILKMFPSTWYVXUO]+", Pattern.CASE_INSENSITIVE); 080 081 /** 082 * Nucleotides a, t, g, c, u 083 */ 084 public static final Pattern NUCLEOTIDE = Pattern.compile("[AGTCU]+", 085 Pattern.CASE_INSENSITIVE); 086 087 /** 088 * Ambiguous nucleotide 089 */ 090 public static final Pattern AMBIGUOUS_NUCLEOTIDE = Pattern.compile( 091 "[AGTCRYMKSWHBVDNU]+", Pattern.CASE_INSENSITIVE); // see IUPAC 092 /** 093 * Non nucleotide 094 */ 095 public static final Pattern NON_NUCLEOTIDE = Pattern.compile("[^AGTCU]+", 096 Pattern.CASE_INSENSITIVE); 097 098 private SequenceUtil() { 099 } // utility class, no instantiation 100 101 /* 102 * public static void write_PirSeq(OutputStream os, FastaSequence seq) 103 * throws IOException { BufferedWriter pir_out = new BufferedWriter(new 104 * OutputStreamWriter(os)); pir_out.write(">P1;" + seq.getId() + 105 * SysPrefs.newlinechar); pir_out.write(seq.getSequence() + 106 * SysPrefs.newlinechar); pir_out.close(); } 107 * 108 * public static void write_FastaSeq(OutputStream os, FastaSequence seq) 109 * throws IOException { BufferedWriter fasta_out = new BufferedWriter( new 110 * OutputStreamWriter(os)); fasta_out.write(">" + seq.getId() + 111 * SysPrefs.newlinechar); fasta_out.write(seq.getSequence() + 112 * SysPrefs.newlinechar); fasta_out.close(); } 113 */ 114 115 /** 116 * @return true is the sequence contains only letters a,c, t, g, u 117 */ 118 public static boolean isNucleotideSequence(final FastaSequence s) { 119 return SequenceUtil.isNonAmbNucleotideSequence(s.getSequence()); 120 } 121 122 /** 123 * Ambiguous DNA chars : AGTCRYMKSWHBVDN // differs from protein in only one 124 * (!) - B char 125 */ 126 public static boolean isNonAmbNucleotideSequence(String sequence) { 127 sequence = SequenceUtil.cleanSequence(sequence); 128 if (SequenceUtil.DIGIT.matcher(sequence).find()) { 129 return false; 130 } 131 if (SequenceUtil.NON_NUCLEOTIDE.matcher(sequence).find()) { 132 return false; 133 /* 134 * System.out.format("I found the text starting at " + 135 * "index %d and ending at index %d.%n", nonDNAmatcher .start(), 136 * nonDNAmatcher.end()); 137 */ 138 } 139 final Matcher DNAmatcher = SequenceUtil.NUCLEOTIDE.matcher(sequence); 140 return DNAmatcher.find(); 141 } 142 143 /** 144 * Removes all whitespace chars in the sequence string 145 * 146 * @param sequence 147 * @return cleaned up sequence 148 */ 149 public static String cleanSequence(String sequence) { 150 assert sequence != null; 151 final Matcher m = SequenceUtil.WHITE_SPACE.matcher(sequence); 152 sequence = m.replaceAll("").toUpperCase(); 153 return sequence; 154 } 155 156 /** 157 * Removes all special characters and digits as well as whitespace chars 158 * from the sequence 159 * 160 * @param sequence 161 * @return cleaned up sequence 162 */ 163 public static String deepCleanSequence(String sequence) { 164 sequence = SequenceUtil.cleanSequence(sequence); 165 sequence = SequenceUtil.DIGIT.matcher(sequence).replaceAll(""); 166 sequence = SequenceUtil.NONWORD.matcher(sequence).replaceAll(""); 167 final Pattern othernonSeqChars = Pattern.compile("[_-]+"); 168 sequence = othernonSeqChars.matcher(sequence).replaceAll(""); 169 return sequence; 170 } 171 172 /** 173 * 174 * @param sequence 175 * @return true is the sequence is a protein sequence, false overwise 176 */ 177 public static boolean isProteinSequence(String sequence) { 178 sequence = SequenceUtil.cleanSequence(sequence); 179 if (SequenceUtil.isNonAmbNucleotideSequence(sequence)) { 180 return false; 181 } 182 if (SequenceUtil.DIGIT.matcher(sequence).find()) { 183 return false; 184 } 185 if (SequenceUtil.NON_AA.matcher(sequence).find()) { 186 logger.info("Found non aa: {}", sequence); 187 return false; 188 } 189 final Matcher protmatcher = SequenceUtil.AA.matcher(sequence); 190 return protmatcher.find(); 191 } 192 193 /** 194 * Check whether the sequence confirms to amboguous protein sequence 195 * 196 * @param sequence 197 * @return return true only if the sequence if ambiguous protein sequence 198 * Return false otherwise. e.g. if the sequence is non-ambiguous 199 * protein or DNA 200 */ 201 public static boolean isAmbiguosProtein(String sequence) { 202 sequence = SequenceUtil.cleanSequence(sequence); 203 if (SequenceUtil.isNonAmbNucleotideSequence(sequence)) { 204 return false; 205 } 206 if (SequenceUtil.DIGIT.matcher(sequence).find()) { 207 return false; 208 } 209 if (SequenceUtil.NON_AA.matcher(sequence).find()) { 210 return false; 211 } 212 if (SequenceUtil.AA.matcher(sequence).find()) { 213 return false; 214 } 215 final Matcher amb_prot = SequenceUtil.AMBIGUOUS_AA.matcher(sequence); 216 return amb_prot.find(); 217 } 218 219 /** 220 * Writes list of FastaSequeces into the outstream formatting the sequence 221 * so that it contains width chars on each line 222 * 223 * @param outstream 224 * @param sequences 225 * @param width 226 * - the maximum number of characters to write in one line 227 * @throws IOException 228 */ 229 public static void writeFasta(final OutputStream outstream, 230 final List<FastaSequence> sequences, final int width) 231 throws IOException { 232 final OutputStreamWriter writer = new OutputStreamWriter(outstream); 233 final BufferedWriter fastawriter = new BufferedWriter(writer); 234 for (final FastaSequence fs : sequences) { 235 fastawriter.write(fs.getFormatedSequence(width)); 236 } 237 outstream.flush(); 238 fastawriter.close(); 239 writer.close(); 240 } 241 242 /** 243 * Reads fasta sequences from inStream into the list of FastaSequence 244 * objects 245 * 246 * @param inStream 247 * from 248 * @return list of FastaSequence objects 249 * @throws IOException 250 */ 251 public static List<FastaSequence> readFasta(final InputStream inStream) 252 throws IOException { 253 final List<FastaSequence> seqs = new ArrayList<>(); 254 255 final BufferedReader infasta = new BufferedReader( 256 new InputStreamReader(inStream, "UTF8"), 16000); 257 final Pattern pattern = Pattern.compile("//s+"); 258 259 String line; 260 String sname = "", seqstr = null; 261 do { 262 line = infasta.readLine(); 263 if ((line == null) || line.startsWith(">")) { 264 if (seqstr != null) { 265 seqs.add(new FastaSequence(sname.substring(1), seqstr)); 266 } 267 sname = line; // remove > 268 seqstr = ""; 269 } else { 270 final String subseq = pattern.matcher(line).replaceAll(""); 271 seqstr += subseq; 272 } 273 } while (line != null); 274 275 infasta.close(); 276 return seqs; 277 } 278 279 /** 280 * Writes FastaSequence in the file, each sequence will take one line only 281 * 282 * @param os 283 * @param sequences 284 * @throws IOException 285 */ 286 public static void writeFasta(final OutputStream os, 287 final List<FastaSequence> sequences) throws IOException { 288 final OutputStreamWriter outWriter = new OutputStreamWriter(os); 289 final BufferedWriter fasta_out = new BufferedWriter(outWriter); 290 for (final FastaSequence fs : sequences) { 291 fasta_out.write(fs.getOnelineFasta()); 292 } 293 fasta_out.close(); 294 outWriter.close(); 295 } 296 297}