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.assertArrayEquals; 021import static org.junit.Assert.assertEquals; 022import static org.junit.Assert.assertTrue; 023 024import java.io.IOException; 025import java.nio.ByteBuffer; 026import org.apache.hadoop.hbase.io.ByteArrayOutputStream; 027import org.apache.hadoop.hbase.testclassification.MiscTests; 028import org.apache.hadoop.hbase.testclassification.SmallTests; 029import org.apache.hadoop.hbase.util.Bytes; 030import org.junit.BeforeClass; 031import org.junit.ClassRule; 032import org.junit.Test; 033import org.junit.experimental.categories.Category; 034 035@Category({ MiscTests.class, SmallTests.class }) 036public class TestIndividualBytesFieldCell { 037 038 @ClassRule 039 public static final HBaseClassTestRule CLASS_RULE = 040 HBaseClassTestRule.forClass(TestIndividualBytesFieldCell.class); 041 042 private static IndividualBytesFieldCell ic0 = null; 043 private static KeyValue kv0 = null; 044 045 @BeforeClass 046 public static void testConstructorAndVerify() { 047 // Immutable inputs 048 byte[] row = Bytes.toBytes("immutable-row"); 049 byte[] family = Bytes.toBytes("immutable-family"); 050 byte[] qualifier = Bytes.toBytes("immutable-qualifier"); 051 byte[] value = Bytes.toBytes("immutable-value"); 052 byte[] tags = Bytes.toBytes("immutable-tags"); 053 054 // Other inputs 055 long timestamp = 5000L; 056 long seqId = 0L; 057 KeyValue.Type type = KeyValue.Type.Put; 058 059 ic0 = new IndividualBytesFieldCell(row, family, qualifier, timestamp, type, seqId, value, tags); 060 kv0 = new KeyValue(row, family, qualifier, timestamp, type, value, tags); 061 062 // Verify if no local copy is made for row, family, qualifier, value or tags. 063 assertTrue(ic0.getRowArray() == row); 064 assertTrue(ic0.getFamilyArray() == family); 065 assertTrue(ic0.getQualifierArray() == qualifier); 066 assertTrue(ic0.getValueArray() == value); 067 assertTrue(ic0.getTagsArray() == tags); 068 069 // Verify others. 070 assertEquals(timestamp, ic0.getTimestamp()); 071 assertEquals(seqId, ic0.getSequenceId()); 072 assertEquals(type.getCode(), ic0.getTypeByte()); 073 074 // Verify offsets of backing byte arrays are always 0. 075 assertEquals(0, ic0.getRowOffset()); 076 assertEquals(0, ic0.getFamilyOffset()); 077 assertEquals(0, ic0.getQualifierOffset()); 078 assertEquals(0, ic0.getValueOffset()); 079 assertEquals(0, ic0.getTagsOffset()); 080 } 081 082 // Verify clone() and deepClone() 083 @Test 084 public void testClone() throws CloneNotSupportedException { 085 // Verify clone. Only shadow copies are made for backing byte arrays. 086 IndividualBytesFieldCell cloned = (IndividualBytesFieldCell) ic0.clone(); 087 assertTrue(cloned.getRowArray() == ic0.getRowArray()); 088 assertTrue(cloned.getFamilyArray() == ic0.getFamilyArray()); 089 assertTrue(cloned.getQualifierArray() == ic0.getQualifierArray()); 090 assertTrue(cloned.getValueArray() == ic0.getValueArray()); 091 assertTrue(cloned.getTagsArray() == ic0.getTagsArray()); 092 093 // Verify if deep clone returns a KeyValue object 094 assertTrue(ic0.deepClone() instanceof KeyValue); 095 } 096 097 /** 098 * Verify KeyValue format related functions: write() and getSerializedSize(). Should have the same 099 * behaviors as {@link KeyValue}. 100 */ 101 @Test 102 public void testWriteIntoKeyValueFormat() throws IOException { 103 // Verify getSerializedSize(). 104 assertEquals(kv0.getSerializedSize(true), ic0.getSerializedSize(true)); // with tags 105 assertEquals(kv0.getSerializedSize(false), ic0.getSerializedSize(false)); // without tags 106 107 // Verify writing into ByteBuffer. 108 ByteBuffer bbufIC = ByteBuffer.allocate(ic0.getSerializedSize(true)); 109 ic0.write(bbufIC, 0); 110 111 ByteBuffer bbufKV = ByteBuffer.allocate(kv0.getSerializedSize(true)); 112 kv0.write(bbufKV, 0); 113 114 assertTrue(bbufIC.equals(bbufKV)); 115 116 // Verify writing into OutputStream. 117 testWriteIntoOutputStream(ic0, kv0, true); // with tags 118 testWriteIntoOutputStream(ic0, kv0, false); // without tags 119 } 120 121 /** 122 * @param ic An instance of IndividualBytesFieldCell to compare. 123 * @param kv An instance of KeyValue to compare. 124 * @param withTags Whether to write tags. 125 */ 126 private void testWriteIntoOutputStream(IndividualBytesFieldCell ic, KeyValue kv, boolean withTags) 127 throws IOException { 128 ByteArrayOutputStream outIC = new ByteArrayOutputStream(ic.getSerializedSize(withTags)); 129 ByteArrayOutputStream outKV = new ByteArrayOutputStream(kv.getSerializedSize(withTags)); 130 131 // compare the number of bytes written 132 assertEquals(kv.write(outKV, withTags), ic.write(outIC, withTags)); 133 // compare the underlying byte array 134 assertArrayEquals(outKV.getBuffer(), outIC.getBuffer()); 135 } 136 137 /** 138 * Verify getXXXArray() and getXXXLength() when family/qualifier/value/tags are null. Should have 139 * the same behaviors as {@link KeyValue}. 140 */ 141 @Test 142 public void testNullFamilyQualifierValueTags() { 143 byte[] row = Bytes.toBytes("row1"); 144 145 long timestamp = 5000L; 146 long seqId = 0L; 147 KeyValue.Type type = KeyValue.Type.Put; 148 149 // Test when following fields are null. 150 byte[] family = null; 151 byte[] qualifier = null; 152 byte[] value = null; 153 byte[] tags = null; 154 155 ExtendedCell ic1 = 156 new IndividualBytesFieldCell(row, family, qualifier, timestamp, type, seqId, value, tags); 157 158 ExtendedCell kv1 = new KeyValue(row, family, qualifier, timestamp, type, value, tags); 159 byte[] familyArrayInKV = 160 Bytes.copy(kv1.getFamilyArray(), kv1.getFamilyOffset(), kv1.getFamilyLength()); 161 byte[] qualifierArrayInKV = 162 Bytes.copy(kv1.getQualifierArray(), kv1.getQualifierOffset(), kv1.getQualifierLength()); 163 byte[] valueArrayInKV = 164 Bytes.copy(kv1.getValueArray(), kv1.getValueOffset(), kv1.getValueLength()); 165 byte[] tagsArrayInKV = Bytes.copy(kv1.getTagsArray(), kv1.getTagsOffset(), kv1.getTagsLength()); 166 167 // getXXXArray() for family, qualifier, value and tags are supposed to return empty byte array, 168 // rather than null. 169 assertArrayEquals(familyArrayInKV, ic1.getFamilyArray()); 170 assertArrayEquals(qualifierArrayInKV, ic1.getQualifierArray()); 171 assertArrayEquals(valueArrayInKV, ic1.getValueArray()); 172 assertArrayEquals(tagsArrayInKV, ic1.getTagsArray()); 173 174 // getXXXLength() for family, qualifier, value and tags are supposed to return 0. 175 assertEquals(kv1.getFamilyLength(), ic1.getFamilyLength()); 176 assertEquals(kv1.getQualifierLength(), ic1.getQualifierLength()); 177 assertEquals(kv1.getValueLength(), ic1.getValueLength()); 178 assertEquals(kv1.getTagsLength(), ic1.getTagsLength()); 179 } 180 181 @Test 182 public void testIfExtendedCellImplemented() { 183 // Verify if ExtendedCell interface is implemented 184 ExtendedCell ec = (ExtendedCell) ic0; 185 ec.deepClone(); // Do something with ec 186 } 187 188 @Test(expected = IllegalArgumentException.class) 189 public void testIllegalRow() { 190 new IndividualBytesFieldCell(Bytes.toBytes("row"), 0, 100, HConstants.EMPTY_BYTE_ARRAY, 0, 0, 191 HConstants.EMPTY_BYTE_ARRAY, 0, 0, 0L, KeyValue.Type.Put, 0, HConstants.EMPTY_BYTE_ARRAY, 0, 192 0, HConstants.EMPTY_BYTE_ARRAY, 0, 0); 193 } 194 195 @Test(expected = IllegalArgumentException.class) 196 public void testIllegalFamily() { 197 new IndividualBytesFieldCell(Bytes.toBytes("row"), 0, 3, Bytes.toBytes("family"), 0, 100, 198 HConstants.EMPTY_BYTE_ARRAY, 0, 0, 0L, KeyValue.Type.Put, 0, HConstants.EMPTY_BYTE_ARRAY, 0, 199 0, HConstants.EMPTY_BYTE_ARRAY, 0, 0); 200 } 201 202 @Test(expected = IllegalArgumentException.class) 203 public void testIllegalQualifier() { 204 new IndividualBytesFieldCell(Bytes.toBytes("row"), 0, 3, Bytes.toBytes("family"), 0, 6, 205 Bytes.toBytes("qualifier"), 0, 100, 0L, KeyValue.Type.Put, 0, HConstants.EMPTY_BYTE_ARRAY, 0, 206 0, HConstants.EMPTY_BYTE_ARRAY, 0, 0); 207 } 208 209 @Test(expected = IllegalArgumentException.class) 210 public void testIllegalTimestamp() { 211 new IndividualBytesFieldCell(Bytes.toBytes("row"), 0, 3, Bytes.toBytes("family"), 0, 6, 212 Bytes.toBytes("qualifier"), 0, 9, -100, KeyValue.Type.Put, 0, HConstants.EMPTY_BYTE_ARRAY, 0, 213 0, HConstants.EMPTY_BYTE_ARRAY, 0, 0); 214 } 215 216 @Test(expected = IllegalArgumentException.class) 217 public void testIllegalValue() { 218 new IndividualBytesFieldCell(Bytes.toBytes("row"), 0, 3, Bytes.toBytes("family"), 0, 6, 219 Bytes.toBytes("qualifier"), 0, 9, 0L, KeyValue.Type.Put, 0, Bytes.toBytes("value"), 0, 100, 220 HConstants.EMPTY_BYTE_ARRAY, 0, 0); 221 } 222 223 @Test(expected = IllegalArgumentException.class) 224 public void testIllegalTags() { 225 new IndividualBytesFieldCell(Bytes.toBytes("row"), 0, 3, Bytes.toBytes("family"), 0, 6, 226 Bytes.toBytes("qualifier"), 0, 9, 0L, KeyValue.Type.Put, 0, Bytes.toBytes("value"), 0, 5, 227 Bytes.toBytes("tags"), 0, 100); 228 } 229 230 @Test 231 public void testWriteTag() throws IOException { 232 byte[] tags = Bytes.toBytes("---tags---"); 233 int tagOffset = 3; 234 int length = 4; 235 IndividualBytesFieldCell cell = new IndividualBytesFieldCell(Bytes.toBytes("row"), 0, 3, 236 Bytes.toBytes("family"), 0, 6, Bytes.toBytes("qualifier"), 0, 9, 0L, KeyValue.Type.Put, 0, 237 Bytes.toBytes("value"), 0, 5, tags, tagOffset, length); 238 239 try (ByteArrayOutputStream output = new ByteArrayOutputStream(300)) { 240 cell.write(output, true); 241 byte[] buf = output.toByteArray(); 242 assertEquals(cell.getSerializedSize(true), buf.length); 243 } 244 } 245 246 @Test 247 public void testWriteValue() throws IOException { 248 byte[] value = Bytes.toBytes("---value---"); 249 int valueOffset = 3; 250 int valueLength = 5; 251 IndividualBytesFieldCell cell = new IndividualBytesFieldCell(Bytes.toBytes("row"), 0, 3, 252 Bytes.toBytes("family"), 0, 6, Bytes.toBytes("qualifier"), 0, 9, 0L, KeyValue.Type.Put, 0, 253 value, valueOffset, valueLength, Bytes.toBytes("value"), 0, 5); 254 255 try (ByteArrayOutputStream output = new ByteArrayOutputStream(300)) { 256 cell.write(output, true); 257 byte[] buf = output.toByteArray(); 258 assertEquals(cell.getSerializedSize(true), buf.length); 259 } 260 } 261}