diff --git a/common/source/java/ch/systemsx/cisd/common/logging/LogAnnotation.java b/common/source/java/ch/systemsx/cisd/common/logging/LogAnnotation.java index 4e3b46f6c9d5813dc0066fab5801382fda94c027..203a32617df22238002fd2c8829e95e69b535ef0 100644 --- a/common/source/java/ch/systemsx/cisd/common/logging/LogAnnotation.java +++ b/common/source/java/ch/systemsx/cisd/common/logging/LogAnnotation.java @@ -24,7 +24,7 @@ import java.lang.annotation.Target; /** * Log annotation for methods invoked by {@link LogInvocationHandler}. - * + * * @author Franz-Josef Elmer */ @Retention(RetentionPolicy.RUNTIME) @@ -32,6 +32,7 @@ import java.lang.annotation.Target; @Inherited public @interface LogAnnotation { - public LogCategory logCategory() default LogCategory.ACCESS; + public LogCategory logCategory() default LogCategory.OPERATION; + public LogLevel logLevel() default LogLevel.UNDEFINED; } diff --git a/common/source/java/ch/systemsx/cisd/common/logging/LogCategory.java b/common/source/java/ch/systemsx/cisd/common/logging/LogCategory.java index e2377645bc7aaddc20fb00c9a8fd1600c5f9c012..23ef465420022f8784c114641b932a8c5185fc1c 100644 --- a/common/source/java/ch/systemsx/cisd/common/logging/LogCategory.java +++ b/common/source/java/ch/systemsx/cisd/common/logging/LogCategory.java @@ -25,9 +25,9 @@ public enum LogCategory { /** Log category for data (read) access events. */ - ACCESS, + ACCESS(true), /** Log category for authentication and authorization events. */ - AUTH, + AUTH(true), /** Log category for log events related to the machine's state (low-level). */ MACHINE, /** @@ -41,6 +41,23 @@ public enum LogCategory /** Log category for (normal) operational events. */ OPERATION, /** Log category for data manipulation events (write access). */ - TRACKING + TRACKING(true); + + final boolean adminLog; + + private LogCategory(boolean adminLog) + { + this.adminLog = adminLog; + } + + private LogCategory() + { + this(false); + } + + public boolean isAdminLog() + { + return adminLog; + } } diff --git a/common/source/java/ch/systemsx/cisd/common/logging/LogFactory.java b/common/source/java/ch/systemsx/cisd/common/logging/LogFactory.java index 5bf4be148d38f594b97dd4a06d5d2e3c1d8d3225..ad2c37b98eaf440dfec184ec2ce009903427a238 100644 --- a/common/source/java/ch/systemsx/cisd/common/logging/LogFactory.java +++ b/common/source/java/ch/systemsx/cisd/common/logging/LogFactory.java @@ -40,6 +40,20 @@ public final class LogFactory return category.name() + "." + clazz.getCanonicalName(); } + /** + * @return The logger name for the given {@link LogCategory}. Needs to be used for admin logs (i.e. + * {@link LogCategory#isAdminLog()} needs to return <code>true</code>). It will contain the name of the + * <var>category</var>. + */ + public static String getLoggerName(LogCategory category) + { + if (category.isAdminLog() == false) + { + throw new IllegalArgumentException("Only admin logs are allowed here, but we got " + category + "."); + } + return category.name() ; + } + /** * @return The logger for the given {@link LogCategory} and {@link Class}. The name of the logger will contain the * name of the <var>category</var>, followed by the canonical name of <var>clazz</var>. @@ -48,4 +62,15 @@ public final class LogFactory { return Logger.getLogger(getLoggerName(category, clazz)); } + + /** + * @return The logger for the given {@link LogCategory}. Needs to be an admin log (i.e. + * {@link LogCategory#isAdminLog()} needs to return <code>true</code>). The name of the logger will + * contain the name of the <var>category</var>. + */ + public static Logger getLogger(LogCategory category) + { + return Logger.getLogger(getLoggerName(category)); + } + } diff --git a/common/source/java/ch/systemsx/cisd/common/logging/LogInvocationHandler.java b/common/source/java/ch/systemsx/cisd/common/logging/LogInvocationHandler.java index bae7d5c69d3b27699632d14bf085b523a247039a..d228981dc0262f91c895e89f5e3663043b1db4e5 100644 --- a/common/source/java/ch/systemsx/cisd/common/logging/LogInvocationHandler.java +++ b/common/source/java/ch/systemsx/cisd/common/logging/LogInvocationHandler.java @@ -26,19 +26,22 @@ import org.apache.log4j.Logger; /** * Invocation handler used to log invocations. - * + * * @author Franz-Josef Elmer */ public final class LogInvocationHandler implements InvocationHandler { private final Object object; + private final String name; + private final Level defaultLogLevel; + private final Class<?> classUsedToNameLogger; /** * Creates a new instance. - * + * * @param object Object whose invocations should be logged. * @param name Meaningful name of <code>object</code>. Will be used in the log message. * @param logLevel The log level to use for normal (successful) events. @@ -102,13 +105,13 @@ public final class LogInvocationHandler implements InvocationHandler } } } - + private Level getLogLevel(Method method) { final LogAnnotation annotation = method.getAnnotation(LogAnnotation.class); if (annotation == null) { - return Level.DEBUG; + return Level.DEBUG; } else if (annotation.logLevel().equals(LogLevel.UNDEFINED)) { return defaultLogLevel; @@ -117,11 +120,11 @@ public final class LogInvocationHandler implements InvocationHandler return Log4jSimpleLogger.toLog4jPriority(annotation.logLevel()); } } - - private Logger createLogger(Method method) + + private Logger createLogger(Method method) { final LogAnnotation annotation = method.getAnnotation(LogAnnotation.class); - final LogCategory logCategory = (annotation == null) ? LogCategory.ACCESS : annotation.logCategory(); + final LogCategory logCategory = (annotation == null) ? LogCategory.OPERATION : annotation.logCategory(); return LogFactory.getLogger(logCategory, classUsedToNameLogger); } } \ No newline at end of file diff --git a/common/source/java/ch/systemsx/cisd/common/logging/LogLevel.java b/common/source/java/ch/systemsx/cisd/common/logging/LogLevel.java index 4062e4ddd25aec4cef5458fd40d63e82e292af6f..efc86d649282c22ecca919851f5ec4d97c6cbdc4 100644 --- a/common/source/java/ch/systemsx/cisd/common/logging/LogLevel.java +++ b/common/source/java/ch/systemsx/cisd/common/logging/LogLevel.java @@ -1,11 +1,11 @@ package ch.systemsx.cisd.common.logging; -/** +/** * A simple form of log levels. * - * @author Bernd RInn + * @author Bernd Rinn */ public enum LogLevel { - UNDEFINED, OFF, TRACE, DEBUG, INFO, WARN, ERROR + UNDEFINED, OFF, TRACE, DEBUG, INFO, WARN, ERROR; } \ No newline at end of file diff --git a/common/source/java/ch/systemsx/cisd/common/parser/DefaultParser.java b/common/source/java/ch/systemsx/cisd/common/parser/DefaultParser.java index 65f2fd2c36f7762870974678d06ab23c14345f45..d0c99bbe4002eded519e2af9ce837d94cf2d6899 100644 --- a/common/source/java/ch/systemsx/cisd/common/parser/DefaultParser.java +++ b/common/source/java/ch/systemsx/cisd/common/parser/DefaultParser.java @@ -73,7 +73,8 @@ public class DefaultParser<E> implements IParser<E> // Parser // - public final List<E> parse(final Iterator<Line> lineIterator, final ILineFilter lineFilter) throws ParsingException + public final List<E> parse(final Iterator<Line> lineIterator, final ILineFilter lineFilter, final long headerLength) + throws ParsingException { final List<E> elements = new ArrayList<E>(); synchronized (lineTokenizer) @@ -90,6 +91,12 @@ public class DefaultParser<E> implements IParser<E> E object = null; try { + if (tokens.length > headerLength) + { + throw new RuntimeException(String.format( + "Line <%s> has more columns (%s) than the header (%s)", number, String + .valueOf(tokens.length), String.valueOf(headerLength))); + } object = createObject(tokens); } catch (final ParserException parserException) { diff --git a/common/source/java/ch/systemsx/cisd/common/parser/IParser.java b/common/source/java/ch/systemsx/cisd/common/parser/IParser.java index c83eb104c156ff817f74cd2781a6dc5a329cbdb1..f3c5df833d39c58878710141d2348ed4b1e67788 100644 --- a/common/source/java/ch/systemsx/cisd/common/parser/IParser.java +++ b/common/source/java/ch/systemsx/cisd/common/parser/IParser.java @@ -32,9 +32,11 @@ public interface IParser<E> * Parses the lines delivered by the specified iterator and creating elements of type <code>E</code>. * * @param lineFilter A filter lines have to pass in order to be parsed. + * @param headerLength number of columns in the header * @return a <code>List</code> of elements. */ - public List<E> parse(final Iterator<Line> lineIterator, final ILineFilter lineFilter) throws ParsingException; + public List<E> parse(final Iterator<Line> lineIterator, final ILineFilter lineFilter, final long headerLength) + throws ParsingException; /** * Sets the <code>IParserObjectFactory</code>. diff --git a/common/source/java/ch/systemsx/cisd/common/parser/TabFileLoader.java b/common/source/java/ch/systemsx/cisd/common/parser/TabFileLoader.java index fcb988a22ff4a4e3d8450a7c543aa910197710ba..2622ce2162f360f17cd77cd1ead484805b3452be 100644 --- a/common/source/java/ch/systemsx/cisd/common/parser/TabFileLoader.java +++ b/common/source/java/ch/systemsx/cisd/common/parser/TabFileLoader.java @@ -35,28 +35,34 @@ import ch.systemsx.cisd.common.exceptions.EnvironmentFailureException; import ch.systemsx.cisd.common.exceptions.UserFailureException; /** - * Convenient class to load a tab file and deliver a list of beans of type <code>T</code>. The following - * formats for the column headers are recognized. - * <ol><li>Column headers in first line: - * <pre> + * Convenient class to load a tab file and deliver a list of beans of type <code>T</code>. The following formats for + * the column headers are recognized. + * <ol> + * <li>Column headers in first line: + * + * <pre> * column1 column2 column2 - * </pre> - * <li>Comment section: - * <pre> + * </pre> + * + * <li>Comment section: + * + * <pre> * # 1. line of comment * # 2. line of comment * # ... * column1 column2 column2 - * </pre> - * <li>Column headers at the end of the comment section: - * <pre> + * </pre> + * + * <li>Column headers at the end of the comment section: + * + * <pre> * # 1. line of comment * # 2. line of comment * # ... * # * #column1 column2 column2 - * </pre> - * + * </pre> + * * </ol> * * @author Franz-Josef Elmer @@ -103,8 +109,7 @@ public class TabFileLoader<T> IOUtils.closeQuietly(reader); } } - - + List<T> load(Reader reader) { List<T> result = new ArrayList<T>(); @@ -128,18 +133,19 @@ public class TabFileLoader<T> } @SuppressWarnings("null") final String headerLine = previousLineHasColumnHeaders ? previousLine.getText().substring(1) : line.getText(); - + final DefaultParser<T> parser = new DefaultParser<T>(); final String[] tokens = StringUtils.split(headerLine, "\t"); + final long headerLength = tokens.length; notUnique(tokens); final IAliasPropertyMapper propertyMapper = new HeaderFilePropertyMapper(tokens); parser.setObjectFactory(factory.createFactory(propertyMapper)); ILineFilter filter = AlwaysAcceptLineFilter.INSTANCE; if (previousLineHasColumnHeaders) { - result.addAll(parser.parse(Arrays.asList(line).iterator(), filter)); + result.addAll(parser.parse(Arrays.asList(line).iterator(), filter, headerLength)); } - result.addAll(parser.parse(lineIterator, filter)); + result.addAll(parser.parse(lineIterator, filter, headerLength)); return result; } @@ -149,6 +155,7 @@ public class TabFileLoader<T> Iterator<Line> iterator = new Iterator<Line>() { private int lineNumber; + public void remove() { lineIterator.remove(); diff --git a/common/sourceTest/java/ch/systemsx/cisd/common/db/DBRestrictionParserTest.java b/common/sourceTest/java/ch/systemsx/cisd/common/db/DBRestrictionParserTest.java index 5ae53f7963c0b1af4fcf28fcbed392d2a63a545a..721b5b75ce02209a732513e9a2a82e7cb7f880fc 100644 --- a/common/sourceTest/java/ch/systemsx/cisd/common/db/DBRestrictionParserTest.java +++ b/common/sourceTest/java/ch/systemsx/cisd/common/db/DBRestrictionParserTest.java @@ -16,7 +16,9 @@ package ch.systemsx.cisd.common.db; -import static org.testng.AssertJUnit.*; +import static org.testng.AssertJUnit.assertEquals; +import static org.testng.AssertJUnit.assertNotNull; +import static org.testng.AssertJUnit.assertTrue; import java.io.File; import java.util.Arrays; @@ -34,8 +36,9 @@ import ch.systemsx.cisd.common.logging.LogMonitoringAppender; import ch.systemsx.cisd.common.utilities.FileUtilities; /** - * Test cases for the @{link DBREstrictionParser}. + * Test cases for the * + * @{link DBREstrictionParser}. * @author Bernd Rinn */ public class DBRestrictionParserTest @@ -65,9 +68,8 @@ public class DBRestrictionParserTest { String invalidDomainStatement = "create domain bla for varchar(0)"; final List<String> domainScript = - Arrays.asList("create table sometable", "create domain user_id as varchar(15)", - invalidDomainStatement, "create domain code as varchar(8)", - "create domain description_80 as varchar(81)"); + Arrays.asList("create table sometable", "create domain user_id as varchar(15)", invalidDomainStatement, + "create domain code as varchar(8)", "create domain description_80 as varchar(81)"); final LogMonitoringAppender appender = LogMonitoringAppender.addAppender(LogCategory.OPERATION, "line \"" + invalidDomainStatement + "\" starts like a domain definition, but key word 'AS' is missing."); @@ -89,9 +91,8 @@ public class DBRestrictionParserTest @Test public void testDefaultKeywordInDomain() { - final List<String> domainScript = - Arrays.asList("create domain vc22 as varchar(22) default 'nothing special'"); - + final List<String> domainScript = Arrays.asList("create domain vc22 as varchar(22) default 'nothing special'"); + final LogMonitoringAppender appender = LogMonitoringAppender.addAppender(LogCategory.OPERATION, "ill-formed"); try { @@ -104,13 +105,12 @@ public class DBRestrictionParserTest LogMonitoringAppender.removeAppender(appender); } } - + @Test public void testDoublePrecisionInDomain() { - final List<String> domainScript = - Arrays.asList("create domain dp as double precision"); - + final List<String> domainScript = Arrays.asList("create domain dp as double precision"); + final LogMonitoringAppender appender = LogMonitoringAppender.addAppender(LogCategory.OPERATION, "ill-formed"); try { @@ -122,13 +122,12 @@ public class DBRestrictionParserTest LogMonitoringAppender.removeAppender(appender); } } - + @Test public void testDoublePrecisionAndDefaultInDomain() { - final List<String> domainScript = - Arrays.asList("create domain dp as double precision default 3.14159"); - + final List<String> domainScript = Arrays.asList("create domain dp as double precision default 3.14159"); + final LogMonitoringAppender appender = LogMonitoringAppender.addAppender(LogCategory.OPERATION, "ill-formed"); try { @@ -140,7 +139,7 @@ public class DBRestrictionParserTest LogMonitoringAppender.removeAppender(appender); } } - + @Test public void testColumnLengths() { diff --git a/common/sourceTest/java/ch/systemsx/cisd/common/parser/DefaultParserTest.java b/common/sourceTest/java/ch/systemsx/cisd/common/parser/DefaultParserTest.java index a8f0c1f8632232bcf75496faf03c9214bfd44f6c..ac0d353cc24cbe74c13092573652086e231df0b0 100644 --- a/common/sourceTest/java/ch/systemsx/cisd/common/parser/DefaultParserTest.java +++ b/common/sourceTest/java/ch/systemsx/cisd/common/parser/DefaultParserTest.java @@ -31,17 +31,19 @@ import org.testng.annotations.Test; */ public final class DefaultParserTest { - private final static List<String> text - = Arrays.asList("", "# This is a comment", "firstName\tlastName\taddress\tcity", - "Charles\tDarwin\tHumboldt Ave. 1865\t4242 Somewhere", - "Albert\tEinstein\tNewton Road 1905\t4711 Princton"); + private final static List<String> text = + Arrays.asList("", "# This is a comment", "firstName\tlastName\taddress\tcity", + "Charles\tDarwin\tHumboldt Ave. 1865\t4242 Somewhere", + "Albert\tEinstein\tNewton Road 1905\t4711 Princton"); + + private final static long HEADER_LENGTH = 4; @Test public final void testParseWithoutFactoryAndHeader() { final IParser<String[]> parser = new DefaultParser<String[]>(); parser.setObjectFactory(IParserObjectFactory.STRING_ARRAY_OBJECT_FACTORY); - final List<String[]> result = parser.parse(createLineIterator(), new HeaderLineFilter()); + final List<String[]> result = parser.parse(createLineIterator(), new HeaderLineFilter(), HEADER_LENGTH); assertEquals(3, result.size()); assertEquals(result.get(0)[0], "firstName"); assertEquals(result.get(1)[1], "Darwin"); @@ -54,7 +56,7 @@ public final class DefaultParserTest { final IParser<String[]> parser = new DefaultParser<String[]>(); parser.setObjectFactory(IParserObjectFactory.STRING_ARRAY_OBJECT_FACTORY); - final List<String[]> result = parser.parse(createLineIterator(), new HeaderLineFilter(3)); + final List<String[]> result = parser.parse(createLineIterator(), new HeaderLineFilter(3), HEADER_LENGTH); assertEquals(2, result.size()); assertEquals(result.get(0)[0], "Charles"); assertEquals(result.get(1)[1], "Einstein"); @@ -77,31 +79,33 @@ public final class DefaultParserTest parser.setObjectFactory(IParserObjectFactory.STRING_ARRAY_OBJECT_FACTORY); try { - parser.parse(createLineIterator(), new HeaderLineFilter(2)); + parser.parse(createLineIterator(), new HeaderLineFilter(2), HEADER_LENGTH); } catch (ParsingException ex) { - assertEquals( - "Creating an object with following tokens '[firstName, lastName, address, city]' failed.", - ex.getMessage()); + assertEquals("Creating an object with following tokens '[firstName, lastName, address, city]' failed.", ex + .getMessage()); assertEquals(3, ex.getLineNumber()); } } - + private Iterator<Line> createLineIterator() { return new Iterator<Line>() { private Iterator<String> iterator = text.iterator(); + private int lineNumber; + public void remove() { throw new UnsupportedOperationException(); } + public Line next() { return new Line(++lineNumber, iterator.next()); } - + public boolean hasNext() { return iterator.hasNext();