2025-01-15 19:30:00 +03:00
|
|
|
/// \author Леонид Юрьев aka Leonid Yuriev <leo@yuriev.ru> \date 2015-2025
|
2024-05-19 22:07:58 +03:00
|
|
|
/// \copyright SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2023-11-15 23:38:31 +03:00
|
|
|
#include "mdbx.h++"
|
|
|
|
#include <array>
|
|
|
|
#include <functional>
|
|
|
|
#include <iostream>
|
|
|
|
#include <random>
|
|
|
|
|
2024-12-11 21:22:04 +03:00
|
|
|
static ::std::ostream &operator<<(::std::ostream &out, const mdbx::cursor::move_operation op) {
|
2023-11-15 23:38:31 +03:00
|
|
|
static const char *const str[] = {"FIRST",
|
|
|
|
"FIRST_DUP",
|
|
|
|
"GET_BOTH",
|
|
|
|
"GET_BOTH_RANGE",
|
|
|
|
"GET_CURRENT",
|
|
|
|
"GET_MULTIPLE",
|
|
|
|
"LAST",
|
|
|
|
"LAST_DUP",
|
|
|
|
"NEXT",
|
|
|
|
"NEXT_DUP",
|
|
|
|
"NEXT_MULTIPLE",
|
|
|
|
"NEXT_NODUP",
|
|
|
|
"PREV",
|
|
|
|
"PREV_DUP",
|
|
|
|
"PREV_NODUP",
|
|
|
|
"SET",
|
|
|
|
"SET_KEY",
|
|
|
|
"SET_RANGE",
|
|
|
|
"PREV_MULTIPLE",
|
|
|
|
"SET_LOWERBOUND",
|
|
|
|
"SET_UPPERBOUND",
|
|
|
|
"TO_KEY_LESSER_THAN",
|
|
|
|
"TO_KEY_LESSER_OR_EQUAL",
|
|
|
|
"TO_KEY_EQUAL",
|
|
|
|
"TO_KEY_GREATER_OR_EQUAL",
|
|
|
|
"TO_KEY_GREATER_THAN",
|
|
|
|
"TO_EXACT_KEY_VALUE_LESSER_THAN",
|
|
|
|
"TO_EXACT_KEY_VALUE_LESSER_OR_EQUAL",
|
|
|
|
"TO_EXACT_KEY_VALUE_EQUAL",
|
|
|
|
"TO_EXACT_KEY_VALUE_GREATER_OR_EQUAL",
|
|
|
|
"TO_EXACT_KEY_VALUE_GREATER_THAN",
|
|
|
|
"TO_PAIR_LESSER_THAN",
|
|
|
|
"TO_PAIR_LESSER_OR_EQUAL",
|
|
|
|
"TO_PAIR_EQUAL",
|
|
|
|
"TO_PAIR_GREATER_OR_EQUAL",
|
|
|
|
"TO_PAIR_GREATER_THAN"};
|
|
|
|
return out << str[op];
|
|
|
|
}
|
|
|
|
|
|
|
|
using buffer = mdbx::default_buffer;
|
|
|
|
using buffer_pair = mdbx::buffer_pair<buffer>;
|
|
|
|
|
|
|
|
std::default_random_engine prng(42);
|
|
|
|
|
|
|
|
static buffer random(const unsigned &value) {
|
|
|
|
switch (prng() % 3) {
|
|
|
|
default:
|
|
|
|
return buffer::hex(value);
|
|
|
|
case 1:
|
|
|
|
return buffer::base64(value);
|
|
|
|
case 2:
|
|
|
|
return buffer::base58(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static buffer random_key() { return random(prng() % 10007); }
|
|
|
|
|
|
|
|
static buffer random_value() { return random(prng() % 47); }
|
|
|
|
|
|
|
|
using predicate = std::function<bool(const mdbx::pair &, const mdbx::pair &)>;
|
|
|
|
|
2024-12-11 21:22:04 +03:00
|
|
|
static bool probe(mdbx::txn txn, mdbx::map_handle dbi, mdbx::cursor::move_operation op, predicate cmp,
|
2023-11-15 23:38:31 +03:00
|
|
|
const buffer_pair &pair) {
|
|
|
|
auto seeker = txn.open_cursor(dbi);
|
|
|
|
auto scanner = seeker.clone();
|
|
|
|
|
2024-12-11 21:22:04 +03:00
|
|
|
const bool scan_backward = op == mdbx::cursor::key_lesser_than || op == mdbx::cursor::key_lesser_or_equal ||
|
|
|
|
op == mdbx::cursor::multi_exactkey_value_lesser_than ||
|
|
|
|
op == mdbx::cursor::multi_exactkey_value_lesser_or_equal ||
|
|
|
|
op == mdbx::cursor::pair_lesser_than || op == mdbx::cursor::pair_lesser_or_equal;
|
2023-11-15 23:38:31 +03:00
|
|
|
|
|
|
|
const bool is_multi = mdbx::is_multi(txn.get_handle_info(dbi).value_mode());
|
|
|
|
|
|
|
|
auto seek_result = seeker.move(op, pair.key, pair.value, false);
|
2024-12-11 21:22:04 +03:00
|
|
|
auto scan_result =
|
|
|
|
scanner.fullscan([cmp, &pair](const mdbx::pair &scan) -> bool { return cmp(scan, pair); }, scan_backward);
|
2023-11-15 23:38:31 +03:00
|
|
|
if (seek_result.done == scan_result &&
|
|
|
|
(!scan_result ||
|
2024-12-11 21:22:04 +03:00
|
|
|
seeker.is_same_position(scanner, op < mdbx::cursor::multi_exactkey_value_lesser_than && is_multi)))
|
2023-11-15 23:38:31 +03:00
|
|
|
return true;
|
|
|
|
|
|
|
|
std::cerr << std::endl;
|
|
|
|
std::cerr << "bug:";
|
|
|
|
std::cerr << std::endl;
|
2024-12-11 21:22:04 +03:00
|
|
|
std::cerr << std::string(is_multi ? "multi" : "single") << "-map, op " << op << ", key " << pair.key << ", value "
|
|
|
|
<< pair.value;
|
2023-11-15 23:38:31 +03:00
|
|
|
std::cerr << std::endl;
|
|
|
|
std::cerr << "\tscanner: ";
|
|
|
|
if (scan_result)
|
2024-12-11 21:22:04 +03:00
|
|
|
std::cerr << " done, key " << scanner.current(false).key << ", value " << scanner.current(false).value;
|
2023-11-15 23:38:31 +03:00
|
|
|
else
|
|
|
|
std::cerr << "not-found";
|
|
|
|
std::cerr << std::endl;
|
2024-12-11 21:22:04 +03:00
|
|
|
std::cerr << "\t seeker: " << (seek_result.done ? " done" : "not-found") << ", key " << seek_result.key
|
|
|
|
<< ", value " << seek_result.value;
|
2023-11-15 23:38:31 +03:00
|
|
|
std::cerr << std::endl;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-12-11 21:22:04 +03:00
|
|
|
static bool probe(mdbx::txn txn, mdbx::map_handle dbi, mdbx::cursor::move_operation op, predicate cmp) {
|
2023-11-15 23:38:31 +03:00
|
|
|
const auto pair = buffer_pair(random_key(), random_value());
|
|
|
|
const bool ok = probe(txn, dbi, op, cmp, pair);
|
|
|
|
#if MDBX_DEBUG
|
|
|
|
if (!ok)
|
|
|
|
// повтор для отладки и поиска причин
|
|
|
|
probe(txn, dbi, op, cmp, pair);
|
|
|
|
#endif /* MDBX_DEBUG */
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool test(mdbx::txn txn, mdbx::map_handle dbi) {
|
|
|
|
bool ok = true;
|
|
|
|
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::key_lesser_than,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
|
|
|
return mdbx_cmp(txn, dbi, l.key, r.key) < 0;
|
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::key_lesser_or_equal,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
|
|
|
return mdbx_cmp(txn, dbi, l.key, r.key) <= 0;
|
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::key_equal,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
|
|
|
return mdbx_cmp(txn, dbi, l.key, r.key) == 0;
|
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::key_greater_or_equal,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
|
|
|
return mdbx_cmp(txn, dbi, l.key, r.key) >= 0;
|
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::key_greater_than,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
|
|
|
return mdbx_cmp(txn, dbi, l.key, r.key) > 0;
|
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::multi_exactkey_value_lesser_than,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
2024-12-11 21:22:04 +03:00
|
|
|
return mdbx_cmp(txn, dbi, l.key, r.key) == 0 && mdbx_dcmp(txn, dbi, l.value, r.value) < 0;
|
2023-11-15 23:38:31 +03:00
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::multi_exactkey_value_lesser_or_equal,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
2024-12-11 21:22:04 +03:00
|
|
|
return mdbx_cmp(txn, dbi, l.key, r.key) == 0 && mdbx_dcmp(txn, dbi, l.value, r.value) <= 0;
|
2023-11-15 23:38:31 +03:00
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::multi_exactkey_value_equal,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
2024-12-11 21:22:04 +03:00
|
|
|
return mdbx_cmp(txn, dbi, l.key, r.key) == 0 && mdbx_dcmp(txn, dbi, l.value, r.value) == 0;
|
2023-11-15 23:38:31 +03:00
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::multi_exactkey_value_greater_or_equal,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
2024-12-11 21:22:04 +03:00
|
|
|
return mdbx_cmp(txn, dbi, l.key, r.key) == 0 && mdbx_dcmp(txn, dbi, l.value, r.value) >= 0;
|
2023-11-15 23:38:31 +03:00
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::multi_exactkey_value_greater,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
2024-12-11 21:22:04 +03:00
|
|
|
return mdbx_cmp(txn, dbi, l.key, r.key) == 0 && mdbx_dcmp(txn, dbi, l.value, r.value) > 0;
|
2023-11-15 23:38:31 +03:00
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::pair_lesser_than,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
|
|
|
auto cmp = mdbx_cmp(txn, dbi, l.key, r.key);
|
|
|
|
if (cmp == 0)
|
|
|
|
cmp = mdbx_dcmp(txn, dbi, l.value, r.value);
|
|
|
|
return cmp < 0;
|
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::pair_lesser_or_equal,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
|
|
|
auto cmp = mdbx_cmp(txn, dbi, l.key, r.key);
|
|
|
|
if (cmp == 0)
|
|
|
|
cmp = mdbx_dcmp(txn, dbi, l.value, r.value);
|
|
|
|
return cmp <= 0;
|
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::pair_equal,
|
2024-12-11 21:22:04 +03:00
|
|
|
[](const mdbx::pair &l, const mdbx::pair &r) -> bool { return l == r; }) &&
|
2023-11-15 23:38:31 +03:00
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::pair_greater_or_equal,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
|
|
|
auto cmp = mdbx_cmp(txn, dbi, l.key, r.key);
|
|
|
|
if (cmp == 0)
|
|
|
|
cmp = mdbx_dcmp(txn, dbi, l.value, r.value);
|
|
|
|
return cmp >= 0;
|
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
ok = probe(txn, dbi, mdbx::cursor::pair_greater_than,
|
|
|
|
[txn, dbi](const mdbx::pair &l, const mdbx::pair &r) -> bool {
|
|
|
|
auto cmp = mdbx_cmp(txn, dbi, l.key, r.key);
|
|
|
|
if (cmp == 0)
|
|
|
|
cmp = mdbx_dcmp(txn, dbi, l.value, r.value);
|
|
|
|
return cmp > 0;
|
|
|
|
}) &&
|
|
|
|
ok;
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, const char *argv[]) {
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
|
|
|
|
2024-09-18 08:01:14 +03:00
|
|
|
mdbx::path db_filename = "test-posi";
|
|
|
|
mdbx::env_managed::remove(db_filename);
|
2024-12-11 21:22:04 +03:00
|
|
|
mdbx::env_managed env(db_filename, mdbx::env_managed::create_parameters(), mdbx::env::operate_parameters(3));
|
2023-11-15 23:38:31 +03:00
|
|
|
|
|
|
|
auto txn = env.start_write();
|
2024-12-11 21:22:04 +03:00
|
|
|
auto single = txn.create_map("single", mdbx::key_mode::usual, mdbx::value_mode::single);
|
|
|
|
auto multi = txn.create_map("multi", mdbx::key_mode::usual, mdbx::value_mode::multi);
|
2023-11-15 23:38:31 +03:00
|
|
|
for (size_t i = 0; i < 1000; ++i) {
|
|
|
|
auto key = random_key();
|
|
|
|
txn.upsert(single, key, random_value());
|
|
|
|
for (auto n = prng() % 5 + 1; n > 0; --n)
|
|
|
|
txn.upsert(multi, key, random_value());
|
|
|
|
}
|
|
|
|
txn.commit_embark_read();
|
|
|
|
|
|
|
|
bool ok = true;
|
|
|
|
for (size_t i = 0; ok && i < 3333; ++i) {
|
|
|
|
ok = test(txn, single) && ok;
|
|
|
|
ok = test(txn, multi) && ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ok) {
|
|
|
|
std::cerr << "Fail\n";
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
std::cout << "OK\n";
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|