2023-03-23 17:27:46 +00:00
/*
* LMDB Abstraction Layer .
* Copyright ( C ) 2023 Yury Gubich < blue @ macaw . me >
*
* 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 3 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 , see < http : //www.gnu.org/licenses/>.
*/
2023-03-20 15:37:13 +00:00
2023-03-21 11:05:54 +00:00
# ifndef LMDBAL_STORAGE_HPP
# define LMDBAL_STORAGE_HPP
2023-03-20 15:37:13 +00:00
# include "storage.h"
# include "exceptions.h"
2023-08-09 17:41:15 +00:00
# define UNUSED(x) (void)(x)
2023-04-07 23:27:53 +00:00
/**
* \ class LMDBAL : : Storage
* \ brief This is a basic key value storage .
*
* \ tparam K type of the keys of the storage
* \ tparam V type of the values of the storage
*
2023-08-20 16:38:29 +00:00
* You can receive an instance of this class calling LMDBAL : : Base : : addStorage ( const std : : string & , bool )
2023-04-07 23:27:53 +00:00
* if the database is yet closed and you ' re defining the storages you ' re going to need .
2023-04-12 15:36:33 +00:00
* Or you can call LMDBAL : : Base : : getStorage ( const std : : string & ) if you didn ' t save a pointer to the storage at first
2023-04-07 23:27:53 +00:00
*
* You are not supposed to instantiate or destory instances of this class yourself !
*/
2023-04-12 15:36:33 +00:00
/**
* \ brief Creates a storage
*
2023-08-15 18:48:19 +00:00
* \ param [ in ] parent - LMDBAL : : Base pointer for the owning database ( borrowed )
* \ param [ in ] name - the name of the storage
2023-08-18 13:31:30 +00:00
* \ param [ in ] duplicates - true if key duplicates are allowed ( false by default )
2023-04-12 15:36:33 +00:00
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-08-18 13:31:30 +00:00
LMDBAL : : Storage < K , V > : : Storage ( Base * parent , const std : : string & name , bool duplicates ) :
2023-08-15 18:48:19 +00:00
iStorage ( parent , name , duplicates ) ,
2023-08-05 20:13:43 +00:00
keySerializer ( ) ,
valueSerializer ( ) ,
cursors ( )
2023-03-20 15:37:13 +00:00
{ }
2023-04-12 15:36:33 +00:00
/**
* \ brief Destroys a storage
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-08-07 21:27:44 +00:00
LMDBAL : : Storage < K , V > : : ~ Storage ( ) {
2023-10-25 22:18:23 +00:00
for ( const std : : pair < const uint32_t , Cursor < K , V > * > & pair : cursors )
pair . second - > dropped ( ) ;
2023-08-07 21:27:44 +00:00
}
2023-03-20 15:37:13 +00:00
2023-04-07 23:27:53 +00:00
/**
* \ brief Adds a key - value record to the storage
*
2023-08-18 15:04:37 +00:00
* Take a note that if the storage already had a record you want to add LMDBAL : : Exist is thrown .
* If your storage doesn ' t support duplicates LMDBAL : : Exist is thrown if the record with the same key already exists in the database .
* If your storage supports duplicates LMDBAL : : Exist is thrown only if the record with the same key < b > AND < / b > already exists in the database .
2023-04-07 23:27:53 +00:00
*
* \ param [ in ] key key of the record
* \ param [ in ] value value of the record
*
* \ exception LMDBAL : : Exist thrown if the storage already has a record with the given key
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-03-21 11:05:54 +00:00
void LMDBAL : : Storage < K , V > : : addRecord ( const K & key , const V & value ) {
2023-03-20 15:37:13 +00:00
ensureOpened ( addRecordMethodName ) ;
TransactionID txn = beginTransaction ( ) ;
2023-03-28 20:45:35 +00:00
try {
2023-04-04 23:27:31 +00:00
Storage < K , V > : : addRecord ( key , value , txn ) ;
2023-03-28 20:45:35 +00:00
} catch ( . . . ) {
abortTransaction ( txn ) ;
throw ;
}
commitTransaction ( txn ) ;
}
2023-04-07 23:27:53 +00:00
/**
2023-10-17 21:06:11 +00:00
* \ brief Adds a key - value record to the storage ( private transaction variant )
2023-04-07 23:27:53 +00:00
*
2023-08-18 15:04:37 +00:00
* Take a note that if the storage already had a record you want to add LMDBAL : : Exist is thrown .
* If your storage doesn ' t support duplicates LMDBAL : : Exist is thrown if the record with the same key already exists in the database .
* If your storage supports duplicates LMDBAL : : Exist is thrown only if the record with the same key < b > AND < / b > already exists in the database .
2023-04-07 23:27:53 +00:00
*
* \ param [ in ] key key of the record
* \ param [ in ] value value of the record
* \ param [ in ] txn transaction ID , needs to be a writable transaction !
*
* \ exception LMDBAL : : Exist thrown if the storage already has a record with the given key
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
void LMDBAL : : Storage < K , V > : : addRecord ( const K & key , const V & value , TransactionID txn ) {
2023-08-05 20:13:43 +00:00
MDB_val lmdbKey = keySerializer . setData ( key ) ;
MDB_val lmdbData = valueSerializer . setData ( value ) ;
2023-03-20 15:37:13 +00:00
2023-08-18 13:31:30 +00:00
unsigned int flags = 0 ;
if ( duplicates )
flags | = MDB_NODUPDATA ;
else
flags | = MDB_NOOVERWRITE ;
2023-08-17 14:45:11 +00:00
2023-11-14 23:15:16 +00:00
int rc = _mdbPut ( txn , lmdbKey , lmdbData , flags ) ;
2023-03-20 15:37:13 +00:00
if ( rc ! = MDB_SUCCESS )
2023-03-28 20:45:35 +00:00
throwDuplicateOrUnknown ( rc , toString ( key ) ) ;
}
2023-10-17 21:06:11 +00:00
/**
* \ brief Adds a key - value record to the storage ( public transaction variant )
*
* This method schedules an addition of a key - value record , but doesn ' t immidiately adds it .
* You can obtain LMDBAL : : WriteTransaction calling LMDBAL : : Base : : beginTransaction ( ) .
*
* Take a note that if the storage already had a record you want to add LMDBAL : : Exist is thrown .
* If your storage doesn ' t support duplicates LMDBAL : : Exist is thrown if the record with the same key already exists in the database .
* If your storage supports duplicates LMDBAL : : Exist is thrown only if the record with the same key < b > AND < / b > already exists in the database .
*
* \ param [ in ] key key of the record
* \ param [ in ] value value of the record
* \ param [ in ] txn transaction
*
* \ exception LMDBAL : : Exist thrown if the storage already has a record with the given key
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
* \ exception LMDBAL : : TransactionTerminated thrown if the passed transaction not active , any action with it ' s inner ID is an error
*/
template < class K , class V >
void LMDBAL : : Storage < K , V > : : addRecord ( const K & key , const V & value , const WriteTransaction & txn ) {
ensureOpened ( addRecordMethodName ) ;
addRecord ( key , value , extractTransactionId ( txn , addRecordMethodName ) ) ;
}
2023-04-07 23:27:53 +00:00
/**
* \ brief Adds a key - value record to the storage , overwrites if it already exists
2023-08-19 19:25:52 +00:00
*
* This method is mostly useless in duplicates mode .
* In this mode it basically does the same thing LMDBAL : : Storage : : addRecord ( ) does ,
* but suppresses LMDBAL : : Exist exception if the record with the same key - value pair existed in the storage .
* In this case just false is returned from the method .
*
2023-04-07 23:27:53 +00:00
* \ param [ in ] key key of the record
* \ param [ in ] value value of the record
2023-08-19 19:25:52 +00:00
*
2023-04-07 23:27:53 +00:00
* \ returns true if the record was added , false otherwise
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
bool LMDBAL : : Storage < K , V > : : forceRecord ( const K & key , const V & value ) {
ensureOpened ( forceRecordMethodName ) ;
TransactionID txn = beginTransaction ( ) ;
bool added ;
try {
2023-04-04 23:27:31 +00:00
added = Storage < K , V > : : forceRecord ( key , value , txn ) ;
2023-03-28 20:45:35 +00:00
} catch ( . . . ) {
abortTransaction ( txn ) ;
throw ;
}
2023-03-20 15:37:13 +00:00
commitTransaction ( txn ) ;
2023-03-28 20:45:35 +00:00
return added ;
2023-03-20 15:37:13 +00:00
}
2023-04-07 23:27:53 +00:00
/**
2023-10-17 21:06:11 +00:00
* \ brief Adds a key - value record to the storage , overwrites if it already exists ( private transaction variant )
2023-04-07 23:27:53 +00:00
*
2023-08-19 19:25:52 +00:00
* This method schedules an addition of a key - value record , but doesn ' t immidiately add it .
2023-04-07 23:27:53 +00:00
*
2023-08-19 19:25:52 +00:00
* This method is mostly useless in duplicates mode .
* In this mode it basically does the same thing LMDBAL : : Storage : : addRecord ( ) does ,
* but suppresses LMDBAL : : Exist exception if the record with the same key - value pair existed in the storage .
* In this case just false is returned from the method .
*
2023-04-07 23:27:53 +00:00
* \ param [ in ] key key of the record
* \ param [ in ] value value of the record
* \ param [ in ] txn transaction ID , needs to be a writable transaction !
* \ returns true if the record was added , false otherwise
*
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-03-28 20:45:35 +00:00
bool LMDBAL : : Storage < K , V > : : forceRecord ( const K & key , const V & value , TransactionID txn ) {
2023-03-20 15:37:13 +00:00
bool added ;
2023-08-19 19:25:52 +00:00
if ( duplicates ) {
try {
addRecord ( key , value , txn ) ;
2023-03-20 15:37:13 +00:00
added = true ;
2023-08-19 19:25:52 +00:00
} catch ( const LMDBAL : : Exist & e ) {
2023-03-20 15:37:13 +00:00
added = false ;
2023-08-19 19:25:52 +00:00
}
} else {
MDB_val lmdbKey = keySerializer . setData ( key ) ;
MDB_val lmdbData ;
2023-11-14 23:15:16 +00:00
int rc = _mdbGet ( txn , lmdbKey , lmdbData ) ;
2023-08-19 19:25:52 +00:00
switch ( rc ) {
case MDB_SUCCESS :
added = false ;
break ;
case MDB_NOTFOUND :
added = true ;
break ;
default :
added = false ;
throwUnknown ( rc ) ;
}
lmdbData = valueSerializer . setData ( value ) ;
2023-11-14 23:15:16 +00:00
rc = _mdbPut ( txn , lmdbKey , lmdbData ) ;
2023-08-19 19:25:52 +00:00
if ( rc ! = MDB_SUCCESS )
2023-03-28 20:45:35 +00:00
throwUnknown ( rc ) ;
2023-03-20 15:37:13 +00:00
}
return added ;
}
2023-10-17 21:06:11 +00:00
/**
* \ brief Adds a key - value record to the storage , overwrites if it already exists ( public transaction variant )
*
* This method schedules an addition of a key - value record , but doesn ' t immidiately add it .
* You can obtain LMDBAL : : WriteTransaction calling LMDBAL : : Base : : beginTransaction ( ) .
* If the record did already exist in the database the actual overwrite will be done only after calling LMDBAL : : WriteTransaction : : commit ( ) .
*
* This method is mostly useless in duplicates mode .
* In this mode it basically does the same thing LMDBAL : : Storage : : addRecord ( ) does ,
* but suppresses LMDBAL : : Exist exception if the record with the same key - value pair existed in the storage .
* In this case just false is returned from the method .
*
* \ param [ in ] key key of the record
* \ param [ in ] value value of the record
* \ param [ in ] txn transaction
* \ returns true if the record was added , false otherwise
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
* \ exception LMDBAL : : TransactionTerminated thrown if the passed transaction not active , any action with it ' s inner ID is an error
*/
template < class K , class V >
bool LMDBAL : : Storage < K , V > : : forceRecord ( const K & key , const V & value , const WriteTransaction & txn ) {
ensureOpened ( forceRecordMethodName ) ;
return forceRecord ( key , value , extractTransactionId ( txn , forceRecordMethodName ) ) ;
}
2023-04-07 23:27:53 +00:00
/**
* \ brief Changes key - value record to the storage .
*
* Take a note that if the storage didn ' t have a record you want to change LMDBAL : : NotFound is thrown
*
2023-08-19 19:25:52 +00:00
* If duplicates mode is enabled this function will find the first entry of the key
* ( which is pretty couterintuitive , see LMDBAL : : Storage : : getRecord ( ) description )
* and change it ' s value to the given one .
* If the given value matches some of the other values for given key the method will throw LMDBAL : : Exist ,
* if no key was found it will still throw LMDBAL : : NotFound .
*
2023-04-07 23:27:53 +00:00
* \ param [ in ] key key of the record
* \ param [ in ] value new value of the record
*
* \ exception LMDBAL : : NotFound thrown if the storage doesn ' t have a record with the given key
2023-08-19 19:25:52 +00:00
* \ exception LMDBAL : : Exist thrown in duplicates mode when the given value matches some of existing values for the given key
2023-04-07 23:27:53 +00:00
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-03-21 11:05:54 +00:00
void LMDBAL : : Storage < K , V > : : changeRecord ( const K & key , const V & value ) {
2023-03-20 15:37:13 +00:00
ensureOpened ( changeRecordMethodName ) ;
TransactionID txn = beginTransaction ( ) ;
2023-03-28 20:45:35 +00:00
try {
2023-04-04 23:27:31 +00:00
Storage < K , V > : : changeRecord ( key , value , txn ) ;
2023-03-28 20:45:35 +00:00
} catch ( . . . ) {
abortTransaction ( txn ) ;
throw ;
}
commitTransaction ( txn ) ;
}
2023-04-07 23:27:53 +00:00
/**
2023-10-17 21:06:11 +00:00
* \ brief Changes key - value record to the storage ( private transaction variant )
2023-04-07 23:27:53 +00:00
*
2023-08-19 19:25:52 +00:00
* This method schedules a modification of a key - value record , but doesn ' t immidiately changes it .
2023-04-07 23:27:53 +00:00
* Take a note that if the storage didn ' t have a record you want to change LMDBAL : : NotFound is thrown
*
2023-08-19 19:25:52 +00:00
* If duplicates mode is enabled this function will find the first entry of the key
* ( which is pretty couterintuitive , see LMDBAL : : Storage : : getRecord ( ) description )
* and change it ' s value to the given one .
* If the given value matches some of the other values for given key the method will throw LMDBAL : : Exist ,
* if no key was found it will still throw LMDBAL : : NotFound .
*
2023-04-07 23:27:53 +00:00
* \ param [ in ] key key of the record
* \ param [ in ] value new value of the record
* \ param [ in ] txn transaction ID , needs to be a writable transaction !
*
* \ exception LMDBAL : : NotFound thrown if the storage doesn ' t have a record with the given key
2023-08-19 19:25:52 +00:00
* \ exception LMDBAL : : Exist thrown in duplicates mode when the given value matches some of existing values for the given key
2023-04-07 23:27:53 +00:00
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
void LMDBAL : : Storage < K , V > : : changeRecord ( const K & key , const V & value , TransactionID txn ) {
2023-04-07 23:27:53 +00:00
MDB_cursor * cursor ;
2023-11-14 23:15:16 +00:00
int rc = _mdbCursorOpen ( txn , & cursor ) ;
2023-04-07 23:27:53 +00:00
if ( rc ! = MDB_SUCCESS )
throwUnknown ( rc ) ;
2023-08-05 20:13:43 +00:00
MDB_val lmdbKey = keySerializer . setData ( key ) ;
2023-08-19 19:25:52 +00:00
MDB_val lmdbData ;
2023-11-14 23:15:16 +00:00
rc = _mdbCursorGet ( cursor , lmdbKey , lmdbData , MDB_SET ) ;
2023-04-07 23:27:53 +00:00
if ( rc ! = MDB_SUCCESS )
throwNotFoundOrUnknown ( rc , toString ( key ) ) ;
2023-03-20 15:37:13 +00:00
2023-08-19 19:25:52 +00:00
MDB_val lmdbNewData = valueSerializer . setData ( value ) ;
bool sameSize = lmdbData . mv_size = = lmdbNewData . mv_size ;
int firstDifferentByte = 0 ;
if ( sameSize ) { //can compare only if they are the same size
firstDifferentByte = memcmp ( lmdbData . mv_data , lmdbNewData . mv_data , lmdbData . mv_size ) ;
if ( firstDifferentByte = = 0 ) { //old and new is the same, nothing to do
2023-11-14 23:15:16 +00:00
_mdbCursorClose ( cursor ) ;
2023-08-19 19:25:52 +00:00
return ;
}
}
unsigned int flags = MDB_CURRENT ;
if ( duplicates & & ( ! sameSize | | firstDifferentByte < 0 ) ) { //if new value is greater than the old one
2023-11-14 23:15:16 +00:00
rc = _mdbCursorDel ( cursor ) ; //we need to initiate duplicates sort, for it to be in the correct place
2023-08-19 19:25:52 +00:00
flags = MDB_NODUPDATA ;
}
if ( rc = = MDB_SUCCESS )
2023-11-14 23:15:16 +00:00
rc = _mdbCursorPut ( cursor , lmdbKey , lmdbNewData , flags ) ;
2023-08-19 19:25:52 +00:00
2023-11-14 23:15:16 +00:00
_mdbCursorClose ( cursor ) ;
2023-03-20 15:37:13 +00:00
if ( rc ! = MDB_SUCCESS )
2023-08-19 19:25:52 +00:00
throwDuplicateOrUnknown ( rc , toString ( key ) ) ;
2023-03-20 15:37:13 +00:00
}
2023-10-17 21:06:11 +00:00
/**
* \ brief Changes key - value record to the storage ( public transaction variant )
*
* This method schedules a modification of a key - value record , but doesn ' t immidiately changes it .
* You can obtain LMDBAL : : WriteTransaction calling LMDBAL : : Base : : beginTransaction ( ) .
*
* Take a note that if the storage didn ' t have a record you want to change LMDBAL : : NotFound is thrown
*
* If duplicates mode is enabled this function will find the first entry of the key
* ( which is pretty couterintuitive , see LMDBAL : : Storage : : getRecord ( ) description )
* and change it ' s value to the given one .
* If the given value matches some of the other values for given key the method will throw LMDBAL : : Exist ,
* if no key was found it will still throw LMDBAL : : NotFound .
*
* \ param [ in ] key key of the record
* \ param [ in ] value new value of the record
* \ param [ in ] txn transaction
*
* \ exception LMDBAL : : NotFound thrown if the storage doesn ' t have a record with the given key
* \ exception LMDBAL : : Exist thrown in duplicates mode when the given value matches some of existing values for the given key
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
* \ exception LMDBAL : : TransactionTerminated thrown if the passed transaction not active , any action with it ' s inner ID is an error
*/
template < class K , class V >
void LMDBAL : : Storage < K , V > : : changeRecord ( const K & key , const V & value , const WriteTransaction & txn ) {
ensureOpened ( changeRecordMethodName ) ;
changeRecord ( key , value , extractTransactionId ( txn , changeRecordMethodName ) ) ;
}
2023-04-07 23:27:53 +00:00
/**
* \ brief Gets the record from the database
2023-08-18 15:04:37 +00:00
* Take a note that if the storage didn ' t have a record you want to get LMDBAL : : NotFound is thrown
2023-04-07 23:27:53 +00:00
*
2023-08-19 19:25:52 +00:00
* If the storage supports duplicates the exact value returned from it depends on comparison function of lmdb .
* It ' s not very straight forward , so , you shouldn ' t really use this method if you use duplicates and you rely on exact result .
* Anyway :
2023-10-15 14:10:27 +00:00
* - if your values are signed or unsigned integer of any size the LOWEST value is returned compared as < b > UNSIGNED < / b > . For example for storage with int32_t as value , from the same key , from the set of values { - 33 , - 1 , 5573 , 77753 } 5573 is returned as it is the lowest by < b > UNSIGNED < / b > comparison .
2023-08-19 19:25:52 +00:00
* - if your values are anything else - they are compared byte by byte as if they are strings , it makes it especially complicated to predict the exact value for float or double templated storages . For strings if makes a bit more sence : if the choise is from " 50 " and " 100 " - " 100 " is returned , because the first byte of the " 100 " is lower than the first byte of the " 50 "
*
2023-04-07 23:27:53 +00:00
* \ param [ in ] key key of the record you look for
* \ returns the value from the storage
*
* \ exception LMDBAL : : NotFound thrown if the storage doesn ' t have a record with the given key
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-03-21 11:05:54 +00:00
V LMDBAL : : Storage < K , V > : : getRecord ( const K & key ) const {
2023-03-20 15:37:13 +00:00
ensureOpened ( getRecordMethodName ) ;
2023-03-28 20:45:35 +00:00
V value ;
2023-04-04 23:27:31 +00:00
Storage < K , V > : : getRecord ( key , value ) ;
2023-03-28 20:45:35 +00:00
return value ;
}
2023-04-07 23:27:53 +00:00
/**
* \ brief Gets the record from the database ( reference variant )
2023-08-18 15:04:37 +00:00
*
* Take a note that if the storage didn ' t have a record you want to get LMDBAL : : NotFound is thrown
*
* If the storage supports duplicates the exact value returned from it depends on comparison function of lmdb .
2023-08-19 19:25:52 +00:00
* It ' s not very straight forward , so , you shouldn ' t really use this method if you use duplicates and you rely on exact result .
2023-08-18 15:04:37 +00:00
* Anyway :
2023-10-15 14:10:27 +00:00
* - if your values are signed or unsigned integer of any size the LOWEST value is returned compared as < b > UNSIGNED < / b > . For example for storage with int32_t as value , from the same key , from the set of values { - 33 , - 1 , 5573 , 77753 } 5573 is returned as it is the lowest by < b > UNSIGNED < / b > comparison .
2023-08-18 15:04:37 +00:00
* - if your values are anything else - they are compared byte by byte as if they are strings , it makes it especially complicated to predict the exact value for float or double templated storages . For strings if makes a bit more sence : if the choise is from " 50 " and " 100 " - " 100 " is returned , because the first byte of the " 100 " is lower than the first byte of the " 50 "
2023-04-07 23:27:53 +00:00
*
* \ param [ in ] key key of the record you look for
* \ param [ out ] value the value from the storage
*
* \ exception LMDBAL : : NotFound thrown if the storage doesn ' t have a record with the given key
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
void LMDBAL : : Storage < K , V > : : getRecord ( const K & key , V & value ) const {
ensureOpened ( getRecordMethodName ) ;
2023-03-20 15:37:13 +00:00
TransactionID txn = beginReadOnlyTransaction ( ) ;
2023-03-28 20:45:35 +00:00
try {
2023-04-04 23:27:31 +00:00
Storage < K , V > : : getRecord ( key , value , txn ) ;
2023-03-28 20:45:35 +00:00
} catch ( . . . ) {
abortTransaction ( txn ) ;
throw ;
}
abortTransaction ( txn ) ;
}
2023-04-07 23:27:53 +00:00
/**
2023-10-17 21:06:11 +00:00
* \ brief Gets the record from the database ( private transaction variant )
2023-04-07 23:27:53 +00:00
*
2023-08-18 15:04:37 +00:00
* Take a note that if the storage didn ' t have a record you want to get LMDBAL : : NotFound is thrown
*
* If the storage supports duplicates the exact value returned from it depends on comparison function of lmdb .
2023-08-19 19:25:52 +00:00
* It ' s not very straight forward , so , you shouldn ' t really use this method if you use duplicates and you rely on exact result .
2023-08-18 15:04:37 +00:00
* Anyway :
2023-10-15 14:10:27 +00:00
* - if your values are signed or unsigned integer of any size the LOWEST value is returned compared as < b > UNSIGNED < / b > . For example for storage with int32_t as value , from the same key , from the set of values { - 33 , - 1 , 5573 , 77753 } 5573 is returned as it is the lowest by < b > UNSIGNED < / b > comparison .
2023-08-18 15:04:37 +00:00
* - if your values are anything else - they are compared byte by byte as if they are strings , it makes it especially complicated to predict the exact value for float or double templated storages . For strings if makes a bit more sence : if the choise is from " 50 " and " 100 " - " 100 " is returned , because the first byte of the " 100 " is lower than the first byte of the " 50 "
2023-04-07 23:27:53 +00:00
*
* \ param [ in ] key key of the record you look for
* \ param [ in ] txn transaction ID , can be read only transaction
* \ returns the value from the storage
*
* \ exception LMDBAL : : NotFound thrown if the storage doesn ' t have a record with the given key
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
V LMDBAL : : Storage < K , V > : : getRecord ( const K & key , TransactionID txn ) const {
V value ;
2023-04-04 23:27:31 +00:00
Storage < K , V > : : getRecord ( key , value , txn ) ;
2023-03-28 20:45:35 +00:00
return value ;
}
2023-04-07 23:27:53 +00:00
/**
2023-10-17 21:06:11 +00:00
* \ brief Gets the record from the database ( public transaction variant )
2023-04-07 23:27:53 +00:00
*
2023-10-17 21:06:11 +00:00
* You can obtain LMDBAL : : Transaction calling LMDBAL : : Base : : beginReadOnlyTransaction ( ) or LMDBAL : : Base : : beginTransaction ( ) .
2023-04-07 23:27:53 +00:00
* If you just want to read data you should prefer LMDBAL : : Base : : beginReadOnlyTransaction ( ) .
*
2023-08-18 15:04:37 +00:00
* Take a note that if the storage didn ' t have a record you want to get LMDBAL : : NotFound is thrown
*
* If the storage supports duplicates the exact value returned from it depends on comparison function of lmdb .
2023-08-19 19:25:52 +00:00
* It ' s not very straight forward , so , you shouldn ' t really use this method if you use duplicates and you rely on exact result .
2023-08-18 15:04:37 +00:00
* Anyway :
2023-10-15 14:10:27 +00:00
* - if your values are signed or unsigned integer of any size the LOWEST value is returned compared as < b > UNSIGNED < / b > . For example for storage with int32_t as value , from the same key , from the set of values { - 33 , - 1 , 5573 , 77753 } 5573 is returned as it is the lowest by < b > UNSIGNED < / b > comparison .
2023-08-18 15:04:37 +00:00
* - if your values are anything else - they are compared byte by byte as if they are strings , it makes it especially complicated to predict the exact value for float or double templated storages . For strings if makes a bit more sence : if the choise is from " 50 " and " 100 " - " 100 " is returned , because the first byte of the " 100 " is lower than the first byte of the " 50 "
2023-04-07 23:27:53 +00:00
*
* \ param [ in ] key key of the record you look for
2023-10-17 21:06:11 +00:00
* \ param [ in ] txn transaction ID , can be read only transaction
* \ returns the value from the storage
*
* \ exception LMDBAL : : NotFound thrown if the storage doesn ' t have a record with the given key
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
* \ exception LMDBAL : : TransactionTerminated thrown if the passed transaction not active , any action with it ' s inner ID is an error
*/
template < class K , class V >
V LMDBAL : : Storage < K , V > : : getRecord ( const K & key , const Transaction & txn ) const {
ensureOpened ( getRecordMethodName ) ;
return getRecord ( key , extractTransactionId ( txn , getRecordMethodName ) ) ;
}
/**
* \ brief Gets the record from the database ( private transaction , reference variant )
*
* Take a note that if the storage didn ' t have a record you want to get LMDBAL : : NotFound is thrown
*
* If the storage supports duplicates the exact value returned from it depends on comparison function of lmdb .
* It ' s not very straight forward , so , you shouldn ' t really use this method if you use duplicates and you rely on exact result .
* Anyway :
* - if your values are signed or unsigned integer of any size the LOWEST value is returned compared as < b > UNSIGNED < / b > . For example for storage with int32_t as value , from the same key , from the set of values { - 33 , - 1 , 5573 , 77753 } 5573 is returned as it is the lowest by < b > UNSIGNED < / b > comparison .
* - if your values are anything else - they are compared byte by byte as if they are strings , it makes it especially complicated to predict the exact value for float or double templated storages . For strings if makes a bit more sence : if the choise is from " 50 " and " 100 " - " 100 " is returned , because the first byte of the " 100 " is lower than the first byte of the " 50 "
*
* \ param [ in ] key key of the record you look for
2023-04-07 23:27:53 +00:00
* \ param [ out ] value the value from the storage
* \ param [ in ] txn transaction ID , can be read only transaction
*
* \ exception LMDBAL : : NotFound thrown if the storage doesn ' t have a record with the given key
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
void LMDBAL : : Storage < K , V > : : getRecord ( const K & key , V & value , TransactionID txn ) const {
2023-08-05 20:13:43 +00:00
MDB_val lmdbKey = keySerializer . setData ( key ) ;
2023-03-20 15:37:13 +00:00
MDB_val lmdbData ;
2023-11-14 23:15:16 +00:00
int rc = _mdbGet ( txn , lmdbKey , lmdbData ) ;
2023-03-20 15:37:13 +00:00
if ( rc ! = MDB_SUCCESS )
2023-03-28 20:45:35 +00:00
throwNotFoundOrUnknown ( rc , toString ( key ) ) ;
2023-03-20 15:37:13 +00:00
2023-08-05 20:13:43 +00:00
valueSerializer . deserialize ( lmdbData , value ) ;
2023-03-20 15:37:13 +00:00
}
2023-10-17 21:06:11 +00:00
/**
* \ brief Gets the record from the database ( public transaction , reference variant )
*
* You can obtain LMDBAL : : Transaction calling LMDBAL : : Base : : beginReadOnlyTransaction ( ) or LMDBAL : : Base : : beginTransaction ( ) .
* If you just want to read data you should prefer LMDBAL : : Base : : beginReadOnlyTransaction ( ) .
*
* Take a note that if the storage didn ' t have a record you want to get LMDBAL : : NotFound is thrown
*
* If the storage supports duplicates the exact value returned from it depends on comparison function of lmdb .
* It ' s not very straight forward , so , you shouldn ' t really use this method if you use duplicates and you rely on exact result .
* Anyway :
* - if your values are signed or unsigned integer of any size the LOWEST value is returned compared as < b > UNSIGNED < / b > . For example for storage with int32_t as value , from the same key , from the set of values { - 33 , - 1 , 5573 , 77753 } 5573 is returned as it is the lowest by < b > UNSIGNED < / b > comparison .
* - if your values are anything else - they are compared byte by byte as if they are strings , it makes it especially complicated to predict the exact value for float or double templated storages . For strings if makes a bit more sence : if the choise is from " 50 " and " 100 " - " 100 " is returned , because the first byte of the " 100 " is lower than the first byte of the " 50 "
*
* \ param [ in ] key key of the record you look for
* \ param [ out ] value the value from the storage
* \ param [ in ] txn transaction ID , can be read only transaction
*
* \ exception LMDBAL : : NotFound thrown if the storage doesn ' t have a record with the given key
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
* \ exception LMDBAL : : TransactionTerminated thrown if the passed transaction not active , any action with it ' s inner ID is an error
*/
template < class K , class V >
void LMDBAL : : Storage < K , V > : : getRecord ( const K & key , V & value , const Transaction & txn ) const {
ensureOpened ( getRecordMethodName ) ;
getRecord ( key , value , extractTransactionId ( txn , getRecordMethodName ) ) ;
}
2023-04-07 23:27:53 +00:00
/**
* \ brief Chechs if storage has value
*
* \ param [ in ] key key of the record you look for
* \ returns true if there was a record with given key , false otherwise
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-03-21 11:05:54 +00:00
bool LMDBAL : : Storage < K , V > : : checkRecord ( const K & key ) const {
2023-03-20 15:37:13 +00:00
ensureOpened ( checkRecordMethodName ) ;
TransactionID txn = beginReadOnlyTransaction ( ) ;
2023-03-28 20:45:35 +00:00
bool result ;
try {
2023-04-04 23:27:31 +00:00
result = Storage < K , V > : : checkRecord ( key , txn ) ;
2023-03-28 20:45:35 +00:00
} catch ( . . . ) {
abortTransaction ( txn ) ;
throw ;
}
abortTransaction ( txn ) ;
return result ;
}
2023-04-07 23:27:53 +00:00
/**
2023-10-17 21:06:11 +00:00
* \ brief Chechs if storage has value ( private transaction variant )
2023-04-07 23:27:53 +00:00
*
* \ param [ in ] key key of the record you look for
* \ param [ in ] txn transaction ID , can be read only transaction
* \ returns true if there was a record with given key , false otherwise
*
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
bool LMDBAL : : Storage < K , V > : : checkRecord ( const K & key , TransactionID txn ) const {
2023-08-05 20:13:43 +00:00
MDB_val lmdbKey = keySerializer . setData ( key ) ;
2023-03-20 15:37:13 +00:00
MDB_val lmdbData ;
2023-11-14 23:15:16 +00:00
int rc = _mdbGet ( txn , lmdbKey , lmdbData ) ;
2023-03-20 15:37:13 +00:00
if ( rc = = MDB_SUCCESS )
return true ;
if ( rc ! = MDB_NOTFOUND )
throwUnknown ( rc ) ;
return false ;
}
2023-10-17 21:06:11 +00:00
/**
* \ brief Chechs if storage has value ( public transaction variant )
*
* You can obtain LMDBAL : : Transaction calling LMDBAL : : Base : : beginReadOnlyTransaction ( ) or LMDBAL : : Base : : beginTransaction ( ) .
* If you just want to read data you should prefer LMDBAL : : Base : : beginReadOnlyTransaction ( ) .
*
* \ param [ in ] key key of the record you look for
* \ param [ in ] txn transaction , can be read only transaction
* \ returns true if there was a record with given key , false otherwise
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
* \ exception LMDBAL : : TransactionTerminated thrown if the passed transaction not active , any action with it ' s inner ID is an error
*/
template < class K , class V >
bool LMDBAL : : Storage < K , V > : : checkRecord ( const K & key , const Transaction & txn ) const {
ensureOpened ( checkRecordMethodName ) ;
return checkRecord ( key , extractTransactionId ( txn , checkRecordMethodName ) ) ;
}
2023-04-09 17:19:23 +00:00
/**
* \ brief Reads whole storage into a map
*
* Basically just reads all database in an std : : map , usefull when you store small storages
*
2023-08-20 16:38:29 +00:00
* In case storage supports duplicates < b > only what lmdb considered to be lowest value < / b >
* ( see LMDBAL : : Storage : : getRecord ( ) description ) is returned in the resulting map
*
2023-04-09 17:19:23 +00:00
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-03-21 11:05:54 +00:00
std : : map < K , V > LMDBAL : : Storage < K , V > : : readAll ( ) const {
2023-03-20 15:37:13 +00:00
ensureOpened ( readAllMethodName ) ;
2023-03-28 20:45:35 +00:00
std : : map < K , V > result ;
2023-04-04 23:27:31 +00:00
Storage < K , V > : : readAll ( result ) ;
2023-03-28 20:45:35 +00:00
return result ;
}
2023-04-09 17:19:23 +00:00
/**
* \ brief Reads whole storage into a map ( reference variant )
*
* Basically just reads all database in an std : : map , usefull when you store small storages
*
2023-08-20 16:38:29 +00:00
* In case storage supports duplicates < b > only what lmdb considered to be lowest value < / b >
* ( see LMDBAL : : Storage : : getRecord ( ) description ) is returned in the resulting map
*
2023-04-09 17:19:23 +00:00
* \ param [ out ] result a map that is going to contain all data
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
void LMDBAL : : Storage < K , V > : : readAll ( std : : map < K , V > & result ) const {
ensureOpened ( readAllMethodName ) ;
2023-03-20 15:37:13 +00:00
TransactionID txn = beginReadOnlyTransaction ( ) ;
2023-03-28 20:45:35 +00:00
try {
2023-04-04 23:27:31 +00:00
Storage < K , V > : : readAll ( result , txn ) ;
2023-03-28 20:45:35 +00:00
} catch ( . . . ) {
abortTransaction ( txn ) ;
throw ;
}
abortTransaction ( txn ) ;
}
2023-04-09 17:19:23 +00:00
/**
2023-10-17 21:06:11 +00:00
* \ brief Reads whole storage into a map ( private transaction variant )
2023-04-09 17:19:23 +00:00
*
* Basically just reads all database in an std : : map , usefull when you store small storages
2023-08-20 16:38:29 +00:00
* In case storage supports duplicates < b > only what lmdb considered to be lowest value < / b >
* ( see LMDBAL : : Storage : : getRecord ( ) description ) is returned in the resulting map
*
2023-04-09 17:19:23 +00:00
* \ param [ in ] txn transaction ID , can be read only transaction
*
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
std : : map < K , V > LMDBAL : : Storage < K , V > : : readAll ( TransactionID txn ) const {
2023-03-20 15:37:13 +00:00
std : : map < K , V > result ;
2023-04-04 23:27:31 +00:00
Storage < K , V > : : readAll ( result , txn ) ;
2023-03-28 20:45:35 +00:00
return result ;
}
2023-04-09 17:19:23 +00:00
/**
2023-10-17 21:06:11 +00:00
* \ brief Reads whole storage into a map ( public transaction variant )
2023-04-09 17:19:23 +00:00
*
* Basically just reads all database in an std : : map , usefull when you store small storages
* You can obtain LMDBAL : : TransactionID calling LMDBAL : : Base : : beginReadOnlyTransaction ( ) or LMDBAL : : Base : : beginTransaction ( ) .
* If you just want to read data you should prefer LMDBAL : : Base : : beginReadOnlyTransaction ( ) .
*
2023-08-20 16:38:29 +00:00
* In case storage supports duplicates < b > only what lmdb considered to be lowest value < / b >
* ( see LMDBAL : : Storage : : getRecord ( ) description ) is returned in the resulting map
*
2023-10-17 21:06:11 +00:00
* \ param [ in ] txn transaction , can be read only transaction
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
* \ exception LMDBAL : : TransactionTerminated thrown if the passed transaction not active , any action with it ' s inner ID is an error
*/
template < class K , class V >
std : : map < K , V > LMDBAL : : Storage < K , V > : : readAll ( const Transaction & txn ) const {
ensureOpened ( readAllMethodName ) ;
return readAll ( extractTransactionId ( txn , readAllMethodName ) ) ;
}
/**
* \ brief Reads whole storage into a map ( private transaction , reference variant )
*
* Basically just reads all database in an std : : map , usefull when you store small storages
* In case storage supports duplicates < b > only what lmdb considered to be lowest value < / b >
* ( see LMDBAL : : Storage : : getRecord ( ) description ) is returned in the resulting map
*
2023-04-09 17:19:23 +00:00
* \ param [ out ] result a map that is going to contain all data
* \ param [ in ] txn transaction ID , can be read only transaction
*
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
void LMDBAL : : Storage < K , V > : : readAll ( std : : map < K , V > & result , TransactionID txn ) const {
2023-03-20 15:37:13 +00:00
MDB_cursor * cursor ;
MDB_val lmdbKey , lmdbData ;
2023-11-14 23:15:16 +00:00
int rc = _mdbCursorOpen ( txn , & cursor ) ;
2023-03-20 15:37:13 +00:00
if ( rc ! = MDB_SUCCESS )
2023-03-28 20:45:35 +00:00
throwUnknown ( rc ) ;
2023-03-20 15:37:13 +00:00
2023-11-14 23:15:16 +00:00
rc = _mdbCursorGet ( cursor , lmdbKey , lmdbData , MDB_FIRST ) ;
2023-03-20 15:37:13 +00:00
while ( rc = = MDB_SUCCESS ) {
2023-03-28 20:45:35 +00:00
K key ;
2023-08-05 20:13:43 +00:00
keySerializer . deserialize ( lmdbKey , key ) ;
2023-08-20 16:38:29 +00:00
std : : pair < typename std : : map < K , V > : : iterator , bool > probe = result . emplace ( key , V { } ) ;
if ( probe . second ) //I do this to avoid overwrites in case duplicates are enabled
valueSerializer . deserialize ( lmdbData , probe . first - > second ) ;
2023-11-14 23:15:16 +00:00
rc = _mdbCursorGet ( cursor , lmdbKey , lmdbData , MDB_NEXT ) ;
2023-03-20 15:37:13 +00:00
}
2023-11-14 23:15:16 +00:00
_mdbCursorClose ( cursor ) ;
2023-03-20 15:37:13 +00:00
if ( rc ! = MDB_NOTFOUND )
throwUnknown ( rc ) ;
}
2023-10-17 21:06:11 +00:00
/**
* \ brief Reads whole storage into a map ( public transaction , reference variant )
*
* Basically just reads all database in an std : : map , usefull when you store small storages
* You can obtain LMDBAL : : Transaction calling LMDBAL : : Base : : beginReadOnlyTransaction ( ) or LMDBAL : : Base : : beginTransaction ( ) .
* If you just want to read data you should prefer LMDBAL : : Base : : beginReadOnlyTransaction ( ) .
*
* In case storage supports duplicates < b > only what lmdb considered to be lowest value < / b >
* ( see LMDBAL : : Storage : : getRecord ( ) description ) is returned in the resulting map
*
* \ param [ out ] result a map that is going to contain all data
* \ param [ in ] txn transaction , can be read only transaction
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
* \ exception LMDBAL : : TransactionTerminated thrown if the passed transaction not active , any action with it ' s inner ID is an error
*/
template < class K , class V >
void LMDBAL : : Storage < K , V > : : readAll ( std : : map < K , V > & result , const Transaction & txn ) const {
ensureOpened ( readAllMethodName ) ;
readAll ( result , extractTransactionId ( txn , readAllMethodName ) ) ;
}
2023-04-09 17:19:23 +00:00
/**
* \ brief Replaces the content of the whole storage with the given
*
2023-08-19 19:25:52 +00:00
* Basically this method drops the database and adds all the records from the given map
2023-04-09 17:19:23 +00:00
*
* \ param [ in ] data new data of the storage
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-03-21 11:05:54 +00:00
void LMDBAL : : Storage < K , V > : : replaceAll ( const std : : map < K , V > & data ) {
2023-03-20 15:37:13 +00:00
ensureOpened ( replaceAllMethodName ) ;
TransactionID txn = beginTransaction ( ) ;
2023-03-28 20:45:35 +00:00
try {
2023-04-04 23:27:31 +00:00
Storage < K , V > : : replaceAll ( data , txn ) ;
2023-03-28 20:45:35 +00:00
} catch ( . . . ) {
abortTransaction ( txn ) ;
throw ;
}
commitTransaction ( txn ) ;
}
2023-04-09 17:19:23 +00:00
/**
2023-10-17 21:06:11 +00:00
* \ brief Replaces the content of the whole storage with the given ( private transaction variant )
2023-04-09 17:19:23 +00:00
*
2023-08-19 19:25:52 +00:00
* Basically this method drops the database and adds all the records from the given map
2023-04-09 17:19:23 +00:00
*
* \ param [ in ] data new data of the storage
* \ param [ in ] txn transaction ID , needs to be a writable transaction !
*
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
void LMDBAL : : Storage < K , V > : : replaceAll ( const std : : map < K , V > & data , TransactionID txn ) {
2023-03-20 15:37:13 +00:00
int rc = drop ( txn ) ;
if ( rc ! = MDB_SUCCESS )
2023-03-28 20:45:35 +00:00
throwUnknown ( rc ) ;
2023-03-20 15:37:13 +00:00
MDB_val lmdbKey , lmdbData ;
for ( const std : : pair < const K , V > & pair : data ) {
2023-08-05 20:13:43 +00:00
lmdbKey = keySerializer . setData ( pair . first ) ;
lmdbData = valueSerializer . setData ( pair . second ) ;
2023-03-20 15:37:13 +00:00
2023-11-14 23:15:16 +00:00
rc = _mdbPut ( txn , lmdbKey , lmdbData , MDB_NOOVERWRITE ) ; //TODO may be appending with cursor makes sence here?
2023-03-20 15:37:13 +00:00
if ( rc ! = MDB_SUCCESS )
2023-03-28 20:45:35 +00:00
throwUnknown ( rc ) ;
2023-03-20 15:37:13 +00:00
}
}
2023-10-17 21:06:11 +00:00
/**
* \ brief Replaces the content of the whole storage with the given ( public transaction variant )
*
* Basically this method drops the database and adds all the records from the given map
* This method schedules a data replacement , but doesn ' t immidiately execute it .
* You can obtain LMDBAL : : Transaction calling LMDBAL : : Base : : beginTransaction ( ) .
*
* \ param [ in ] data new data of the storage
* \ param [ in ] txn transaction
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
* \ exception LMDBAL : : TransactionTerminated thrown if the passed transaction not active , any action with it ' s inner ID is an error
*/
template < class K , class V >
void LMDBAL : : Storage < K , V > : : replaceAll ( const std : : map < K , V > & data , const WriteTransaction & txn ) {
ensureOpened ( replaceAllMethodName ) ;
replaceAll ( data , extractTransactionId ( txn , replaceAllMethodName ) ) ;
}
2023-04-09 17:19:23 +00:00
/**
* \ brief Adds records in bulk
*
* \ param [ in ] data the data to be added
2023-08-19 19:25:52 +00:00
* \ param [ in ] overwrite if false method throws LMDBAL : : Exist on repeated key , if true - overwrites it
2023-04-09 17:19:23 +00:00
* \ returns new actual amount of records in the storage
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Exist thrown if overwrite = = false and at least one of the keys of data already exists in the storage
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-03-21 11:05:54 +00:00
uint32_t LMDBAL : : Storage < K , V > : : addRecords ( const std : : map < K , V > & data , bool overwrite ) {
2023-03-20 15:37:13 +00:00
ensureOpened ( addRecordsMethodName ) ;
TransactionID txn = beginTransaction ( ) ;
2023-03-28 20:45:35 +00:00
uint32_t amount ;
try {
2023-04-04 23:27:31 +00:00
amount = Storage < K , V > : : addRecords ( data , txn , overwrite ) ;
2023-03-28 20:45:35 +00:00
} catch ( . . . ) {
abortTransaction ( txn ) ;
throw ;
}
commitTransaction ( txn ) ;
return amount ;
}
2023-04-09 17:19:23 +00:00
/**
2023-10-17 21:06:11 +00:00
* \ brief Adds records in bulk ( private transaction variant )
2023-04-09 17:19:23 +00:00
*
2023-08-19 19:25:52 +00:00
* This method schedules a data addition , but doesn ' t immidiately execute it .
2023-04-09 17:19:23 +00:00
*
* \ param [ in ] data the data to be added
* \ param [ in ] txn transaction ID , needs to be a writable transaction !
2023-08-19 19:25:52 +00:00
* \ param [ in ] overwrite if false method throws LMDBAL : : Exist on repeated key , if true - overwrites it
2023-04-09 17:19:23 +00:00
* \ returns new actual amount of records in the storage
*
* \ exception LMDBAL : : Exist thrown if overwrite = = false and at least one of the keys of data already exists in the storage
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
uint32_t LMDBAL : : Storage < K , V > : : addRecords ( const std : : map < K , V > & data , TransactionID txn , bool overwrite ) {
2023-03-20 15:37:13 +00:00
MDB_val lmdbKey , lmdbData ;
int rc ;
for ( const std : : pair < const K , V > & pair : data ) {
2023-08-05 20:13:43 +00:00
lmdbKey = keySerializer . setData ( pair . first ) ;
lmdbData = valueSerializer . setData ( pair . second ) ;
2023-03-20 15:37:13 +00:00
2023-11-14 23:15:16 +00:00
rc = _mdbPut ( txn , lmdbKey , lmdbData , overwrite ? 0 : MDB_NOOVERWRITE ) ;
2023-04-05 23:01:24 +00:00
if ( rc = = MDB_KEYEXIST )
throwDuplicate ( toString ( pair . first ) ) ;
2023-03-20 15:37:13 +00:00
if ( rc ! = MDB_SUCCESS )
2023-03-28 20:45:35 +00:00
throwUnknown ( rc ) ;
2023-03-20 15:37:13 +00:00
}
MDB_stat stat ;
2023-11-14 23:15:16 +00:00
rc = _mdbStat ( txn , stat ) ;
2023-03-20 15:37:13 +00:00
if ( rc ! = MDB_SUCCESS )
2023-03-28 20:45:35 +00:00
throwUnknown ( rc ) ;
2023-03-20 15:37:13 +00:00
2023-03-28 20:45:35 +00:00
return stat . ms_entries ;
2023-03-20 15:37:13 +00:00
}
2023-10-17 21:06:11 +00:00
/**
* \ brief Adds records in bulk ( public transaction variant )
*
* This method schedules a data addition , but doesn ' t immidiately execute it .
* You can obtain LMDBAL : : Transaction calling LMDBAL : : Base : : beginTransaction ( ) .
*
* \ param [ in ] data the data to be added
* \ param [ in ] txn transaction
* \ param [ in ] overwrite if false method throws LMDBAL : : Exist on repeated key , if true - overwrites it
* \ returns new actual amount of records in the storage
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Exist thrown if overwrite = = false and at least one of the keys of data already exists in the storage
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
* \ exception LMDBAL : : TransactionTerminated thrown if the passed transaction not active , any action with it ' s inner ID is an error
*/
template < class K , class V >
uint32_t LMDBAL : : Storage < K , V > : : addRecords ( const std : : map < K , V > & data , const WriteTransaction & txn , bool overwrite ) {
ensureOpened ( addRecordsMethodName ) ;
return addRecords ( data , extractTransactionId ( txn , addRecordsMethodName ) , overwrite ) ;
}
2023-04-09 17:19:23 +00:00
/**
* \ brief Removes one of the records
*
* Take a note that if the storage didn ' t have a record you want to remove LMDBAL : : NotFound is thrown
*
* \ param [ in ] key key of the record you wish to be removed
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : NotFound thrown if the record with given key wasn ' t found
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-03-21 11:05:54 +00:00
void LMDBAL : : Storage < K , V > : : removeRecord ( const K & key ) {
2023-03-20 15:37:13 +00:00
ensureOpened ( removeRecordMethodName ) ;
TransactionID txn = beginTransaction ( ) ;
2023-03-28 20:45:35 +00:00
try {
2023-04-04 23:27:31 +00:00
Storage < K , V > : : removeRecord ( key , txn ) ;
2023-03-28 20:45:35 +00:00
} catch ( . . . ) {
abortTransaction ( txn ) ;
throw ;
}
commitTransaction ( txn ) ;
}
2023-04-09 17:19:23 +00:00
/**
2023-10-17 21:06:11 +00:00
* \ brief Removes one of the records ( private transaction variant )
2023-04-09 17:19:23 +00:00
*
* Take a note that if the storage didn ' t have a record you want to remove LMDBAL : : NotFound is thrown
2023-08-19 19:25:52 +00:00
* This method schedules a record removal , but doesn ' t immidiately execute it .
2023-04-09 17:19:23 +00:00
*
2023-08-10 23:07:12 +00:00
* \ param [ in ] key key of the record you wish to be removed
2023-04-09 17:19:23 +00:00
* \ param [ in ] txn transaction ID , needs to be a writable transaction !
*
* \ exception LMDBAL : : NotFound thrown if the record with given key wasn ' t found
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
*/
2023-03-28 20:45:35 +00:00
template < class K , class V >
void LMDBAL : : Storage < K , V > : : removeRecord ( const K & key , TransactionID txn ) {
2023-08-05 20:13:43 +00:00
MDB_val lmdbKey = keySerializer . setData ( key ) ;
2023-11-14 23:15:16 +00:00
int rc = _mdbDel ( txn , lmdbKey ) ;
2023-03-20 15:37:13 +00:00
if ( rc ! = MDB_SUCCESS )
2023-03-28 20:45:35 +00:00
throwNotFoundOrUnknown ( rc , toString ( key ) ) ;
2023-03-20 15:37:13 +00:00
}
2023-10-17 21:06:11 +00:00
/**
* \ brief Removes one of the records ( transaction variant )
*
* Take a note that if the storage didn ' t have a record you want to remove LMDBAL : : NotFound is thrown
* This method schedules a record removal , but doesn ' t immidiately execute it .
* You can obtain LMDBAL : : Transaction calling LMDBAL : : Base : : beginTransaction ( ) .
*
* \ param [ in ] key key of the record you wish to be removed
* \ param [ in ] txn transaction ID
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : NotFound thrown if the record with given key wasn ' t found
* \ exception LMDBAL : : Unknown thrown if something unexpected happend within lmdb
* \ exception LMDBAL : : TransactionTerminated thrown if the passed transaction not active , any action with it ' s inner ID is an error
*/
template < class K , class V >
void LMDBAL : : Storage < K , V > : : removeRecord ( const K & key , const WriteTransaction & txn ) {
ensureOpened ( removeRecordMethodName ) ;
removeRecord ( key , extractTransactionId ( txn , removeRecordMethodName ) ) ;
}
2023-04-09 17:19:23 +00:00
/**
2023-08-19 19:25:52 +00:00
* \ brief A private virtual method I need to open each storage in the database
2023-04-09 17:19:23 +00:00
*
2023-04-12 15:36:33 +00:00
* \ param [ in ] transaction - lmdb transaction to call < a class = " el " href= " http : //www.lmdb.tech/doc/group__mdb.html#gac08cad5b096925642ca359a6d6f0562a">mdb_dbi_open</a>
* \ returns MDB_SUCCESS if everything went smooth or MDB_ < error > - like error code
2023-04-09 17:19:23 +00:00
*/
2023-03-20 15:37:13 +00:00
template < class K , class V >
2023-08-05 20:13:43 +00:00
int LMDBAL : : Storage < K , V > : : open ( MDB_txn * transaction ) {
2023-08-15 18:48:19 +00:00
return makeStorage < K , V > ( transaction , duplicates ) ;
2023-03-20 15:37:13 +00:00
}
2023-08-05 20:13:43 +00:00
/**
2023-08-19 19:25:52 +00:00
* \ brief A private virtual method I need to close each storage in the database
2023-08-05 20:13:43 +00:00
*/
template < class K , class V >
void LMDBAL : : Storage < K , V > : : close ( ) {
2023-10-25 22:18:23 +00:00
for ( const std : : pair < const uint32_t , Cursor < K , V > * > & pair : cursors )
pair . second - > terminated ( ) ;
2023-08-05 20:13:43 +00:00
iStorage : : close ( ) ;
}
2023-08-10 23:07:12 +00:00
/**
* \ brief Creates cursor
*
2023-11-01 22:45:42 +00:00
* This is a legitimate way to aquire cursor to a storage .
* Aquired cursor is RAII safe , automatic destructor will free everything it occupied .
*
2023-08-10 23:07:12 +00:00
* \ returns LMDBAL : : Cursor for this storage and returs you a pointer to a created cursor
*/
2023-08-05 20:13:43 +00:00
template < class K , class V >
2023-10-25 22:18:23 +00:00
LMDBAL : : Cursor < K , V > LMDBAL : : Storage < K , V > : : createCursor ( ) {
return Cursor < K , V > ( this ) ;
2023-08-05 20:13:43 +00:00
}
2023-11-01 22:45:42 +00:00
/**
* \ brief Frees cursor
*
* This is a legitimate way to free cursor .
* You don ' t actually need to do it manually ,
* you can just reassign cursor , let it be destroyed by leaving the scope
* or call LMDBAL : : Cursor < K , V > : : drop , but you may if you wish .
*
* \ param [ in ] cursor cursor you wish to destroy
*
* \ exception LMDBAL : : Unknown thrown if you try to destroy a cursor this storage didn ' t create
*/
template < class K , class V >
void LMDBAL : : Storage < K , V > : : destroyCursor ( LMDBAL : : Cursor < K , V > & cursor ) {
typename std : : map < uint32_t , Cursor < K , V > * > : : iterator itr = cursors . find ( cursor . id ) ;
if ( itr = = cursors . end ( ) )
throwUnknown ( " An attempt to destroy a cursor the storage doesn't own " ) ;
cursor . close ( ) ;
cursors . erase ( itr ) ;
cursor . freed ( ) ;
}
2023-08-15 18:48:19 +00:00
/**
* \ brief Reads current storage flags it was opened with
*
2023-08-19 19:25:52 +00:00
* This method exists mostly for testing purposes
2023-08-15 18:48:19 +00:00
*
* \ returns Third out parameter of < a href = " http://www.lmdb.tech/doc/group__internal.html#ga95ba4cb721035478a8705e57b91ae4d4 " > mdb_dbi_flags < / a > function
*
* \ exception LMDBAL : : Closed thrown if the database was not opened
* \ exception LMDBAL : : Unknown if the result of < a href = " http://www.lmdb.tech/doc/group__internal.html#ga95ba4cb721035478a8705e57b91ae4d4 " > mdb_dbi_flags < / a > was not successfull
*/
template < class K , class V >
uint32_t LMDBAL : : Storage < K , V > : : flags ( ) const {
ensureOpened ( flagsMethodName ) ;
uint32_t result ;
TransactionID txn = beginReadOnlyTransaction ( ) ;
2023-11-14 23:15:16 +00:00
int res = _mdbFlags ( txn , result ) ;
2023-08-15 18:48:19 +00:00
abortTransaction ( txn ) ;
if ( res ! = MDB_SUCCESS )
throwUnknown ( res ) ;
return result ;
}
2023-08-10 23:07:12 +00:00
/**
2023-08-19 19:25:52 +00:00
* \ brief A private virtual method that cursor calls when he reads a record , does nothing here but populates the LMDBAL : : Cache
2023-08-10 23:07:12 +00:00
*
* \ param [ in ] key a key of discovered record
* \ param [ in ] value a value of discovered record
*/
2023-08-09 17:41:15 +00:00
template < class K , class V >
void LMDBAL : : Storage < K , V > : : discoveredRecord ( const K & key , const V & value ) const {
UNUSED ( key ) ;
UNUSED ( value ) ;
}
2023-08-10 23:07:12 +00:00
/**
2023-08-19 19:25:52 +00:00
* \ brief A private virtual method that cursor calls when he reads a record , does nothing here but populates the LMDBAL : : Cache
2023-08-10 23:07:12 +00:00
*
* \ param [ in ] key a key of discovered record
* \ param [ in ] value a value of discovered record
* \ param [ in ] txn TransactionID under which the dicovery happened , to avoid not commited changes collisions
*/
2023-08-09 17:41:15 +00:00
template < class K , class V >
void LMDBAL : : Storage < K , V > : : discoveredRecord ( const K & key , const V & value , TransactionID txn ) const {
UNUSED ( key ) ;
UNUSED ( value ) ;
UNUSED ( txn ) ;
}
2023-04-12 15:36:33 +00:00
/**
* \ brief A functiion to actually open < a class = " el " href= " http : //www.lmdb.tech/doc/group__mdb.html#gadbe68a06c448dfb62da16443d251a78b">MDB_dbi</a> storage
*
* \ tparam K type of keys in opening storage
*
* \ param [ in ] transaction - lmdb transaction to call < a class = " el " href= " http : //www.lmdb.tech/doc/group__mdb.html#gac08cad5b096925642ca359a6d6f0562a">mdb_dbi_open</a>, must be a writable transaction!
2023-08-18 13:31:30 +00:00
* \ param [ in ] duplicates - true if key duplicates are allowed ( false by default )
2023-08-15 18:48:19 +00:00
*
2023-04-12 15:36:33 +00:00
* \ returns MDB_SUCCESS if everything went smooth or MDB_ < error > - like error code
*
2023-08-15 18:48:19 +00:00
* This is a way to optimise database using MDB_INTEGERKEY flag ,
* when the key is actually kind of an integer
2023-04-12 15:36:33 +00:00
* This infrastructure also allowes us to customize mdb_dbi_open call in the future
*/
2023-08-15 18:48:19 +00:00
template < class K , class V >
2023-08-18 13:31:30 +00:00
inline int LMDBAL : : iStorage : : makeStorage ( MDB_txn * transaction , bool duplicates ) {
2023-08-15 18:48:19 +00:00
unsigned int flags = MDB_CREATE ;
if constexpr ( std : : is_integral < K > : : value )
flags | = MDB_INTEGERKEY ;
2023-03-20 15:37:13 +00:00
2023-08-18 13:31:30 +00:00
if ( duplicates ) {
2023-08-15 18:48:19 +00:00
flags | = MDB_DUPSORT ;
2023-03-20 15:37:13 +00:00
2023-08-18 15:04:37 +00:00
if constexpr ( std : : is_scalar < V > : : value )
2023-08-17 14:45:11 +00:00
flags | = MDB_DUPFIXED ;
2023-08-18 15:04:37 +00:00
if constexpr (
std : : is_same < V , uint32_t > : : value | |
std : : is_same < V , int32_t > : : value | |
std : : is_same < V , uint64_t > : : value | |
std : : is_same < V , int64_t > : : value
) //for some reason lmdb breaks if it's not one of these types in MDB_DUPFIXED mode
flags | = MDB_INTEGERDUP ;
2023-08-15 18:48:19 +00:00
}
2023-03-20 15:37:13 +00:00
2023-11-14 23:15:16 +00:00
return _mdbOpen ( transaction , flags ) ;
2023-03-20 15:37:13 +00:00
}
2023-04-09 17:19:23 +00:00
/**
* \ brief A method to cast a value ( which can be a value or a key ) to string .
*
* This function is mainly used in exceptions , to report which key was duplicated or not found .
* You can define your own specializations to this function in case std : : to_string doesn ' t cover your case
*
* \ param [ in ] value a value that should be converted to string
* \ returns a string presentation of value
*/
2023-03-20 15:37:13 +00:00
template < class T >
2023-03-21 11:05:54 +00:00
inline std : : string LMDBAL : : iStorage : : toString ( const T & value ) {
2023-03-20 15:37:13 +00:00
return std : : to_string ( value ) ;
}
2023-04-12 15:36:33 +00:00
/**
* \ brief A method to cast a value ( which can be a value or a key ) to string .
*
* QString spectialization
*
* \ param [ in ] value a value that should be converted to string
* \ returns a string presentation of value
*/
2023-03-20 15:37:13 +00:00
template < >
2023-03-21 11:05:54 +00:00
inline std : : string LMDBAL : : iStorage : : toString ( const QString & value ) {
2023-03-20 15:37:13 +00:00
return value . toStdString ( ) ;
}
2023-04-12 15:36:33 +00:00
/**
* \ brief A method to cast a value ( which can be a value or a key ) to string .
*
* std : : string spectialization
*
* \ param [ in ] value a value that should be converted to string
* \ returns a string presentation of value
*/
2023-03-20 15:37:13 +00:00
template < >
2023-03-21 11:05:54 +00:00
inline std : : string LMDBAL : : iStorage : : toString ( const std : : string & value ) {
2023-03-20 15:37:13 +00:00
return value ;
}
2023-03-21 11:05:54 +00:00
# endif //LMDBAL_STORAGE_HPP