mirror of
https://github.com/MariaDB/server.git
synced 2025-01-22 14:54:20 +01:00
353 lines
9.8 KiB
C++
353 lines
9.8 KiB
C++
/* Copyright (C) 2003 MySQL AB
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
#include <NDBT.hpp>
|
|
#include <NDBT_Test.hpp>
|
|
#include <HugoTransactions.hpp>
|
|
#include <UtilTransactions.hpp>
|
|
#include <random.h>
|
|
#include <NdbConfig.hpp>
|
|
#include <signaldata/DumpStateOrd.hpp>
|
|
|
|
#define TIMEOUT 3000
|
|
|
|
Uint32 g_org_timeout = 3000;
|
|
|
|
int
|
|
setTransactionTimeout(NDBT_Context* ctx, NDBT_Step* step){
|
|
NdbRestarter restarter;
|
|
|
|
NdbConfig conf(GETNDB(step)->getNodeId()+1);
|
|
unsigned int nodeId = conf.getMasterNodeId();
|
|
if (!conf.getProperty(nodeId,
|
|
NODE_TYPE_DB,
|
|
CFG_DB_TRANSACTION_INACTIVE_TIMEOUT,
|
|
&g_org_timeout)){
|
|
return NDBT_FAILED;
|
|
}
|
|
|
|
int val[] = { DumpStateOrd::TcSetApplTransactionTimeout, TIMEOUT };
|
|
if(restarter.dumpStateAllNodes(val, 2) != 0){
|
|
return NDBT_FAILED;
|
|
}
|
|
|
|
return NDBT_OK;
|
|
}
|
|
|
|
int
|
|
resetTransactionTimeout(NDBT_Context* ctx, NDBT_Step* step){
|
|
NdbRestarter restarter;
|
|
|
|
int val[] = { DumpStateOrd::TcSetApplTransactionTimeout, g_org_timeout };
|
|
if(restarter.dumpStateAllNodes(val, 2) != 0){
|
|
return NDBT_FAILED;
|
|
}
|
|
|
|
return NDBT_OK;
|
|
}
|
|
|
|
|
|
int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
|
|
|
|
int records = ctx->getNumRecords();
|
|
HugoTransactions hugoTrans(*ctx->getTab());
|
|
if (hugoTrans.loadTable(GETNDB(step), records) != 0){
|
|
return NDBT_FAILED;
|
|
}
|
|
return NDBT_OK;
|
|
}
|
|
|
|
int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
|
|
int records = ctx->getNumRecords();
|
|
|
|
UtilTransactions utilTrans(*ctx->getTab());
|
|
if (utilTrans.clearTable2(GETNDB(step), records) != 0){
|
|
return NDBT_FAILED;
|
|
}
|
|
return NDBT_OK;
|
|
}
|
|
|
|
|
|
#define CHECK(b) if (!(b)) { \
|
|
ndbout << "ERR: "<< step->getName() \
|
|
<< " failed on line " << __LINE__ << endl; \
|
|
result = NDBT_FAILED; \
|
|
break; }
|
|
|
|
int runTimeoutTrans2(NDBT_Context* ctx, NDBT_Step* step){
|
|
int result = NDBT_OK;
|
|
int loops = ctx->getNumLoops();
|
|
int stepNo = step->getStepNo();
|
|
int mul1 = ctx->getProperty("Op1", (Uint32)0);
|
|
int mul2 = ctx->getProperty("Op2", (Uint32)0);
|
|
int records = ctx->getNumRecords();
|
|
|
|
int minSleep = (int)(TIMEOUT * 1.5);
|
|
int maxSleep = TIMEOUT * 2;
|
|
|
|
HugoOperations hugoOps(*ctx->getTab());
|
|
Ndb* pNdb = GETNDB(step);
|
|
|
|
for (int l = 0; l<loops && !ctx->isTestStopped() && result == NDBT_OK; l++){
|
|
|
|
int op1 = 0 + (l + stepNo) * mul1;
|
|
int op2 = 0 + (l + stepNo) * mul2;
|
|
|
|
op1 = (op1 % 5);
|
|
op2 = (op2 % 5);
|
|
|
|
ndbout << stepNo << ": TransactionInactiveTimeout="<< TIMEOUT
|
|
<< ", minSleep="<<minSleep
|
|
<< ", maxSleep="<<maxSleep
|
|
<< ", op1=" << op1
|
|
<< ", op2=" << op2 << endl;;
|
|
|
|
do{
|
|
// Commit transaction
|
|
CHECK(hugoOps.startTransaction(pNdb) == 0);
|
|
|
|
switch(op1){
|
|
case 0:
|
|
break;
|
|
case 1:
|
|
if(hugoOps.pkReadRecord(pNdb, stepNo) != 0){
|
|
g_err << stepNo << ": Fail" << __LINE__ << endl;
|
|
result = NDBT_FAILED; break;
|
|
}
|
|
break;
|
|
case 2:
|
|
if(hugoOps.pkUpdateRecord(pNdb, stepNo) != 0){
|
|
g_err << stepNo << ": Fail" << __LINE__ << endl;
|
|
result = NDBT_FAILED; break;
|
|
}
|
|
break;
|
|
case 3:
|
|
if(hugoOps.pkDeleteRecord(pNdb, stepNo) != 0){
|
|
g_err << stepNo << ": Fail" << __LINE__ << endl;
|
|
result = NDBT_FAILED; break;
|
|
}
|
|
break;
|
|
case 4:
|
|
if(hugoOps.pkInsertRecord(pNdb, stepNo+records+l) != 0){
|
|
g_err << stepNo << ": Fail" << __LINE__ << endl;
|
|
result = NDBT_FAILED; break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if(result != NDBT_OK)
|
|
break;
|
|
|
|
int res = hugoOps.execute_NoCommit(pNdb);
|
|
if(res != 0){
|
|
g_err << stepNo << ": Fail" << __LINE__ << endl;
|
|
result = NDBT_FAILED; break;
|
|
}
|
|
|
|
int sleep = minSleep + myRandom48(maxSleep-minSleep);
|
|
ndbout << stepNo << ": Sleeping for "<< sleep << " milliseconds" << endl;
|
|
NdbSleep_MilliSleep(sleep);
|
|
|
|
switch(op2){
|
|
case 0:
|
|
break;
|
|
case 1:
|
|
if(hugoOps.pkReadRecord(pNdb, stepNo) != 0){
|
|
g_err << stepNo << ": Fail" << __LINE__ << endl;
|
|
result = NDBT_FAILED; break;
|
|
}
|
|
break;
|
|
case 2:
|
|
if(hugoOps.pkUpdateRecord(pNdb, stepNo) != 0){
|
|
g_err << stepNo << ": Fail" << __LINE__ << endl;
|
|
result = NDBT_FAILED; break;
|
|
}
|
|
break;
|
|
case 3:
|
|
if(hugoOps.pkDeleteRecord(pNdb, stepNo) != 0){
|
|
g_err << stepNo << ": Fail" << __LINE__ << endl;
|
|
result = NDBT_FAILED; break;
|
|
}
|
|
break;
|
|
case 4:
|
|
if(hugoOps.pkInsertRecord(pNdb, stepNo+2*records+l) != 0){
|
|
g_err << stepNo << ": Fail" << __LINE__ << endl;
|
|
result = NDBT_FAILED; break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
// Expect that transaction has timed-out
|
|
res = hugoOps.execute_Commit(pNdb);
|
|
if(op1 != 0 && res != 266){
|
|
g_err << stepNo << ": Fail: " << res << "!= 237, op1="
|
|
<< op1 << ", op2=" << op2 << endl;
|
|
result = NDBT_FAILED; break;
|
|
}
|
|
|
|
} while(false);
|
|
|
|
hugoOps.closeTransaction(pNdb);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
int runDontTimeoutTrans(NDBT_Context* ctx, NDBT_Step* step){
|
|
int result = NDBT_OK;
|
|
int loops = ctx->getNumLoops();
|
|
int stepNo = step->getStepNo();
|
|
|
|
int maxSleep = (int)(TIMEOUT * 0.5);
|
|
ndbout << "TransactionInactiveTimeout="<< TIMEOUT
|
|
<< ", maxSleep="<<maxSleep<<endl;
|
|
|
|
|
|
HugoOperations hugoOps(*ctx->getTab());
|
|
Ndb* pNdb = GETNDB(step);
|
|
|
|
for (int l = 0; l < loops && result == NDBT_OK; l++){
|
|
|
|
do{
|
|
// Commit transaction
|
|
CHECK(hugoOps.startTransaction(pNdb) == 0);
|
|
CHECK(hugoOps.pkReadRecord(pNdb, stepNo) == 0);
|
|
CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
|
|
|
|
int sleep = myRandom48(maxSleep);
|
|
ndbout << "Sleeping for " << sleep << " milliseconds" << endl;
|
|
NdbSleep_MilliSleep(sleep);
|
|
|
|
// Expect that transaction has NOT timed-out
|
|
CHECK(hugoOps.execute_Commit(pNdb) == 0);
|
|
|
|
} while(false);
|
|
|
|
hugoOps.closeTransaction(pNdb);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
int runBuddyTransNoTimeout(NDBT_Context* ctx, NDBT_Step* step){
|
|
int result = NDBT_OK;
|
|
int loops = ctx->getNumLoops();
|
|
int records = ctx->getNumRecords();
|
|
int stepNo = step->getStepNo();
|
|
int maxSleep = (int)(TIMEOUT * 0.3);
|
|
ndbout << "TransactionInactiveTimeout="<< TIMEOUT
|
|
<< ", maxSleep="<<maxSleep<<endl;
|
|
|
|
HugoOperations hugoOps(*ctx->getTab());
|
|
Ndb* pNdb = GETNDB(step);
|
|
|
|
for (int l = 1; l < loops && result == NDBT_OK; l++){
|
|
|
|
do{
|
|
// Start an insert trans
|
|
CHECK(hugoOps.startTransaction(pNdb) == 0);
|
|
int recordNo = records + (stepNo*loops) + l;
|
|
CHECK(hugoOps.pkInsertRecord(pNdb, recordNo) == 0);
|
|
CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
|
|
|
|
for (int i = 0; i < 3; i++){
|
|
// Perform buddy scan reads
|
|
CHECK((hugoOps.scanReadRecords(pNdb)) == 0);
|
|
CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
|
|
|
|
int sleep = myRandom48(maxSleep);
|
|
ndbout << "Sleeping for " << sleep << " milliseconds" << endl;
|
|
NdbSleep_MilliSleep(sleep);
|
|
}
|
|
|
|
// Expect that transaction has NOT timed-out
|
|
CHECK(hugoOps.execute_Commit(pNdb) == 0);
|
|
|
|
} while(false);
|
|
|
|
hugoOps.closeTransaction(pNdb);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
NDBT_TESTSUITE(testTimeout);
|
|
TESTCASE("DontTimeoutTransaction",
|
|
"Test that the transaction does not timeout "\
|
|
"if we sleep during the transaction. Use a sleep "\
|
|
"value which is smaller than TransactionInactiveTimeout"){
|
|
INITIALIZER(runLoadTable);
|
|
INITIALIZER(setTransactionTimeout);
|
|
STEPS(runDontTimeoutTrans, 1);
|
|
FINALIZER(resetTransactionTimeout);
|
|
FINALIZER(runClearTable);
|
|
}
|
|
TESTCASE("DontTimeoutTransaction5",
|
|
"Test that the transaction does not timeout "\
|
|
"if we sleep during the transaction. Use a sleep "\
|
|
"value which is smaller than TransactionInactiveTimeout" \
|
|
"Five simultaneous threads"){
|
|
INITIALIZER(runLoadTable);
|
|
INITIALIZER(setTransactionTimeout);
|
|
STEPS(runDontTimeoutTrans, 5);
|
|
FINALIZER(resetTransactionTimeout);
|
|
FINALIZER(runClearTable);
|
|
}
|
|
TESTCASE("TimeoutRandTransaction",
|
|
"Test that the transaction does timeout "\
|
|
"if we sleep during the transaction. Use a sleep "\
|
|
"value which is larger than TransactionInactiveTimeout"){
|
|
INITIALIZER(runLoadTable);
|
|
INITIALIZER(setTransactionTimeout);
|
|
TC_PROPERTY("Op1", 7);
|
|
TC_PROPERTY("Op2", 11);
|
|
STEPS(runTimeoutTrans2, 5);
|
|
FINALIZER(resetTransactionTimeout);
|
|
FINALIZER(runClearTable);
|
|
}
|
|
TESTCASE("BuddyTransNoTimeout",
|
|
"Start a transaction and perform an insert with NoCommit. " \
|
|
"Start a buddy transaction wich performs long running scans " \
|
|
"and sleeps. " \
|
|
"The total sleep time is longer than TransactionInactiveTimeout" \
|
|
"Commit the first transaction, it should not have timed out."){
|
|
INITIALIZER(runLoadTable);
|
|
INITIALIZER(setTransactionTimeout);
|
|
STEPS(runBuddyTransNoTimeout, 1);
|
|
FINALIZER(resetTransactionTimeout);
|
|
FINALIZER(runClearTable);
|
|
}
|
|
TESTCASE("BuddyTransNoTimeout5",
|
|
"Start a transaction and perform an insert with NoCommit. " \
|
|
"Start a buddy transaction wich performs long running scans " \
|
|
"and sleeps. " \
|
|
"The total sleep time is longer than TransactionInactiveTimeout" \
|
|
"Commit the first transaction, it should not have timed out." \
|
|
"Five simultaneous threads"){
|
|
INITIALIZER(runLoadTable);
|
|
INITIALIZER(setTransactionTimeout);
|
|
STEPS(runBuddyTransNoTimeout, 5);
|
|
FINALIZER(resetTransactionTimeout);
|
|
FINALIZER(runClearTable);
|
|
}
|
|
NDBT_TESTSUITE_END(testTimeout);
|
|
|
|
int main(int argc, const char** argv){
|
|
ndb_init();
|
|
myRandom48Init(NdbTick_CurrentMillisecond());
|
|
return testTimeout.execute(argc, argv);
|
|
}
|
|
|