http://git-wip-us.apache.org/repos/asf/phoenix/blob/1e777b2b/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/NonTxIndexBuilderTest.java ---------------------------------------------------------------------- diff --git a/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/NonTxIndexBuilderTest.java b/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/NonTxIndexBuilderTest.java new file mode 100644 index 0000000..7c8575f --- /dev/null +++ b/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/NonTxIndexBuilderTest.java @@ -0,0 +1,341 @@ +/* + * 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 org.apache.phoenix.hbase.index.covered; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotEquals; +import static org.junit.Assert.assertTrue; + +import java.io.IOException; +import java.sql.Connection; +import java.sql.DriverManager; +import java.util.Arrays; +import java.util.Collection; +import java.util.Collections; +import java.util.List; +import java.util.NavigableMap; +import java.util.Properties; + +import org.apache.hadoop.conf.Configuration; +import org.apache.hadoop.hbase.Cell; +import org.apache.hadoop.hbase.CellUtil; +import org.apache.hadoop.hbase.HRegionInfo; +import org.apache.hadoop.hbase.KeyValue; +import org.apache.hadoop.hbase.KeyValue.Type; +import org.apache.hadoop.hbase.client.Mutation; +import org.apache.hadoop.hbase.client.Put; +import org.apache.hadoop.hbase.client.Scan; +import org.apache.hadoop.hbase.coprocessor.RegionCoprocessorEnvironment; +import org.apache.hadoop.hbase.io.ImmutableBytesWritable; +import org.apache.hadoop.hbase.io.TimeRange; +import org.apache.hadoop.hbase.regionserver.HRegion; +import org.apache.hadoop.hbase.regionserver.RegionScanner; +import org.apache.hadoop.hbase.util.Bytes; +import org.apache.hadoop.hbase.util.Pair; +import org.apache.phoenix.coprocessor.BaseRegionScanner; +import org.apache.phoenix.hbase.index.MultiMutation; +import org.apache.phoenix.hbase.index.covered.data.LocalTable; +import org.apache.phoenix.hbase.index.covered.update.ColumnTracker; +import org.apache.phoenix.hbase.index.util.GenericKeyValueBuilder; +import org.apache.phoenix.hbase.index.util.ImmutableBytesPtr; +import org.apache.phoenix.hbase.index.util.IndexManagementUtil; +import org.apache.phoenix.index.IndexMaintainer; +import org.apache.phoenix.index.PhoenixIndexCodec; +import org.apache.phoenix.index.PhoenixIndexMetaData; +import org.apache.phoenix.jdbc.PhoenixConnection; +import org.apache.phoenix.query.BaseConnectionlessQueryTest; +import org.apache.phoenix.query.QueryConstants; +import org.apache.phoenix.query.QueryServices; +import org.apache.phoenix.schema.PTable; +import org.apache.phoenix.schema.PTableKey; +import org.apache.phoenix.util.PropertiesUtil; +import org.apache.phoenix.util.TestUtil; +import org.junit.Before; +import org.junit.Test; +import org.mockito.Mockito; +import org.mockito.invocation.InvocationOnMock; +import org.mockito.stubbing.Answer; + +import com.google.common.base.Optional; +import com.google.common.base.Predicate; +import com.google.common.collect.Iterables; +import com.google.common.collect.Lists; + +public class NonTxIndexBuilderTest extends BaseConnectionlessQueryTest { + private static final String TEST_TABLE_STRING = "TEST_TABLE"; + private static final String TEST_TABLE_DDL = "CREATE TABLE IF NOT EXISTS " + + TEST_TABLE_STRING + " (\n" + + " ORGANIZATION_ID CHAR(4) NOT NULL,\n" + + " ENTITY_ID CHAR(7) NOT NULL,\n" + + " SCORE INTEGER,\n" + + " LAST_UPDATE_TIME TIMESTAMP\n" + + " CONSTRAINT TEST_TABLE_PK PRIMARY KEY (\n" + + " ORGANIZATION_ID,\n" + + " ENTITY_ID\n" + + " )\n" + + ") VERSIONS=1, MULTI_TENANT=TRUE"; + private static final String TEST_TABLE_INDEX_STRING = "TEST_TABLE_SCORE"; + private static final String TEST_TABLE_INDEX_DDL = "CREATE INDEX IF NOT EXISTS " + + TEST_TABLE_INDEX_STRING + + " ON " + TEST_TABLE_STRING + " (SCORE DESC, ENTITY_ID DESC)"; + private static final byte[] ROW = Bytes.toBytes("org1entity1"); //length 4 + 7 (see ddl) + private static final String FAM_STRING = QueryConstants.DEFAULT_COLUMN_FAMILY; + private static final byte[] FAM = Bytes.toBytes(FAM_STRING); + private static final byte[] INDEXED_QUALIFIER = Bytes.toBytes("SCORE"); + private static final byte[] VALUE_1 = Bytes.toBytes(111); + private static final byte[] VALUE_2 = Bytes.toBytes(222); + private static final byte[] VALUE_3 = Bytes.toBytes(333); + private static final byte[] VALUE_4 = Bytes.toBytes(444); + private static final byte PUT_TYPE = KeyValue.Type.Put.getCode(); + + private NonTxIndexBuilder indexBuilder; + private PhoenixIndexMetaData mockIndexMetaData; + // Put your current row state in here - the index builder will read from this in LocalTable + // to determine whether the index has changed. + // Whatever we return here should match the table DDL (e.g. length of column value) + private List<Cell> currentRowCells; + + /** + * Test setup so that {@link NonTxIndexBuilder#getIndexUpdate(Mutation, IndexMetaData)} can be + * called, where any read requests to + * {@link LocalTable#getCurrentRowState(Mutation, Collection, boolean)} are read from our test + * field 'currentRowCells' + */ + @Before + public void setup() throws Exception { + RegionCoprocessorEnvironment env = Mockito.mock(RegionCoprocessorEnvironment.class); + Configuration conf = new Configuration(false); + conf.set(NonTxIndexBuilder.CODEC_CLASS_NAME_KEY, PhoenixIndexCodec.class.getName()); + Mockito.when(env.getConfiguration()).thenReturn(conf); + + // the following is used by LocalTable#getCurrentRowState() + HRegion mockRegion = Mockito.mock(HRegion.class); + Mockito.when(env.getRegion()).thenReturn(mockRegion); + + Mockito.when(mockRegion.getScanner(Mockito.any(Scan.class))) + .thenAnswer(new Answer<RegionScanner>() { + @Override + public RegionScanner answer(InvocationOnMock invocation) throws Throwable { + Scan sArg = (Scan) invocation.getArguments()[0]; + TimeRange timeRange = sArg.getTimeRange(); + return getMockTimeRangeRegionScanner(timeRange); + } + }); + + // the following is called by PhoenixIndexCodec#getIndexUpserts() , getIndexDeletes() + HRegionInfo mockRegionInfo = Mockito.mock(HRegionInfo.class); + Mockito.when(mockRegion.getRegionInfo()).thenReturn(mockRegionInfo); + Mockito.when(mockRegionInfo.getStartKey()).thenReturn(Bytes.toBytes("a")); + Mockito.when(mockRegionInfo.getEndKey()).thenReturn(Bytes.toBytes("z")); + + mockIndexMetaData = Mockito.mock(PhoenixIndexMetaData.class); + Mockito.when(mockIndexMetaData.isImmutableRows()).thenReturn(false); + Mockito.when(mockIndexMetaData.getIndexMaintainers()) + .thenReturn(Collections.singletonList(getTestIndexMaintainer())); + + indexBuilder = new NonTxIndexBuilder(); + indexBuilder.setup(env); + } + + // returns a RegionScanner which filters currentRowCells using the given TimeRange. + // This is called from LocalTable#getCurrentRowState() + // If testIndexMetaData.ignoreNewerMutations() is not set, default TimeRange is 0 to + // Long.MAX_VALUE + private RegionScanner getMockTimeRangeRegionScanner(final TimeRange timeRange) { + return new BaseRegionScanner(Mockito.mock(RegionScanner.class)) { + @Override + public boolean next(List<Cell> results) throws IOException { + for (Cell cell : currentRowCells) { + if (cell.getTimestamp() >= timeRange.getMin() + && cell.getTimestamp() < timeRange.getMax()) { + results.add(cell); + } + } + return false; // indicate no more results + } + }; + } + + private IndexMaintainer getTestIndexMaintainer() throws Exception { + Properties props = PropertiesUtil.deepCopy(TestUtil.TEST_PROPERTIES); + // disable column encoding, makes debugging easier + props.put(QueryServices.DEFAULT_COLUMN_ENCODED_BYTES_ATRRIB, "0"); + Connection conn = DriverManager.getConnection(getUrl(), props); + try { + conn.setAutoCommit(true); + conn.createStatement().execute(TEST_TABLE_DDL); + conn.createStatement().execute(TEST_TABLE_INDEX_DDL); + PhoenixConnection pconn = conn.unwrap(PhoenixConnection.class); + PTable table = pconn.getTable(new PTableKey(pconn.getTenantId(), TEST_TABLE_STRING)); + ImmutableBytesWritable ptr = new ImmutableBytesWritable(); + table.getIndexMaintainers(ptr, pconn); + List<IndexMaintainer> indexMaintainerList = + IndexMaintainer.deserialize(ptr, GenericKeyValueBuilder.INSTANCE, true); + assertEquals(1, indexMaintainerList.size()); + IndexMaintainer indexMaintainer = indexMaintainerList.get(0); + return indexMaintainer; + } finally { + conn.close(); + } + } + + /** + * Tests that updating an indexed column results in a DeleteFamily (prior index cell) and a Put + * (new index cell) + */ + @Test + public void testGetMutableIndexUpdate() throws IOException { + setCurrentRowState(FAM, INDEXED_QUALIFIER, 1, VALUE_1); + + // update ts and value + Put put = new Put(ROW); + put.addImmutable(FAM, INDEXED_QUALIFIER, 2, VALUE_2); + MultiMutation mutation = new MultiMutation(new ImmutableBytesPtr(ROW)); + mutation.addAll(put); + + Collection<Pair<Mutation, byte[]>> indexUpdates = + indexBuilder.getIndexUpdate(mutation, mockIndexMetaData); + assertEquals(2, indexUpdates.size()); + assertContains(indexUpdates, 2, ROW, KeyValue.Type.DeleteFamily, FAM, + new byte[0] /* qual not needed */, 2); + assertContains(indexUpdates, ColumnTracker.NO_NEWER_PRIMARY_TABLE_ENTRY_TIMESTAMP, ROW, + KeyValue.Type.Put, FAM, QueryConstants.EMPTY_COLUMN_BYTES, 2); + } + + /** + * Tests a partial rebuild of a row with multiple versions. 3 versions of the row in data table, + * and we rebuild the index starting from time t=2 + * + * There should be one index row version per data row version. + */ + @Test + public void testRebuildMultipleVersionRow() throws IOException { + // when doing a rebuild, we are replaying mutations so we want to ignore newer mutations + // see LocalTable#getCurrentRowState() + Mockito.when(mockIndexMetaData.ignoreNewerMutations()).thenReturn(true); + + // the current row state has 3 versions, but if we rebuild as of t=2, scanner in LocalTable + // should only return first + Cell currentCell1 = CellUtil.createCell(ROW, FAM, INDEXED_QUALIFIER, 1, PUT_TYPE, VALUE_1); + Cell currentCell2 = CellUtil.createCell(ROW, FAM, INDEXED_QUALIFIER, 2, PUT_TYPE, VALUE_2); + Cell currentCell3 = CellUtil.createCell(ROW, FAM, INDEXED_QUALIFIER, 3, PUT_TYPE, VALUE_3); + Cell currentCell4 = CellUtil.createCell(ROW, FAM, INDEXED_QUALIFIER, 4, PUT_TYPE, VALUE_4); + setCurrentRowState(Arrays.asList(currentCell4, currentCell3, currentCell2, currentCell1)); + + // rebuilder replays mutations starting from t=2 + MultiMutation mutation = new MultiMutation(new ImmutableBytesPtr(ROW)); + Put put = new Put(ROW); + put.addImmutable(FAM, INDEXED_QUALIFIER, 4, VALUE_4); + mutation.addAll(put); + put = new Put(ROW); + put.addImmutable(FAM, INDEXED_QUALIFIER, 3, VALUE_3); + mutation.addAll(put); + put = new Put(ROW); + put.addImmutable(FAM, INDEXED_QUALIFIER, 2, VALUE_2); + mutation.addAll(put); + + Collection<Pair<Mutation, byte[]>> indexUpdates = Lists.newArrayList(); + for (Mutation m : IndexManagementUtil.flattenMutationsByTimestamp(Collections.singletonList(mutation))) { + indexUpdates.addAll(indexBuilder.getIndexUpdate(m, mockIndexMetaData)); + } + // 3 puts and 3 deletes (one to hide existing index row for VALUE_1, and two to hide index + // rows for VALUE_2, VALUE_3) + assertEquals(6, indexUpdates.size()); + + assertContains(indexUpdates, 2, ROW, KeyValue.Type.DeleteFamily, FAM, + new byte[0] /* qual not needed */, 2); + assertContains(indexUpdates, ColumnTracker.NO_NEWER_PRIMARY_TABLE_ENTRY_TIMESTAMP, ROW, + KeyValue.Type.Put, FAM, QueryConstants.EMPTY_COLUMN_BYTES, 2); + assertContains(indexUpdates, 3, ROW, KeyValue.Type.DeleteFamily, FAM, + new byte[0] /* qual not needed */, 3); + assertContains(indexUpdates, ColumnTracker.NO_NEWER_PRIMARY_TABLE_ENTRY_TIMESTAMP, ROW, + KeyValue.Type.Put, FAM, QueryConstants.EMPTY_COLUMN_BYTES, 3); + assertContains(indexUpdates, 4, ROW, KeyValue.Type.DeleteFamily, FAM, + new byte[0] /* qual not needed */, 4); + assertContains(indexUpdates, ColumnTracker.NO_NEWER_PRIMARY_TABLE_ENTRY_TIMESTAMP, ROW, + KeyValue.Type.Put, FAM, QueryConstants.EMPTY_COLUMN_BYTES, 4); + } + + /** + * Tests getting an index update for a mutation with 200 versions Before, the issue PHOENIX-3807 + * was causing this test to take >90 seconds, so here we set a timeout of 5 seconds + */ + @Test(timeout = 10000) + public void testManyVersions() throws IOException { + // when doing a rebuild, we are replaying mutations so we want to ignore newer mutations + // see LocalTable#getCurrentRowState() + Mockito.when(mockIndexMetaData.ignoreNewerMutations()).thenReturn(true); + MultiMutation mutation = getMultipleVersionMutation(200); + currentRowCells = mutation.getFamilyCellMap().get(FAM); + + Collection<Pair<Mutation, byte[]>> indexUpdates = Lists.newArrayList(); + for (Mutation m : IndexManagementUtil.flattenMutationsByTimestamp(Collections.singletonList(mutation))) { + indexUpdates.addAll(indexBuilder.getIndexUpdate(m, mockIndexMetaData)); + } + assertNotEquals(0, indexUpdates.size()); + } + + // Assert that the given collection of indexUpdates contains the given cell + private void assertContains(Collection<Pair<Mutation, byte[]>> indexUpdates, + final long mutationTs, final byte[] row, final Type cellType, final byte[] fam, + final byte[] qual, final long cellTs) { + Predicate<Pair<Mutation, byte[]>> hasCellPredicate = + new Predicate<Pair<Mutation, byte[]>>() { + @Override + public boolean apply(Pair<Mutation, byte[]> input) { + assertEquals(TEST_TABLE_INDEX_STRING, Bytes.toString(input.getSecond())); + Mutation mutation = input.getFirst(); + if (mutationTs == mutation.getTimeStamp()) { + NavigableMap<byte[], List<Cell>> familyCellMap = + mutation.getFamilyCellMap(); + Cell updateCell = familyCellMap.get(fam).get(0); + if (cellType == KeyValue.Type.codeToType(updateCell.getTypeByte()) + && Bytes.compareTo(fam, CellUtil.cloneFamily(updateCell)) == 0 + && Bytes.compareTo(qual, + CellUtil.cloneQualifier(updateCell)) == 0 + && cellTs == updateCell.getTimestamp()) { + return true; + } + } + return false; + } + }; + Optional<Pair<Mutation, byte[]>> tryFind = + Iterables.tryFind(indexUpdates, hasCellPredicate); + assertTrue(tryFind.isPresent()); + } + + private void setCurrentRowState(byte[] fam2, byte[] indexedQualifier, int i, byte[] value1) { + Cell cell = CellUtil.createCell(ROW, FAM, INDEXED_QUALIFIER, 1, PUT_TYPE, VALUE_1); + currentRowCells = Collections.singletonList(cell); + } + + private void setCurrentRowState(List<Cell> cells) { + currentRowCells = cells; + } + + private MultiMutation getMultipleVersionMutation(int versions) { + MultiMutation mutation = new MultiMutation(new ImmutableBytesPtr(ROW)); + for (int i = versions - 1; i >= 0; i--) { + Put put = new Put(ROW); + put.addImmutable(FAM, INDEXED_QUALIFIER, i, Bytes.toBytes(i)); + mutation.addAll(put); + } + return mutation; + } +}
http://git-wip-us.apache.org/repos/asf/phoenix/blob/1e777b2b/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestCoveredColumns.java ---------------------------------------------------------------------- diff --git a/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestCoveredColumns.java b/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestCoveredColumns.java deleted file mode 100644 index dd5223b..0000000 --- a/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestCoveredColumns.java +++ /dev/null @@ -1,46 +0,0 @@ -/* - * 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 org.apache.phoenix.hbase.index.covered; - -import static org.junit.Assert.assertEquals; - -import java.util.Arrays; - -import org.apache.hadoop.hbase.util.Bytes; -import org.junit.Test; - -import org.apache.phoenix.hbase.index.covered.CoveredColumns; -import org.apache.phoenix.hbase.index.covered.update.ColumnReference; - -public class TestCoveredColumns { - - private static final byte[] fam = Bytes.toBytes("fam"); - private static final byte[] qual = Bytes.toBytes("qual"); - - @Test - public void testCovering() { - ColumnReference ref = new ColumnReference(fam, qual); - CoveredColumns columns = new CoveredColumns(); - assertEquals("Should have only found a single column to cover", 1, columns - .findNonCoveredColumns(Arrays.asList(ref)).size()); - - columns.addColumn(ref); - assertEquals("Shouldn't have any columns to cover", 0, - columns.findNonCoveredColumns(Arrays.asList(ref)).size()); - } -} \ No newline at end of file http://git-wip-us.apache.org/repos/asf/phoenix/blob/1e777b2b/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestLocalTableState.java ---------------------------------------------------------------------- diff --git a/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestLocalTableState.java b/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestLocalTableState.java deleted file mode 100644 index d58012f..0000000 --- a/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestLocalTableState.java +++ /dev/null @@ -1,293 +0,0 @@ -/* - * 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 org.apache.phoenix.hbase.index.covered; - -import static org.junit.Assert.assertEquals; - -import java.util.Arrays; -import java.util.List; - -import org.apache.hadoop.conf.Configuration; -import org.apache.hadoop.hbase.KeyValue; -import org.apache.hadoop.hbase.KeyValue.Type; -import org.apache.hadoop.hbase.KeyValueUtil; -import org.apache.hadoop.hbase.client.Put; -import org.apache.hadoop.hbase.client.Scan; -import org.apache.hadoop.hbase.coprocessor.RegionCoprocessorEnvironment; -import org.apache.hadoop.hbase.regionserver.HRegion; -import org.apache.hadoop.hbase.regionserver.RegionScanner; -import org.apache.hadoop.hbase.util.Bytes; -import org.apache.hadoop.hbase.util.Pair; -import org.apache.phoenix.hbase.index.covered.data.LocalHBaseState; -import org.apache.phoenix.hbase.index.covered.data.LocalTable; -import org.apache.phoenix.hbase.index.covered.update.ColumnReference; -import org.apache.phoenix.hbase.index.scanner.Scanner; -import org.apache.phoenix.hbase.index.scanner.ScannerBuilder.CoveredDeleteScanner; -import org.junit.Test; -import org.mockito.Mockito; -import org.mockito.invocation.InvocationOnMock; -import org.mockito.stubbing.Answer; - -/** - * - */ -public class TestLocalTableState { - - private static final byte[] row = Bytes.toBytes("row"); - private static final byte[] fam = Bytes.toBytes("fam"); - private static final byte[] qual = Bytes.toBytes("qual"); - private static final byte[] val = Bytes.toBytes("val"); - private static final long ts = 10; - private static final IndexMetaData indexMetaData = new IndexMetaData() { - - @Override - public boolean isImmutableRows() { - return false; - } - - @Override - public boolean ignoreNewerMutations() { - return false; - } - - }; - - @SuppressWarnings("unchecked") - @Test - public void testCorrectOrderingWithLazyLoadingColumns() throws Exception { - Put m = new Put(row); - m.add(fam, qual, ts, val); - // setup mocks - Configuration conf = new Configuration(false); - RegionCoprocessorEnvironment env = Mockito.mock(RegionCoprocessorEnvironment.class); - Mockito.when(env.getConfiguration()).thenReturn(conf); - - HRegion region = Mockito.mock(HRegion.class); - Mockito.when(env.getRegion()).thenReturn(region); - RegionScanner scanner = Mockito.mock(RegionScanner.class); - Mockito.when(region.getScanner(Mockito.any(Scan.class))).thenReturn(scanner); - final byte[] stored = Bytes.toBytes("stored-value"); - Mockito.when(scanner.next(Mockito.any(List.class))).thenAnswer(new Answer<Boolean>() { - @Override - public Boolean answer(InvocationOnMock invocation) throws Throwable { - List<KeyValue> list = (List<KeyValue>) invocation.getArguments()[0]; - KeyValue kv = new KeyValue(row, fam, qual, ts, Type.Put, stored); - kv.setMvccVersion(0); - list.add(kv); - return false; - } - }); - - - LocalHBaseState state = new LocalTable(env); - LocalTableState table = new LocalTableState(env, state, m); - //add the kvs from the mutation - table.addPendingUpdates(KeyValueUtil.ensureKeyValues(m.get(fam, qual))); - - // setup the lookup - ColumnReference col = new ColumnReference(fam, qual); - table.setCurrentTimestamp(ts); - //check that our value still shows up first on scan, even though this is a lazy load - Pair<CoveredDeleteScanner, IndexUpdate> p = table.getIndexedColumnsTableState(Arrays.asList(col), false, false, indexMetaData); - Scanner s = p.getFirst(); - assertEquals("Didn't get the pending mutation's value first", m.get(fam, qual).get(0), s.next()); - } - - public static final class ScannerCreatedException extends RuntimeException { - ScannerCreatedException(String msg) { - super(msg); - } - } - - @Test(expected = ScannerCreatedException.class) - public void testScannerForMutableRows() throws Exception { - IndexMetaData indexMetaData = new IndexMetaData() { - - @Override - public boolean isImmutableRows() { - return false; - } - - @Override - public boolean ignoreNewerMutations() { - return false; - } - - }; - Put m = new Put(row); - m.add(fam, qual, ts, val); - // setup mocks - Configuration conf = new Configuration(false); - RegionCoprocessorEnvironment env = Mockito.mock(RegionCoprocessorEnvironment.class); - Mockito.when(env.getConfiguration()).thenReturn(conf); - - HRegion region = Mockito.mock(HRegion.class); - Mockito.when(env.getRegion()).thenReturn(region); - Mockito.when(region.getScanner(Mockito.any(Scan.class))).thenThrow(new ScannerCreatedException("Should not open scanner when data is immutable")); - - LocalHBaseState state = new LocalTable(env); - LocalTableState table = new LocalTableState(env, state, m); - //add the kvs from the mutation - table.addPendingUpdates(KeyValueUtil.ensureKeyValues(m.get(fam, qual))); - - // setup the lookup - ColumnReference col = new ColumnReference(fam, qual); - table.setCurrentTimestamp(ts); - table.getIndexedColumnsTableState(Arrays.asList(col), false, false, indexMetaData); - } - - @Test - public void testNoScannerForImmutableRows() throws Exception { - IndexMetaData indexMetaData = new IndexMetaData() { - - @Override - public boolean isImmutableRows() { - return true; - } - - @Override - public boolean ignoreNewerMutations() { - return false; - } - - }; - Put m = new Put(row); - m.add(fam, qual, ts, val); - // setup mocks - Configuration conf = new Configuration(false); - RegionCoprocessorEnvironment env = Mockito.mock(RegionCoprocessorEnvironment.class); - Mockito.when(env.getConfiguration()).thenReturn(conf); - - HRegion region = Mockito.mock(HRegion.class); - Mockito.when(env.getRegion()).thenReturn(region); - Mockito.when(region.getScanner(Mockito.any(Scan.class))).thenThrow(new ScannerCreatedException("Should not open scanner when data is immutable")); - - LocalHBaseState state = new LocalTable(env); - LocalTableState table = new LocalTableState(env, state, m); - //add the kvs from the mutation - table.addPendingUpdates(KeyValueUtil.ensureKeyValues(m.get(fam, qual))); - - // setup the lookup - ColumnReference col = new ColumnReference(fam, qual); - table.setCurrentTimestamp(ts); - //check that our value still shows up first on scan, even though this is a lazy load - Pair<CoveredDeleteScanner, IndexUpdate> p = table.getIndexedColumnsTableState(Arrays.asList(col), false, false, indexMetaData); - Scanner s = p.getFirst(); - assertEquals("Didn't get the pending mutation's value first", m.get(fam, qual).get(0), s.next()); - } - - /** - * Test that we correctly rollback the state of keyvalue - * @throws Exception - */ - @Test - @SuppressWarnings("unchecked") - public void testCorrectRollback() throws Exception { - Put m = new Put(row); - m.add(fam, qual, ts, val); - // setup mocks - RegionCoprocessorEnvironment env = Mockito.mock(RegionCoprocessorEnvironment.class); - - HRegion region = Mockito.mock(HRegion.class); - Mockito.when(env.getRegion()).thenReturn(region); - RegionScanner scanner = Mockito.mock(RegionScanner.class); - Mockito.when(region.getScanner(Mockito.any(Scan.class))).thenReturn(scanner); - final byte[] stored = Bytes.toBytes("stored-value"); - final KeyValue storedKv = new KeyValue(row, fam, qual, ts, Type.Put, stored); - storedKv.setMvccVersion(2); - Mockito.when(scanner.next(Mockito.any(List.class))).thenAnswer(new Answer<Boolean>() { - @Override - public Boolean answer(InvocationOnMock invocation) throws Throwable { - List<KeyValue> list = (List<KeyValue>) invocation.getArguments()[0]; - - list.add(storedKv); - return false; - } - }); - LocalHBaseState state = new LocalTable(env); - LocalTableState table = new LocalTableState(env, state, m); - // add the kvs from the mutation - KeyValue kv = KeyValueUtil.ensureKeyValue(m.get(fam, qual).get(0)); - kv.setMvccVersion(0); - table.addPendingUpdates(kv); - - // setup the lookup - ColumnReference col = new ColumnReference(fam, qual); - table.setCurrentTimestamp(ts); - // check that the value is there - Pair<CoveredDeleteScanner, IndexUpdate> p = table.getIndexedColumnsTableState(Arrays.asList(col), false, false, indexMetaData); - Scanner s = p.getFirst(); - assertEquals("Didn't get the pending mutation's value first", kv, s.next()); - - // rollback that value - table.rollback(Arrays.asList(kv)); - p = table.getIndexedColumnsTableState(Arrays.asList(col), false, false, indexMetaData); - s = p.getFirst(); - assertEquals("Didn't correctly rollback the row - still found it!", null, s.next()); - Mockito.verify(env, Mockito.times(1)).getRegion(); - Mockito.verify(region, Mockito.times(1)).getScanner(Mockito.any(Scan.class)); - } - - @SuppressWarnings("unchecked") - @Test - public void testOnlyLoadsRequestedColumns() throws Exception { - // setup mocks - RegionCoprocessorEnvironment env = Mockito.mock(RegionCoprocessorEnvironment.class); - - HRegion region = Mockito.mock(HRegion.class); - Mockito.when(env.getRegion()).thenReturn(region); - RegionScanner scanner = Mockito.mock(RegionScanner.class); - Mockito.when(region.getScanner(Mockito.any(Scan.class))).thenReturn(scanner); - final KeyValue storedKv = - new KeyValue(row, fam, qual, ts, Type.Put, Bytes.toBytes("stored-value")); - storedKv.setMvccVersion(2); - Mockito.when(scanner.next(Mockito.any(List.class))).thenAnswer(new Answer<Boolean>() { - @Override - public Boolean answer(InvocationOnMock invocation) throws Throwable { - List<KeyValue> list = (List<KeyValue>) invocation.getArguments()[0]; - - list.add(storedKv); - return false; - } - }); - LocalHBaseState state = new LocalTable(env); - Put pendingUpdate = new Put(row); - pendingUpdate.add(fam, qual, ts, val); - LocalTableState table = new LocalTableState(env, state, pendingUpdate); - - // do the lookup for the given column - ColumnReference col = new ColumnReference(fam, qual); - table.setCurrentTimestamp(ts); - // check that the value is there - Pair<CoveredDeleteScanner, IndexUpdate> p = table.getIndexedColumnsTableState(Arrays.asList(col), false, false, indexMetaData); - Scanner s = p.getFirst(); - // make sure it read the table the one time - assertEquals("Didn't get the stored keyvalue!", storedKv, s.next()); - - // on the second lookup it shouldn't access the underlying table again - the cached columns - // should know they are done - p = table.getIndexedColumnsTableState(Arrays.asList(col), false, false, indexMetaData); - s = p.getFirst(); - assertEquals("Lost already loaded update!", storedKv, s.next()); - Mockito.verify(env, Mockito.times(1)).getRegion(); - Mockito.verify(region, Mockito.times(1)).getScanner(Mockito.any(Scan.class)); - } - - // TODO add test here for making sure multiple column references with the same column family don't - // cause an infinite loop -} http://git-wip-us.apache.org/repos/asf/phoenix/blob/1e777b2b/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestNonTxIndexBuilder.java ---------------------------------------------------------------------- diff --git a/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestNonTxIndexBuilder.java b/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestNonTxIndexBuilder.java deleted file mode 100644 index e01c8f6..0000000 --- a/phoenix-core/src/test/java/org/apache/phoenix/hbase/index/covered/TestNonTxIndexBuilder.java +++ /dev/null @@ -1,335 +0,0 @@ -/* - * 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 org.apache.phoenix.hbase.index.covered; - -import static org.junit.Assert.assertEquals; -import static org.junit.Assert.assertNotEquals; -import static org.junit.Assert.assertTrue; - -import java.io.IOException; -import java.sql.Connection; -import java.sql.DriverManager; -import java.util.Arrays; -import java.util.Collection; -import java.util.Collections; -import java.util.List; -import java.util.NavigableMap; -import java.util.Properties; - -import org.apache.hadoop.conf.Configuration; -import org.apache.hadoop.hbase.Cell; -import org.apache.hadoop.hbase.CellUtil; -import org.apache.hadoop.hbase.HRegionInfo; -import org.apache.hadoop.hbase.KeyValue; -import org.apache.hadoop.hbase.KeyValue.Type; -import org.apache.hadoop.hbase.client.Mutation; -import org.apache.hadoop.hbase.client.Put; -import org.apache.hadoop.hbase.client.Scan; -import org.apache.hadoop.hbase.coprocessor.RegionCoprocessorEnvironment; -import org.apache.hadoop.hbase.io.ImmutableBytesWritable; -import org.apache.hadoop.hbase.io.TimeRange; -import org.apache.hadoop.hbase.regionserver.HRegion; -import org.apache.hadoop.hbase.regionserver.RegionScanner; -import org.apache.hadoop.hbase.util.Bytes; -import org.apache.hadoop.hbase.util.Pair; -import org.apache.phoenix.coprocessor.BaseRegionScanner; -import org.apache.phoenix.hbase.index.MultiMutation; -import org.apache.phoenix.hbase.index.covered.data.LocalTable; -import org.apache.phoenix.hbase.index.covered.update.ColumnTracker; -import org.apache.phoenix.hbase.index.util.GenericKeyValueBuilder; -import org.apache.phoenix.hbase.index.util.ImmutableBytesPtr; -import org.apache.phoenix.index.IndexMaintainer; -import org.apache.phoenix.index.PhoenixIndexCodec; -import org.apache.phoenix.index.PhoenixIndexMetaData; -import org.apache.phoenix.jdbc.PhoenixConnection; -import org.apache.phoenix.query.BaseConnectionlessQueryTest; -import org.apache.phoenix.query.QueryConstants; -import org.apache.phoenix.query.QueryServices; -import org.apache.phoenix.schema.PTable; -import org.apache.phoenix.schema.PTableKey; -import org.apache.phoenix.util.PropertiesUtil; -import org.apache.phoenix.util.TestUtil; -import org.junit.Before; -import org.junit.Test; -import org.mockito.Mockito; -import org.mockito.invocation.InvocationOnMock; -import org.mockito.stubbing.Answer; - -import com.google.common.base.Optional; -import com.google.common.base.Predicate; -import com.google.common.collect.Iterables; - -public class TestNonTxIndexBuilder extends BaseConnectionlessQueryTest { - private static final String TEST_TABLE_STRING = "TEST_TABLE"; - private static final String TEST_TABLE_DDL = "CREATE TABLE IF NOT EXISTS " + - TEST_TABLE_STRING + " (\n" + - " ORGANIZATION_ID CHAR(4) NOT NULL,\n" + - " ENTITY_ID CHAR(7) NOT NULL,\n" + - " SCORE INTEGER,\n" + - " LAST_UPDATE_TIME TIMESTAMP\n" + - " CONSTRAINT TEST_TABLE_PK PRIMARY KEY (\n" + - " ORGANIZATION_ID,\n" + - " ENTITY_ID\n" + - " )\n" + - ") VERSIONS=1, MULTI_TENANT=TRUE"; - private static final String TEST_TABLE_INDEX_STRING = "TEST_TABLE_SCORE"; - private static final String TEST_TABLE_INDEX_DDL = "CREATE INDEX IF NOT EXISTS " + - TEST_TABLE_INDEX_STRING - + " ON " + TEST_TABLE_STRING + " (SCORE DESC, ENTITY_ID DESC)"; - private static final byte[] ROW = Bytes.toBytes("org1entity1"); //length 4 + 7 (see ddl) - private static final String FAM_STRING = QueryConstants.DEFAULT_COLUMN_FAMILY; - private static final byte[] FAM = Bytes.toBytes(FAM_STRING); - private static final byte[] INDEXED_QUALIFIER = Bytes.toBytes("SCORE"); - private static final byte[] VALUE_1 = Bytes.toBytes(111); - private static final byte[] VALUE_2 = Bytes.toBytes(222); - private static final byte[] VALUE_3 = Bytes.toBytes(333); - private static final byte[] VALUE_4 = Bytes.toBytes(444); - private static final byte PUT_TYPE = KeyValue.Type.Put.getCode(); - - private NonTxIndexBuilder indexBuilder; - private PhoenixIndexMetaData mockIndexMetaData; - // Put your current row state in here - the index builder will read from this in LocalTable - // to determine whether the index has changed. - // Whatever we return here should match the table DDL (e.g. length of column value) - private List<Cell> currentRowCells; - - /** - * Test setup so that {@link NonTxIndexBuilder#getIndexUpdate(Mutation, IndexMetaData)} can be - * called, where any read requests to - * {@link LocalTable#getCurrentRowState(Mutation, Collection, boolean)} are read from our test - * field 'currentRowCells' - */ - @Before - public void setup() throws Exception { - RegionCoprocessorEnvironment env = Mockito.mock(RegionCoprocessorEnvironment.class); - Configuration conf = new Configuration(false); - conf.set(NonTxIndexBuilder.CODEC_CLASS_NAME_KEY, PhoenixIndexCodec.class.getName()); - Mockito.when(env.getConfiguration()).thenReturn(conf); - - // the following is used by LocalTable#getCurrentRowState() - HRegion mockRegion = Mockito.mock(HRegion.class); - Mockito.when(env.getRegion()).thenReturn(mockRegion); - - Mockito.when(mockRegion.getScanner(Mockito.any(Scan.class))) - .thenAnswer(new Answer<RegionScanner>() { - @Override - public RegionScanner answer(InvocationOnMock invocation) throws Throwable { - Scan sArg = (Scan) invocation.getArguments()[0]; - TimeRange timeRange = sArg.getTimeRange(); - return getMockTimeRangeRegionScanner(timeRange); - } - }); - - // the following is called by PhoenixIndexCodec#getIndexUpserts() , getIndexDeletes() - HRegionInfo mockRegionInfo = Mockito.mock(HRegionInfo.class); - Mockito.when(mockRegion.getRegionInfo()).thenReturn(mockRegionInfo); - Mockito.when(mockRegionInfo.getStartKey()).thenReturn(Bytes.toBytes("a")); - Mockito.when(mockRegionInfo.getEndKey()).thenReturn(Bytes.toBytes("z")); - - mockIndexMetaData = Mockito.mock(PhoenixIndexMetaData.class); - Mockito.when(mockIndexMetaData.isImmutableRows()).thenReturn(false); - Mockito.when(mockIndexMetaData.getIndexMaintainers()) - .thenReturn(Collections.singletonList(getTestIndexMaintainer())); - - indexBuilder = new NonTxIndexBuilder(); - indexBuilder.setup(env); - } - - // returns a RegionScanner which filters currentRowCells using the given TimeRange. - // This is called from LocalTable#getCurrentRowState() - // If testIndexMetaData.ignoreNewerMutations() is not set, default TimeRange is 0 to - // Long.MAX_VALUE - private RegionScanner getMockTimeRangeRegionScanner(final TimeRange timeRange) { - return new BaseRegionScanner(Mockito.mock(RegionScanner.class)) { - @Override - public boolean next(List<Cell> results) throws IOException { - for (Cell cell : currentRowCells) { - if (cell.getTimestamp() >= timeRange.getMin() - && cell.getTimestamp() < timeRange.getMax()) { - results.add(cell); - } - } - return false; // indicate no more results - } - }; - } - - private IndexMaintainer getTestIndexMaintainer() throws Exception { - Properties props = PropertiesUtil.deepCopy(TestUtil.TEST_PROPERTIES); - // disable column encoding, makes debugging easier - props.put(QueryServices.DEFAULT_COLUMN_ENCODED_BYTES_ATRRIB, "0"); - Connection conn = DriverManager.getConnection(getUrl(), props); - try { - conn.setAutoCommit(true); - conn.createStatement().execute(TEST_TABLE_DDL); - conn.createStatement().execute(TEST_TABLE_INDEX_DDL); - PhoenixConnection pconn = conn.unwrap(PhoenixConnection.class); - PTable table = pconn.getTable(new PTableKey(pconn.getTenantId(), TEST_TABLE_STRING)); - ImmutableBytesWritable ptr = new ImmutableBytesWritable(); - table.getIndexMaintainers(ptr, pconn); - List<IndexMaintainer> indexMaintainerList = - IndexMaintainer.deserialize(ptr, GenericKeyValueBuilder.INSTANCE, true); - assertEquals(1, indexMaintainerList.size()); - IndexMaintainer indexMaintainer = indexMaintainerList.get(0); - return indexMaintainer; - } finally { - conn.close(); - } - } - - /** - * Tests that updating an indexed column results in a DeleteFamily (prior index cell) and a Put - * (new index cell) - */ - @Test - public void testGetMutableIndexUpdate() throws IOException { - setCurrentRowState(FAM, INDEXED_QUALIFIER, 1, VALUE_1); - - // update ts and value - Put put = new Put(ROW); - put.addImmutable(FAM, INDEXED_QUALIFIER, 2, VALUE_2); - MultiMutation mutation = new MultiMutation(new ImmutableBytesPtr(ROW)); - mutation.addAll(put); - - Collection<Pair<Mutation, byte[]>> indexUpdates = - indexBuilder.getIndexUpdate(mutation, mockIndexMetaData); - assertEquals(2, indexUpdates.size()); - assertContains(indexUpdates, 2, ROW, KeyValue.Type.DeleteFamily, FAM, - new byte[0] /* qual not needed */, 2); - assertContains(indexUpdates, ColumnTracker.NO_NEWER_PRIMARY_TABLE_ENTRY_TIMESTAMP, ROW, - KeyValue.Type.Put, FAM, QueryConstants.EMPTY_COLUMN_BYTES, 2); - } - - /** - * Tests a partial rebuild of a row with multiple versions. 3 versions of the row in data table, - * and we rebuild the index starting from time t=2 - * - * There should be one index row version per data row version. - */ - @Test - public void testRebuildMultipleVersionRow() throws IOException { - // when doing a rebuild, we are replaying mutations so we want to ignore newer mutations - // see LocalTable#getCurrentRowState() - Mockito.when(mockIndexMetaData.ignoreNewerMutations()).thenReturn(true); - - // the current row state has 3 versions, but if we rebuild as of t=2, scanner in LocalTable - // should only return first - Cell currentCell1 = CellUtil.createCell(ROW, FAM, INDEXED_QUALIFIER, 1, PUT_TYPE, VALUE_1); - Cell currentCell2 = CellUtil.createCell(ROW, FAM, INDEXED_QUALIFIER, 2, PUT_TYPE, VALUE_2); - Cell currentCell3 = CellUtil.createCell(ROW, FAM, INDEXED_QUALIFIER, 3, PUT_TYPE, VALUE_3); - Cell currentCell4 = CellUtil.createCell(ROW, FAM, INDEXED_QUALIFIER, 4, PUT_TYPE, VALUE_4); - setCurrentRowState(Arrays.asList(currentCell4, currentCell3, currentCell2, currentCell1)); - - // rebuilder replays mutations starting from t=2 - MultiMutation mutation = new MultiMutation(new ImmutableBytesPtr(ROW)); - Put put = new Put(ROW); - put.addImmutable(FAM, INDEXED_QUALIFIER, 4, VALUE_4); - mutation.addAll(put); - put = new Put(ROW); - put.addImmutable(FAM, INDEXED_QUALIFIER, 3, VALUE_3); - mutation.addAll(put); - put = new Put(ROW); - put.addImmutable(FAM, INDEXED_QUALIFIER, 2, VALUE_2); - mutation.addAll(put); - - Collection<Pair<Mutation, byte[]>> indexUpdates = - indexBuilder.getIndexUpdate(mutation, mockIndexMetaData); - // 3 puts and 3 deletes (one to hide existing index row for VALUE_1, and two to hide index - // rows for VALUE_2, VALUE_3) - assertEquals(6, indexUpdates.size()); - - assertContains(indexUpdates, 2, ROW, KeyValue.Type.DeleteFamily, FAM, - new byte[0] /* qual not needed */, 2); - assertContains(indexUpdates, ColumnTracker.NO_NEWER_PRIMARY_TABLE_ENTRY_TIMESTAMP, ROW, - KeyValue.Type.Put, FAM, QueryConstants.EMPTY_COLUMN_BYTES, 2); - assertContains(indexUpdates, 3, ROW, KeyValue.Type.DeleteFamily, FAM, - new byte[0] /* qual not needed */, 3); - assertContains(indexUpdates, ColumnTracker.NO_NEWER_PRIMARY_TABLE_ENTRY_TIMESTAMP, ROW, - KeyValue.Type.Put, FAM, QueryConstants.EMPTY_COLUMN_BYTES, 3); - assertContains(indexUpdates, 4, ROW, KeyValue.Type.DeleteFamily, FAM, - new byte[0] /* qual not needed */, 4); - assertContains(indexUpdates, ColumnTracker.NO_NEWER_PRIMARY_TABLE_ENTRY_TIMESTAMP, ROW, - KeyValue.Type.Put, FAM, QueryConstants.EMPTY_COLUMN_BYTES, 4); - } - - /** - * Tests getting an index update for a mutation with 200 versions Before, the issue PHOENIX-3807 - * was causing this test to take >90 seconds, so here we set a timeout of 5 seconds - */ - @Test(timeout = 10000) - public void testManyVersions() throws IOException { - // when doing a rebuild, we are replaying mutations so we want to ignore newer mutations - // see LocalTable#getCurrentRowState() - Mockito.when(mockIndexMetaData.ignoreNewerMutations()).thenReturn(true); - MultiMutation mutation = getMultipleVersionMutation(200); - currentRowCells = mutation.getFamilyCellMap().get(FAM); - - Collection<Pair<Mutation, byte[]>> indexUpdates = - indexBuilder.getIndexUpdate(mutation, mockIndexMetaData); - assertNotEquals(0, indexUpdates.size()); - } - - // Assert that the given collection of indexUpdates contains the given cell - private void assertContains(Collection<Pair<Mutation, byte[]>> indexUpdates, - final long mutationTs, final byte[] row, final Type cellType, final byte[] fam, - final byte[] qual, final long cellTs) { - Predicate<Pair<Mutation, byte[]>> hasCellPredicate = - new Predicate<Pair<Mutation, byte[]>>() { - @Override - public boolean apply(Pair<Mutation, byte[]> input) { - assertEquals(TEST_TABLE_INDEX_STRING, Bytes.toString(input.getSecond())); - Mutation mutation = input.getFirst(); - if (mutationTs == mutation.getTimeStamp()) { - NavigableMap<byte[], List<Cell>> familyCellMap = - mutation.getFamilyCellMap(); - Cell updateCell = familyCellMap.get(fam).get(0); - if (cellType == KeyValue.Type.codeToType(updateCell.getTypeByte()) - && Bytes.compareTo(fam, CellUtil.cloneFamily(updateCell)) == 0 - && Bytes.compareTo(qual, - CellUtil.cloneQualifier(updateCell)) == 0 - && cellTs == updateCell.getTimestamp()) { - return true; - } - } - return false; - } - }; - Optional<Pair<Mutation, byte[]>> tryFind = - Iterables.tryFind(indexUpdates, hasCellPredicate); - assertTrue(tryFind.isPresent()); - } - - private void setCurrentRowState(byte[] fam2, byte[] indexedQualifier, int i, byte[] value1) { - Cell cell = CellUtil.createCell(ROW, FAM, INDEXED_QUALIFIER, 1, PUT_TYPE, VALUE_1); - currentRowCells = Collections.singletonList(cell); - } - - private void setCurrentRowState(List<Cell> cells) { - currentRowCells = cells; - } - - private MultiMutation getMultipleVersionMutation(int versions) { - MultiMutation mutation = new MultiMutation(new ImmutableBytesPtr(ROW)); - for (int i = versions - 1; i >= 0; i--) { - Put put = new Put(ROW); - put.addImmutable(FAM, INDEXED_QUALIFIER, i, Bytes.toBytes(i)); - mutation.addAll(put); - } - return mutation; - } -} http://git-wip-us.apache.org/repos/asf/phoenix/blob/1e777b2b/phoenix-core/src/test/java/org/apache/phoenix/util/IndexScrutiny.java ---------------------------------------------------------------------- diff --git a/phoenix-core/src/test/java/org/apache/phoenix/util/IndexScrutiny.java b/phoenix-core/src/test/java/org/apache/phoenix/util/IndexScrutiny.java new file mode 100644 index 0000000..c78658d --- /dev/null +++ b/phoenix-core/src/test/java/org/apache/phoenix/util/IndexScrutiny.java @@ -0,0 +1,144 @@ +/* + * 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 org.apache.phoenix.util; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertTrue; + +import java.sql.Connection; +import java.sql.PreparedStatement; +import java.sql.ResultSet; +import java.sql.ResultSetMetaData; +import java.sql.SQLException; + +import org.apache.phoenix.jdbc.PhoenixConnection; +import org.apache.phoenix.schema.PColumn; +import org.apache.phoenix.schema.PTable; +import org.apache.phoenix.schema.PTableKey; +import org.apache.phoenix.schema.types.PDataType; + +import com.google.common.base.Objects; + +public class IndexScrutiny { + + public static long scrutinizeIndex(Connection conn, String fullTableName, String fullIndexName) throws SQLException { + PhoenixConnection pconn = conn.unwrap(PhoenixConnection.class); + PTable ptable = pconn.getTable(new PTableKey(pconn.getTenantId(), fullTableName)); + PTable pindex = pconn.getTable(new PTableKey(pconn.getTenantId(), fullIndexName)); + StringBuilder indexQueryBuf = new StringBuilder("SELECT "); + for (PColumn dcol : ptable.getPKColumns()) { + indexQueryBuf.append("CAST(\"" + IndexUtil.getIndexColumnName(dcol) + "\" AS " + dcol.getDataType().getSqlTypeName() + ")"); + indexQueryBuf.append(","); + } + for (PColumn icol : pindex.getColumns()) { + PColumn dcol = IndexUtil.getDataColumn(ptable, icol.getName().getString()); + if (SchemaUtil.isPKColumn(icol) && !SchemaUtil.isPKColumn(dcol)) { + indexQueryBuf.append("CAST (\"" + icol.getName().getString() + "\" AS " + dcol.getDataType().getSqlTypeName() + ")"); + indexQueryBuf.append(","); + } + } + for (PColumn icol : pindex.getColumns()) { + if (!SchemaUtil.isPKColumn(icol)) { + PColumn dcol = IndexUtil.getDataColumn(ptable, icol.getName().getString()); + indexQueryBuf.append("CAST (\"" + icol.getName().getString() + "\" AS " + dcol.getDataType().getSqlTypeName() + ")"); + indexQueryBuf.append(","); + } + } + indexQueryBuf.setLength(indexQueryBuf.length()-1); + indexQueryBuf.append("\nFROM " + fullIndexName); + + StringBuilder tableQueryBuf = new StringBuilder("SELECT "); + for (PColumn dcol : ptable.getPKColumns()) { + tableQueryBuf.append("\"" + dcol.getName().getString() + "\""); + tableQueryBuf.append(","); + } + for (PColumn icol : pindex.getColumns()) { + PColumn dcol = IndexUtil.getDataColumn(ptable, icol.getName().getString()); + if (SchemaUtil.isPKColumn(icol) && !SchemaUtil.isPKColumn(dcol)) { + if (dcol.getFamilyName() != null) { + tableQueryBuf.append("\"" + dcol.getFamilyName().getString() + "\""); + tableQueryBuf.append("."); + } + tableQueryBuf.append("\"" + dcol.getName().getString() + "\""); + tableQueryBuf.append(","); + } + } + for (PColumn icol : pindex.getColumns()) { + if (!SchemaUtil.isPKColumn(icol)) { + PColumn dcol = IndexUtil.getDataColumn(ptable, icol.getName().getString()); + if (dcol.getFamilyName() != null) { + tableQueryBuf.append("\"" + dcol.getFamilyName().getString() + "\""); + tableQueryBuf.append("."); + } + tableQueryBuf.append("\"" + dcol.getName().getString() + "\""); + tableQueryBuf.append(","); + } + } + tableQueryBuf.setLength(tableQueryBuf.length()-1); + tableQueryBuf.append("\nFROM " + fullTableName + "\nWHERE ("); + for (PColumn dcol : ptable.getPKColumns()) { + tableQueryBuf.append("\"" + dcol.getName().getString() + "\""); + tableQueryBuf.append(","); + } + tableQueryBuf.setLength(tableQueryBuf.length()-1); + tableQueryBuf.append(") = (("); + for (int i = 0; i < ptable.getPKColumns().size(); i++) { + tableQueryBuf.append("?"); + tableQueryBuf.append(","); + } + tableQueryBuf.setLength(tableQueryBuf.length()-1); + tableQueryBuf.append("))"); + + String tableQuery = tableQueryBuf.toString(); + PreparedStatement istmt = conn.prepareStatement(tableQuery); + + String indexQuery = indexQueryBuf.toString(); + ResultSet irs = conn.createStatement().executeQuery(indexQuery); + ResultSetMetaData irsmd = irs.getMetaData(); + long icount = 0; + while (irs.next()) { + icount++; + StringBuilder pkBuf = new StringBuilder("("); + for (int i = 0; i < ptable.getPKColumns().size(); i++) { + PColumn dcol = ptable.getPKColumns().get(i); + Object pkVal = irs.getObject(i+1); + PDataType pkType = PDataType.fromTypeId(irsmd.getColumnType(i + 1)); + istmt.setObject(i+1, pkVal, dcol.getDataType().getSqlType()); + pkBuf.append(pkType.toStringLiteral(pkVal)); + pkBuf.append(","); + } + pkBuf.setLength(pkBuf.length()-1); + pkBuf.append(")"); + ResultSet drs = istmt.executeQuery(); + ResultSetMetaData drsmd = drs.getMetaData(); + assertTrue("Expected to find PK in data table: " + pkBuf, drs.next()); + for (int i = 0; i < irsmd.getColumnCount(); i++) { + Object iVal = irs.getObject(i + 1); + PDataType iType = PDataType.fromTypeId(irsmd.getColumnType(i + 1)); + Object dVal = drs.getObject(i + 1); + PDataType dType = PDataType.fromTypeId(drsmd.getColumnType(i + 1)); + assertTrue("Expected equality for " + drsmd.getColumnName(i + 1) + ", but " + iType.toStringLiteral(iVal) + "!=" + dType.toStringLiteral(dVal), Objects.equal(iVal, dVal)); + } + } + + long dcount = TestUtil.getRowCount(conn, fullTableName); + assertEquals("Expected data table row count to match", dcount, icount); + return dcount; + } + +} http://git-wip-us.apache.org/repos/asf/phoenix/blob/1e777b2b/phoenix-core/src/test/java/org/apache/phoenix/util/TestUtil.java ---------------------------------------------------------------------- diff --git a/phoenix-core/src/test/java/org/apache/phoenix/util/TestUtil.java b/phoenix-core/src/test/java/org/apache/phoenix/util/TestUtil.java index af924f1..6001f59 100644 --- a/phoenix-core/src/test/java/org/apache/phoenix/util/TestUtil.java +++ b/phoenix-core/src/test/java/org/apache/phoenix/util/TestUtil.java @@ -41,7 +41,6 @@ import java.sql.Date; import java.sql.DriverManager; import java.sql.PreparedStatement; import java.sql.ResultSet; -import java.sql.ResultSetMetaData; import java.sql.SQLException; import java.sql.Statement; import java.util.ArrayList; @@ -122,7 +121,6 @@ import org.apache.phoenix.schema.stats.GuidePostsKey; import org.apache.phoenix.schema.tuple.Tuple; import org.apache.phoenix.schema.types.PDataType; -import com.google.common.base.Objects; import com.google.common.collect.Lists; @@ -882,112 +880,7 @@ public class TestUtil { } } - public static long scrutinizeIndex(Connection conn, String fullTableName, String fullIndexName) throws SQLException { - PhoenixConnection pconn = conn.unwrap(PhoenixConnection.class); - PTable ptable = pconn.getTable(new PTableKey(pconn.getTenantId(), fullTableName)); - PTable pindex = pconn.getTable(new PTableKey(pconn.getTenantId(), fullIndexName)); - StringBuilder indexQueryBuf = new StringBuilder("SELECT "); - for (PColumn dcol : ptable.getPKColumns()) { - indexQueryBuf.append("CAST(\"" + IndexUtil.getIndexColumnName(dcol) + "\" AS " + dcol.getDataType().getSqlTypeName() + ")"); - indexQueryBuf.append(","); - } - for (PColumn icol : pindex.getColumns()) { - PColumn dcol = IndexUtil.getDataColumn(ptable, icol.getName().getString()); - if (SchemaUtil.isPKColumn(icol) && !SchemaUtil.isPKColumn(dcol)) { - indexQueryBuf.append("CAST (\"" + icol.getName().getString() + "\" AS " + dcol.getDataType().getSqlTypeName() + ")"); - indexQueryBuf.append(","); - } - } - for (PColumn icol : pindex.getColumns()) { - if (!SchemaUtil.isPKColumn(icol)) { - PColumn dcol = IndexUtil.getDataColumn(ptable, icol.getName().getString()); - indexQueryBuf.append("CAST (\"" + icol.getName().getString() + "\" AS " + dcol.getDataType().getSqlTypeName() + ")"); - indexQueryBuf.append(","); - } - } - indexQueryBuf.setLength(indexQueryBuf.length()-1); - indexQueryBuf.append("\nFROM " + fullIndexName); - - StringBuilder tableQueryBuf = new StringBuilder("SELECT "); - for (PColumn dcol : ptable.getPKColumns()) { - tableQueryBuf.append("\"" + dcol.getName().getString() + "\""); - tableQueryBuf.append(","); - } - for (PColumn icol : pindex.getColumns()) { - PColumn dcol = IndexUtil.getDataColumn(ptable, icol.getName().getString()); - if (SchemaUtil.isPKColumn(icol) && !SchemaUtil.isPKColumn(dcol)) { - if (dcol.getFamilyName() != null) { - tableQueryBuf.append("\"" + dcol.getFamilyName().getString() + "\""); - tableQueryBuf.append("."); - } - tableQueryBuf.append("\"" + dcol.getName().getString() + "\""); - tableQueryBuf.append(","); - } - } - for (PColumn icol : pindex.getColumns()) { - if (!SchemaUtil.isPKColumn(icol)) { - PColumn dcol = IndexUtil.getDataColumn(ptable, icol.getName().getString()); - if (dcol.getFamilyName() != null) { - tableQueryBuf.append("\"" + dcol.getFamilyName().getString() + "\""); - tableQueryBuf.append("."); - } - tableQueryBuf.append("\"" + dcol.getName().getString() + "\""); - tableQueryBuf.append(","); - } - } - tableQueryBuf.setLength(tableQueryBuf.length()-1); - tableQueryBuf.append("\nFROM " + fullTableName + "\nWHERE ("); - for (PColumn dcol : ptable.getPKColumns()) { - tableQueryBuf.append("\"" + dcol.getName().getString() + "\""); - tableQueryBuf.append(","); - } - tableQueryBuf.setLength(tableQueryBuf.length()-1); - tableQueryBuf.append(") = (("); - for (int i = 0; i < ptable.getPKColumns().size(); i++) { - tableQueryBuf.append("?"); - tableQueryBuf.append(","); - } - tableQueryBuf.setLength(tableQueryBuf.length()-1); - tableQueryBuf.append("))"); - - String tableQuery = tableQueryBuf.toString(); - PreparedStatement istmt = conn.prepareStatement(tableQuery); - - String indexQuery = indexQueryBuf.toString(); - ResultSet irs = conn.createStatement().executeQuery(indexQuery); - ResultSetMetaData irsmd = irs.getMetaData(); - long icount = 0; - while (irs.next()) { - icount++; - StringBuilder pkBuf = new StringBuilder("("); - for (int i = 0; i < ptable.getPKColumns().size(); i++) { - PColumn dcol = ptable.getPKColumns().get(i); - Object pkVal = irs.getObject(i+1); - PDataType pkType = PDataType.fromTypeId(irsmd.getColumnType(i + 1)); - istmt.setObject(i+1, pkVal, dcol.getDataType().getSqlType()); - pkBuf.append(pkType.toStringLiteral(pkVal)); - pkBuf.append(","); - } - pkBuf.setLength(pkBuf.length()-1); - pkBuf.append(")"); - ResultSet drs = istmt.executeQuery(); - ResultSetMetaData drsmd = drs.getMetaData(); - assertTrue("Expected to find PK in data table: " + pkBuf, drs.next()); - for (int i = 0; i < irsmd.getColumnCount(); i++) { - Object iVal = irs.getObject(i + 1); - PDataType iType = PDataType.fromTypeId(irsmd.getColumnType(i + 1)); - Object dVal = drs.getObject(i + 1); - PDataType dType = PDataType.fromTypeId(drsmd.getColumnType(i + 1)); - assertTrue("Expected equality for " + drsmd.getColumnName(i + 1) + ", but " + iType.toStringLiteral(iVal) + "!=" + dType.toStringLiteral(dVal), Objects.equal(iVal, dVal)); - } - } - - long dcount = getRowCount(conn, fullTableName); - assertEquals("Expected data table row count to match", dcount, icount); - return dcount; - } - - private static long getRowCount(Connection conn, String tableName) throws SQLException { + public static long getRowCount(Connection conn, String tableName) throws SQLException { ResultSet rs = conn.createStatement().executeQuery("SELECT /*+ NO_INDEX */ count(*) FROM " + tableName); assertTrue(rs.next()); return rs.getLong(1);