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.io.encoding;
019
020import static org.junit.Assert.assertEquals;
021import static org.junit.Assert.assertTrue;
022import static org.junit.Assert.fail;
023
024import java.io.ByteArrayInputStream;
025import java.io.DataInputStream;
026import java.io.DataOutputStream;
027import java.io.IOException;
028import java.nio.ByteBuffer;
029import java.util.ArrayList;
030import java.util.Collection;
031import java.util.List;
032import java.util.Random;
033import org.apache.hadoop.hbase.ArrayBackedTag;
034import org.apache.hadoop.hbase.Cell;
035import org.apache.hadoop.hbase.CellComparatorImpl;
036import org.apache.hadoop.hbase.CellUtil;
037import org.apache.hadoop.hbase.HBaseClassTestRule;
038import org.apache.hadoop.hbase.HBaseTestingUtility;
039import org.apache.hadoop.hbase.HConstants;
040import org.apache.hadoop.hbase.KeyValue;
041import org.apache.hadoop.hbase.KeyValue.Type;
042import org.apache.hadoop.hbase.PrivateCellUtil;
043import org.apache.hadoop.hbase.Tag;
044import org.apache.hadoop.hbase.io.ByteArrayOutputStream;
045import org.apache.hadoop.hbase.io.compress.Compression;
046import org.apache.hadoop.hbase.io.compress.Compression.Algorithm;
047import org.apache.hadoop.hbase.io.hfile.HFileContext;
048import org.apache.hadoop.hbase.io.hfile.HFileContextBuilder;
049import org.apache.hadoop.hbase.nio.SingleByteBuff;
050import org.apache.hadoop.hbase.testclassification.IOTests;
051import org.apache.hadoop.hbase.testclassification.LargeTests;
052import org.apache.hadoop.hbase.util.Bytes;
053import org.apache.hadoop.hbase.util.RedundantKVGenerator;
054import org.junit.Assert;
055import org.junit.ClassRule;
056import org.junit.Test;
057import org.junit.experimental.categories.Category;
058import org.junit.runner.RunWith;
059import org.junit.runners.Parameterized;
060import org.junit.runners.Parameterized.Parameters;
061import org.slf4j.Logger;
062import org.slf4j.LoggerFactory;
063
064/**
065 * Test all of the data block encoding algorithms for correctness. Most of the
066 * class generate data which will test different branches in code.
067 */
068@Category({IOTests.class, LargeTests.class})
069@RunWith(Parameterized.class)
070public class TestDataBlockEncoders {
071
072  @ClassRule
073  public static final HBaseClassTestRule CLASS_RULE =
074      HBaseClassTestRule.forClass(TestDataBlockEncoders.class);
075
076  private static final Logger LOG = LoggerFactory.getLogger(TestDataBlockEncoders.class);
077
078  private static int NUMBER_OF_KV = 10000;
079  private static int NUM_RANDOM_SEEKS = 1000;
080
081  private static int ENCODED_DATA_OFFSET = HConstants.HFILEBLOCK_HEADER_SIZE
082      + DataBlockEncoding.ID_SIZE;
083  static final byte[] HFILEBLOCK_DUMMY_HEADER = new byte[HConstants.HFILEBLOCK_HEADER_SIZE];
084
085  private RedundantKVGenerator generator = new RedundantKVGenerator();
086  private Random randomizer = new Random(42L);
087
088  private final boolean includesMemstoreTS;
089  private final boolean includesTags;
090  private final boolean useOffheapData;
091
092  @Parameters
093  public static Collection<Object[]> parameters() {
094    return HBaseTestingUtility.memStoreTSTagsAndOffheapCombination();
095  }
096
097  public TestDataBlockEncoders(boolean includesMemstoreTS, boolean includesTag,
098      boolean useOffheapData) {
099    this.includesMemstoreTS = includesMemstoreTS;
100    this.includesTags = includesTag;
101    this.useOffheapData = useOffheapData;
102  }
103
104  private HFileBlockEncodingContext getEncodingContext(Compression.Algorithm algo,
105      DataBlockEncoding encoding) {
106    DataBlockEncoder encoder = encoding.getEncoder();
107    HFileContext meta = new HFileContextBuilder()
108                        .withHBaseCheckSum(false)
109                        .withIncludesMvcc(includesMemstoreTS)
110                        .withIncludesTags(includesTags)
111                        .withCompression(algo).build();
112    if (encoder != null) {
113      return encoder.newDataBlockEncodingContext(encoding, HFILEBLOCK_DUMMY_HEADER, meta);
114    } else {
115      return new HFileBlockDefaultEncodingContext(encoding, HFILEBLOCK_DUMMY_HEADER, meta);
116    }
117  }
118
119  /**
120   * Test data block encoding of empty KeyValue.
121   *
122   * @throws IOException
123   *           On test failure.
124   */
125  @Test
126  public void testEmptyKeyValues() throws IOException {
127    List<KeyValue> kvList = new ArrayList<>();
128    byte[] row = new byte[0];
129    byte[] family = new byte[0];
130    byte[] qualifier = new byte[0];
131    byte[] value = new byte[0];
132    if (!includesTags) {
133      kvList.add(new KeyValue(row, family, qualifier, 0L, value));
134      kvList.add(new KeyValue(row, family, qualifier, 0L, value));
135    } else {
136      byte[] metaValue1 = Bytes.toBytes("metaValue1");
137      byte[] metaValue2 = Bytes.toBytes("metaValue2");
138      kvList.add(new KeyValue(row, family, qualifier, 0L, value,
139          new Tag[] { new ArrayBackedTag((byte) 1, metaValue1) }));
140      kvList.add(new KeyValue(row, family, qualifier, 0L, value,
141          new Tag[] { new ArrayBackedTag((byte) 1, metaValue2) }));
142    }
143    testEncodersOnDataset(kvList, includesMemstoreTS, includesTags);
144  }
145
146  /**
147   * Test KeyValues with negative timestamp.
148   *
149   * @throws IOException
150   *           On test failure.
151   */
152  @Test
153  public void testNegativeTimestamps() throws IOException {
154    List<KeyValue> kvList = new ArrayList<>();
155    byte[] row = new byte[0];
156    byte[] family = new byte[0];
157    byte[] qualifier = new byte[0];
158    byte[] value = new byte[0];
159    if (includesTags) {
160      byte[] metaValue1 = Bytes.toBytes("metaValue1");
161      byte[] metaValue2 = Bytes.toBytes("metaValue2");
162      kvList.add(new KeyValue(row, family, qualifier, 0L, value,
163          new Tag[] { new ArrayBackedTag((byte) 1, metaValue1) }));
164      kvList.add(new KeyValue(row, family, qualifier, 0L, value,
165          new Tag[] { new ArrayBackedTag((byte) 1, metaValue2) }));
166    } else {
167      kvList.add(new KeyValue(row, family, qualifier, -1L, Type.Put, value));
168      kvList.add(new KeyValue(row, family, qualifier, -2L, Type.Put, value));
169    }
170    testEncodersOnDataset(kvList, includesMemstoreTS, includesTags);
171  }
172
173
174  /**
175   * Test whether compression -> decompression gives the consistent results on
176   * pseudorandom sample.
177   * @throws IOException On test failure.
178   */
179  @Test
180  public void testExecutionOnSample() throws IOException {
181    List<KeyValue> kvList = generator.generateTestKeyValues(NUMBER_OF_KV, includesTags);
182    testEncodersOnDataset(kvList, includesMemstoreTS, includesTags);
183  }
184
185  /**
186   * Test seeking while file is encoded.
187   */
188  @Test
189  public void testSeekingOnSample() throws IOException {
190    List<KeyValue> sampleKv = generator.generateTestKeyValues(NUMBER_OF_KV, includesTags);
191
192    // create all seekers
193    List<DataBlockEncoder.EncodedSeeker> encodedSeekers = new ArrayList<>();
194    for (DataBlockEncoding encoding : DataBlockEncoding.values()) {
195      LOG.info("Encoding: " + encoding);
196      DataBlockEncoder encoder = encoding.getEncoder();
197      if (encoder == null) {
198        continue;
199      }
200      LOG.info("Encoder: " + encoder);
201      ByteBuffer encodedBuffer = encodeKeyValues(encoding, sampleKv,
202          getEncodingContext(Compression.Algorithm.NONE, encoding), this.useOffheapData);
203      HFileContext meta = new HFileContextBuilder()
204                          .withHBaseCheckSum(false)
205                          .withIncludesMvcc(includesMemstoreTS)
206                          .withIncludesTags(includesTags)
207                          .withCompression(Compression.Algorithm.NONE)
208                          .build();
209      DataBlockEncoder.EncodedSeeker seeker =
210        encoder.createSeeker(encoder.newDataBlockDecodingContext(meta));
211      seeker.setCurrentBuffer(new SingleByteBuff(encodedBuffer));
212      encodedSeekers.add(seeker);
213    }
214    LOG.info("Testing it!");
215    // test it!
216    // try a few random seeks
217    for (boolean seekBefore : new boolean[] { false, true }) {
218      for (int i = 0; i < NUM_RANDOM_SEEKS; ++i) {
219        int keyValueId;
220        if (!seekBefore) {
221          keyValueId = randomizer.nextInt(sampleKv.size());
222        } else {
223          keyValueId = randomizer.nextInt(sampleKv.size() - 1) + 1;
224        }
225
226        KeyValue keyValue = sampleKv.get(keyValueId);
227        checkSeekingConsistency(encodedSeekers, seekBefore, keyValue);
228      }
229    }
230
231    // check edge cases
232    LOG.info("Checking edge cases");
233    checkSeekingConsistency(encodedSeekers, false, sampleKv.get(0));
234    for (boolean seekBefore : new boolean[] { false, true }) {
235      checkSeekingConsistency(encodedSeekers, seekBefore, sampleKv.get(sampleKv.size() - 1));
236      KeyValue midKv = sampleKv.get(sampleKv.size() / 2);
237      Cell lastMidKv =PrivateCellUtil.createLastOnRowCol(midKv);
238      checkSeekingConsistency(encodedSeekers, seekBefore, lastMidKv);
239    }
240    LOG.info("Done");
241  }
242
243  static ByteBuffer encodeKeyValues(DataBlockEncoding encoding, List<KeyValue> kvs,
244      HFileBlockEncodingContext encodingContext, boolean useOffheapData) throws IOException {
245    DataBlockEncoder encoder = encoding.getEncoder();
246    ByteArrayOutputStream baos = new ByteArrayOutputStream();
247    baos.write(HFILEBLOCK_DUMMY_HEADER);
248    DataOutputStream dos = new DataOutputStream(baos);
249    encoder.startBlockEncoding(encodingContext, dos);
250    for (KeyValue kv : kvs) {
251      encoder.encode(kv, encodingContext, dos);
252    }
253    encoder.endBlockEncoding(encodingContext, dos, baos.getBuffer());
254    byte[] encodedData = new byte[baos.size() - ENCODED_DATA_OFFSET];
255    System.arraycopy(baos.toByteArray(), ENCODED_DATA_OFFSET, encodedData, 0, encodedData.length);
256    if (useOffheapData) {
257      ByteBuffer bb = ByteBuffer.allocateDirect(encodedData.length);
258      bb.put(encodedData);
259      bb.rewind();
260      return bb;
261    }
262    return ByteBuffer.wrap(encodedData);
263  }
264
265  @Test
266  public void testNextOnSample() throws IOException {
267    List<KeyValue> sampleKv = generator.generateTestKeyValues(NUMBER_OF_KV, includesTags);
268
269    for (DataBlockEncoding encoding : DataBlockEncoding.values()) {
270      if (encoding.getEncoder() == null) {
271        continue;
272      }
273      DataBlockEncoder encoder = encoding.getEncoder();
274      ByteBuffer encodedBuffer = encodeKeyValues(encoding, sampleKv,
275          getEncodingContext(Compression.Algorithm.NONE, encoding), this.useOffheapData);
276      HFileContext meta = new HFileContextBuilder()
277                          .withHBaseCheckSum(false)
278                          .withIncludesMvcc(includesMemstoreTS)
279                          .withIncludesTags(includesTags)
280                          .withCompression(Compression.Algorithm.NONE)
281                          .build();
282      DataBlockEncoder.EncodedSeeker seeker =
283        encoder.createSeeker(encoder.newDataBlockDecodingContext(meta));
284      seeker.setCurrentBuffer(new SingleByteBuff(encodedBuffer));
285      int i = 0;
286      do {
287        KeyValue expectedKeyValue = sampleKv.get(i);
288        Cell cell = seeker.getCell();
289        if (PrivateCellUtil.compareKeyIgnoresMvcc(CellComparatorImpl.COMPARATOR, expectedKeyValue,
290          cell) != 0) {
291          int commonPrefix = PrivateCellUtil
292              .findCommonPrefixInFlatKey(expectedKeyValue, cell, false, true);
293          fail(String.format("next() produces wrong results "
294              + "encoder: %s i: %d commonPrefix: %d" + "\n expected %s\n actual      %s", encoder
295              .toString(), i, commonPrefix, Bytes.toStringBinary(expectedKeyValue.getBuffer(),
296              expectedKeyValue.getKeyOffset(), expectedKeyValue.getKeyLength()), CellUtil.toString(
297              cell, false)));
298        }
299        i++;
300      } while (seeker.next());
301    }
302  }
303
304  /**
305   * Test whether the decompression of first key is implemented correctly.
306   * @throws IOException
307   */
308  @Test
309  public void testFirstKeyInBlockOnSample() throws IOException {
310    List<KeyValue> sampleKv = generator.generateTestKeyValues(NUMBER_OF_KV, includesTags);
311
312    for (DataBlockEncoding encoding : DataBlockEncoding.values()) {
313      if (encoding.getEncoder() == null) {
314        continue;
315      }
316      DataBlockEncoder encoder = encoding.getEncoder();
317      ByteBuffer encodedBuffer = encodeKeyValues(encoding, sampleKv,
318          getEncodingContext(Compression.Algorithm.NONE, encoding), this.useOffheapData);
319      Cell key = encoder.getFirstKeyCellInBlock(new SingleByteBuff(encodedBuffer));
320      KeyValue firstKv = sampleKv.get(0);
321      if (0 != PrivateCellUtil.compareKeyIgnoresMvcc(CellComparatorImpl.COMPARATOR, key, firstKv)) {
322        int commonPrefix = PrivateCellUtil.findCommonPrefixInFlatKey(key, firstKv, false, true);
323        fail(String.format("Bug in '%s' commonPrefix %d", encoder.toString(), commonPrefix));
324      }
325    }
326  }
327
328  @Test
329  public void testRowIndexWithTagsButNoTagsInCell() throws IOException {
330    List<KeyValue> kvList = new ArrayList<>();
331    byte[] row = new byte[0];
332    byte[] family = new byte[0];
333    byte[] qualifier = new byte[0];
334    byte[] value = new byte[0];
335    KeyValue expectedKV = new KeyValue(row, family, qualifier, 1L, Type.Put, value);
336    kvList.add(expectedKV);
337    DataBlockEncoding encoding = DataBlockEncoding.ROW_INDEX_V1;
338    DataBlockEncoder encoder = encoding.getEncoder();
339    ByteBuffer encodedBuffer =
340        encodeKeyValues(encoding, kvList, getEncodingContext(Algorithm.NONE, encoding), false);
341    HFileContext meta =
342        new HFileContextBuilder().withHBaseCheckSum(false).withIncludesMvcc(includesMemstoreTS)
343            .withIncludesTags(includesTags).withCompression(Compression.Algorithm.NONE).build();
344    DataBlockEncoder.EncodedSeeker seeker =
345      encoder.createSeeker(encoder.newDataBlockDecodingContext(meta));
346    seeker.setCurrentBuffer(new SingleByteBuff(encodedBuffer));
347    Cell cell = seeker.getCell();
348    Assert.assertEquals(expectedKV.getLength(), ((KeyValue) cell).getLength());
349  }
350
351  private void checkSeekingConsistency(List<DataBlockEncoder.EncodedSeeker> encodedSeekers,
352      boolean seekBefore, Cell keyValue) {
353    Cell expectedKeyValue = null;
354    ByteBuffer expectedKey = null;
355    ByteBuffer expectedValue = null;
356    for (DataBlockEncoder.EncodedSeeker seeker : encodedSeekers) {
357      seeker.seekToKeyInBlock(keyValue, seekBefore);
358      seeker.rewind();
359
360      Cell actualKeyValue = seeker.getCell();
361      ByteBuffer actualKey = null;
362      actualKey = ByteBuffer.wrap(((KeyValue) seeker.getKey()).getKey());
363      ByteBuffer actualValue = seeker.getValueShallowCopy();
364
365      if (expectedKeyValue != null) {
366        assertTrue(CellUtil.equals(expectedKeyValue, actualKeyValue));
367      } else {
368        expectedKeyValue = actualKeyValue;
369      }
370
371      if (expectedKey != null) {
372        assertEquals(expectedKey, actualKey);
373      } else {
374        expectedKey = actualKey;
375      }
376
377      if (expectedValue != null) {
378        assertEquals(expectedValue, actualValue);
379      } else {
380        expectedValue = actualValue;
381      }
382    }
383  }
384
385  private void testEncodersOnDataset(List<KeyValue> kvList, boolean includesMemstoreTS,
386      boolean includesTags) throws IOException {
387    ByteBuffer unencodedDataBuf = RedundantKVGenerator.convertKvToByteBuffer(kvList,
388        includesMemstoreTS);
389    HFileContext fileContext = new HFileContextBuilder().withIncludesMvcc(includesMemstoreTS)
390        .withIncludesTags(includesTags).build();
391    for (DataBlockEncoding encoding : DataBlockEncoding.values()) {
392      DataBlockEncoder encoder = encoding.getEncoder();
393      if (encoder == null) {
394        continue;
395      }
396      HFileBlockEncodingContext encodingContext = new HFileBlockDefaultEncodingContext(encoding,
397          HFILEBLOCK_DUMMY_HEADER, fileContext);
398
399      ByteArrayOutputStream baos = new ByteArrayOutputStream();
400      baos.write(HFILEBLOCK_DUMMY_HEADER);
401      DataOutputStream dos = new DataOutputStream(baos);
402      encoder.startBlockEncoding(encodingContext, dos);
403      for (KeyValue kv : kvList) {
404        encoder.encode(kv, encodingContext, dos);
405      }
406      encoder.endBlockEncoding(encodingContext, dos, baos.getBuffer());
407      byte[] encodedData = baos.toByteArray();
408
409      testAlgorithm(encodedData, unencodedDataBuf, encoder);
410    }
411  }
412
413  @Test
414  public void testZeroByte() throws IOException {
415    List<KeyValue> kvList = new ArrayList<>();
416    byte[] row = Bytes.toBytes("abcd");
417    byte[] family = new byte[] { 'f' };
418    byte[] qualifier0 = new byte[] { 'b' };
419    byte[] qualifier1 = new byte[] { 'c' };
420    byte[] value0 = new byte[] { 'd' };
421    byte[] value1 = new byte[] { 0x00 };
422    if (includesTags) {
423      kvList.add(new KeyValue(row, family, qualifier0, 0, value0,
424          new Tag[] { new ArrayBackedTag((byte) 1, "value1") }));
425      kvList.add(new KeyValue(row, family, qualifier1, 0, value1,
426          new Tag[] { new ArrayBackedTag((byte) 1, "value1") }));
427    } else {
428      kvList.add(new KeyValue(row, family, qualifier0, 0, Type.Put, value0));
429      kvList.add(new KeyValue(row, family, qualifier1, 0, Type.Put, value1));
430    }
431    testEncodersOnDataset(kvList, includesMemstoreTS, includesTags);
432  }
433
434  private void testAlgorithm(byte[] encodedData, ByteBuffer unencodedDataBuf,
435      DataBlockEncoder encoder) throws IOException {
436    // decode
437    ByteArrayInputStream bais = new ByteArrayInputStream(encodedData, ENCODED_DATA_OFFSET,
438        encodedData.length - ENCODED_DATA_OFFSET);
439    DataInputStream dis = new DataInputStream(bais);
440    ByteBuffer actualDataset;
441    HFileContext meta = new HFileContextBuilder().withHBaseCheckSum(false)
442        .withIncludesMvcc(includesMemstoreTS).withIncludesTags(includesTags)
443        .withCompression(Compression.Algorithm.NONE).build();
444    actualDataset = encoder.decodeKeyValues(dis, encoder.newDataBlockDecodingContext(meta));
445    actualDataset.rewind();
446
447    // this is because in case of prefix tree the decoded stream will not have
448    // the
449    // mvcc in it.
450    assertEquals("Encoding -> decoding gives different results for " + encoder,
451        Bytes.toStringBinary(unencodedDataBuf), Bytes.toStringBinary(actualDataset));
452  }
453}