mirror of
https://github.com/MariaDB/server.git
synced 2025-01-24 15:54:37 +01:00
Merge mysql.com:/home/stewart/Documents/MySQL/5.0/ndb
into mysql.com:/home/stewart/Documents/MySQL/5.0/ndb-dynamic-port
This commit is contained in:
commit
c4de1e0d14
16 changed files with 367 additions and 273 deletions
|
@ -94,6 +94,7 @@ jcole@sarvik.tfr.cafe.ee
|
|||
jcole@tetra.spaceapes.com
|
||||
jimw@mysql.com
|
||||
joerg@mysql.com
|
||||
jon@gigan.
|
||||
joreland@mysql.com
|
||||
jorge@linux.jorge.mysql.com
|
||||
jplindst@t41.(none)
|
||||
|
|
|
@ -532,7 +532,7 @@ os_mem_alloc_large(
|
|||
ibool assert_on_error) /* in: if TRUE, we crash mysqld if the memory
|
||||
cannot be allocated */
|
||||
{
|
||||
#ifdef UNIV_LINUX
|
||||
#ifdef HAVE_LARGE_PAGES
|
||||
ulint size;
|
||||
int shmid;
|
||||
void *ptr = NULL;
|
||||
|
@ -541,7 +541,8 @@ os_mem_alloc_large(
|
|||
if (!os_use_large_pages || !os_large_page_size) {
|
||||
goto skip;
|
||||
}
|
||||
|
||||
|
||||
#ifdef UNIV_LINUX
|
||||
/* Align block size to os_large_page_size */
|
||||
size = ((n - 1) & ~(os_large_page_size - 1)) + os_large_page_size;
|
||||
|
||||
|
@ -561,6 +562,7 @@ os_mem_alloc_large(
|
|||
*/
|
||||
shmctl(shmid, IPC_RMID, &buf);
|
||||
}
|
||||
#endif
|
||||
|
||||
if (ptr) {
|
||||
if (set_to_zero) {
|
||||
|
@ -573,8 +575,8 @@ os_mem_alloc_large(
|
|||
}
|
||||
|
||||
fprintf(stderr, "InnoDB HugeTLB: Warning: Using conventional memory pool\n");
|
||||
#endif
|
||||
skip:
|
||||
#endif /* HAVE_LARGE_PAGES */
|
||||
|
||||
return(ut_malloc_low(n, set_to_zero, assert_on_error));
|
||||
}
|
||||
|
@ -587,8 +589,12 @@ os_mem_free_large(
|
|||
/*=================*/
|
||||
void *ptr) /* in: number of bytes */
|
||||
{
|
||||
#ifdef HAVE_LARGE_PAGES
|
||||
if (os_use_large_pages && os_large_page_size
|
||||
#ifdef UNIV_LINUX
|
||||
if (os_use_large_pages && os_large_page_size && !shmdt(ptr)) {
|
||||
&& !shmdt(ptr)
|
||||
#endif
|
||||
) {
|
||||
return;
|
||||
}
|
||||
#endif
|
||||
|
|
|
@ -59,7 +59,7 @@ EXTRACT_PRIVATE = NO
|
|||
# If the EXTRACT_STATIC tag is set to YES all static members of a file
|
||||
# will be included in the documentation.
|
||||
|
||||
EXTRACT_STATIC = NO
|
||||
EXTRACT_STATIC = YES
|
||||
|
||||
# If the HIDE_UNDOC_MEMBERS tag is set to YES, Doxygen will hide all
|
||||
# undocumented members of documented classes, files or namespaces.
|
||||
|
@ -146,7 +146,7 @@ HIDE_SCOPE_NAMES = NO
|
|||
# will generate a verbatim copy of the header file for each class for
|
||||
# which an include is specified. Set to NO to disable this.
|
||||
|
||||
VERBATIM_HEADERS = YES
|
||||
VERBATIM_HEADERS = NO
|
||||
|
||||
# If the SHOW_INCLUDE_FILES tag is set to YES (the default) then Doxygen
|
||||
# will put list of the files that are included by a file in the documentation
|
||||
|
|
|
@ -52,7 +52,7 @@ EXTRACT_ALL = YES
|
|||
# If the EXTRACT_PRIVATE tag is set to YES all private members of a class
|
||||
# will be included in the documentation.
|
||||
|
||||
EXTRACT_PRIVATE = YES
|
||||
EXTRACT_PRIVATE = NO
|
||||
|
||||
# If the EXTRACT_STATIC tag is set to YES all static members of a file
|
||||
# will be included in the documentation.
|
||||
|
@ -157,7 +157,7 @@ HIDE_SCOPE_NAMES = NO
|
|||
# will generate a verbatim copy of the header file for each class for
|
||||
# which an include is specified. Set to NO to disable this.
|
||||
|
||||
VERBATIM_HEADERS = YES
|
||||
VERBATIM_HEADERS = NO
|
||||
|
||||
# If the SHOW_INCLUDE_FILES tag is set to YES (the default) then Doxygen
|
||||
# will put list of the files that are included by a file in the documentation
|
||||
|
|
|
@ -59,7 +59,7 @@ EXTRACT_PRIVATE = NO
|
|||
# If the EXTRACT_STATIC tag is set to YES all static members of a file
|
||||
# will be included in the documentation.
|
||||
|
||||
EXTRACT_STATIC = NO
|
||||
EXTRACT_STATIC = YES
|
||||
|
||||
# If the HIDE_UNDOC_MEMBERS tag is set to YES, Doxygen will hide all
|
||||
# undocumented members of documented classes, files or namespaces.
|
||||
|
@ -146,7 +146,7 @@ HIDE_SCOPE_NAMES = NO
|
|||
# will generate a verbatim copy of the header file for each class for
|
||||
# which an include is specified. Set to NO to disable this.
|
||||
|
||||
VERBATIM_HEADERS = YES
|
||||
VERBATIM_HEADERS = NO
|
||||
|
||||
# If the SHOW_INCLUDE_FILES tag is set to YES (the default) then Doxygen
|
||||
# will put list of the files that are included by a file in the documentation
|
||||
|
|
|
@ -52,7 +52,7 @@ EXTRACT_ALL = YES
|
|||
# If the EXTRACT_PRIVATE tag is set to YES all private members of a class
|
||||
# will be included in the documentation.
|
||||
|
||||
EXTRACT_PRIVATE = YES
|
||||
EXTRACT_PRIVATE = NO
|
||||
|
||||
# If the EXTRACT_STATIC tag is set to YES all static members of a file
|
||||
# will be included in the documentation.
|
||||
|
@ -157,7 +157,7 @@ HIDE_SCOPE_NAMES = NO
|
|||
# will generate a verbatim copy of the header file for each class for
|
||||
# which an include is specified. Set to NO to disable this.
|
||||
|
||||
VERBATIM_HEADERS = YES
|
||||
VERBATIM_HEADERS = NO
|
||||
|
||||
# If the SHOW_INCLUDE_FILES tag is set to YES (the default) then Doxygen
|
||||
# will put list of the files that are included by a file in the documentation
|
||||
|
|
|
@ -52,7 +52,7 @@ EXTRACT_ALL = YES
|
|||
# If the EXTRACT_PRIVATE tag is set to YES all private members of a class
|
||||
# will be included in the documentation.
|
||||
|
||||
EXTRACT_PRIVATE = YES
|
||||
EXTRACT_PRIVATE = NO
|
||||
|
||||
# If the EXTRACT_STATIC tag is set to YES all static members of a file
|
||||
# will be included in the documentation.
|
||||
|
@ -157,7 +157,7 @@ HIDE_SCOPE_NAMES = NO
|
|||
# will generate a verbatim copy of the header file for each class for
|
||||
# which an include is specified. Set to NO to disable this.
|
||||
|
||||
VERBATIM_HEADERS = YES
|
||||
VERBATIM_HEADERS = NO
|
||||
|
||||
# If the SHOW_INCLUDE_FILES tag is set to YES (the default) then Doxygen
|
||||
# will put list of the files that are included by a file in the documentation
|
||||
|
|
|
@ -17,30 +17,37 @@
|
|||
/**
|
||||
@mainpage NDB API Programmers' Guide
|
||||
|
||||
This guide assumes a basic familiarity with NDB Cluster concepts.
|
||||
This guide assumes a basic familiarity with MySQL Cluster concepts.
|
||||
Some of the fundamental ones are described in section @ref secConcepts.
|
||||
|
||||
The <em>NDB API</em> is an NDB Cluster application interface
|
||||
that implements both synchronous and asynchronous transactions.
|
||||
The <em>NDB API</em> is an MySQL Cluster application interface
|
||||
that implements transactions.
|
||||
The NDB API consists of the following fundamental classes:
|
||||
- Ndb_cluster_connection class representing a connection to a cluster,
|
||||
- Ndb is the main class representing the database,
|
||||
- NdbConnection represents a transaction,
|
||||
- NdbOperation represents a transaction operation using primary key,
|
||||
- NdbIndexOperation represents a transaction operation using a secondary
|
||||
index,
|
||||
- NdbOperation represents a operation using primary key,
|
||||
- NdbScanOperation represents a operation performing a full table scan.
|
||||
- NdbIndexOperation represents a operation using a unique hash index,
|
||||
- NdbIndexScanOperation represents a operation performing a scan using
|
||||
an ordered index,
|
||||
- NdbRecAttr represents the value of an attribute, and
|
||||
- NdbDictionary represents meta information about tables and attributes.
|
||||
- NdbError represents an error condition
|
||||
- NdbError contains a specification of an error.
|
||||
There are also some auxiliary classes.
|
||||
|
||||
The main structure of an application program is as follows:
|
||||
-# Construct and connect to a cluster using the Ndb_cluster_connection
|
||||
object.
|
||||
-# Construct and initialize Ndb object(s).
|
||||
-# Define and execute (synchronous or asynchronous) transactions.
|
||||
-# Define and execute transactions using NdbConnection and Ndb*Operation.
|
||||
-# Delete Ndb objects
|
||||
-# Delete connection to cluster
|
||||
|
||||
The main structure of a transaction is as follows:
|
||||
-# Start transaction
|
||||
-# Add and define operations (associated with the transaction)
|
||||
-# Start transaction, a NdbConnection
|
||||
-# Add and define operations (associated with the transaction),
|
||||
Ndb*Operation
|
||||
-# Execute transaction
|
||||
|
||||
The execute can be of two different types,
|
||||
|
@ -71,8 +78,8 @@
|
|||
At this step the transaction is being defined.
|
||||
It is not yet sent to the NDB kernel.
|
||||
-# Add and define operations to the transaction
|
||||
(using NdbConnection::getNdbOperation and
|
||||
methods from class NdbOperation).
|
||||
(using NdbConnection::getNdb*Operation and
|
||||
methods from class Ndb*Operation).
|
||||
The transaction is still not sent to the NDB kernel.
|
||||
-# Execute the transaction (using NdbConnection::execute).
|
||||
-# Close the transaction (using Ndb::closeTransaction).
|
||||
|
@ -82,20 +89,21 @@
|
|||
To execute several parallel synchronous transactions, one can either
|
||||
use multiple Ndb objects in several threads or start multiple
|
||||
applications programs.
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
Another way to execute several parallel transactions is to use
|
||||
asynchronous transactions.
|
||||
|
||||
#endif
|
||||
|
||||
@section secNdbOperations Operations
|
||||
|
||||
Each transaction (NdbConnection object) consist of a list of
|
||||
operations (NdbOperation or NdbIndexOperation objects.
|
||||
NdbIndexOperation is used for accessing tables through secondary indexes).
|
||||
operations (Ndb*Operation objects).
|
||||
Operations are of two different kinds:
|
||||
-# standard operations, and
|
||||
-# interpreted program operations.
|
||||
|
||||
<h3>Standard Operations</h3>
|
||||
<h3>Single row operations</h3>
|
||||
After the operation is created using NdbConnection::getNdbOperation
|
||||
(or NdbConnection::getNdbIndexOperation),
|
||||
it is defined in the following three steps:
|
||||
|
@ -106,36 +114,42 @@
|
|||
-# Specify attribute actions
|
||||
(e.g. using NdbOperation::getValue)
|
||||
|
||||
Example code (using an NdbOperation):
|
||||
Example code (using an NdbOperation and excluding error handling):
|
||||
@code
|
||||
MyOperation = MyConnection->getNdbOperation("MYTABLENAME"); // 1. Create
|
||||
if (MyOperation == NULL) APIERROR(MyConnection->getNdbError());
|
||||
// 1. Create
|
||||
MyOperation= MyConnection->getNdbOperation("MYTABLENAME");
|
||||
|
||||
MyOperation->readTuple(); // 2. Define type of operation
|
||||
MyOperation->equal("ATTR1", i); // 3. Specify Search Conditions
|
||||
// 2. Define type of operation and lock mode
|
||||
MyOperation->readTuple(NdbOperation::LM_Read);
|
||||
|
||||
// 3. Specify Search Conditions
|
||||
MyOperation->equal("ATTR1", i);
|
||||
|
||||
MyRecAttr = MyOperation->getValue("ATTR2", NULL); // 4. Attribute Actions
|
||||
if (MyRecAttr == NULL) APIERROR(MyConnection->getNdbError());
|
||||
// 4. Attribute Actions
|
||||
MyRecAttr= MyOperation->getValue("ATTR2", NULL);
|
||||
@endcode
|
||||
For more examples, see @ref ndbapi_example1.cpp and @ref ndbapi_example2.cpp.
|
||||
For more examples, see @ref ndbapi_example1.cpp and
|
||||
@ref ndbapi_example2.cpp.
|
||||
|
||||
Example code using an NdbIndexOperation:
|
||||
Example code (using an NdbIndexOperation and excluding error handling):
|
||||
@code
|
||||
MyOperation = // 1. Create
|
||||
MyConnection->getNdbIndexOperation("MYINDEX", "MYTABLENAME");
|
||||
if (MyOperation == NULL) APIERROR(MyConnection->getNdbError());
|
||||
// 1. Create
|
||||
MyOperation= MyConnection->getNdbIndexOperation("MYINDEX", "MYTABLENAME");
|
||||
|
||||
MyOperation->readTuple(); // 2. Define type of operation
|
||||
MyOperation->equal("ATTR1", i); // 3. Specify Search Conditions
|
||||
// 2. Define type of operation and lock mode
|
||||
MyOperation->readTuple(NdbOperation::LM_Read);
|
||||
|
||||
MyRecAttr = MyOperation->getValue("ATTR2", NULL); // 4. Attribute Actions
|
||||
if (MyRecAttr == NULL) APIERROR(MyConnection->getNdbError());
|
||||
// 3. Specify Search Conditions
|
||||
MyOperation->equal("ATTR1", i);
|
||||
|
||||
// 4. Attribute Actions
|
||||
MyRecAttr = MyOperation->getValue("ATTR2", NULL);
|
||||
@endcode
|
||||
For more examples, see @ref ndbapi_example4.cpp.
|
||||
|
||||
|
||||
<h4>Step 1: Define Standard Operation Type</h4>
|
||||
The following types of standard operations exist:
|
||||
<h4>Step 1: Define single row operation type</h4>
|
||||
The following types of operations exist:
|
||||
-# NdbOperation::insertTuple :
|
||||
inserts a non-existing tuple
|
||||
-# NdbOperation::writeTuple :
|
||||
|
@ -146,79 +160,26 @@
|
|||
-# NdbOperation::deleteTuple :
|
||||
deletes an existing tuple
|
||||
-# NdbOperation::readTuple :
|
||||
reads an existing tuple
|
||||
-# NdbOperation::readTupleExclusive :
|
||||
reads an existing tuple using an exclusive lock
|
||||
-# NdbOperation::simpleRead :
|
||||
reads an existing tuple (using shared read lock),
|
||||
but releases lock immediately after read
|
||||
-# NdbOperation::committedRead :
|
||||
reads committed tuple
|
||||
-# NdbOperation::dirtyUpdate :
|
||||
updates an existing tuple, but releases lock immediately
|
||||
after read (uses dirty lock)
|
||||
-# NdbOperation::dirtyWrite :
|
||||
updates or writes a tuple, but releases lock immediately
|
||||
after read (uses dirty lock)
|
||||
reads an existing tuple with specified lock mode
|
||||
|
||||
All of these operations operate on the unique tuple key.
|
||||
(When NdbIndexOperation is used then all of these operations
|
||||
operate on a defined secondary index.)
|
||||
|
||||
|
||||
Some comments:
|
||||
- NdbOperation::simpleRead and
|
||||
NdbOperation::committedRead can execute on the same transaction
|
||||
as the above operations but will release its locks immediately
|
||||
after reading the tuple.
|
||||
NdbOperation::simpleRead will always read the latest version
|
||||
of the tuple.
|
||||
Thus it will wait until it can acquire a shared read lock on
|
||||
the tuple.
|
||||
NdbOperation::committedRead will read the latest committed
|
||||
version of the tuple.
|
||||
<br>
|
||||
Both NdbOperation::simpleRead and NdbOperation::committedRead
|
||||
are examples of consistent reads which are not repeatable.
|
||||
All reads read the latest version if updates were made by the same
|
||||
transaction.
|
||||
Errors on simple read are only reported by the NdbOperation object.
|
||||
These error codes are not transferred to the NdbConnection object.
|
||||
- NdbOperation::dirtyUpdate and NdbOperation::dirtyWrite
|
||||
will execute in the same transaction
|
||||
but will release the lock immediately after updating the
|
||||
tuple.
|
||||
It will wait on the lock until it can acquire an exclusive
|
||||
write lock.
|
||||
In a replicated version of NDB Cluster NdbOperation::dirtyUpdate
|
||||
can lead to inconsistency between the replicas.
|
||||
Examples of when it could be used is
|
||||
to update statistical counters on tuples which are "hot-spots".
|
||||
operate on a defined unique hash index.)
|
||||
|
||||
@note If you want to define multiple operations within the same transaction,
|
||||
then you need to call NdbConnection::getNdbOperation
|
||||
(or NdbConnection::getNdbIndexOperation) for each
|
||||
then you need to call NdbConnection::getNdb*Operation for each
|
||||
operation.
|
||||
|
||||
|
||||
<h4>Step 2: Specify Search Conditions</h4>
|
||||
The search condition is used to select tuples.
|
||||
(In the current NdbIndexOperation implementation
|
||||
this means setting the value of
|
||||
the secondary index attributes of the wanted tuple.)
|
||||
|
||||
If a tuple identity is used, then NdbOperation::setTupleId
|
||||
is used to define the search key when inserting new tuples.
|
||||
Otherwise, NdbOperation::equal is used.
|
||||
|
||||
For NdbOperation::insertTuple it is also allowed to define the
|
||||
For NdbOperation::insertTuple it is also allowed to define the
|
||||
search key by using NdbOperation::setValue.
|
||||
The NDB API will automatically detect that it is
|
||||
supposed to use NdbOperation::equal instead.
|
||||
For NdbOperation::insertTuple it is not necessary to use
|
||||
NdbOperation::setValue on key attributes before other attributes.
|
||||
|
||||
|
||||
<h4>Step 3: Specify Attribute Actions</h4>
|
||||
Now it is time to define which attributes should be read or updated.
|
||||
Deletes can neither read nor set values, read can only read values and
|
||||
|
@ -495,7 +456,7 @@
|
|||
should match the automatic numbering to make it easier to
|
||||
debug the interpreted program.
|
||||
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
@section secAsync Asynchronous Transactions
|
||||
The asynchronous interface is used to increase the speed of
|
||||
transaction executing by better utilizing the connection
|
||||
|
@ -583,7 +544,7 @@
|
|||
|
||||
The poll method returns the number of transactions that
|
||||
have finished processing and executed their callback methods.
|
||||
|
||||
|
||||
@note When an asynchronous transaction has been started and sent to
|
||||
the NDB kernel, it is not allowed to execute any methods on
|
||||
objects belonging to this transaction until the transaction
|
||||
|
@ -595,7 +556,7 @@
|
|||
|
||||
More about how transactions are send the NDB Kernel is
|
||||
available in section @ref secAdapt.
|
||||
|
||||
#endif
|
||||
|
||||
@section secError Error Handling
|
||||
|
||||
|
@ -671,6 +632,7 @@
|
|||
* @include ndbapi_example4.cpp
|
||||
*/
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
/**
|
||||
* @page select_all.cpp select_all.cpp
|
||||
* @include select_all.cpp
|
||||
|
@ -680,6 +642,7 @@
|
|||
* @page ndbapi_async.cpp ndbapi_async.cpp
|
||||
* @include ndbapi_async.cpp
|
||||
*/
|
||||
#endif
|
||||
|
||||
/**
|
||||
* @page ndbapi_scan.cpp ndbapi_scan.cpp
|
||||
|
@ -691,8 +654,7 @@
|
|||
@page secAdapt Adaptive Send Algorithm
|
||||
|
||||
At the time of "sending" the transaction
|
||||
(using NdbConnection::execute, NdbConnection::executeAsynch,
|
||||
Ndb::sendPreparedTransactions, or Ndb::sendPollNdb), the transactions
|
||||
(using NdbConnection::execute), the transactions
|
||||
are in reality <em>not</em> immediately transfered to the NDB Kernel.
|
||||
Instead, the "sent" transactions are only kept in a
|
||||
special send list (buffer) in the Ndb object to which they belong.
|
||||
|
@ -847,12 +809,56 @@
|
|||
then a timeout error occurs.
|
||||
|
||||
Concurrent transactions (parallel application programs, thread-based
|
||||
applications, or applications with asynchronous transactions)
|
||||
applications)
|
||||
sometimes deadlock when they try to access the same information.
|
||||
Applications need to be programmed so that timeout errors
|
||||
occurring due to deadlocks are handled. This generally
|
||||
means that the transaction encountering timeout
|
||||
should be rolled back and restarted.
|
||||
|
||||
@section secHint Hints and performance
|
||||
|
||||
NDB API can be hinted to select a particular transaction coordinator.
|
||||
The default method is round robin where each set of new transactions
|
||||
is placed on the next NDB kernel node.
|
||||
By providing a distribution key (usually the primary key
|
||||
of the mostly used table of the transaction) for a record
|
||||
the transaction will be placed on the node where the primary replica
|
||||
of that record resides.
|
||||
Note that this is only a hint, the system can
|
||||
be under reconfiguration and then the NDB API
|
||||
will use select the transaction coordinator without using
|
||||
this hint.
|
||||
|
||||
Placing the transaction coordinator close
|
||||
to the actual data used in the transaction can in many cases
|
||||
improve performance significantly. This is particularly true for
|
||||
systems using TCP/IP. A system using Solaris and a 500 MHz processor
|
||||
has a cost model for TCP/IP communication which is:
|
||||
|
||||
30 microseconds + (100 nanoseconds * no of Bytes)
|
||||
|
||||
This means that if we can ensure that we use "popular" links we increase
|
||||
buffering and thus drastically reduce the communication cost.
|
||||
Systems using SCI has a different cost model which is:
|
||||
|
||||
5 microseconds + (10 nanoseconds * no of Bytes)
|
||||
|
||||
Thus SCI systems are much less dependent on selection of
|
||||
transaction coordinators.
|
||||
Typically TCP/IP systems spend 30-60% of the time during communication,
|
||||
whereas SCI systems typically spend 5-10% of the time during
|
||||
communication.
|
||||
Thus SCI means that less care from the NDB API programmer is
|
||||
needed and great scalability can be achieved even for applications using
|
||||
data from many parts of the database.
|
||||
|
||||
A simple example is an application that uses many simple updates where
|
||||
a transaction needs to update one record.
|
||||
This record has a 32 bit primary key,
|
||||
which is also the distribution key.
|
||||
Then the keyData will be the address of the integer
|
||||
of the primary key and keyLen will be 4.
|
||||
*/
|
||||
|
||||
#ifndef Ndb_H
|
||||
|
@ -945,6 +951,11 @@ public:
|
|||
* Semaphores, mutexes and so forth are easy ways of issuing memory
|
||||
* barriers without having to bother about the memory barrier concept.
|
||||
*
|
||||
*/
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
// to be documented later
|
||||
/*
|
||||
* If one Ndb object is used to handle parallel transactions through the
|
||||
* asynchronous programming interface, please read the notes regarding
|
||||
* asynchronous transactions (Section @ref secAsync).
|
||||
|
@ -955,6 +966,8 @@ public:
|
|||
* asynchronous transaction or the methods for
|
||||
* synchronous transactions but not both.
|
||||
*/
|
||||
#endif
|
||||
|
||||
class Ndb
|
||||
{
|
||||
friend class NdbReceiver;
|
||||
|
@ -976,29 +989,30 @@ public:
|
|||
* @{
|
||||
*/
|
||||
/**
|
||||
* The starting point of your application code is to create an
|
||||
* Ndb object.
|
||||
* This object represents the NDB kernel and is the main
|
||||
* object used in interaction with the NDB kernel.
|
||||
* The Ndb object represents a connection to a database.
|
||||
*
|
||||
* @note the init() method must be called before it may be used
|
||||
*
|
||||
* @param ndb_cluster_connection is a connection to a cluster containing
|
||||
* the database to be used
|
||||
* @param aCatalogName is the name of the catalog you want to use.
|
||||
* @note The catalog name provides a name space for the tables and
|
||||
* indexes created in any connection from the Ndb object.
|
||||
* @param aSchemaName is the name of the schema you
|
||||
* want to use. It is optional and defaults to the "def" schema.
|
||||
* want to use.
|
||||
* @note The schema name provides an additional name space
|
||||
* for the tables and indexes created in a given catalog.
|
||||
* @note The methods get/setDatabaseName and get/setDatabaseSchemaName
|
||||
* are equivalent to get/setCatalogName and get/setSchemaName.
|
||||
* The get/setDatabaseName and get/setDatabaseSchemaName are
|
||||
* deprecated.
|
||||
*/
|
||||
Ndb(const char* aCatalogName = "", const char* aSchemaName = "def");
|
||||
Ndb(Ndb_cluster_connection *ndb_cluster_connection,
|
||||
const char* aCatalogName = "", const char* aSchemaName = "def");
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
// depricated
|
||||
Ndb(const char* aCatalogName = "", const char* aSchemaName = "def");
|
||||
#endif
|
||||
~Ndb();
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
/**
|
||||
* The current catalog name can be fetched by getCatalogName.
|
||||
*
|
||||
|
@ -1026,7 +1040,7 @@ public:
|
|||
* @param aSchemaName is the new name of the current schema
|
||||
*/
|
||||
void setSchemaName(const char * aSchemaName);
|
||||
|
||||
#endif
|
||||
|
||||
/**
|
||||
* The current database name can be fetched by getDatabaseName.
|
||||
|
@ -1057,22 +1071,22 @@ public:
|
|||
void setDatabaseSchemaName(const char * aDatabaseSchemaName);
|
||||
|
||||
/**
|
||||
* Before anything else it is necessary to initialize (start)
|
||||
* the Ndb object.
|
||||
* Initializes the Ndb object
|
||||
*
|
||||
* @param maxNoOfTransactions
|
||||
* Maximum number of parallel
|
||||
* NdbConnection objects that should be handled by the Ndb object.
|
||||
* A value larger than 1024 will be downgraded to 1024.
|
||||
* This means that one Ndb object can handle at most 1024 parallel
|
||||
* transactions.
|
||||
* @return 0 if successful, -1 otherwise.
|
||||
* NdbConnection objects that can be handled by the Ndb object.
|
||||
* Maximum value is 1024.
|
||||
*
|
||||
* @note The internal implementation multiplies this value
|
||||
* with 3.
|
||||
* @note each scan or index scan operation uses one extra
|
||||
* NdbConnection object
|
||||
*
|
||||
* @return 0 if successful, -1 otherwise.
|
||||
*/
|
||||
int init(int maxNoOfTransactions = 4);
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
// depricated
|
||||
/**
|
||||
* Wait for Ndb object to successfully set-up connections to
|
||||
* the NDB kernel.
|
||||
|
@ -1085,8 +1099,8 @@ public:
|
|||
* @return 0: Ndb is ready and timeout has not occurred.<br>
|
||||
* -1: Timeout has expired
|
||||
*/
|
||||
|
||||
int waitUntilReady(int timeout = 60);
|
||||
#endif
|
||||
|
||||
/** @} *********************************************************************/
|
||||
|
||||
|
@ -1096,30 +1110,55 @@ public:
|
|||
*/
|
||||
|
||||
/**
|
||||
* Query the database for schema information
|
||||
* (without performing any transaction).
|
||||
* Get an object for retrieving or manipulating database schema information
|
||||
*
|
||||
* @note this object operates outside any transaction
|
||||
*
|
||||
* @return Object containing meta information about all tables
|
||||
* in NDB Cluster.
|
||||
*/
|
||||
class NdbDictionary::Dictionary* getDictionary() const;
|
||||
|
||||
NdbEventOperation* createEventOperation(const char* eventName,
|
||||
const int bufferLength);
|
||||
int dropEventOperation(NdbEventOperation*);
|
||||
void monitorEvent(NdbEventOperation *, NdbEventCallback, void*);
|
||||
int pollEvents(int aMillisecondNumber);
|
||||
|
||||
/** @} *********************************************************************/
|
||||
|
||||
/**
|
||||
* @name Event subscriptions
|
||||
* @{
|
||||
*/
|
||||
|
||||
/**
|
||||
* Get the application node identity.
|
||||
* Create a subcription to an event defined in the database
|
||||
*
|
||||
* Each node (DB nodes, Applications, and Management Servers)
|
||||
* has its own node identity in the NDB Cluster.
|
||||
* See documentation for the management server configuration file.
|
||||
* @param eventName
|
||||
* unique identifier of the event
|
||||
* @param bufferLength
|
||||
* buffer size for storing event data
|
||||
*
|
||||
* @return Node id of this application.
|
||||
* @return Object representing an event, NULL on failure
|
||||
*/
|
||||
int getNodeId();
|
||||
NdbEventOperation* createEventOperation(const char* eventName,
|
||||
const int bufferLength);
|
||||
/**
|
||||
* Drop a subscription to an event
|
||||
*
|
||||
* @param eventName
|
||||
* unique identifier of the event
|
||||
*
|
||||
* @return 0 on success
|
||||
*/
|
||||
int dropEventOperation(NdbEventOperation* eventName);
|
||||
|
||||
/**
|
||||
* Wait for an event to occur. Will return as soon as an event
|
||||
* is detected on any of the created events.
|
||||
*
|
||||
* @param aMillisecondNumber
|
||||
* maximum time to wait
|
||||
*
|
||||
* @return the number of events that has occured, -1 on failure
|
||||
*/
|
||||
int pollEvents(int aMillisecondNumber);
|
||||
|
||||
/** @} *********************************************************************/
|
||||
|
||||
|
@ -1129,71 +1168,19 @@ public:
|
|||
*/
|
||||
|
||||
/**
|
||||
* This method returns an NdbConnection which caters for the transaction.
|
||||
* When the transaction is completed it must be closed.
|
||||
* The Ndb::closeTransaction also return the NdbConnection object
|
||||
* and all other memory related to the transaction.
|
||||
* Failure to close the transaction will lead to memory leakage.
|
||||
* The transaction must be closed independent of its outcome, i.e.
|
||||
* even if there is an error.
|
||||
* Start a transaction
|
||||
*
|
||||
* @note When the transaction is completed it must be closed using
|
||||
* Ndb::closeTransaction or NdbConnection::close.
|
||||
* The transaction must be closed independent of its outcome, i.e.
|
||||
* even if there is an error.
|
||||
*
|
||||
* @param prio Not implemented
|
||||
* @param keyData Pointer to partition key to be used for deciding
|
||||
* which node to run the Transaction Coordinator on
|
||||
* @param keyLen Length of partition key expressed in bytes
|
||||
*
|
||||
* NDB API can be hinted to select a particular transaction coordinator.
|
||||
* The default method is round robin where each set of new transactions
|
||||
* is placed on the next NDB kernel node.
|
||||
* By providing a distribution key (usually the primary key
|
||||
* of the mostly used table of the transaction) for a record
|
||||
* the transaction will be placed on the node where the primary replica
|
||||
* of that record resides.
|
||||
* Note that this is only a hint, the system can
|
||||
* be under reconfiguration and then the NDB API
|
||||
* will use select the transaction coordinator without using
|
||||
* this hint.
|
||||
*
|
||||
* Placing the transaction coordinator close
|
||||
* to the actual data used in the transaction can in many cases
|
||||
* improve performance significantly. This is particularly true for
|
||||
* systems using TCP/IP. A system using Solaris and a 500 MHz processor
|
||||
* has a cost model for TCP/IP communication which is:
|
||||
*
|
||||
* 30 microseconds + (100 nanoseconds * no of Bytes)
|
||||
*
|
||||
* This means that if we can ensure that we use "popular" links we increase
|
||||
* buffering and thus drastically reduce the communication cost.
|
||||
* Systems using SCI has a different cost model which is:
|
||||
*
|
||||
* 5 microseconds + (10 nanoseconds * no of Bytes)
|
||||
*
|
||||
* Thus SCI systems are much less dependent on selection of
|
||||
* transaction coordinators.
|
||||
* Typically TCP/IP systems spend 30-60% of the time during communication,
|
||||
* whereas SCI systems typically spend 5-10% of the time during
|
||||
* communication.
|
||||
* Thus SCI means that less care from the NDB API programmer is
|
||||
* needed and great scalability can be achieved even for applications using
|
||||
* data from many parts of the database.
|
||||
*
|
||||
* A simple example is an application that uses many simple updates where
|
||||
* a transaction needs to update one record.
|
||||
* This record has a 32 bit primary key,
|
||||
* which is also the distribution key.
|
||||
* Then the keyData will be the address of the integer
|
||||
* of the primary key and keyLen will be 4.
|
||||
*
|
||||
* @note Transaction priorities are not yet supported.
|
||||
*
|
||||
* @param prio The priority of the transaction.<br>
|
||||
* Priority 0 is the highest priority and is used
|
||||
* for short transactions with requirements on low delay.<br>
|
||||
* Priority 1 is a medium priority for short transactions.
|
||||
* <br>
|
||||
* Priority 2 is a medium priority for long transactions.<br>
|
||||
* Priority 3 is a low priority for long transactions.<br>
|
||||
* <em>This parameter is not currently used,
|
||||
* and can be set to any value</em>
|
||||
* @param keyData Pointer to distribution key
|
||||
* @param keyLen Length of distribution key expressed in bytes
|
||||
*
|
||||
* @return NdbConnection object, or NULL if method failed.
|
||||
* @return NdbConnection object, or NULL on failure.
|
||||
*/
|
||||
NdbConnection* startTransaction(Uint32 prio = 0,
|
||||
const char * keyData = 0,
|
||||
|
@ -1233,7 +1220,10 @@ public:
|
|||
#endif
|
||||
|
||||
/**
|
||||
* When a transactions is completed, the transaction has to be closed.
|
||||
* Close a transaction.
|
||||
*
|
||||
* @note should be called after the transaction has completed, irrespective
|
||||
* of success or failure
|
||||
*
|
||||
* @note It is not allowed to call Ndb::closeTransaction after sending the
|
||||
* transaction asynchronously with either
|
||||
|
@ -1245,10 +1235,11 @@ public:
|
|||
* If the transaction is not committed it will be aborted.
|
||||
*/
|
||||
void closeTransaction(NdbConnection* aConnection);
|
||||
|
||||
|
||||
/** @} *********************************************************************/
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
// to be documented later
|
||||
/**
|
||||
* @name Asynchronous Transactions
|
||||
* @{
|
||||
|
@ -1259,11 +1250,10 @@ public:
|
|||
* Will return as soon as at least 'minNoOfEventsToWakeUp'
|
||||
* of them have completed, or the maximum time given as timeout has passed.
|
||||
*
|
||||
* @param aMillisecondNumber Maximum time to wait for transactions
|
||||
* to complete.
|
||||
* Polling without wait is achieved by setting the
|
||||
* timer to zero.
|
||||
* Time is expressed in milliseconds.
|
||||
* @param aMillisecondNumber
|
||||
* Maximum time to wait for transactions to complete. Polling
|
||||
* without wait is achieved by setting the timer to zero.
|
||||
* Time is expressed in milliseconds.
|
||||
* @param minNoOfEventsToWakeup Minimum number of transactions
|
||||
* which has to wake up before the poll-call will return.
|
||||
* If minNoOfEventsToWakeup is
|
||||
|
@ -1325,6 +1315,7 @@ public:
|
|||
int sendPollNdb(int aMillisecondNumber = WAITFOR_RESPONSE_TIMEOUT,
|
||||
int minNoOfEventsToWakeup = 1,
|
||||
int forceSend = 0);
|
||||
#endif
|
||||
|
||||
/** @} *********************************************************************/
|
||||
|
||||
|
@ -1336,7 +1327,7 @@ public:
|
|||
/**
|
||||
* Get the NdbError object
|
||||
*
|
||||
* The NdbError object is valid until you call a new NDB API method.
|
||||
* @note The NdbError object is valid until a new NDB API method is called.
|
||||
*/
|
||||
const NdbError & getNdbError() const;
|
||||
|
||||
|
@ -1348,37 +1339,36 @@ public:
|
|||
const NdbError & getNdbError(int errorCode);
|
||||
|
||||
|
||||
/** @} *********************************************************************/
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
/**
|
||||
* Get the application node identity.
|
||||
*
|
||||
* @return Node id of this application.
|
||||
*/
|
||||
int getNodeId();
|
||||
|
||||
/**
|
||||
* setConnectString
|
||||
* @param connectString - the connectString has the following format:
|
||||
* @code
|
||||
* "nodeid=<ID>;host=host://<HOSTNAME>:<PORT>;
|
||||
* host=host://<HOSTNAME2>:<PORT>;..."
|
||||
* @endcode
|
||||
* or
|
||||
* @code
|
||||
* "nodeid=<ID>;host=<HOSTNAME>:<PORT>;host=<HOSTNAME2>:<PORT>;..."
|
||||
* @endcode
|
||||
*
|
||||
* @param connectString - see MySQL ref manual for format
|
||||
*/
|
||||
static void setConnectString(const char * connectString);
|
||||
|
||||
bool usingFullyQualifiedNames();
|
||||
|
||||
/** @} *********************************************************************/
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
|
||||
/**
|
||||
* Different types of tampering with the NDB Cluster.
|
||||
* <b>Only for debugging purposes only.</b>
|
||||
*/
|
||||
enum TamperType {
|
||||
LockGlbChp = 1, ///< Lock GCP
|
||||
UnlockGlbChp, ///< Unlock GCP
|
||||
CrashNode, ///< Crash an NDB node
|
||||
ReadRestartGCI, ///< Request the restart GCI id from NDB Cluster
|
||||
InsertError ///< Execute an error in NDB Cluster
|
||||
///< (may crash system)
|
||||
LockGlbChp = 1, ///< Lock GCP
|
||||
UnlockGlbChp, ///< Unlock GCP
|
||||
CrashNode, ///< Crash an NDB node
|
||||
ReadRestartGCI, ///< Request the restart GCI id from NDB Cluster
|
||||
InsertError ///< Execute an error in NDB Cluster
|
||||
///< (may crash system)
|
||||
};
|
||||
|
||||
/**
|
||||
|
@ -1397,9 +1387,7 @@ public:
|
|||
* on type of tampering.
|
||||
*/
|
||||
int NdbTamper(TamperType aAction, int aNode);
|
||||
#endif
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
/**
|
||||
* Return a unique tuple id for a table. The id sequence is
|
||||
* ascending but may contain gaps.
|
||||
|
@ -1429,9 +1417,7 @@ public:
|
|||
bool increase);
|
||||
bool setTupleIdInNdb(Uint32 aTableId, Uint64 val, bool increase);
|
||||
Uint64 opTupleIdOnNdb(Uint32 aTableId, Uint64 opValue, Uint32 op);
|
||||
#endif
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
/**
|
||||
*/
|
||||
NdbConnection* hupp( NdbConnection* );
|
||||
|
|
|
@ -31,6 +31,8 @@ class Ndb;
|
|||
class NdbBlob;
|
||||
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
// to be documented later
|
||||
/**
|
||||
* NdbAsynchCallback functions are used when executing asynchronous
|
||||
* transactions (using NdbConnection::executeAsynchPrepare, or
|
||||
|
@ -39,6 +41,7 @@ class NdbBlob;
|
|||
* See @ref secAsync for more information.
|
||||
*/
|
||||
typedef void (* NdbAsynchCallback)(int, NdbConnection*, void*);
|
||||
#endif
|
||||
|
||||
/**
|
||||
* Commit type of transaction
|
||||
|
@ -184,7 +187,8 @@ public:
|
|||
* @note All operations within the same transaction need to
|
||||
* be initialized with this method.
|
||||
*
|
||||
* @param aTable A table object (fetched by NdbDictionary::Dictionary::getTable)
|
||||
* @param aTable
|
||||
* A table object (fetched by NdbDictionary::Dictionary::getTable)
|
||||
* @return Pointer to an NdbOperation object if successful, otherwise NULL.
|
||||
*/
|
||||
NdbOperation* getNdbOperation(const NdbDictionary::Table * aTable);
|
||||
|
@ -204,7 +208,8 @@ public:
|
|||
* get the NdbConnection object which
|
||||
* was fetched by startTransaction pointing to this operation.
|
||||
*
|
||||
* @param aTable A table object (fetched by NdbDictionary::Dictionary::getTable)
|
||||
* @param aTable
|
||||
* A table object (fetched by NdbDictionary::Dictionary::getTable)
|
||||
* @return pointer to an NdbOperation object if successful, otherwise NULL
|
||||
*/
|
||||
NdbScanOperation* getNdbScanOperation(const NdbDictionary::Table * aTable);
|
||||
|
@ -226,12 +231,15 @@ public:
|
|||
* get the NdbConnection object which
|
||||
* was fetched by startTransaction pointing to this operation.
|
||||
*
|
||||
* @param anIndex An index object (fetched by NdbDictionary::Dictionary::getIndex).
|
||||
* @param aTable A table object (fetched by NdbDictionary::Dictionary::getTable).
|
||||
* @param anIndex
|
||||
An index object (fetched by NdbDictionary::Dictionary::getIndex).
|
||||
* @param aTable
|
||||
A table object (fetched by NdbDictionary::Dictionary::getTable).
|
||||
* @return pointer to an NdbOperation object if successful, otherwise NULL
|
||||
*/
|
||||
NdbIndexScanOperation* getNdbIndexScanOperation(const NdbDictionary::Index * anIndex,
|
||||
const NdbDictionary::Table * aTable);
|
||||
NdbIndexScanOperation* getNdbIndexScanOperation
|
||||
(const NdbDictionary::Index * anIndex,
|
||||
const NdbDictionary::Table * aTable);
|
||||
|
||||
/**
|
||||
* Get an operation from NdbIndexOperation idlelist and
|
||||
|
@ -251,8 +259,10 @@ public:
|
|||
* get the NdbConnection object that
|
||||
* was fetched by startTransaction pointing to this operation.
|
||||
*
|
||||
* @param anIndex An index object (fetched by NdbDictionary::Dictionary::getIndex).
|
||||
* @param aTable A table object (fetched by NdbDictionary::Dictionary::getTable).
|
||||
* @param anIndex
|
||||
* An index object (fetched by NdbDictionary::Dictionary::getIndex).
|
||||
* @param aTable
|
||||
* A table object (fetched by NdbDictionary::Dictionary::getTable).
|
||||
* @return Pointer to an NdbIndexOperation object if
|
||||
* successful, otherwise NULL
|
||||
*/
|
||||
|
@ -289,6 +299,8 @@ public:
|
|||
AbortOption abortOption = AbortOnError,
|
||||
int force = 0 );
|
||||
|
||||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
// to be documented later
|
||||
/**
|
||||
* Prepare an asynchronous transaction.
|
||||
*
|
||||
|
@ -334,7 +346,7 @@ public:
|
|||
NdbAsynchCallback aCallback,
|
||||
void* anyObject,
|
||||
AbortOption abortOption = AbortOnError);
|
||||
|
||||
#endif
|
||||
/**
|
||||
* Refresh
|
||||
* Update timeout counter of this transaction
|
||||
|
@ -397,14 +409,14 @@ public:
|
|||
* (Note that there has to be an NdbConnection::execute call
|
||||
* with Ndb::Commit for the GCI to be available.)
|
||||
*/
|
||||
int getGCI();
|
||||
int getGCI();
|
||||
|
||||
/**
|
||||
* Get transaction identity.
|
||||
*
|
||||
* @return Transaction id.
|
||||
*/
|
||||
Uint64 getTransactionId();
|
||||
Uint64 getTransactionId();
|
||||
|
||||
/**
|
||||
* Returns the commit status of the transaction.
|
||||
|
|
|
@ -166,7 +166,7 @@ public:
|
|||
* The builtin column types
|
||||
*/
|
||||
enum Type {
|
||||
Undefined=0,///< Undefined
|
||||
Undefined=0, ///< Undefined
|
||||
Tinyint, ///< 8 bit. 1 byte signed integer, can be used in array
|
||||
Tinyunsigned, ///< 8 bit. 1 byte unsigned integer, can be used in array
|
||||
Smallint, ///< 16 bit. 2 byte signed integer, can be used in array
|
||||
|
@ -374,16 +374,11 @@ public:
|
|||
#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
|
||||
const Table * getBlobTable() const;
|
||||
|
||||
/**
|
||||
* @name ODBC Specific methods
|
||||
* @{
|
||||
*/
|
||||
void setAutoIncrement(bool);
|
||||
void setAutoIncrement(bool);
|
||||
bool getAutoIncrement() const;
|
||||
void setAutoIncrementInitialValue(Uint64 val);
|
||||
void setDefaultValue(const char*);
|
||||
const char* getDefaultValue() const;
|
||||
/** @} *******************************************************************/
|
||||
|
||||
static const Column * FRAGMENT;
|
||||
static const Column * ROW_COUNT;
|
||||
|
|
|
@ -125,6 +125,16 @@ int runCreateTheTable(NDBT_Context* ctx, NDBT_Step* step){
|
|||
return NDBT_OK;
|
||||
}
|
||||
|
||||
int runDropTheTable(NDBT_Context* ctx, NDBT_Step* step){
|
||||
Ndb* pNdb = GETNDB(step);
|
||||
const NdbDictionary::Table* pTab = ctx->getTab();
|
||||
|
||||
// Try to create table in db
|
||||
pNdb->getDictionary()->dropTable(pTab->getName());
|
||||
|
||||
return NDBT_OK;
|
||||
}
|
||||
|
||||
int runCreateTableWhenDbIsFull(NDBT_Context* ctx, NDBT_Step* step){
|
||||
Ndb* pNdb = GETNDB(step);
|
||||
int result = NDBT_OK;
|
||||
|
@ -1584,7 +1594,7 @@ TESTCASE("CreateTableWhenDbIsFull",
|
|||
INITIALIZER(runFillTable);
|
||||
INITIALIZER(runCreateTableWhenDbIsFull);
|
||||
INITIALIZER(runDropTableWhenDbIsFull);
|
||||
FINALIZER(runClearTable);
|
||||
FINALIZER(runDropTheTable);
|
||||
}
|
||||
TESTCASE("FragmentTypeSingle",
|
||||
"Create the table with fragment type Single\n"){
|
||||
|
|
|
@ -1277,7 +1277,7 @@ TESTCASE("CreateLoadDrop_O",
|
|||
TESTCASE("NFNR1",
|
||||
"Test that indexes are correctly maintained during node fail and node restart"){
|
||||
TC_PROPERTY("LoggedIndexes", (unsigned)0);
|
||||
//TC_PROPERTY("Threads", 2);
|
||||
TC_PROPERTY("PauseThreads", 2);
|
||||
INITIALIZER(runClearTable);
|
||||
INITIALIZER(createRandomIndex);
|
||||
INITIALIZER(runLoadTable);
|
||||
|
@ -1292,6 +1292,7 @@ TESTCASE("NFNR1_O",
|
|||
"Test that indexes are correctly maintained during node fail and node restart"){
|
||||
TC_PROPERTY("OrderedIndex", 1);
|
||||
TC_PROPERTY("LoggedIndexes", (unsigned)0);
|
||||
TC_PROPERTY("PauseThreads", 2);
|
||||
INITIALIZER(runClearTable);
|
||||
INITIALIZER(createRandomIndex);
|
||||
INITIALIZER(runLoadTable);
|
||||
|
@ -1305,6 +1306,7 @@ TESTCASE("NFNR1_O",
|
|||
TESTCASE("NFNR2",
|
||||
"Test that indexes are correctly maintained during node fail and node restart"){
|
||||
TC_PROPERTY("LoggedIndexes", (unsigned)0);
|
||||
TC_PROPERTY("PauseThreads", 2);
|
||||
INITIALIZER(runClearTable);
|
||||
INITIALIZER(createRandomIndex);
|
||||
INITIALIZER(createPkIndex);
|
||||
|
@ -1321,6 +1323,7 @@ TESTCASE("NFNR2_O",
|
|||
"Test that indexes are correctly maintained during node fail and node restart"){
|
||||
TC_PROPERTY("OrderedIndex", 1);
|
||||
TC_PROPERTY("LoggedIndexes", (unsigned)0);
|
||||
TC_PROPERTY("PauseThreads", 2);
|
||||
INITIALIZER(runClearTable);
|
||||
INITIALIZER(createRandomIndex);
|
||||
INITIALIZER(createPkIndex);
|
||||
|
@ -1336,6 +1339,7 @@ TESTCASE("NFNR2_O",
|
|||
TESTCASE("NFNR3",
|
||||
"Test that indexes are correctly maintained during node fail and node restart"){
|
||||
TC_PROPERTY("LoggedIndexes", (unsigned)0);
|
||||
TC_PROPERTY("PauseThreads", 2);
|
||||
INITIALIZER(runClearTable);
|
||||
INITIALIZER(createRandomIndex);
|
||||
INITIALIZER(createPkIndex);
|
||||
|
@ -1351,6 +1355,7 @@ TESTCASE("NFNR3_O",
|
|||
"Test that indexes are correctly maintained during node fail and node restart"){
|
||||
TC_PROPERTY("OrderedIndex", 1);
|
||||
TC_PROPERTY("LoggedIndexes", (unsigned)0);
|
||||
TC_PROPERTY("PauseThreads", 2);
|
||||
INITIALIZER(runClearTable);
|
||||
INITIALIZER(createRandomIndex);
|
||||
INITIALIZER(createPkIndex);
|
||||
|
@ -1365,6 +1370,7 @@ TESTCASE("NFNR3_O",
|
|||
TESTCASE("NFNR4",
|
||||
"Test that indexes are correctly maintained during node fail and node restart"){
|
||||
TC_PROPERTY("LoggedIndexes", (unsigned)0);
|
||||
TC_PROPERTY("PauseThreads", 4);
|
||||
INITIALIZER(runClearTable);
|
||||
INITIALIZER(createRandomIndex);
|
||||
INITIALIZER(createPkIndex);
|
||||
|
@ -1383,6 +1389,7 @@ TESTCASE("NFNR4_O",
|
|||
"Test that indexes are correctly maintained during node fail and node restart"){
|
||||
TC_PROPERTY("OrderedIndex", 1);
|
||||
TC_PROPERTY("LoggedIndexes", (unsigned)0);
|
||||
TC_PROPERTY("PauseThreads", 4);
|
||||
INITIALIZER(runClearTable);
|
||||
INITIALIZER(createRandomIndex);
|
||||
INITIALIZER(createPkIndex);
|
||||
|
|
|
@ -799,13 +799,13 @@ int runUpdateWithoutValues(NDBT_Context* ctx, NDBT_Step* step){
|
|||
|
||||
// Dont' call any setValues
|
||||
|
||||
// Execute should not work
|
||||
// Execute should work
|
||||
int check = pCon->execute(Commit);
|
||||
if (check == 0){
|
||||
ndbout << "execute worked" << endl;
|
||||
result = NDBT_FAILED;
|
||||
} else {
|
||||
ERR(pCon->getNdbError());
|
||||
result = NDBT_FAILED;
|
||||
}
|
||||
|
||||
pNdb->closeTransaction(pCon);
|
||||
|
|
|
@ -465,9 +465,73 @@ max-time: 150000
|
|||
cmd: testOperations
|
||||
args:
|
||||
|
||||
max-time: 150000
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args:
|
||||
args: T1
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T2
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T3
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T4
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T5
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T6
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T7
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T8
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T9
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T10
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T11
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T12
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T13
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: T14
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: I1
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: I2
|
||||
|
||||
max-time: 1500
|
||||
cmd: testTransactions
|
||||
args: I3
|
||||
|
||||
max-time: 1500
|
||||
cmd: testRestartGci
|
||||
|
@ -477,7 +541,7 @@ max-time: 600
|
|||
cmd: testBlobs
|
||||
args:
|
||||
|
||||
max-time: 2500
|
||||
max-time: 5000
|
||||
cmd: testOIBasic
|
||||
args:
|
||||
|
||||
|
|
|
@ -1097,7 +1097,7 @@ HugoTransactions::pkReadRecords(Ndb* pNdb,
|
|||
pNdb->closeTransaction(pTrans);
|
||||
return NDBT_FAILED;
|
||||
}
|
||||
} else{
|
||||
} else {
|
||||
if(pIndexScanOp)
|
||||
{
|
||||
int rows_found = 0;
|
||||
|
@ -1759,7 +1759,7 @@ HugoTransactions::indexReadRecords(Ndb* pNdb,
|
|||
pNdb->closeTransaction(pTrans);
|
||||
return NDBT_FAILED;
|
||||
}
|
||||
check = 0;
|
||||
check = sOp->readTuples();
|
||||
}
|
||||
|
||||
if( check == -1 ) {
|
||||
|
@ -1948,7 +1948,7 @@ HugoTransactions::indexUpdateRecords(Ndb* pNdb,
|
|||
}
|
||||
|
||||
if(ordered && check != 0){
|
||||
g_err << "Row: " << r << " not found!!" << endl;
|
||||
g_err << check << " - Row: " << r << " not found!!" << endl;
|
||||
pNdb->closeTransaction(pTrans);
|
||||
return NDBT_FAILED;
|
||||
}
|
||||
|
|
|
@ -629,7 +629,7 @@ UtilTransactions::scanAndCompareUniqueIndex(Ndb* pNdb,
|
|||
parallelism = 1;
|
||||
|
||||
while (true){
|
||||
|
||||
restart:
|
||||
if (retryAttempt >= retryMax){
|
||||
g_info << "ERROR: has retried this operation " << retryAttempt
|
||||
<< " times, failing!" << endl;
|
||||
|
@ -719,11 +719,26 @@ UtilTransactions::scanAndCompareUniqueIndex(Ndb* pNdb,
|
|||
|
||||
// ndbout << row.c_str().c_str() << endl;
|
||||
|
||||
|
||||
if (readRowFromTableAndIndex(pNdb,
|
||||
pTrans,
|
||||
pIndex,
|
||||
row) != NDBT_OK){
|
||||
|
||||
while((eof= pOp->nextResult(false)) == 0);
|
||||
if(eof == 2)
|
||||
eof = pOp->nextResult(true); // this should give -1
|
||||
if(eof == -1)
|
||||
{
|
||||
const NdbError err = pTrans->getNdbError();
|
||||
|
||||
if (err.status == NdbError::TemporaryError){
|
||||
ERR(err);
|
||||
pNdb->closeTransaction(pTrans);
|
||||
NdbSleep_MilliSleep(50);
|
||||
retryAttempt++;
|
||||
goto restart;
|
||||
}
|
||||
}
|
||||
pNdb->closeTransaction(pTrans);
|
||||
return NDBT_FAILED;
|
||||
}
|
||||
|
@ -736,7 +751,6 @@ UtilTransactions::scanAndCompareUniqueIndex(Ndb* pNdb,
|
|||
pNdb->closeTransaction(pTrans);
|
||||
NdbSleep_MilliSleep(50);
|
||||
retryAttempt++;
|
||||
rows--;
|
||||
continue;
|
||||
}
|
||||
ERR(err);
|
||||
|
@ -811,7 +825,6 @@ UtilTransactions::readRowFromTableAndIndex(Ndb* pNdb,
|
|||
check = pOp->readTuple();
|
||||
if( check == -1 ) {
|
||||
ERR(pTrans1->getNdbError());
|
||||
pNdb->closeTransaction(pTrans1);
|
||||
goto close_all;
|
||||
}
|
||||
|
||||
|
@ -943,7 +956,7 @@ UtilTransactions::readRowFromTableAndIndex(Ndb* pNdb,
|
|||
#if VERBOSE
|
||||
printf("\n");
|
||||
#endif
|
||||
|
||||
scanTrans->refresh();
|
||||
check = pTrans1->execute(Commit);
|
||||
if( check == -1 ) {
|
||||
const NdbError err = pTrans1->getNdbError();
|
||||
|
|
Loading…
Add table
Reference in a new issue