mirror of
https://github.com/MariaDB/server.git
synced 2025-07-29 05:21:33 +03:00
ndb -
1) New testcase Check every combination of ins/upd/del of length 5 Check reading savepoint's 2) Fix 1 liner in acc wrt committing read ndb/include/ndbapi/NdbConnection.hpp: Make testcase friend ndb/src/kernel/blocks/dbacc/DbaccMain.cpp: Fix so that committing a READ can _not_ result in setting elementIsDisappeared ndb/test/include/NDBT_Test.hpp: Make copy of testcase name ndb/test/ndbapi/testOperations.cpp: New testcase Check every combination of ins/upd/del of length 5 Check reading savepoint's ndb/test/src/HugoOperations.cpp: Close transaction in destructor ndb/test/src/NDBT_Test.cpp: Make copy of testcase name
This commit is contained in:
@ -687,6 +687,8 @@ private:
|
|||||||
|
|
||||||
void remove_list(NdbOperation*& head, NdbOperation*);
|
void remove_list(NdbOperation*& head, NdbOperation*);
|
||||||
void define_scan_op(NdbIndexScanOperation*);
|
void define_scan_op(NdbIndexScanOperation*);
|
||||||
|
|
||||||
|
friend int runOperations(class NDBT_Context*, class NDBT_Step*);
|
||||||
};
|
};
|
||||||
|
|
||||||
inline
|
inline
|
||||||
|
@ -5704,7 +5704,8 @@ void Dbacc::commitOperation(Signal* signal)
|
|||||||
Uint32 tmp2Olq;
|
Uint32 tmp2Olq;
|
||||||
|
|
||||||
if ((operationRecPtr.p->commitDeleteCheckFlag == ZFALSE) &&
|
if ((operationRecPtr.p->commitDeleteCheckFlag == ZFALSE) &&
|
||||||
(operationRecPtr.p->operation != ZSCAN_OP)) {
|
(operationRecPtr.p->operation != ZSCAN_OP) &&
|
||||||
|
(operationRecPtr.p->operation != ZREAD)) {
|
||||||
jam();
|
jam();
|
||||||
/* This method is used to check whether the end result of the transaction
|
/* This method is used to check whether the end result of the transaction
|
||||||
will be to delete the tuple. In this case all operation will be marked
|
will be to delete the tuple. In this case all operation will be marked
|
||||||
|
@ -188,7 +188,7 @@ public:
|
|||||||
NDBT_TestCase(NDBT_TestSuite* psuite,
|
NDBT_TestCase(NDBT_TestSuite* psuite,
|
||||||
const char* name,
|
const char* name,
|
||||||
const char* comment);
|
const char* comment);
|
||||||
virtual ~NDBT_TestCase(){}
|
virtual ~NDBT_TestCase() {}
|
||||||
|
|
||||||
// This is the default executor of a test case
|
// This is the default executor of a test case
|
||||||
// When a test case is executed it will need to be suplied with a number of
|
// When a test case is executed it will need to be suplied with a number of
|
||||||
@ -225,6 +225,8 @@ protected:
|
|||||||
void stopTimer(NDBT_Context*);
|
void stopTimer(NDBT_Context*);
|
||||||
void printTimer(NDBT_Context*);
|
void printTimer(NDBT_Context*);
|
||||||
|
|
||||||
|
BaseString _name;
|
||||||
|
BaseString _comment;
|
||||||
const char* name;
|
const char* name;
|
||||||
const char* comment;
|
const char* comment;
|
||||||
NDBT_TestSuite* suite;
|
NDBT_TestSuite* suite;
|
||||||
|
@ -98,6 +98,11 @@ OperationTestCase matrix[] = {
|
|||||||
result = NDBT_FAILED; \
|
result = NDBT_FAILED; \
|
||||||
break; }
|
break; }
|
||||||
|
|
||||||
|
#define C3(b) if (!(b)) { \
|
||||||
|
g_err << "ERR: "<< step->getName() \
|
||||||
|
<< " failed on line " << __LINE__ << endl; \
|
||||||
|
abort(); return NDBT_FAILED; }
|
||||||
|
|
||||||
int
|
int
|
||||||
runOp(HugoOperations & hugoOps,
|
runOp(HugoOperations & hugoOps,
|
||||||
Ndb * pNdb,
|
Ndb * pNdb,
|
||||||
@ -228,11 +233,287 @@ runClearTable(NDBT_Context* ctx, NDBT_Step* step){
|
|||||||
return NDBT_OK;
|
return NDBT_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum OPS { o_DONE= 0, o_INS= 1, o_UPD= 2, o_DEL= 3 };
|
||||||
|
typedef Vector<OPS> Sequence;
|
||||||
|
|
||||||
|
static
|
||||||
|
bool
|
||||||
|
valid(const Sequence& s)
|
||||||
|
{
|
||||||
|
if(s.size() == 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
for(size_t i = 1; i<s.size(); i++)
|
||||||
|
{
|
||||||
|
switch(s[i]){
|
||||||
|
case o_INS:
|
||||||
|
if(s[i-1] != o_DEL)
|
||||||
|
return false;
|
||||||
|
break;
|
||||||
|
case o_UPD:
|
||||||
|
case o_DEL:
|
||||||
|
if(s[i-1] == o_DEL)
|
||||||
|
return false;
|
||||||
|
break;
|
||||||
|
case o_DONE:
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
NdbOut& operator<<(NdbOut& out, const Sequence& s)
|
||||||
|
{
|
||||||
|
out << "[ ";
|
||||||
|
for(size_t i = 0; i<s.size(); i++)
|
||||||
|
{
|
||||||
|
switch(s[i]){
|
||||||
|
case o_INS:
|
||||||
|
out << "INS ";
|
||||||
|
break;
|
||||||
|
case o_DEL:
|
||||||
|
out << "DEL ";
|
||||||
|
break;
|
||||||
|
case o_UPD:
|
||||||
|
out << "UPD ";
|
||||||
|
break;
|
||||||
|
case o_DONE:
|
||||||
|
abort();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
out << "]";
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
void
|
||||||
|
generate(Sequence& out, int no)
|
||||||
|
{
|
||||||
|
while(no & 3)
|
||||||
|
{
|
||||||
|
out.push_back((OPS)(no & 3));
|
||||||
|
no >>= 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
void
|
||||||
|
generate(Vector<int>& out, size_t len)
|
||||||
|
{
|
||||||
|
int max= 1;
|
||||||
|
while(len)
|
||||||
|
{
|
||||||
|
max <<= 2;
|
||||||
|
len--;
|
||||||
|
}
|
||||||
|
|
||||||
|
len= 1;
|
||||||
|
for(int i = 0; i<max; i++)
|
||||||
|
{
|
||||||
|
Sequence tmp;
|
||||||
|
generate(tmp, i);
|
||||||
|
|
||||||
|
if(tmp.size() >= len && valid(tmp))
|
||||||
|
{
|
||||||
|
out.push_back(i);
|
||||||
|
len= tmp.size();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
//ndbout << "DISCARD: " << tmp << endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
runOperations(NDBT_Context* ctx, NDBT_Step* step)
|
||||||
|
{
|
||||||
|
const Uint32 DUMMY = 0;
|
||||||
|
const Uint32 ROW = 1;
|
||||||
|
|
||||||
|
int tmp;
|
||||||
|
Ndb* pNdb = GETNDB(step);
|
||||||
|
|
||||||
|
Uint32 seqNo = ctx->getProperty("Sequence", (Uint32)0);
|
||||||
|
Uint32 no_wait = NdbOperation::LM_CommittedRead*
|
||||||
|
ctx->getProperty("NoWait", (Uint32)1);
|
||||||
|
|
||||||
|
if(seqNo == 0)
|
||||||
|
{
|
||||||
|
return NDBT_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
Sequence seq;
|
||||||
|
generate(seq, seqNo);
|
||||||
|
|
||||||
|
{
|
||||||
|
// Dummy row
|
||||||
|
HugoOperations hugoOps(*ctx->getTab());
|
||||||
|
C3(hugoOps.startTransaction(pNdb) == 0);
|
||||||
|
C3(hugoOps.pkInsertRecord(pNdb, DUMMY, 1, 0) == 0);
|
||||||
|
C3(hugoOps.execute_Commit(pNdb) == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
const bool inital_row= (seq[0] != o_INS);
|
||||||
|
if(inital_row)
|
||||||
|
{
|
||||||
|
HugoOperations hugoOps(*ctx->getTab());
|
||||||
|
C3(hugoOps.startTransaction(pNdb) == 0);
|
||||||
|
C3(hugoOps.pkInsertRecord(pNdb, ROW, 1, 0) == 0);
|
||||||
|
C3(hugoOps.execute_Commit(pNdb) == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
HugoOperations trans1(*ctx->getTab());
|
||||||
|
C3(trans1.startTransaction(pNdb) == 0);
|
||||||
|
for(size_t i = 0; i<seq.size(); i++)
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Perform operation
|
||||||
|
*/
|
||||||
|
switch(seq[i]){
|
||||||
|
case o_INS:
|
||||||
|
C3(trans1.pkInsertRecord(pNdb, ROW, 1, i+1) == 0);
|
||||||
|
break;
|
||||||
|
case o_UPD:
|
||||||
|
C3(trans1.pkUpdateRecord(pNdb, ROW, 1, i+1) == 0);
|
||||||
|
break;
|
||||||
|
case o_DEL:
|
||||||
|
C3(trans1.pkDeleteRecord(pNdb, ROW, 1) == 0);
|
||||||
|
break;
|
||||||
|
case o_DONE:
|
||||||
|
abort();
|
||||||
|
}
|
||||||
|
C3(trans1.execute_NoCommit(pNdb) == 0);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Verify other transaction
|
||||||
|
*/
|
||||||
|
for(size_t j = no_wait; j<3; j++)
|
||||||
|
{
|
||||||
|
HugoOperations other(*ctx->getTab());
|
||||||
|
C3(other.startTransaction(pNdb) == 0);
|
||||||
|
C3(other.pkReadRecord(pNdb, ROW, 1, (NdbOperation::LockMode)j) == 0);
|
||||||
|
tmp= other.execute_Commit(pNdb);
|
||||||
|
if(j == NdbOperation::LM_CommittedRead)
|
||||||
|
{
|
||||||
|
C3(inital_row? tmp==0 && other.verifyUpdatesValue(0) == 0 : tmp==626);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
C3(tmp == 266);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Verify savepoint read
|
||||||
|
*/
|
||||||
|
Uint64 transactionId= trans1.getTransaction()->getTransactionId();
|
||||||
|
for(size_t k=0; k<=i+1; k++)
|
||||||
|
{
|
||||||
|
for(size_t j = 0; j<3; j++)
|
||||||
|
{
|
||||||
|
const NdbOperation::LockMode lm= (NdbOperation::LockMode)j;
|
||||||
|
|
||||||
|
HugoOperations same(*ctx->getTab());
|
||||||
|
C3(same.startTransaction(pNdb) == 0);
|
||||||
|
same.getTransaction()->setTransactionId(transactionId); // Cheat
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Increase savepoint to <em>k</em>
|
||||||
|
*/
|
||||||
|
for(size_t l = 1; l<=k; l++)
|
||||||
|
{
|
||||||
|
C3(same.pkReadRecord(pNdb, DUMMY, 1, lm) == 0); // Read dummy row
|
||||||
|
C3(same.execute_NoCommit(pNdb) == 0);
|
||||||
|
g_info << "savepoint: " << l << endl;
|
||||||
|
}
|
||||||
|
|
||||||
|
g_info << "op(" << k << ", " << i << "): "
|
||||||
|
<< " lock mode " << lm << endl;
|
||||||
|
|
||||||
|
C3(same.pkReadRecord(pNdb, ROW, 1, lm) == 0); // Read real row
|
||||||
|
tmp= same.execute_Commit(pNdb);
|
||||||
|
if(k == 0)
|
||||||
|
{
|
||||||
|
if(inital_row)
|
||||||
|
{
|
||||||
|
C3(tmp == 0 && same.verifyUpdatesValue(0) == 0);
|
||||||
|
} else
|
||||||
|
{
|
||||||
|
C3(tmp == 626);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
switch(seq[k-1]){
|
||||||
|
case o_INS:
|
||||||
|
case o_UPD:
|
||||||
|
C3(tmp == 0 && same.verifyUpdatesValue(k) == 0);
|
||||||
|
break;
|
||||||
|
case o_DEL:
|
||||||
|
C3(tmp == 626);
|
||||||
|
break;
|
||||||
|
case o_DONE:
|
||||||
|
abort();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
C3(trans1.execute_Commit(pNdb) == 0);
|
||||||
|
|
||||||
|
return NDBT_OK;
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
main(int argc, const char** argv){
|
main(int argc, const char** argv){
|
||||||
ndb_init();
|
ndb_init();
|
||||||
|
|
||||||
|
Vector<int> tmp;
|
||||||
|
generate(tmp, 5);
|
||||||
|
|
||||||
NDBT_TestSuite ts("testOperations");
|
NDBT_TestSuite ts("testOperations");
|
||||||
|
for(size_t i = 0; i<tmp.size(); i++)
|
||||||
|
{
|
||||||
|
BaseString name;
|
||||||
|
Sequence s;
|
||||||
|
generate(s, tmp[i]);
|
||||||
|
for(size_t j = 0; j<s.size(); j++){
|
||||||
|
switch(s[j]){
|
||||||
|
case o_INS:
|
||||||
|
name.append("_INS");
|
||||||
|
break;
|
||||||
|
case o_DEL:
|
||||||
|
name.append("_DEL");
|
||||||
|
break;
|
||||||
|
case o_UPD:
|
||||||
|
name.append("_UPD");
|
||||||
|
break;
|
||||||
|
case o_DONE:
|
||||||
|
abort();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
NDBT_TestCaseImpl1 *pt = new NDBT_TestCaseImpl1(&ts,
|
||||||
|
name.c_str()+1, "");
|
||||||
|
|
||||||
|
pt->setProperty("Sequence", tmp[i]);
|
||||||
|
pt->addInitializer(new NDBT_Initializer(pt,
|
||||||
|
"runClearTable",
|
||||||
|
runClearTable));
|
||||||
|
|
||||||
|
pt->addStep(new NDBT_ParallelStep(pt,
|
||||||
|
name.c_str()+1,
|
||||||
|
runOperations));
|
||||||
|
|
||||||
|
pt->addFinalizer(new NDBT_Finalizer(pt,
|
||||||
|
"runClearTable",
|
||||||
|
runClearTable));
|
||||||
|
|
||||||
|
ts.addTest(pt);
|
||||||
|
}
|
||||||
|
|
||||||
for(Uint32 i = 0; i<sizeof(matrix)/sizeof(matrix[0]); i++){
|
for(Uint32 i = 0; i<sizeof(matrix)/sizeof(matrix[0]); i++){
|
||||||
NDBT_TestCaseImpl1 *pt = new NDBT_TestCaseImpl1(&ts, matrix[i].name, "");
|
NDBT_TestCaseImpl1 *pt = new NDBT_TestCaseImpl1(&ts, matrix[i].name, "");
|
||||||
|
|
||||||
@ -270,3 +551,5 @@ main(int argc, const char** argv){
|
|||||||
return ts.execute(argc, argv);
|
return ts.execute(argc, argv);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template class Vector<OPS>;
|
||||||
|
template class Vector<Sequence>;
|
||||||
|
@ -401,6 +401,10 @@ HugoOperations::HugoOperations(const NdbDictionary::Table& _tab):
|
|||||||
|
|
||||||
HugoOperations::~HugoOperations(){
|
HugoOperations::~HugoOperations(){
|
||||||
deallocRows();
|
deallocRows();
|
||||||
|
if (pTrans != NULL){
|
||||||
|
pTrans->close();
|
||||||
|
pTrans = NULL;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -327,13 +327,17 @@ NDBT_Finalizer::NDBT_Finalizer(NDBT_TestCase* ptest,
|
|||||||
NDBT_TestCase::NDBT_TestCase(NDBT_TestSuite* psuite,
|
NDBT_TestCase::NDBT_TestCase(NDBT_TestSuite* psuite,
|
||||||
const char* pname,
|
const char* pname,
|
||||||
const char* pcomment) :
|
const char* pcomment) :
|
||||||
name(pname) ,
|
name(strdup(pname)) ,
|
||||||
comment(pcomment),
|
comment(strdup(pcomment)),
|
||||||
suite(psuite){
|
suite(psuite)
|
||||||
|
{
|
||||||
|
_name.assign(pname);
|
||||||
|
_comment.assign(pcomment);
|
||||||
|
name= _name.c_str();
|
||||||
|
comment= _comment.c_str();
|
||||||
assert(suite != NULL);
|
assert(suite != NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
NDBT_TestCaseImpl1::NDBT_TestCaseImpl1(NDBT_TestSuite* psuite,
|
NDBT_TestCaseImpl1::NDBT_TestCaseImpl1(NDBT_TestSuite* psuite,
|
||||||
const char* pname,
|
const char* pname,
|
||||||
const char* pcomment) :
|
const char* pcomment) :
|
||||||
|
Reference in New Issue
Block a user