HBASE-11788 HBase is not deleting the cell when a Put with a KeyValue, KeyValue.Type.Delete is submitted
This commit is contained in:
parent
9bd9b26a0e
commit
095a0e9aa2
|
@ -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
|
||||||
*
|
*
|
||||||
|
|
|
@ -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();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue