blob: 5ff3d0173e863b629c2ee85d62f06d50a56dc2db [file] [log] [blame]
/*
* 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.
*/
#pragma once
#ifndef GEODE_INTEGRATION_TEST_THINCLIENTTRANSACTIONS_H_
#define GEODE_INTEGRATION_TEST_THINCLIENTTRANSACTIONS_H_
#include "fw_dunit.hpp"
#include <string>
#include <geode/TransactionId.hpp>
#include <geode/CacheTransactionManager.hpp>
#define ROOT_NAME "ThinClientTransactions"
#define ROOT_SCOPE DISTRIBUTED_ACK
#include "CacheHelper.hpp"
#include "util/concurrent/binary_semaphore.hpp"
namespace { // NOLINT(google-build-namespaces)
using apache::geode::client::binary_semaphore;
using apache::geode::client::CacheableKey;
using apache::geode::client::CacheableString;
using apache::geode::client::CacheHelper;
using apache::geode::client::CacheServerException;
using apache::geode::client::EntryExistsException;
using apache::geode::client::EntryNotFoundException;
using apache::geode::client::IllegalStateException;
using apache::geode::client::Properties;
using apache::geode::client::TransactionException;
using apache::geode::client::TransactionId;
#define CLIENT1 s1p1
#define CLIENT2 s1p2
#define SERVER1 s2p1
#define SERVER2 s2p2
#define CREATE_TWICE_KEY "__create_twice_key"
#define CREATE_TWICE_VALUE "__create_twice_value"
CacheHelper* cacheHelper = nullptr;
static bool isLocalServer = false;
static bool isLocator = false;
static int numberOfLocators = 0;
const std::string locatorsG =
CacheHelper::getLocatorHostPort(isLocator, isLocalServer, numberOfLocators);
void initClient(const bool isthinClient) {
if (cacheHelper == nullptr) {
auto config = Properties::create();
config->insert("suspended-tx-timeout", std::chrono::minutes(1));
cacheHelper = new CacheHelper(isthinClient, config);
}
ASSERT(cacheHelper, "Failed to create a CacheHelper client instance.");
}
void cleanProc() {
if (cacheHelper != nullptr) {
delete cacheHelper;
cacheHelper = nullptr;
}
}
CacheHelper* getHelper() {
ASSERT(cacheHelper != nullptr, "No cacheHelper initialized.");
return cacheHelper;
}
void _verifyEntry(const char* name, const char* key, const char* val,
bool noKey) {
// Verify key and value exist in this region, in this process.
const char* value = val ? val : "";
std::string msg;
if (noKey) {
msg =
std::string("Verify key ") + key + " does not exist in region " + name;
} else if (!val) {
msg = std::string("Verify value for key ") + key +
" does not exist in region " + name;
} else {
msg = std::string("Verify value for key ") + key + " is: " + value +
" in region " + name;
}
LOG(msg);
auto regPtr = getHelper()->getRegion(name);
ASSERT(regPtr != nullptr, "Region not found.");
auto keyPtr = CacheableKey::create(key);
// if the region is no ack, then we may need to wait...
if (noKey == false) { // need to find the key!
ASSERT(regPtr->containsKey(keyPtr), "Key not found in region.");
}
if (val != nullptr) { // need to have a value!
ASSERT(regPtr->containsValueForKey(keyPtr), "Value not found in region.");
}
// loop up to MAX times, testing condition
uint32_t MAX = 100;
uint32_t SLEEP = 10; // milliseconds
uint32_t containsKeyCnt = 0;
uint32_t containsValueCnt = 0;
uint32_t testValueCnt = 0;
for (int i = MAX; i >= 0; i--) {
if (noKey) {
if (regPtr->containsKey(keyPtr)) {
containsKeyCnt++;
} else {
break;
}
ASSERT(containsKeyCnt < MAX, "Key found in region.");
}
if (val == nullptr) {
if (regPtr->containsValueForKey(keyPtr)) {
containsValueCnt++;
} else {
break;
}
ASSERT(containsValueCnt < MAX, "Value found in region.");
}
if (val != nullptr) {
auto checkPtr =
std::dynamic_pointer_cast<CacheableString>(regPtr->get(keyPtr));
ASSERT(checkPtr != nullptr, "Value Ptr should not be null.");
LOG("In verify loop, get returned " + checkPtr->value() + " for key " +
key);
if (strcmp(checkPtr->value().c_str(), value) != 0) {
testValueCnt++;
} else {
break;
}
ASSERT(testValueCnt < MAX, "Incorrect value found.");
}
dunit::sleep(SLEEP);
}
}
#define verifyEntry(x, y, z) _verifyEntry(x, y, z, __LINE__)
void _verifyEntry(const char* name, const char* key, const char* val,
int line) {
LOG(std::string("verifyEntry() called from ") + std::to_string(line) + "\n");
_verifyEntry(name, key, val, false);
LOG("Entry verified.");
}
void createRegion(const char* name, bool ackMode, const char* endpoints,
bool clientNotificationEnabled = false,
bool cachingEnable = true) {
LOG("createRegion() entered.");
std::cout << "Creating region -- " << name << " ackMode is " << ackMode
<< "\n"
<< std::flush;
auto regPtr = getHelper()->createRegion(name, ackMode, cachingEnable, nullptr,
endpoints, clientNotificationEnabled);
ASSERT(regPtr != nullptr, "Failed to create region.");
LOG("Region created.");
}
void createPooledRegion(const std::string& name, bool ackMode,
const std::string& locators,
const std::string& poolname,
bool clientNotificationEnabled = false,
bool cachingEnable = true) {
LOG("createRegion_Pool() entered.");
std::cout << "Creating region -- " << name << " ackMode is " << ackMode
<< "\n"
<< std::flush;
auto regPtr =
getHelper()->createPooledRegion(name, ackMode, locators, poolname,
cachingEnable, clientNotificationEnabled);
ASSERT(regPtr != nullptr, "Failed to create region.");
LOG("Pooled Region created.");
}
void createPooledRegionSticky(const std::string& name, bool ackMode,
const std::string& locators,
const std::string& poolname,
bool clientNotificationEnabled = false,
bool cachingEnable = true) {
LOG("createRegion_Pool() entered.");
std::cout << "Creating region -- " << name << " ackMode is " << ackMode
<< "\n"
<< std::flush;
auto regPtr = getHelper()->createPooledRegionSticky(
name, ackMode, locators, poolname, cachingEnable,
clientNotificationEnabled);
ASSERT(regPtr != nullptr, "Failed to create region.");
LOG("Pooled Region created.");
}
void createEntry(const char* name, const char* key, const char* value) {
LOG("createEntry() entered.");
std::cout << "Creating entry -- key: " << key << " value: " << value
<< " in region " << name << "\n"
<< std::flush;
// Create entry, verify entry is correct
auto keyPtr = CacheableKey::create(key);
auto valPtr = CacheableString::create(value);
auto regPtr = getHelper()->getRegion(name);
ASSERT(regPtr != nullptr, "Region not found.");
ASSERT(!regPtr->containsKey(keyPtr),
"Key should not have been found in region.");
ASSERT(!regPtr->containsValueForKey(keyPtr),
"Value should not have been found in region.");
// regPtr->create( keyPtr, valPtr );
regPtr->put(keyPtr, valPtr);
LOG("Created entry.");
// verifyEntry( name, key, value );
LOG("Entry created.");
}
void createEntryTwice(const char* name, const char* key, const char* value) {
LOG("createEntryTwice() entered.");
LOG(std::string("Creating entry -- key: ") + key + " value : " + value +
" in region " + name);
auto keyPtr = CacheableKey::create(key);
auto valPtr = CacheableString::create(value);
auto regPtr = getHelper()->getRegion(name);
regPtr->create(keyPtr, valPtr);
try {
regPtr->create(keyPtr, valPtr);
} catch (const EntryExistsException& geodeExcp) {
LOG(geodeExcp.what());
LOG("createEntryTwice() Clean Exit.");
return;
}
ASSERT(false,
"Creating key twice is not allowed and while doing that exception was "
"not thrown");
return; // This return will never reach
}
void updateEntry(const char* name, const char* key, const char* value) {
LOG("updateEntry() entered.");
std::cout << "Updating entry -- key: " << key << " value: " << value
<< " in region " << name << "\n"
<< std::flush;
// Update entry, verify entry is correct
auto keyPtr = CacheableKey::create(key);
auto valPtr = CacheableString::create(value);
auto regPtr = getHelper()->getRegion(name);
ASSERT(regPtr != nullptr, "Region not found.");
ASSERT(regPtr->containsKey(keyPtr), "Key should have been found in region.");
ASSERT(regPtr->containsValueForKey(keyPtr),
"Value should have been found in region.");
regPtr->put(keyPtr, valPtr);
LOG("Put entry.");
verifyEntry(name, key, value);
LOG("Entry updated.");
}
void doGetAgain(const char* name, const char* key, const char* value) {
LOG("doGetAgain() entered.");
std::cout << "Netsearching for entry -- key: " << key << " value: " << value
<< " in region " << name << "\n"
<< std::flush;
// Get entry created in Process A, verify entry is correct
auto keyPtr = CacheableKey::create(key);
auto regPtr = getHelper()->getRegion(name);
std::cout << "netsearch region " << regPtr->getName() << "\n" << std::flush;
ASSERT(regPtr != nullptr, "Region not found.");
auto checkPtr = std::dynamic_pointer_cast<CacheableString>(
regPtr->get(keyPtr)); // force a netsearch
if (checkPtr != nullptr) {
LOG("checkPtr is not null");
LOG(std::string("In doGetAgain, get returned ") + checkPtr->value() +
" for key " + key);
} else {
LOG("checkPtr is nullptr");
}
verifyEntry(name, key, value);
LOG("GetAgain complete.");
}
void doNetsearch(const char* name, const char* key, const char* value) {
LOG("doNetsearch() entered.");
std::cout << "Netsearching for entry -- key: " << key
<< " expecting value: " << value << " in region " << name << "\n"
<< std::flush;
static int count = 0;
// Get entry created in Process A, verify entry is correct
auto keyPtr = CacheableKey::create(key);
auto regPtr = getHelper()->getRegion(name);
std::cout << "netsearch region " << regPtr->getName() << "\n" << std::flush;
ASSERT(regPtr != nullptr, "Region not found.");
if (count == 0) {
ASSERT(!regPtr->containsKey(keyPtr),
"Key should not have been found in region.");
ASSERT(!regPtr->containsValueForKey(keyPtr),
"Value should not have been found in region.");
count++;
}
auto checkPtr = std::dynamic_pointer_cast<CacheableString>(
regPtr->get(keyPtr)); // force a netsearch
if (checkPtr != nullptr) {
LOG("checkPtr is not null");
LOG(std::string("In net search, get returned ") + checkPtr->value() +
" for key " + key);
} else {
LOG("checkPtr is nullptr");
}
verifyEntry(name, key, value);
LOG("Netsearch complete.");
}
const char* keys[] = {"Key-1", "Key-2", "Key-3", "Key-4",
"Key-5", "Key-6", "Key-7"};
const char* vals[] = {"Value-1", "Value-2", "Value-3", "Value-4",
"Value-5", "Value-6", "Value-7"};
const char* nvals[] = {"New Value-1", "New Value-2", "New Value-3",
"New Value-4", "New Value-5", "New Value-6",
"New Value-7"};
const char* regionNames[] = {"DistRegionAck", "DistRegionNoAck", "testregion"};
const bool USE_ACK = true;
const bool NO_ACK = false;
#include "LocatorHelper.hpp"
#define THREADERRORCHECK(x, y) \
do { \
if (!(x)) { \
failed_ = true; \
error_ = y; \
return; \
} \
} while (0)
class SuspendTransactionThread {
public:
SuspendTransactionThread(bool sleep, binary_semaphore& event)
: sleep_{sleep}, sem_{event} {}
void run() {
LOG(" In SuspendTransactionThread");
auto txManager = getHelper()->getCache()->getCacheTransactionManager();
txManager->begin();
createEntry(regionNames[0], keys[4], vals[4]);
createEntry(regionNames[1], keys[5], vals[5]);
tx_ = &txManager->getTransactionId();
if (sleep_) {
sem_.acquire();
std::this_thread::sleep_for(std::chrono::seconds{5});
}
tx_ = &txManager->suspend();
LOG(" Out SuspendTransactionThread");
getHelper()
->getCache()
->getPoolManager()
.find("__TESTPOOL1_")
->releaseThreadLocalConnection();
}
void start() {
thread_ = std::thread{[this]() { run(); }};
}
void stop() {
if (thread_.joinable()) {
thread_.join();
}
}
TransactionId& getSuspendedTx() { return *tx_; }
protected:
bool sleep_;
std::thread thread_;
binary_semaphore& sem_;
TransactionId* tx_{nullptr};
};
class ResumeTransactionThread {
public:
ResumeTransactionThread(TransactionId& tx, bool commit, bool sleep,
binary_semaphore& event)
: tx_(tx), commit_{commit}, sleep_(sleep), failed_{false}, sem_{event} {}
void run() {
LOG("In ResumeTransactionThread");
auto regPtr0 = getHelper()->getRegion(regionNames[0]);
THREADERRORCHECK(regPtr0 != nullptr,
"In ResumeTransactionThread - Region not found.");
auto keyPtr4 = CacheableKey::create(keys[4]);
auto keyPtr5 = CacheableKey::create(keys[5]);
auto keyPtr6 = CacheableKey::create(keys[6]);
auto regPtr1 = getHelper()->getRegion(regionNames[1]);
THREADERRORCHECK(regPtr1 != nullptr,
"In ResumeTransactionThread - Region not found.");
THREADERRORCHECK(!regPtr0->containsKeyOnServer(keyPtr4),
"In ResumeTransactionThread - Key should not have been "
"found in region.");
THREADERRORCHECK(!regPtr1->containsKeyOnServer(keyPtr5),
"In ResumeTransactionThread - Key should not have been "
"found in region.");
auto txManager = getHelper()->getCache()->getCacheTransactionManager();
if (sleep_) {
THREADERRORCHECK(!txManager->isSuspended(tx_),
"In ResumeTransactionThread - the transaction should "
"NOT be in suspended state");
} else {
THREADERRORCHECK(txManager->isSuspended(tx_),
"In ResumeTransactionThread - the transaction should be "
"in suspended state");
}
THREADERRORCHECK(
txManager->exists(tx_),
"In ResumeTransactionThread - the transaction should exist");
if (sleep_) {
sem_.release();
txManager->tryResume(tx_, std::chrono::seconds(30));
} else {
txManager->resume(tx_);
}
THREADERRORCHECK(
regPtr0->containsKeyOnServer(keyPtr4),
"In ResumeTransactionThread - Key should have been found in region.");
THREADERRORCHECK(
regPtr1->containsKeyOnServer(keyPtr5),
"In ResumeTransactionThread - Key should have been found in region.");
createEntry(regionNames[1], keys[6], vals[6]);
if (commit_) {
txManager->commit();
THREADERRORCHECK(
regPtr0->containsKeyOnServer(keyPtr4),
"In ResumeTransactionThread - Key should have been found in region.");
THREADERRORCHECK(
regPtr1->containsKeyOnServer(keyPtr5),
"In ResumeTransactionThread - Key should have been found in region.");
THREADERRORCHECK(
regPtr1->containsKeyOnServer(keyPtr6),
"In ResumeTransactionThread - Key should have been found in region.");
} else {
txManager->rollback();
THREADERRORCHECK(!regPtr0->containsKeyOnServer(keyPtr4),
"In ResumeTransactionThread - Key should not have been "
"found in region.");
THREADERRORCHECK(!regPtr1->containsKeyOnServer(keyPtr5),
"In ResumeTransactionThread - Key should not have been "
"found in region.");
THREADERRORCHECK(!regPtr1->containsKeyOnServer(keyPtr6),
"In ResumeTransactionThread - Key should not have been "
"found in region.");
}
if (commit_) {
regPtr1->destroy(keyPtr6);
regPtr1->destroy(keyPtr5);
regPtr0->destroy(keyPtr4);
THREADERRORCHECK(!regPtr1->containsKeyOnServer(keyPtr6),
"In ResumeTransactionThread - Key should not have been "
"found in region.");
THREADERRORCHECK(!regPtr1->containsKeyOnServer(keyPtr5),
"In ResumeTransactionThread - Key should not have been "
"found in region.");
THREADERRORCHECK(!regPtr0->containsKeyOnServer(keyPtr4),
"In ResumeTransactionThread - Key should not have been "
"found in region.");
} else {
try {
regPtr1->destroy(keyPtr6);
FAIL("Should have got EntryNotFoundException for keyPtr6");
} catch (EntryNotFoundException& /*ex*/) {
LOG("Got expected EntryNotFoundException for keyPtr6");
}
try {
regPtr1->destroy(keyPtr5);
FAIL("Should have got EntryNotFoundException for keyPtr5");
} catch (EntryNotFoundException& /*ex*/) {
LOG("Got expected EntryNotFoundException for keyPtr5");
}
try {
regPtr0->destroy(keyPtr4);
FAIL("Should have got EntryNotFoundException for keyPtr4");
} catch (EntryNotFoundException& /*ex*/) {
LOG("Got expected EntryNotFoundException for keyPtr4");
}
}
getHelper()
->getCache()
->getPoolManager()
.find("__TESTPOOL1_")
->releaseThreadLocalConnection();
LOG(" Out ResumeTransactionThread");
}
void start() {
thread_ = std::thread{[this]() { run(); }};
}
void stop() {
if (thread_.joinable()) {
thread_.join();
}
}
bool isFailed() { return failed_; }
std::string getError() { return error_; }
protected:
TransactionId& tx_;
bool commit_;
bool sleep_;
bool failed_;
std::string error_;
std::thread thread_;
binary_semaphore& sem_;
};
DUNIT_TASK_DEFINITION(SERVER1, CreateServer1)
{
if (isLocalServer) CacheHelper::initServer(1);
LOG("SERVER1 started");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, SuspendResumeCommit)
{
auto txManager = getHelper()->getCache()->getCacheTransactionManager();
auto regPtr0 = getHelper()->getRegion(regionNames[0]);
ASSERT(regPtr0 != nullptr, "In SuspendResumeCommit - Region not found.");
auto regPtr1 = getHelper()->getRegion(regionNames[1]);
ASSERT(regPtr1 != nullptr, "In SuspendResumeCommit - Region not found.");
auto keyPtr4 = CacheableKey::create(keys[4]);
auto keyPtr5 = CacheableKey::create(keys[5]);
auto keyPtr6 = CacheableKey::create(keys[6]);
txManager->begin();
createEntry(regionNames[0], keys[4], vals[4]);
createEntry(regionNames[1], keys[5], vals[5]);
auto& m_suspendedTransaction = txManager->suspend();
ASSERT(
!regPtr0->containsKeyOnServer(keyPtr4),
"In SuspendResumeCommit - Key should not have been found in region.");
ASSERT(
!regPtr1->containsKeyOnServer(keyPtr5),
"In SuspendResumeCommit - Key should not have been found in region.");
ASSERT(txManager->isSuspended(m_suspendedTransaction),
"In SuspendResumeCommit - the transaction should be in suspended "
"state");
ASSERT(txManager->exists(m_suspendedTransaction),
"In SuspendResumeCommit - the transaction should exist");
txManager->resume(m_suspendedTransaction);
ASSERT(
!txManager->tryResume(m_suspendedTransaction),
"SuspendResumeRollback: Transaction shouldnt have been resumed again");
ASSERT(regPtr0->containsKeyOnServer(keyPtr4),
"In SuspendResumeCommit - Key should have been found in region.");
ASSERT(regPtr1->containsKeyOnServer(keyPtr5),
"In SuspendResumeCommit - Key should have been found in region.");
createEntry(regionNames[1], keys[6], vals[6]);
txManager->commit();
ASSERT(regPtr0->containsKeyOnServer(keyPtr4),
"In SuspendResumeCommit - Key should have been found in region.");
ASSERT(regPtr1->containsKeyOnServer(keyPtr5),
"In SuspendResumeCommit - Key should have been found in region.");
ASSERT(regPtr1->containsKeyOnServer(keyPtr6),
"In SuspendResumeCommit - Key should have been found in region.");
regPtr1->destroy(keyPtr6);
regPtr1->destroy(keyPtr5);
regPtr0->destroy(keyPtr4);
ASSERT(!txManager->isSuspended(m_suspendedTransaction),
"In SuspendResumeCommit the transaction should NOT present");
ASSERT(!txManager->exists(m_suspendedTransaction),
"In SuspendResumeCommit - the transaction should NOT exist");
ASSERT(!txManager->tryResume(m_suspendedTransaction),
"In SuspendResumeCommit - the transaction should NOT have been "
"resumed");
bool resumeExc = false;
try {
txManager->resume(m_suspendedTransaction);
} catch (const IllegalStateException&) {
resumeExc = true;
}
ASSERT(resumeExc,
"SuspendResumeCommit: Transaction shouldnt have been resumed");
bool threwTransactionException = false;
try {
txManager->suspend();
} catch (const TransactionException&) {
threwTransactionException = true;
}
ASSERT(threwTransactionException,
"SuspendResumeCommit: Transaction shouldnt have been suspended");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, SuspendTimeOut)
{
auto txManager = getHelper()->getCache()->getCacheTransactionManager();
auto keyPtr4 = CacheableKey::create(keys[4]);
auto keyPtr5 = CacheableKey::create(keys[5]);
auto regPtr0 = getHelper()->getRegion(regionNames[0]);
ASSERT(regPtr0 != nullptr, "In SuspendTimeOut - Region not found.");
txManager->begin();
createEntry(regionNames[0], keys[4], vals[4]);
auto& tid1 = txManager->suspend();
txManager->begin();
createEntry(regionNames[0], keys[5], vals[5]);
auto& tid2 = txManager->suspend();
txManager->resume(tid1);
createEntry(regionNames[0], keys[6], vals[6]);
txManager->commit();
ASSERT(txManager->isSuspended(tid2),
"In SuspendTimeOut the transaction should be present");
ASSERT(txManager->exists(tid2),
"In SuspendTimeOut - the transaction should exist");
std::this_thread::sleep_for(std::chrono::seconds(65));
ASSERT(!txManager->tryResume(tid2),
"In SuspendTimeOut - the transaction should NOT have been resumed");
ASSERT(!txManager->isSuspended(tid2),
"In SuspendTimeOut the transaction should NOT present");
ASSERT(!txManager->exists(tid2),
"In SuspendTimeOut - the transaction should NOT exist");
ASSERT(regPtr0->containsKeyOnServer(keyPtr4),
"In SuspendTimeOut - Key should have been found in region.");
ASSERT(!regPtr0->containsKeyOnServer(keyPtr5),
"In SuspendTimeOut - Key should not have been found in region.");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, SuspendResumeRollback)
{
auto txManager = getHelper()->getCache()->getCacheTransactionManager();
auto keyPtr4 = CacheableKey::create(keys[4]);
auto keyPtr5 = CacheableKey::create(keys[5]);
auto keyPtr6 = CacheableKey::create(keys[6]);
auto regPtr0 = getHelper()->getRegion(regionNames[0]);
ASSERT(regPtr0 != nullptr, "In SuspendResumeRollback - Region not found.");
auto regPtr1 = getHelper()->getRegion(regionNames[1]);
ASSERT(regPtr1 != nullptr, "In SuspendResumeRollback - Region not found.");
txManager->begin();
createEntry(regionNames[0], keys[4], vals[4]);
createEntry(regionNames[1], keys[5], vals[5]);
auto& m_suspendedTransaction = txManager->suspend();
ASSERT(
!regPtr0->containsKeyOnServer(keyPtr4),
"In SuspendResumeRollback - Key should not have been found in region.");
ASSERT(
!regPtr1->containsKeyOnServer(keyPtr5),
"In SuspendResumeRollback - Key should not have been found in region.");
ASSERT(txManager->isSuspended(m_suspendedTransaction),
"In SuspendResumeRollback the transaction should be in suspended "
"state");
ASSERT(txManager->exists(m_suspendedTransaction),
"In SuspendResumeRollback - the transaction should exist");
txManager->resume(m_suspendedTransaction);
ASSERT(regPtr0->containsKeyOnServer(keyPtr4),
"In SuspendResumeRollback - Key should have been found in region.");
ASSERT(regPtr1->containsKeyOnServer(keyPtr5),
"In SuspendResumeRollback - Key should have been found in region.");
createEntry(regionNames[1], keys[6], vals[6]);
txManager->rollback();
ASSERT(
!regPtr0->containsKeyOnServer(keyPtr4),
"In SuspendResumeRollback - Key should not have been found in region.");
ASSERT(
!regPtr1->containsKeyOnServer(keyPtr5),
"In SuspendResumeRollback - Key should not have been found in region.");
ASSERT(
!regPtr1->containsKeyOnServer(keyPtr6),
"In SuspendResumeRollback - Key should not have been found in region.");
try {
regPtr1->destroy(keyPtr6);
FAIL("Should have got EntryNotFoundException for keyPtr6");
} catch (EntryNotFoundException& /*ex*/) {
LOG("Got expected EntryNotFoundException for keyPtr6");
}
try {
regPtr1->destroy(keyPtr5);
FAIL("Should have got EntryNotFoundException for keyPtr5");
} catch (EntryNotFoundException& /*ex*/) {
LOG("Got expected EntryNotFoundException for keyPtr5");
}
try {
regPtr0->destroy(keyPtr4);
FAIL("Should have got EntryNotFoundException for keyPtr4");
} catch (EntryNotFoundException& /*ex*/) {
LOG("Got expected EntryNotFoundException for keyPtr4");
}
ASSERT(!txManager->isSuspended(m_suspendedTransaction),
"In SuspendResumeRollback the transaction should NOT present");
ASSERT(!txManager->exists(m_suspendedTransaction),
"In SuspendResumeRollback - the transaction should NOT exist");
ASSERT(!txManager->tryResume(m_suspendedTransaction),
"In SuspendResumeRollback - the transaction should NOT have been "
"resumed");
bool resumeExc = false;
try {
txManager->resume(m_suspendedTransaction);
} catch (const IllegalStateException&) {
resumeExc = true;
}
ASSERT(resumeExc,
"SuspendResumeRollback: Transaction shouldnt have been resumed");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, SuspendResumeInThread)
{
LOG("Start suspend thread and resume thread and rollback immedidately");
{
binary_semaphore event{0};
SuspendTransactionThread suspend{false, event};
suspend.start();
std::this_thread::sleep_for(std::chrono::seconds{2});
ResumeTransactionThread resume{suspend.getSuspendedTx(), false, false,
event};
resume.start();
suspend.stop();
resume.stop();
ASSERT(!resume.isFailed(), resume.getError());
}
LOG("Start suspend thread and resume thread and commit immedidately");
{
binary_semaphore event{0};
SuspendTransactionThread suspend{false, event};
suspend.start();
std::this_thread::sleep_for(std::chrono::seconds{2});
ResumeTransactionThread resume{suspend.getSuspendedTx(), true, false,
event};
resume.start();
suspend.stop();
resume.stop();
ASSERT(!resume.isFailed(), resume.getError());
}
LOG("Start suspend thread and tryresume thread with rollback. make "
"tryResume to sleep");
{
binary_semaphore event{0};
SuspendTransactionThread suspend{true, event};
suspend.start();
std::this_thread::sleep_for(std::chrono::seconds{2});
ResumeTransactionThread resume{suspend.getSuspendedTx(), false, true,
event};
resume.start();
suspend.stop();
resume.stop();
ASSERT(!resume.isFailed(), resume.getError());
}
LOG("Start suspend thread and tryresume thread with commit. make "
"tryResume to sleep");
{
binary_semaphore event{0};
SuspendTransactionThread suspend{true, event};
suspend.start();
std::this_thread::sleep_for(std::chrono::seconds{2});
LOG("suspendTh->activate();");
ResumeTransactionThread resume{suspend.getSuspendedTx(), true, true,
event};
resume.start();
suspend.stop();
resume.stop();
ASSERT(!resume.isFailed(), resume.getError());
}
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, CreateNonexistentServerRegion_Pooled_Locator)
{
initClient(true);
createPooledRegion("non-region", USE_ACK, locatorsG, "__TESTPOOL1_");
try {
createEntry("non-region", keys[0], vals[0]);
FAIL(
"Expected exception when doing operations on a non-existent region.");
} catch (const CacheServerException& ex) {
std::cout << "Got expected CacheServerException when performing operation on a non-existent region: " << ex.what() << "\n";
}
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1,
CreateNonexistentServerRegion_Pooled_Locator_Sticky)
{
initClient(true);
createPooledRegionSticky("non-region", USE_ACK, locatorsG, "__TESTPOOL1_");
try {
createEntry("non-region", keys[0], vals[0]);
FAIL(
"Expected exception when doing operations on a non-existent region.");
} catch (const CacheServerException& ex) {
std::cout << "Got expected CacheServerException when performing operation on a non-existent region: " << ex.what() << "\n";
}
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, CreateClient1PooledRegionWithoutSticky)
{
createPooledRegion(regionNames[0], USE_ACK, locatorsG, "__TESTPOOL1_");
createPooledRegion(regionNames[1], NO_ACK, locatorsG, "__TESTPOOL1_");
LOG("StepOne_Pooled complete.");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, CreateClient1PooledRegionWithSticky)
{
createPooledRegionSticky(regionNames[0], USE_ACK, locatorsG,
"__TESTPOOL1_");
createPooledRegionSticky(regionNames[1], NO_ACK, locatorsG, "__TESTPOOL1_");
LOG("StepOne_Pooled_Locator_Sticky complete.");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT2, CreateClient2PooledRegionWithoutSticky)
{
initClient(true);
createPooledRegion(regionNames[0], USE_ACK, locatorsG, "__TESTPOOL1_");
createPooledRegion(regionNames[1], NO_ACK, locatorsG, "__TESTPOOL1_");
LOG("StepTwo complete.");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT2, CreateClient2PooledRegionWithSticky)
{
initClient(true);
createPooledRegionSticky(regionNames[0], USE_ACK, locatorsG,
"__TESTPOOL1_");
createPooledRegionSticky(regionNames[1], NO_ACK, locatorsG, "__TESTPOOL1_");
LOG("StepTwo complete.");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, CreateClient1Entries)
{
auto txManager = getHelper()->getCache()->getCacheTransactionManager();
txManager->begin();
createEntry(regionNames[0], keys[0], vals[0]);
createEntry(regionNames[1], keys[2], vals[2]);
txManager->commit();
LOG("StepThree complete.");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT2, CreateClient2Entries)
{
doNetsearch(regionNames[0], keys[0], vals[0]);
doNetsearch(regionNames[1], keys[2], vals[2]);
auto txManager = getHelper()->getCache()->getCacheTransactionManager();
txManager->begin();
createEntry(regionNames[0], keys[1], vals[1]);
createEntry(regionNames[1], keys[3], vals[3]);
txManager->commit();
verifyEntry(regionNames[0], keys[1], vals[1]);
verifyEntry(regionNames[1], keys[3], vals[3]);
LOG("StepFour complete.");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, UpdateClient1Entries)
{
auto reg0 = getHelper()->getRegion(regionNames[0]);
auto reg1 = getHelper()->getRegion(regionNames[1]);
auto vec0 = reg0->serverKeys();
auto vec1 = reg1->serverKeys();
ASSERT(vec0.size() == 2, "Should have 2 keys in first region.");
ASSERT(vec1.size() == 2, "Should have 2 keys in second region.");
std::string key0, key1;
key0 = vec0[0]->toString().c_str();
key1 = vec0[1]->toString().c_str();
ASSERT(key0 != key1, "The two keys should be different in first region.");
ASSERT(key0 == keys[0] || key0 == keys[1],
"Unexpected key in first region.");
ASSERT(key1 == keys[0] || key1 == keys[1],
"Unexpected key in first region.");
key0 = vec1[0]->toString().c_str();
key1 = vec1[1]->toString().c_str();
ASSERT(key0 != key1, "The two keys should be different in second region.");
ASSERT(key0 == keys[2] || key0 == keys[3],
"Unexpected key in second region.");
ASSERT(key1 == keys[2] || key1 == keys[3],
"Unexpected key in second region.");
doNetsearch(regionNames[0], keys[1], vals[1]);
doNetsearch(regionNames[1], keys[3], vals[3]);
updateEntry(regionNames[0], keys[0], nvals[0]);
updateEntry(regionNames[1], keys[2], nvals[2]);
LOG("StepFive complete.");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT2, UpdateClient2Entries)
{
doNetsearch(regionNames[0], keys[0], vals[0]);
doNetsearch(regionNames[1], keys[2], vals[2]);
auto txManager = getHelper()->getCache()->getCacheTransactionManager();
txManager->begin();
updateEntry(regionNames[0], keys[1], nvals[1]);
updateEntry(regionNames[1], keys[3], nvals[3]);
txManager->commit();
LOG("StepSix complete.");
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, CreateClient1EntryTwice)
{ createEntryTwice(regionNames[0], CREATE_TWICE_KEY, CREATE_TWICE_VALUE); }
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, CreateClient1KeyThriceWithoutSticky)
{
createPooledRegion(regionNames[2], NO_ACK, locatorsG, "__TESTPOOL1_", false,
false);
auto reg = getHelper()->getRegion(regionNames[2]);
LOG("REGION Created with Caching Enabled false");
auto keyPtr = CacheableKey::create(CREATE_TWICE_KEY);
auto valPtr = CacheableString::create(CREATE_TWICE_VALUE);
try {
reg->create(keyPtr, valPtr);
char message[200];
LOG(std::string("First create on Key ") + CREATE_TWICE_KEY);
LOG(message);
reg->create(keyPtr, valPtr);
LOG(std::string("Second create on Key ") + CREATE_TWICE_KEY);
LOG(message);
reg->create(keyPtr, valPtr);
LOG(std::string("Third create on Key ") + CREATE_TWICE_KEY);
LOG(message);
} catch (const EntryExistsException& geodeExcp) {
LOG(geodeExcp.what());
ASSERT(false,
"Creating KEY Twice on a caching-enabled false region should be "
"allowed.");
}
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, CreateClient1KeyThriceWithSticky)
{
createPooledRegionSticky(regionNames[2], NO_ACK, locatorsG, "__TESTPOOL1_",
false, false);
auto reg = getHelper()->getRegion(regionNames[2]);
LOG("REGION Created with Caching Enabled false");
auto keyPtr = CacheableKey::create(CREATE_TWICE_KEY);
auto valPtr = CacheableString::create(CREATE_TWICE_VALUE);
auto reg0 = getHelper()->getRegion(regionNames[0]);
auto reg1 = getHelper()->getRegion(regionNames[1]);
reg0->localInvalidate(CacheableKey::create(keys[1]));
reg1->localInvalidate(CacheableKey::create(keys[3]));
auto pool = getHelper()->getCache()->getPoolManager().find("__TESTPOOL1_");
ASSERT(pool != nullptr, "Pool Should have been found");
doNetsearch(regionNames[0], keys[1], nvals[1]);
doNetsearch(regionNames[1], keys[3], nvals[3]);
pool->releaseThreadLocalConnection();
updateEntry(regionNames[0], keys[0], nvals[0]);
updateEntry(regionNames[1], keys[2], nvals[2]);
pool->releaseThreadLocalConnection();
try {
reg->create(keyPtr, valPtr);
char message[200];
LOG(std::string("First create on Key ") + CREATE_TWICE_KEY);
LOG(message);
reg->create(keyPtr, valPtr);
LOG(std::string("Second create on Key ") + CREATE_TWICE_KEY);
LOG(message);
reg->create(keyPtr, valPtr);
LOG(std::string("Third create on Key ") + CREATE_TWICE_KEY);
LOG(message);
} catch (const EntryExistsException& geodeExcp) {
LOG(geodeExcp.what());
ASSERT(false,
"Creating KEY Twice on a caching-enabled false region should be "
"allowed.");
}
}
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT1, CloseCache1)
{ cleanProc(); }
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(CLIENT2, CloseCache2)
{ cleanProc(); }
END_TASK_DEFINITION
DUNIT_TASK_DEFINITION(SERVER1, CloseServer1)
{
if (isLocalServer) {
CacheHelper::closeServer(1);
LOG("SERVER1 stopped");
}
}
END_TASK_DEFINITION
} // namespace
#endif // GEODE_INTEGRATION_TEST_THINCLIENTTRANSACTIONS_H_