001/**
002 * Licensed to the Apache Software Foundation (ASF) under one
003 * or more contributor license agreements.  See the NOTICE file
004 * distributed with this work for additional information
005 * regarding copyright ownership.  The ASF licenses this file
006 * to you under the Apache License, Version 2.0 (the
007 * "License"); you may not use this file except in compliance
008 * with the License.  You may obtain a copy of the License at
009 *
010 *     http://www.apache.org/licenses/LICENSE-2.0
011 *
012 * Unless required by applicable law or agreed to in writing, software
013 * distributed under the License is distributed on an "AS IS" BASIS,
014 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
015 * See the License for the specific language governing permissions and
016 * limitations under the License.
017 */
018package org.apache.hadoop.hbase;
019
020import static org.junit.Assert.assertEquals;
021import static org.junit.Assert.assertTrue;
022
023import java.nio.ByteBuffer;
024import java.util.Collections;
025import java.util.Set;
026import java.util.TreeSet;
027
028import org.apache.hadoop.hbase.KeyValue.Type;
029import org.apache.hadoop.hbase.testclassification.MiscTests;
030import org.apache.hadoop.hbase.testclassification.SmallTests;
031import org.apache.hadoop.hbase.util.Bytes;
032import org.junit.ClassRule;
033import org.junit.Test;
034import org.junit.experimental.categories.Category;
035
036@Category({MiscTests.class, SmallTests.class})
037public class TestCellComparator {
038
039  @ClassRule
040  public static final HBaseClassTestRule CLASS_RULE =
041      HBaseClassTestRule.forClass(TestCellComparator.class);
042
043  private CellComparator comparator = CellComparator.getInstance();
044  byte[] row1 = Bytes.toBytes("row1");
045  byte[] row2 = Bytes.toBytes("row2");
046  byte[] row_1_0 = Bytes.toBytes("row10");
047
048  byte[] fam1 = Bytes.toBytes("fam1");
049  byte[] fam2 = Bytes.toBytes("fam2");
050  byte[] fam_1_2 = Bytes.toBytes("fam12");
051
052  byte[] qual1 = Bytes.toBytes("qual1");
053  byte[] qual2 = Bytes.toBytes("qual2");
054
055  byte[] val = Bytes.toBytes("val");
056
057  @Test
058  public void testCompareCells() {
059    KeyValue kv1 = new KeyValue(row1, fam1, qual1, val);
060    KeyValue kv2 = new KeyValue(row2, fam1, qual1, val);
061    assertTrue((comparator.compare(kv1, kv2)) < 0);
062
063    kv1 = new KeyValue(row1, fam2, qual1, val);
064    kv2 = new KeyValue(row1, fam1, qual1, val);
065    assertTrue((comparator.compareFamilies(kv1, kv2) > 0));
066
067    kv1 = new KeyValue(row1, fam1, qual1, 1L, val);
068    kv2 = new KeyValue(row1, fam1, qual1, 2L, val);
069    assertTrue((comparator.compare(kv1, kv2) > 0));
070
071    kv1 = new KeyValue(row1, fam1, qual1, 1L, Type.Put);
072    kv2 = new KeyValue(row1, fam1, qual1, 1L, Type.Maximum);
073    assertTrue((comparator.compare(kv1, kv2) > 0));
074
075    kv1 = new KeyValue(row1, fam1, qual1, 1L, Type.Put);
076    kv2 = new KeyValue(row1, fam1, qual1, 1L, Type.Put);
077    assertTrue((CellUtil.equals(kv1, kv2)));
078  }
079
080  @Test
081  public void testCompareCellWithKey() throws Exception {
082    KeyValue kv1 = new KeyValue(row1, fam1, qual1, val);
083    KeyValue kv2 = new KeyValue(row2, fam1, qual1, val);
084    assertTrue(
085      (PrivateCellUtil.compare(comparator, kv1, kv2.getKey(), 0, kv2.getKey().length)) < 0);
086
087    kv1 = new KeyValue(row1, fam2, qual1, val);
088    kv2 = new KeyValue(row1, fam1, qual1, val);
089    assertTrue(
090      (PrivateCellUtil.compare(comparator, kv1, kv2.getKey(), 0, kv2.getKey().length)) > 0);
091
092    kv1 = new KeyValue(row1, fam1, qual1, 1L, val);
093    kv2 = new KeyValue(row1, fam1, qual1, 2L, val);
094    assertTrue(
095      (PrivateCellUtil.compare(comparator, kv1, kv2.getKey(), 0, kv2.getKey().length)) > 0);
096
097    kv1 = new KeyValue(row1, fam1, qual1, 1L, Type.Put);
098    kv2 = new KeyValue(row1, fam1, qual1, 1L, Type.Maximum);
099    assertTrue(
100      (PrivateCellUtil.compare(comparator, kv1, kv2.getKey(), 0, kv2.getKey().length)) > 0);
101
102    kv1 = new KeyValue(row1, fam1, qual1, 1L, Type.Put);
103    kv2 = new KeyValue(row1, fam1, qual1, 1L, Type.Put);
104    assertTrue(
105      (PrivateCellUtil.compare(comparator, kv1, kv2.getKey(), 0, kv2.getKey().length)) == 0);
106  }
107
108  @Test
109  public void testCompareByteBufferedCell() {
110    byte[] r1 = Bytes.toBytes("row1");
111    byte[] r2 = Bytes.toBytes("row2");
112    byte[] f1 = Bytes.toBytes("cf1");
113    byte[] q1 = Bytes.toBytes("qual1");
114    byte[] q2 = Bytes.toBytes("qual2");
115    byte[] v = Bytes.toBytes("val1");
116    KeyValue kv = new KeyValue(r1, f1, q1, v);
117    ByteBuffer buffer = ByteBuffer.wrap(kv.getBuffer());
118    Cell bbCell1 = new ByteBufferKeyValue(buffer, 0, buffer.remaining());
119    kv = new KeyValue(r2, f1, q1, v);
120    buffer = ByteBuffer.wrap(kv.getBuffer());
121    Cell bbCell2 = new ByteBufferKeyValue(buffer, 0, buffer.remaining());
122    // compareColumns not on CellComparator so use Impl directly
123    assertEquals(0, CellComparatorImpl.COMPARATOR.compareColumns(bbCell1, bbCell2));
124    assertEquals(0, CellComparatorImpl.COMPARATOR.compareColumns(bbCell1, kv));
125    kv = new KeyValue(r2, f1, q2, v);
126    buffer = ByteBuffer.wrap(kv.getBuffer());
127    Cell bbCell3 = new ByteBufferKeyValue(buffer, 0, buffer.remaining());
128    assertEquals(0, comparator.compareFamilies(bbCell2, bbCell3));
129    assertTrue(comparator.compareQualifiers(bbCell2, bbCell3) < 0);
130    assertTrue(CellComparatorImpl.COMPARATOR.compareColumns(bbCell2, bbCell3) < 0);
131
132    assertEquals(0, comparator.compareRows(bbCell2, bbCell3));
133    assertTrue(comparator.compareRows(bbCell1, bbCell2) < 0);
134  }
135
136  /**
137   * Test meta comparisons using our new ByteBufferKeyValue Cell type, the type we use everywhere
138   * in 2.0.
139   */
140  @Test
141  public void testMetaComparisons() throws Exception {
142    long now = System.currentTimeMillis();
143
144    // Meta compares
145    Cell aaa = createByteBufferKeyValueFromKeyValue(new KeyValue(
146        Bytes.toBytes("TestScanMultipleVersions,row_0500,1236020145502"), now));
147    Cell bbb = createByteBufferKeyValueFromKeyValue(new KeyValue(
148        Bytes.toBytes("TestScanMultipleVersions,,99999999999999"), now));
149    CellComparator c = MetaCellComparator.META_COMPARATOR;
150    assertTrue(c.compare(bbb, aaa) < 0);
151
152    Cell ccc = createByteBufferKeyValueFromKeyValue(
153        new KeyValue(Bytes.toBytes("TestScanMultipleVersions,,1236023996656"),
154        Bytes.toBytes("info"), Bytes.toBytes("regioninfo"), 1236024396271L,
155        (byte[])null));
156    assertTrue(c.compare(ccc, bbb) < 0);
157
158    Cell x = createByteBufferKeyValueFromKeyValue(
159        new KeyValue(Bytes.toBytes("TestScanMultipleVersions,row_0500,1236034574162"),
160        Bytes.toBytes("info"), Bytes.toBytes(""), 9223372036854775807L,
161        (byte[])null));
162    Cell y = createByteBufferKeyValueFromKeyValue(
163        new KeyValue(Bytes.toBytes("TestScanMultipleVersions,row_0500,1236034574162"),
164        Bytes.toBytes("info"), Bytes.toBytes("regioninfo"), 1236034574912L,
165        (byte[])null));
166    assertTrue(c.compare(x, y) < 0);
167  }
168
169  private static Cell createByteBufferKeyValueFromKeyValue(KeyValue kv) {
170    ByteBuffer bb = ByteBuffer.wrap(kv.getBuffer());
171    return new ByteBufferKeyValue(bb, 0, bb.remaining());
172  }
173
174  /**
175   * More tests using ByteBufferKeyValue copied over from TestKeyValue which uses old KVs only.
176   */
177  @Test
178  public void testMetaComparisons2() {
179    long now = System.currentTimeMillis();
180    CellComparator c = MetaCellComparator.META_COMPARATOR;
181    assertTrue(c.compare(createByteBufferKeyValueFromKeyValue(new KeyValue(
182            Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",a,,0,1"), now)),
183        createByteBufferKeyValueFromKeyValue(new KeyValue(
184            Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",a,,0,1"), now))) == 0);
185    Cell a = createByteBufferKeyValueFromKeyValue(new KeyValue(
186        Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",a,,0,1"), now));
187    Cell b = createByteBufferKeyValueFromKeyValue(new KeyValue(
188        Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",a,,0,2"), now));
189    assertTrue(c.compare(a, b) < 0);
190    assertTrue(c.compare(createByteBufferKeyValueFromKeyValue(new KeyValue(
191            Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",a,,0,2"), now)),
192        createByteBufferKeyValueFromKeyValue(new KeyValue(
193            Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",a,,0,1"), now))) > 0);
194    assertTrue(c.compare(createByteBufferKeyValueFromKeyValue(new KeyValue(
195            Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",,1"), now)),
196        createByteBufferKeyValueFromKeyValue(new KeyValue(
197            Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",,1"), now))) == 0);
198    assertTrue(c.compare(createByteBufferKeyValueFromKeyValue(new KeyValue(
199            Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",,1"), now)),
200        createByteBufferKeyValueFromKeyValue(new KeyValue(
201            Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",,2"), now))) < 0);
202    assertTrue(c.compare(createByteBufferKeyValueFromKeyValue(new KeyValue(
203            Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",,2"), now)),
204        createByteBufferKeyValueFromKeyValue(new KeyValue(
205            Bytes.toBytes(TableName.META_TABLE_NAME.getNameAsString()+",,1"), now))) > 0);
206  }
207
208  @Test
209  public void testBinaryKeys() throws Exception {
210    Set<Cell> set = new TreeSet<>(CellComparatorImpl.COMPARATOR);
211    final byte [] fam = Bytes.toBytes("col");
212    final byte [] qf = Bytes.toBytes("umn");
213    final byte [] nb = new byte[0];
214    Cell [] keys = {
215        createByteBufferKeyValueFromKeyValue(
216            new KeyValue(Bytes.toBytes("aaaaa,\u0000\u0000,2"), fam, qf, 2, nb)),
217        createByteBufferKeyValueFromKeyValue(
218            new KeyValue(Bytes.toBytes("aaaaa,\u0001,3"), fam, qf, 3, nb)),
219        createByteBufferKeyValueFromKeyValue(
220            new KeyValue(Bytes.toBytes("aaaaa,,1"), fam, qf, 1, nb)),
221        createByteBufferKeyValueFromKeyValue(
222            new KeyValue(Bytes.toBytes("aaaaa,\u1000,5"), fam, qf, 5, nb)),
223        createByteBufferKeyValueFromKeyValue(
224            new KeyValue(Bytes.toBytes("aaaaa,a,4"), fam, qf, 4, nb)),
225        createByteBufferKeyValueFromKeyValue(
226            new KeyValue(Bytes.toBytes("a,a,0"), fam, qf, 0, nb)),
227    };
228    // Add to set with bad comparator
229    Collections.addAll(set, keys);
230    // This will output the keys incorrectly.
231    boolean assertion = false;
232    int count = 0;
233    try {
234      for (Cell k: set) {
235        assertTrue("count=" + count + ", " + k.toString(), count++ == k.getTimestamp());
236      }
237    } catch (AssertionError e) {
238      // Expected
239      assertion = true;
240    }
241    assertTrue(assertion);
242    // Make set with good comparator
243    set = new TreeSet<>(MetaCellComparator.META_COMPARATOR);
244    Collections.addAll(set, keys);
245    count = 0;
246    for (Cell k: set) {
247      assertTrue("count=" + count + ", " + k.toString(), count++ == k.getTimestamp());
248    }
249  }
250}