mirror of
https://gitflic.ru/project/erthink/libmdbx.git
synced 2025-01-26 01:56:05 +00:00
mdbx-tests: добавление extra/txn
.
This commit is contained in:
parent
dc98f06d2c
commit
10ac9a9c50
@ -306,6 +306,7 @@ else()
|
|||||||
add_extra_test(crunched_delete TIMEOUT 10800)
|
add_extra_test(crunched_delete TIMEOUT 10800)
|
||||||
add_extra_test(dbi)
|
add_extra_test(dbi)
|
||||||
add_extra_test(open)
|
add_extra_test(open)
|
||||||
|
add_extra_test(txn)
|
||||||
endif()
|
endif()
|
||||||
add_extra_test(hex_base64_base58)
|
add_extra_test(hex_base64_base58)
|
||||||
endif()
|
endif()
|
||||||
|
292
test/extra/txn.c++
Normal file
292
test/extra/txn.c++
Normal file
@ -0,0 +1,292 @@
|
|||||||
|
#include "mdbx.h++"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
#if !defined(__cpp_lib_latch) && __cpp_lib_latch < 201907L
|
||||||
|
|
||||||
|
int main(int argc, const char *argv[]) {
|
||||||
|
(void)argc;
|
||||||
|
(void)argv;
|
||||||
|
std::cout << "FAKE-OK (since no C++20 std::thread and/or std::latch)\n";
|
||||||
|
return EXIT_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
#include <latch>
|
||||||
|
#include <thread>
|
||||||
|
|
||||||
|
static char log_buffer[1024];
|
||||||
|
|
||||||
|
static void logger_nofmt(MDBX_log_level_t loglevel, const char *function, int line, const char *msg,
|
||||||
|
unsigned length) noexcept {
|
||||||
|
(void)length;
|
||||||
|
(void)loglevel;
|
||||||
|
fprintf(stdout, "%s:%u %s", function, line, msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, const char *argv[]) {
|
||||||
|
(void)argc;
|
||||||
|
(void)argv;
|
||||||
|
bool ok = true;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
mdbx_setup_debug_nofmt(MDBX_LOG_VERBOSE, MDBX_DBG_ASSERT, logger_nofmt, log_buffer, sizeof(log_buffer));
|
||||||
|
|
||||||
|
mdbx::path path = "test-txn";
|
||||||
|
mdbx::env::remove(path);
|
||||||
|
mdbx::env::operate_parameters operateParameters(100, 10);
|
||||||
|
|
||||||
|
{
|
||||||
|
mdbx::env_managed::create_parameters createParameters;
|
||||||
|
createParameters.geometry.make_dynamic(21 * mdbx::env::geometry::MiB, 84 * mdbx::env::geometry::MiB);
|
||||||
|
|
||||||
|
operateParameters.options.no_sticky_threads = false;
|
||||||
|
mdbx::env_managed env(path, createParameters, operateParameters);
|
||||||
|
auto txn = env.start_write(false);
|
||||||
|
/* mdbx::map_handle testHandle = */ txn.create_map("xyz", mdbx::key_mode::usual, mdbx::value_mode::single);
|
||||||
|
txn.commit();
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
txn = env.start_write();
|
||||||
|
MDBX_txn *c_txn = txn;
|
||||||
|
err = mdbx_txn_reset(txn);
|
||||||
|
assert(err == MDBX_EINVAL);
|
||||||
|
ok = ok && err == MDBX_EINVAL;
|
||||||
|
|
||||||
|
err = mdbx_txn_break(txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
|
||||||
|
err = mdbx_txn_commit(txn);
|
||||||
|
assert(err == MDBX_RESULT_TRUE);
|
||||||
|
ok = ok && err == MDBX_RESULT_TRUE;
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
assert(c_txn == (const MDBX_txn *)txn);
|
||||||
|
|
||||||
|
err = mdbx_txn_break(txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
|
||||||
|
err = mdbx_txn_reset(txn);
|
||||||
|
assert(err == MDBX_EINVAL);
|
||||||
|
ok = ok && err == MDBX_EINVAL;
|
||||||
|
|
||||||
|
err = mdbx_txn_commit(txn);
|
||||||
|
assert(err == MDBX_RESULT_TRUE);
|
||||||
|
ok = ok && err == MDBX_RESULT_TRUE;
|
||||||
|
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
//-------------------------------------
|
||||||
|
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
assert(c_txn == (const MDBX_txn *)txn);
|
||||||
|
txn.commit();
|
||||||
|
|
||||||
|
err = mdbx_txn_reset(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
err = mdbx_txn_break(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
|
||||||
|
//=====================================
|
||||||
|
|
||||||
|
txn = env.start_read();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.make_broken();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.reset_reading();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.abort();
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
|
||||||
|
txn = env.start_read();
|
||||||
|
txn.reset_reading();
|
||||||
|
txn.make_broken();
|
||||||
|
txn.abort();
|
||||||
|
|
||||||
|
//=====================================
|
||||||
|
|
||||||
|
std::latch s(1);
|
||||||
|
txn = env.start_read();
|
||||||
|
c_txn = txn;
|
||||||
|
|
||||||
|
std::thread t([&]() {
|
||||||
|
s.wait();
|
||||||
|
err = mdbx_txn_reset(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
err = mdbx_txn_break(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
err = mdbx_txn_commit(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
});
|
||||||
|
|
||||||
|
s.count_down();
|
||||||
|
t.join();
|
||||||
|
}
|
||||||
|
|
||||||
|
//=====================================
|
||||||
|
//=====================================
|
||||||
|
|
||||||
|
{
|
||||||
|
operateParameters.options.no_sticky_threads = true;
|
||||||
|
operateParameters.options.nested_write_transactions = true;
|
||||||
|
mdbx::env_managed env(path, operateParameters);
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
auto txn = env.start_write();
|
||||||
|
MDBX_txn *c_txn = txn;
|
||||||
|
err = mdbx_txn_reset(txn);
|
||||||
|
assert(err == MDBX_EINVAL);
|
||||||
|
ok = ok && err == MDBX_EINVAL;
|
||||||
|
|
||||||
|
err = mdbx_txn_break(txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
|
||||||
|
err = mdbx_txn_commit(txn);
|
||||||
|
assert(err == MDBX_RESULT_TRUE);
|
||||||
|
ok = ok && err == MDBX_RESULT_TRUE;
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
assert(c_txn == (const MDBX_txn *)txn);
|
||||||
|
|
||||||
|
err = mdbx_txn_break(txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
|
||||||
|
err = mdbx_txn_reset(txn);
|
||||||
|
assert(err == MDBX_EINVAL);
|
||||||
|
ok = ok && err == MDBX_EINVAL;
|
||||||
|
|
||||||
|
err = mdbx_txn_commit(txn);
|
||||||
|
assert(err == MDBX_RESULT_TRUE);
|
||||||
|
ok = ok && err == MDBX_RESULT_TRUE;
|
||||||
|
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
//-------------------------------------
|
||||||
|
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
assert(c_txn == (const MDBX_txn *)txn);
|
||||||
|
txn.commit();
|
||||||
|
|
||||||
|
err = mdbx_txn_reset(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
err = mdbx_txn_break(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
|
||||||
|
//=====================================
|
||||||
|
|
||||||
|
txn = env.start_read();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.make_broken();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.reset_reading();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.abort();
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
|
||||||
|
txn = env.start_read();
|
||||||
|
txn.reset_reading();
|
||||||
|
txn.make_broken();
|
||||||
|
txn.abort();
|
||||||
|
|
||||||
|
//=====================================
|
||||||
|
|
||||||
|
std::latch s1(1), s2(1), s3(1);
|
||||||
|
txn = env.start_read();
|
||||||
|
c_txn = txn;
|
||||||
|
|
||||||
|
std::thread t([&]() {
|
||||||
|
s1.wait();
|
||||||
|
err = mdbx_txn_break(c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
err = mdbx_txn_reset(c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
txn.renew_reading();
|
||||||
|
s2.count_down();
|
||||||
|
|
||||||
|
s3.wait();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
err = mdbx_txn_commit(c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
c_txn = txn;
|
||||||
|
err = mdbx_txn_commit(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
err = mdbx_txn_break(c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
err = mdbx_txn_reset(c_txn);
|
||||||
|
assert(err == MDBX_EINVAL);
|
||||||
|
ok = ok && err == MDBX_EINVAL;
|
||||||
|
});
|
||||||
|
|
||||||
|
s1.count_down();
|
||||||
|
s2.wait();
|
||||||
|
txn.commit();
|
||||||
|
txn = env.start_write();
|
||||||
|
s3.count_down();
|
||||||
|
|
||||||
|
t.join();
|
||||||
|
txn.abort();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << (ok ? "OK\n" : "FAIL\n");
|
||||||
|
return ok ? EXIT_SUCCESS : EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* __cpp_lib_latch */
|
Loading…
Reference in New Issue
Block a user