/* CSVLexer.java is a generated file. You probably want to
* edit CSVLexer.lex to make changes. Use JFlex to generate it.
* JFlex may be obtained from
* the JFlex website.
* This file was tested to work with jflex 1.4 (and may not
* work with more recent version because it needs a skeleton file)
* Run:
* jflex --skel csv.jflex.skel CSVLexer.lex
* You will then have a file called CSVLexer.java
*/
/*
* Read files in comma separated value format.
* Copyright (C) 2001-2004 Stephen Ostermiller
* http://ostermiller.org/contact.pl?regarding=Java+Utilities
*
* 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 2 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.
*
* See COPYING.TXT for details.
*/
package com.Ostermiller.util;
import java.io.*;
/**
* Read files in comma separated value format.
*
* The use of this class is no longer recommended. It is now recommended that you use
* com.Ostermiller.util.CSVParser instead. That class, has a cleaner API, and methods
* for returning all the values on a line in a String array.
*
* CSV is a file format used as a portable representation of a database.
* Each line is one entry or record and the fields in a record are separated by commas.
* Commas may be preceded or followed by arbitrary space and/or tab characters which are
* ignored.
*
* If field includes a comma or a new line, the whole field must be surrounded with double quotes. * When the field is in quotes, any quote literals must be escaped by \" Backslash * literals must be escaped by \\. Otherwise a backslash an the character following it * will be treated as the following character, ie."\n" is equivelent to "n". Other escape * sequences may be set using the setEscapes() method. Text that comes after quotes that have * been closed but come before the next comma will be ignored. *
* Empty fields are returned as as String of length zero: "". The following line has four empty
* fields and two non-empty fields in it. There is an empty field on each end, and two in the
* middle.
*
,second,, ,fifth,*
* Blank lines are always ignored. Other lines will be ignored if they start with a * comment character as set by the setCommentStart() method. *
* An example of how CVSLexer might be used: *
* CSVLexer shredder = new CSVLexer(System.in); * shredder.setCommentStart("#;!"); * shredder.setEscapes("nrtf", "\n\r\t\f"); * String t; * while ((t = shredder.getNextToken()) != null) { * System.out.println("" + shredder.getLineNumber() + " " + t); * } ** * @author Stephen Ostermiller http://ostermiller.org/contact.pl?regarding=Java+Utilities * @since ostermillerutils 1.00.00 */ %% %public %class CSVLexer %function getNextToken %type String %{ /** * Prints out tokens and line numbers from a file or System.in. * If no arguments are given, System.in will be used for input. * If more arguments are given, the first argument will be used as * the name of the file to use as input * * @param args program arguments, of which the first is a filename * * @since ostermillerutils 1.00.00 */ public static void main(String[] args){ InputStream in; try { if (args.length > 0){ File f = new File(args[0]); if (f.exists()){ if (f.canRead()){ in = new FileInputStream(f); } else { throw new IOException("Could not open " + args[0]); } } else { throw new IOException("Could not find " + args[0]); } } else { in = System.in; } CSVLexer shredder = new CSVLexer(in); shredder.setCommentStart("#;!"); shredder.setEscapes("nrtf", "\n\r\t\f"); String t; while ((t = shredder.getNextToken()) != null) { System.out.println("" + shredder.getLineNumber() + " " + t); } } catch (IOException e){ System.out.println(e.getMessage()); } } private char delimiter = ','; private char quote = '\"'; /** * Checks that zzcmap_instance is an instance variable (not just * a pointer to a static variable). If it is a pointer to a static * variable, it will be cloned. * * @since ostermillerutils 1.00.00 */ private void ensureCharacterMapIsInstance(){ if (ZZ_CMAP == zzcmap_instance){ zzcmap_instance = new char[ZZ_CMAP.length]; System.arraycopy(ZZ_CMAP, 0, zzcmap_instance, 0, ZZ_CMAP.length); } } /** * Ensures that the given character is not used for some special purpose * in parsing. This method should be called before setting some character * to be a delimiter so that the parsing doesn't break. Examples of bad * characters are quotes, commas, and whitespace. * * @since ostermillerutils 1.00.00 */ private boolean charIsSafe(char c){ // There are two character classes that one could use as a delimiter. // The first would be the class that most characters are in. These // are normally data. The second is the class that the tab is usually in. return (zzcmap_instance[c] == ZZ_CMAP['a'] || zzcmap_instance[c] == ZZ_CMAP['\t']); } /** * Change the character classes of the two given characters. This * will make the state machine behave as if the characters were switched * when they are encountered in the input. * * @param old the old character, its value will be returned to initial * @param two second character * * @since ostermillerutils 1.00.00 */ private void updateCharacterClasses(char oldChar, char newChar){ // before modifying the character map, make sure it isn't static. ensureCharacterMapIsInstance(); // make the newChar behave like the oldChar zzcmap_instance[newChar] = zzcmap_instance[oldChar]; // make the oldChar behave like it isn't special. switch (oldChar){ case ',': case '"': { // These should act as normal character, // not delimiters or quotes right now. zzcmap_instance[oldChar] = ZZ_CMAP['a']; } break; default: { // Set the it back to the way it would act // if not used as a delimiter or quote. zzcmap_instance[oldChar] = ZZ_CMAP[oldChar]; } break; } } /** * Change this Lexer so that it uses a new delimiter. *
* The initial character is a comma, the delimiter cannot be changed * to a quote or other character that has special meaning in CSV. * * @param newDelim delimiter to which to switch. * @throws BadDelimiterException if the character cannot be used as a delimiter. * * @since ostermillerutils 1.00.00 */ public void changeDelimiter(char newDelim) throws BadDelimiterException { if (newDelim == delimiter) return; // no need to do anything. if (!charIsSafe(newDelim)){ throw new BadDelimiterException(newDelim + " is not a safe delimiter."); } updateCharacterClasses(delimiter, newDelim); // keep a record of the current delimiter. delimiter = newDelim; } /** * Change this Lexer so that it uses a new character for quoting. *
* The initial character is a double quote ("), the delimiter cannot be changed
* to a comma or other character that has special meaning in CSV.
*
* @param newQuote character to use for quoting.
* @throws BadQuoteException if the character cannot be used as a quote.
*
* @since ostermillerutils 1.00.00
*/
public void changeQuote(char newQuote) throws BadQuoteException {
if (newQuote == quote) return; // no need to do anything.
if (!charIsSafe(newQuote)){
throw new BadQuoteException(newQuote + " is not a safe quote.");
}
updateCharacterClasses(quote, newQuote);
// keep a record of the current quote.
quote = newQuote;
}
private String escapes = "";
private String replacements = "";
/**
* Specify escape sequences and their replacements.
* Escape sequences set here are in addition to \\ and \".
* \\ and \" are always valid escape sequences. This method
* allows standard escape sequenced to be used. For example
* "\n" can be set to be a newline rather than an 'n'.
* A common way to call this method might be:
* New line breaks that occur in the middle of a token are not
* counted in the line number count.
*
* If no tokens have been returned, the line number is undefined.
*
* @return line number of the last token.
*
* @since ostermillerutils 1.00.00
*/
public int getLineNumber(){
return lines;
}
%}
%unicode
%state BEFORE
%state AFTER
%state COMMENT
Blank=([ ])
Tab=([\t])
FF=([\f])
CR=([\r])
LF=([\n])
EOL=({CR}|{LF}|{CR}{LF})
NonBreakingWS=({Blank}|{Tab}|{FF})
/* To change the default delimeter, change the comma in the next five lines */
Separator=([\,])
NotCommaSpaceQuote=([^\t\f\r\n\,\" ])
NotCommaSpace=([^\t\f\r\n\, ])
NotCommaEOL=([^\,\r\n])
IgnoreAfter=(([^\r\n\,])*)
FalseLiteral=([\"]([^\"\\]|([\\][^]))*)
StringLiteral=({FalseLiteral}[\"])
Value=({NotCommaSpaceQuote}(({NotCommaEOL}*){NotCommaSpace})?)
%%
* setEscapes("nrtf", "\n\r\t\f");
* which would set the escape sequences to be the Java escape
* sequences. Characters that follow a \ that are not escape
* sequences will still be interpreted as that character.
* The two arguemnts to this method must be the same length. If
* they are not, the longer of the two will be truncated.
*
* @param escapes a list of characters that will represent escape sequences.
* @param replacements the list of repacement characters for those escape sequences.
*
* @since ostermillerutils 1.00.00
*/
public void setEscapes(String escapes, String replacements){
int length = escapes.length();
if (replacements.length() < length){
length = replacements.length();
}
this.escapes = escapes.substring(0, length);
this.replacements = replacements.substring(0, length);
}
private String unescape(String s){
if (s.indexOf('\\') == -1){
return s.substring(1, s.length()-1);
}
StringBuffer sb = new StringBuffer(s.length());
for (int i=1; i # Comment
* ; Another Comment
* ! Yet another comment
* By default there are no comments in CVS files. Commas and quotes may not be
* used to indicate comment lines.
*
* @param commentDelims list of characters a comment line may start with.
*
* @since ostermillerutils 1.00.00
*/
public void setCommentStart(String commentDelims){
this.commentDelims = commentDelims;
}
private int addLine = 1;
private int lines = 0;
/**
* Get the line number that the last token came from.
*