HBASE-11788 HBase is not deleting the cell when a Put with a KeyValue, KeyValue.Type.Delete is submitted

This commit is contained in:
Srikanth Srungarapu 2014-08-26 11:25:24 -07:00 committed by Jimmy Xiang
parent 9bd9b26a0e
commit 095a0e9aa2
2 changed files with 162 additions and 32 deletions

View File

@ -18,24 +18,17 @@
package org.apache.hadoop.hbase.protobuf; package org.apache.hadoop.hbase.protobuf;
import static org.apache.hadoop.hbase.protobuf.generated.HBaseProtos.RegionSpecifier.RegionSpecifierType.REGION_NAME; import com.google.common.collect.ArrayListMultimap;
import com.google.common.collect.ListMultimap;
import java.io.ByteArrayOutputStream; import com.google.common.collect.Lists;
import java.io.IOException; import com.google.protobuf.ByteString;
import java.lang.reflect.Constructor; import com.google.protobuf.InvalidProtocolBufferException;
import java.lang.reflect.InvocationTargetException; import com.google.protobuf.Message;
import java.lang.reflect.Method; import com.google.protobuf.Parser;
import java.lang.reflect.ParameterizedType; import com.google.protobuf.RpcChannel;
import java.lang.reflect.Type; import com.google.protobuf.Service;
import java.nio.ByteBuffer; import com.google.protobuf.ServiceException;
import java.util.ArrayList; import com.google.protobuf.TextFormat;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.NavigableSet;
import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.Path; import org.apache.hadoop.fs.Path;
import org.apache.hadoop.hbase.Cell; import org.apache.hadoop.hbase.Cell;
@ -138,17 +131,23 @@ import org.apache.hadoop.io.Text;
import org.apache.hadoop.ipc.RemoteException; import org.apache.hadoop.ipc.RemoteException;
import org.apache.hadoop.security.token.Token; import org.apache.hadoop.security.token.Token;
import com.google.common.collect.ArrayListMultimap; import java.io.ByteArrayOutputStream;
import com.google.common.collect.ListMultimap; import java.io.IOException;
import com.google.common.collect.Lists; import java.lang.reflect.Constructor;
import com.google.protobuf.ByteString; import java.lang.reflect.InvocationTargetException;
import com.google.protobuf.InvalidProtocolBufferException; import java.lang.reflect.Method;
import com.google.protobuf.Message; import java.lang.reflect.ParameterizedType;
import com.google.protobuf.Parser; import java.lang.reflect.Type;
import com.google.protobuf.RpcChannel; import java.nio.ByteBuffer;
import com.google.protobuf.Service; import java.util.ArrayList;
import com.google.protobuf.ServiceException; import java.util.Collection;
import com.google.protobuf.TextFormat; import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.NavigableSet;
import static org.apache.hadoop.hbase.protobuf.generated.HBaseProtos.RegionSpecifier.RegionSpecifierType.REGION_NAME;
/** /**
* Protobufs utility. * Protobufs utility.
@ -579,9 +578,21 @@ public final class ProtobufUtil {
for(int i = 0; i< array.length; i++) { for(int i = 0; i< array.length; i++) {
tagArray[i] = (Tag)array[i]; tagArray[i] = (Tag)array[i];
} }
put.addImmutable(family, qualifier, ts, value, tagArray); if(qv.hasDeleteType()) {
byte[] qual = qv.hasQualifier() ? qv.getQualifier().toByteArray() : null;
put.add(new KeyValue(proto.getRow().toByteArray(), family, qual, ts,
fromDeleteType(qv.getDeleteType()), null, tags));
} else {
put.addImmutable(family, qualifier, ts, value, tagArray);
}
} else { } else {
put.addImmutable(family, qualifier, ts, value); if(qv.hasDeleteType()) {
byte[] qual = qv.hasQualifier() ? qv.getQualifier().toByteArray() : null;
put.add(new KeyValue(proto.getRow().toByteArray(), family, qual, ts,
fromDeleteType(qv.getDeleteType())));
} else{
put.addImmutable(family, qualifier, ts, value);
}
} }
} }
} }
@ -1167,7 +1178,7 @@ public final class ProtobufUtil {
valueBuilder.setTags(ByteStringer.wrap(kv.getTagsArray(), kv.getTagsOffset(), valueBuilder.setTags(ByteStringer.wrap(kv.getTagsArray(), kv.getTagsOffset(),
kv.getTagsLength())); kv.getTagsLength()));
} }
if (type == MutationType.DELETE) { if (type == MutationType.DELETE || (type == MutationType.PUT && CellUtil.isDelete(kv))) {
KeyValue.Type keyValueType = KeyValue.Type.codeToType(kv.getType()); KeyValue.Type keyValueType = KeyValue.Type.codeToType(kv.getType());
valueBuilder.setDeleteType(toDeleteType(keyValueType)); valueBuilder.setDeleteType(toDeleteType(keyValueType));
} }
@ -1469,6 +1480,29 @@ public final class ProtobufUtil {
} }
} }
/**
* Convert a protocol buffer DeleteType to delete KeyValue type.
*
* @param protocol buffer DeleteType
* @return type
* @throws IOException
*/
public static KeyValue.Type fromDeleteType(
DeleteType type) throws IOException {
switch (type) {
case DELETE_ONE_VERSION:
return KeyValue.Type.Delete;
case DELETE_MULTIPLE_VERSIONS:
return KeyValue.Type.DeleteColumn;
case DELETE_FAMILY:
return KeyValue.Type.DeleteFamily;
case DELETE_FAMILY_VERSION:
return KeyValue.Type.DeleteFamilyVersion;
default:
throw new IOException("Unknown delete type: " + type);
}
}
/** /**
* Convert a stringified protocol buffer exception Parameter to a Java Exception * Convert a stringified protocol buffer exception Parameter to a Java Exception
* *

View File

@ -0,0 +1,96 @@
/**
*
* 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.hadoop.hbase.client;
import org.apache.hadoop.hbase.*;
import org.apache.hadoop.hbase.util.Bytes;
import org.junit.AfterClass;
import org.junit.BeforeClass;
import org.junit.Test;
import org.junit.experimental.categories.Category;
import static org.junit.Assert.assertTrue;
@Category(MediumTests.class)
public class TestPutWithDelete {
private static final HBaseTestingUtility TEST_UTIL = new HBaseTestingUtility();
/**
* @throws java.lang.Exception
*/
@BeforeClass
public static void setUpBeforeClass() throws Exception {
TEST_UTIL.startMiniCluster();
}
/**
* @throws java.lang.Exception
*/
@AfterClass
public static void tearDownAfterClass() throws Exception {
TEST_UTIL.shutdownMiniCluster();
}
@Test
public void testHbasePutDeleteCell() throws Exception {
final TableName tableName = TableName.valueOf("TestPutWithDelete");
final byte[] rowKey = Bytes.toBytes("12345");
final byte[] family = Bytes.toBytes("cf");
HTable table = TEST_UTIL.createTable(tableName, family);
TEST_UTIL.waitTableAvailable(tableName.getName(), 5000);
try {
// put one row
Put put = new Put(rowKey);
put.add(family, Bytes.toBytes("A"), Bytes.toBytes("a"));
put.add(family, Bytes.toBytes("B"), Bytes.toBytes("b"));
put.add(family, Bytes.toBytes("C"), Bytes.toBytes("c"));
table.put(put);
// get row back and assert the values
Get get = new Get(rowKey);
Result result = table.get(get);
assertTrue("Column A value should be a",
Bytes.toString(result.getValue(family, Bytes.toBytes("A"))).equals("a"));
assertTrue("Column B value should be b",
Bytes.toString(result.getValue(family, Bytes.toBytes("B"))).equals("b"));
assertTrue("Column C value should be c",
Bytes.toString(result.getValue(family, Bytes.toBytes("C"))).equals("c"));
// put the same row again with C column deleted
put = new Put(rowKey);
put.add(family, Bytes.toBytes("A"), Bytes.toBytes("a"));
put.add(family, Bytes.toBytes("B"), Bytes.toBytes("b"));
KeyValue marker = new KeyValue(rowKey, family, Bytes.toBytes("C"),
HConstants.LATEST_TIMESTAMP, KeyValue.Type.DeleteColumn);
put.add(marker);
table.put(put);
// get row back and assert the values
get = new Get(rowKey);
result = table.get(get);
assertTrue("Column A value should be a",
Bytes.toString(result.getValue(family, Bytes.toBytes("A"))).equals("a"));
assertTrue("Column B value should be b",
Bytes.toString(result.getValue(family, Bytes.toBytes("B"))).equals("b"));
assertTrue("Column C should not exist",
result.getValue(family, Bytes.toBytes("C")) == null);
} finally {
table.close();
}
}
}