Skip to content

Commit

Permalink
Extend StringAppendTESTOperator to use string delimiter of variable l…
Browse files Browse the repository at this point in the history
…ength
  • Loading branch information
azagrebin committed Jan 31, 2019
1 parent f438b98 commit 94f52f9
Show file tree
Hide file tree
Showing 6 changed files with 93 additions and 25 deletions.
22 changes: 22 additions & 0 deletions java/rocksjni/merge_operator.cc
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,28 @@ jlong Java_org_rocksdb_StringAppendOperator_newSharedStringAppendOperator(
return reinterpret_cast<jlong>(sptr_string_append_op);
}

/*
* Class: org_rocksdb_StringAppendOperator
* Method: newSharedStringAppendTESTOperator
* Signature: ([B)J
*/
jlong Java_org_rocksdb_StringAppendOperator_newSharedStringAppendTESTOperator(
JNIEnv* env, jclass /*jclazz*/, jbyteArray jdelim) {
jboolean has_exception = JNI_FALSE;
std::string delim = rocksdb::JniUtil::byteString<std::string>(
env, jdelim,
[](const char* str, const size_t len) { return std::string(str, len); },
&has_exception);
if (has_exception == JNI_TRUE) {
// exception occurred
return 0;
}

auto* sptr_string_append_test_op = new std::shared_ptr<rocksdb::MergeOperator>(
rocksdb::MergeOperators::CreateStringAppendTESTOperator(delim));
return reinterpret_cast<jlong>(sptr_string_append_test_op);
}

/*
* Class: org_rocksdb_StringAppendOperator
* Method: disposeInternal
Expand Down
15 changes: 15 additions & 0 deletions java/src/main/java/org/rocksdb/StringAppendOperator.java
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@

package org.rocksdb;

import java.nio.charset.Charset;

/**
* StringAppendOperator is a merge operator that concatenates
* two strings.
Expand All @@ -18,6 +20,19 @@ public StringAppendOperator(char delim) {
super(newSharedStringAppendOperator(delim));
}

public StringAppendOperator(byte[] delim) {
super(newSharedStringAppendTESTOperator(delim));
}

public StringAppendOperator(String delim) {
this(delim.getBytes());
}

public StringAppendOperator(String delim, Charset charset) {
this(delim.getBytes(charset));
}

private native static long newSharedStringAppendOperator(final char delim);
private native static long newSharedStringAppendTESTOperator(final byte[] delim);
@Override protected final native void disposeInternal(final long handle);
}
28 changes: 28 additions & 0 deletions java/src/test/java/org/rocksdb/MergeTest.java
Original file line number Diff line number Diff line change
Expand Up @@ -108,6 +108,34 @@ public void operatorOption()
}
}

@Test
public void stringDelimiter() throws RocksDBException {
stringDelimiter("DELIM");
stringDelimiter("");
}

private void stringDelimiter(String delim) throws RocksDBException {
try (final StringAppendOperator stringAppendOperator = new StringAppendOperator(delim.getBytes());
final Options opt = new Options()
.setCreateIfMissing(true)
.setMergeOperator(stringAppendOperator);
final RocksDB db = RocksDB.open(opt, dbFolder.getRoot().getAbsolutePath())) {
// Writing aa under key
db.put("key".getBytes(), "aa".getBytes());

// Writing bb under key
db.merge("key".getBytes(), "bb".getBytes());

// Writing empty under key
db.merge("key".getBytes(), "".getBytes());

final byte[] value = db.get("key".getBytes());
final String strValue = new String(value);

assertThat(strValue).isEqualTo("aa" + delim + "bb" + delim);
}
}

@Test
public void cFOperatorOption()
throws InterruptedException, RocksDBException {
Expand Down
1 change: 1 addition & 0 deletions utilities/merge_operators.h
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ class MergeOperators {
static std::shared_ptr<MergeOperator> CreateStringAppendOperator();
static std::shared_ptr<MergeOperator> CreateStringAppendOperator(char delim_char);
static std::shared_ptr<MergeOperator> CreateStringAppendTESTOperator();
static std::shared_ptr<MergeOperator> CreateStringAppendTESTOperator(std::string delim);
static std::shared_ptr<MergeOperator> CreateMaxOperator();
static std::shared_ptr<MergeOperator> CreateBytesXOROperator();

Expand Down
28 changes: 14 additions & 14 deletions utilities/merge_operators/string_append/stringappend2.cc
Original file line number Diff line number Diff line change
Expand Up @@ -15,11 +15,6 @@

namespace rocksdb {

// Constructor: also specify the delimiter character.
StringAppendTESTOperator::StringAppendTESTOperator(char delim_char)
: delim_(delim_char) {
}

// Implementation for the merge operation (concatenates two strings)
bool StringAppendTESTOperator::FullMergeV2(
const MergeOperationInput& merge_in,
Expand All @@ -37,7 +32,7 @@ bool StringAppendTESTOperator::FullMergeV2(
size_t numBytes = 0;
for (auto it = merge_in.operand_list.begin();
it != merge_in.operand_list.end(); ++it) {
numBytes += it->size() + 1; // Plus 1 for the delimiter
numBytes += it->size() + delim_.size(); // Plus one delimiter
}

// Only print the delimiter after the first entry has been printed
Expand All @@ -48,20 +43,20 @@ bool StringAppendTESTOperator::FullMergeV2(
merge_out->new_value.reserve(numBytes + merge_in.existing_value->size());
merge_out->new_value.append(merge_in.existing_value->data(),
merge_in.existing_value->size());
printDelim = true;
printDelim = !delim_.empty();
} else if (numBytes) {
merge_out->new_value.reserve(
numBytes - 1); // Minus 1 since we have one less delimiter
numBytes - delim_.size()); // Minus 1 delimiter since we have one less delimiter
}

// Concatenate the sequence of strings (and add a delimiter between each)
for (auto it = merge_in.operand_list.begin();
it != merge_in.operand_list.end(); ++it) {
if (printDelim) {
merge_out->new_value.append(1, delim_);
merge_out->new_value.append(delim_);
}
merge_out->new_value.append(it->data(), it->size());
printDelim = true;
printDelim = !delim_.empty();
}

return true;
Expand All @@ -87,17 +82,17 @@ bool StringAppendTESTOperator::_AssocPartialMergeMulti(
// Determine and reserve correct size for *new_value.
size_t size = 0;
for (const auto& operand : operand_list) {
size += operand.size();
size += operand.size() + delim_.size();
}
size += operand_list.size() - 1; // Delimiters
size -= delim_.size(); // since we have one less delimiter
new_value->reserve(size);

// Apply concatenation
new_value->assign(operand_list.front().data(), operand_list.front().size());

for (std::deque<Slice>::const_iterator it = operand_list.begin() + 1;
for (auto it = operand_list.begin() + 1;
it != operand_list.end(); ++it) {
new_value->append(1, delim_);
new_value->append(delim_);
new_value->append(it->data(), it->size());
}

Expand All @@ -114,4 +109,9 @@ MergeOperators::CreateStringAppendTESTOperator() {
return std::make_shared<StringAppendTESTOperator>(',');
}

std::shared_ptr<MergeOperator>
MergeOperators::CreateStringAppendTESTOperator(std::string delim) {
return std::make_shared<StringAppendTESTOperator>(delim);
}

} // namespace rocksdb
24 changes: 13 additions & 11 deletions utilities/merge_operators/string_append/stringappend2.h
Original file line number Diff line number Diff line change
Expand Up @@ -13,26 +13,28 @@
#pragma once
#include <deque>
#include <string>

#include <utility>
#include "rocksdb/merge_operator.h"
#include "rocksdb/slice.h"

namespace rocksdb {

class StringAppendTESTOperator : public MergeOperator {
public:
// Constructor with delimiter
explicit StringAppendTESTOperator(char delim_char);
// Constructor with string delimiter
explicit StringAppendTESTOperator(std::string delim_str) : delim_(std::move(delim_str)) {};

// Constructor with char delimiter
explicit StringAppendTESTOperator(char delim_char) : delim_(std::string(1, delim_char)) {};

virtual bool FullMergeV2(const MergeOperationInput& merge_in,
MergeOperationOutput* merge_out) const override;
bool FullMergeV2(const MergeOperationInput& merge_in,
MergeOperationOutput* merge_out) const override;

virtual bool PartialMergeMulti(const Slice& key,
const std::deque<Slice>& operand_list,
std::string* new_value, Logger* logger) const
override;
bool PartialMergeMulti(const Slice& key,
const std::deque<Slice>& operand_list,
std::string* new_value, Logger* logger) const override;

virtual const char* Name() const override;
const char* Name() const override;

private:
// A version of PartialMerge that actually performs "partial merging".
Expand All @@ -41,7 +43,7 @@ class StringAppendTESTOperator : public MergeOperator {
const std::deque<Slice>& operand_list,
std::string* new_value, Logger* logger) const;

char delim_; // The delimiter is inserted between elements
std::string delim_; // The delimiter is inserted between elements

};

Expand Down

0 comments on commit 94f52f9

Please sign in to comment.