diff options
-rw-r--r-- | CREDITS.txt | 1 | ||||
-rw-r--r-- | src/changes/changes.xml | 5 | ||||
-rw-r--r-- | src/java/com/healthmarketscience/jackcess/RowFilter.java | 203 | ||||
-rw-r--r-- | test/src/java/com/healthmarketscience/jackcess/BigIndexTest.java | 4 | ||||
-rw-r--r-- | test/src/java/com/healthmarketscience/jackcess/IndexTest.java | 4 | ||||
-rw-r--r-- | test/src/java/com/healthmarketscience/jackcess/RowFilterTest.java | 112 |
6 files changed, 321 insertions, 8 deletions
diff --git a/CREDITS.txt b/CREDITS.txt index c77b80e..78b627b 100644 --- a/CREDITS.txt +++ b/CREDITS.txt @@ -4,3 +4,4 @@ Mitchell J. Friedman - Added support for additional JDBC data types James Ahlborn - Added support for NUMERIC data type Jon Iles - Added support for reading table definitions that span multiple pages James Schopp - added support for reading currency columns +Patricia Donaldson - contributed RowFilter class diff --git a/src/changes/changes.xml b/src/changes/changes.xml index 332abbc..8b9ed1f 100644 --- a/src/changes/changes.xml +++ b/src/changes/changes.xml @@ -5,6 +5,11 @@ <author email="jahlborn@users.sf.net">James Ahlborn</author> </properties> <body> + <release version="1.1.18" date="TBD"> + <action dev="jahlborn" type="add"> + Add RowFilter contributed by Patricia Donaldson. + </action> + </release> <release version="1.1.17" date="2008-09-23"> <action dev="jahlborn" type="fix" issue="2043499"> Fix simple index handling of tail index pages. diff --git a/src/java/com/healthmarketscience/jackcess/RowFilter.java b/src/java/com/healthmarketscience/jackcess/RowFilter.java new file mode 100644 index 0000000..c498273 --- /dev/null +++ b/src/java/com/healthmarketscience/jackcess/RowFilter.java @@ -0,0 +1,203 @@ +/* +Copyright (c) 2007 Health Market Science, Inc. + +This library is free software; you can redistribute it and/or +modify it under the terms of the GNU Lesser General Public +License as published by the Free Software Foundation; either +version 2.1 of the License, or (at your option) any later version. + +This library 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 +Lesser General Public License for more details. + +You should have received a copy of the GNU Lesser General Public +License along with this library; if not, write to the Free Software +Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 +USA + +You can contact Health Market Science at info@healthmarketscience.com +or at the following address: + +Health Market Science +2700 Horizon Drive +Suite 200 +King of Prussia, PA 19406 +*/ + +package com.healthmarketscience.jackcess; + +import java.util.Iterator; +import java.util.Map; + +import org.apache.commons.lang.ObjectUtils; + + +/** + * The RowFilter class encapsulates a filter test for a table row. This can + * be used by the {@link #apply(Iterable)} method to create an Iterable over a + * table which returns only rows matching some criteria. + * + * @author Patricia Donaldson, Xerox Corporation + */ +public abstract class RowFilter +{ + + /** + * Returns {@code true} if the given table row matches the Filter criteria, + * {@code false} otherwise. + * @param row current row to test for inclusion in the filter + */ + public abstract boolean matches(Map<String, Object> row); + + /** + * Returns an iterable which filters the given iterable based on this + * filter. + * + * @param iterable row iterable to filter + * + * @return a filtering iterable + */ + public Iterable<Map<String, Object>> apply( + Iterable<Map<String, Object>> iterable) + { + return new FilterIterable(iterable); + } + + + /** + * Creates a filter based on a row pattern. + * + * @param rowPattern Map from column names to the values to be matched. + * A table row will match the target if + * {@code ObjectUtil.equals(rowPattern.get(s), row.get(s))} + * for all column names in the pattern map. + * @return a filter which matches table rows which match the values in the + * row pattern + */ + public static RowFilter matchPattern(final Map<String, Object> rowPattern) + { + return new RowFilter() { + @Override + public boolean matches(Map<String, Object> row) + { + for(Map.Entry<String,Object> e : rowPattern.entrySet()) { + if(!ObjectUtils.equals(e.getValue(), row.get(e.getKey()))) { + return false; + } + } + return true; + } + }; + } + + /** + * Creates a filter based on a single value row pattern. + * + * @param columnPattern column to be matched + * @param valuePattern value to be matched. + * A table row will match the target if + * {@code ObjectUtil.equals(valuePattern, row.get(columnPattern.getName()))}. + * @return a filter which matches table rows which match the value in the + * row pattern + */ + public static RowFilter matchPattern(final Column columnPattern, final Object valuePattern) + { + return new RowFilter() { + @Override + public boolean matches(Map<String, Object> row) + { + return ObjectUtils.equals(valuePattern, row.get(columnPattern.getName())); + } + }; + } + + /** + * Creates a filter which inverts the sense of the given filter (rows which + * are matched by the given filter will not be matched by the returned + * filter, and vice versa). + * + * @param filter filter which to invert + * + * @return a RowFilter which matches rows not matched by the given filter + */ + public static RowFilter invert(final RowFilter filter) + { + return new RowFilter() { + @Override + public boolean matches(Map<String, Object> row) + { + return !filter.matches(row); + } + }; + } + + + /** + * Returns an iterable which filters the given iterable based on the given + * rowFilter. + * + * @param rowFilter the filter criteria, may be {@code null} + * @param iterable row iterable to filter + * + * @return a filtering iterable (or the given iterable if a {@code null} + * filter was given) + */ + public static Iterable<Map<String, Object>> apply( + RowFilter rowFilter, + Iterable<Map<String, Object>> iterable) + { + return((rowFilter != null) ? rowFilter.apply(iterable) : iterable); + } + + + /** + * Iterable which creates a filtered view of a another row iterable. + */ + private class FilterIterable implements Iterable<Map<String, Object>> + { + private final Iterable<Map<String, Object>> _iterable; + + private FilterIterable(Iterable<Map<String, Object>> iterable) + { + _iterable = iterable; + } + + + /** + * Returns an iterator which iterates through the rows of the underlying + * iterable, returning only rows for which the {@link RowFilter#matches} + * method returns {@code true} + */ + public Iterator<Map<String, Object>> iterator() + { + return new Iterator<Map<String, Object>>() { + private final Iterator<Map<String, Object>> _iter = + _iterable.iterator(); + private Map<String, Object> _next; + + public boolean hasNext() { + while(_iter.hasNext()) { + _next = _iter.next(); + if(RowFilter.this.matches(_next)) { + return true; + } + } + _next = null; + return false; + } + + public Map<String, Object> next() { + return _next; + } + + public void remove() { + throw new UnsupportedOperationException(); + } + + }; + } + + } + +} diff --git a/test/src/java/com/healthmarketscience/jackcess/BigIndexTest.java b/test/src/java/com/healthmarketscience/jackcess/BigIndexTest.java index 628fc05..212bae8 100644 --- a/test/src/java/com/healthmarketscience/jackcess/BigIndexTest.java +++ b/test/src/java/com/healthmarketscience/jackcess/BigIndexTest.java @@ -45,10 +45,6 @@ public class BigIndexTest extends TestCase { private String _oldBigIndexValue = null; - /** - * Creates a new <code>IndexTest</code> instance. - * - */ public BigIndexTest(String name) { super(name); } diff --git a/test/src/java/com/healthmarketscience/jackcess/IndexTest.java b/test/src/java/com/healthmarketscience/jackcess/IndexTest.java index 351e2d4..d746844 100644 --- a/test/src/java/com/healthmarketscience/jackcess/IndexTest.java +++ b/test/src/java/com/healthmarketscience/jackcess/IndexTest.java @@ -46,10 +46,6 @@ import static com.healthmarketscience.jackcess.DatabaseTest.*; */ public class IndexTest extends TestCase { - /** - * Creates a new <code>IndexTest</code> instance. - * - */ public IndexTest(String name) { super(name); } diff --git a/test/src/java/com/healthmarketscience/jackcess/RowFilterTest.java b/test/src/java/com/healthmarketscience/jackcess/RowFilterTest.java new file mode 100644 index 0000000..8d9b510 --- /dev/null +++ b/test/src/java/com/healthmarketscience/jackcess/RowFilterTest.java @@ -0,0 +1,112 @@ +/* +Copyright (c) 2008 Health Market Science, Inc. + +This library is free software; you can redistribute it and/or +modify it under the terms of the GNU Lesser General Public +License as published by the Free Software Foundation; either +version 2.1 of the License, or (at your option) any later version. + +This library 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 +Lesser General Public License for more details. + +You should have received a copy of the GNU Lesser General Public +License along with this library; if not, write to the Free Software +Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 +USA + +You can contact Health Market Science at info@healthmarketscience.com +or at the following address: + +Health Market Science +2700 Horizon Drive +Suite 200 +King of Prussia, PA 19406 +*/ + +package com.healthmarketscience.jackcess; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.Map; + +import junit.framework.TestCase; + +import static com.healthmarketscience.jackcess.DatabaseTest.*; + +/** + * @author James Ahlborn + */ +public class RowFilterTest extends TestCase +{ + private static final String ID_COL = "id"; + private static final String COL1 = "col1"; + private static final String COL2 = "col2"; + private static final String COL3 = "col3"; + + + public RowFilterTest(String name) { + super(name); + } + + @SuppressWarnings("unchecked") + public void testFilter() throws Exception + { + Map<String,Object> row0 = createExpectedRow(ID_COL, 0, COL1, "foo", COL2, 13, COL3, "bar"); + Map<String,Object> row1 = createExpectedRow(ID_COL, 1, COL1, "bar", COL2, 42, COL3, null); + Map<String,Object> row2 = createExpectedRow(ID_COL, 2, COL1, "foo", COL2, 55, COL3, "bar"); + Map<String,Object> row3 = createExpectedRow(ID_COL, 3, COL1, "baz", COL2, 42, COL3, "bar"); + Map<String,Object> row4 = createExpectedRow(ID_COL, 4, COL1, "foo", COL2, 13, COL3, null); + Map<String,Object> row5 = createExpectedRow(ID_COL, 5, COL1, "bla", COL2, 13, COL3, "bar"); + + + List<Map<String,Object>> rows = Arrays.asList(row0, row1, row2, row3, row4, row5); + + assertEquals(Arrays.asList(row0, row2, row4), + toList(RowFilter.matchPattern( + new ColumnBuilder(COL1, DataType.TEXT).toColumn(), + "foo").apply(rows))); + assertEquals(Arrays.asList(row1, row3, row5), + toList(RowFilter.invert( + RowFilter.matchPattern( + new ColumnBuilder(COL1, DataType.TEXT).toColumn(), + "foo")).apply(rows))); + + assertEquals(Arrays.asList(row0, row2, row4), + toList(RowFilter.matchPattern( + createExpectedRow(COL1, "foo")) + .apply(rows))); + assertEquals(Arrays.asList(row0, row2), + toList(RowFilter.matchPattern( + createExpectedRow(COL1, "foo", COL3, "bar")) + .apply(rows))); + assertEquals(Arrays.asList(row4), + toList(RowFilter.matchPattern( + createExpectedRow(COL1, "foo", COL3, null)) + .apply(rows))); + assertEquals(Arrays.asList(row0, row4, row5), + toList(RowFilter.matchPattern( + createExpectedRow(COL2, 13)) + .apply(rows))); + assertEquals(Arrays.asList(row1), + toList(RowFilter.matchPattern(row1) + .apply(rows))); + + assertEquals(rows, toList(RowFilter.apply(null, rows))); + assertEquals(Arrays.asList(row1), + toList(RowFilter.apply(RowFilter.matchPattern(row1), + rows))); + } + + private List<Map<String,Object>> toList(Iterable<Map<String,Object>> rows) + { + List<Map<String,Object>> rowList = new ArrayList<Map<String,Object>>(); + for(Map<String,Object> row : rows) { + rowList.add(row); + } + return rowList; + } + +} |