base_lmdb_store.pyx 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713
  1. import logging
  2. import os
  3. import threading
  4. import multiprocessing
  5. from contextlib import contextmanager
  6. from os import makedirs, path
  7. from shutil import rmtree
  8. from lakesuperior import env, wsgi
  9. from lakesuperior.cy_include cimport cylmdb as lmdb
  10. from libc cimport errno
  11. from cpython.mem cimport PyMem_Malloc, PyMem_Realloc, PyMem_Free
  12. from cython.parallel import parallel, prange
  13. logger = logging.getLogger(__name__)
  14. cdef void _check(int rc, str message='') except *:
  15. """
  16. Check return code.
  17. """
  18. if rc == lmdb.MDB_NOTFOUND:
  19. raise KeyNotFoundError()
  20. if rc == lmdb.MDB_KEYEXIST:
  21. raise KeyExistsError()
  22. if rc != lmdb.MDB_SUCCESS:
  23. out_msg = (
  24. message + '\nInternal error ({}): '.format(rc)
  25. if len(message) else 'LMDB Error ({}): '.format(rc))
  26. out_msg += lmdb.mdb_strerror(rc).decode()
  27. raise LmdbError(out_msg)
  28. class LmdbError(Exception):
  29. pass
  30. class KeyNotFoundError(LmdbError):
  31. pass
  32. class KeyExistsError(LmdbError):
  33. pass
  34. cdef class BaseLmdbStore:
  35. """
  36. Generic LMDB store abstract class.
  37. This class contains convenience method to create an LMDB store for any
  38. purpose and provides some convenience methods to wrap cursors and
  39. transactions into contexts.
  40. Example usage::
  41. >>> class MyStore(BaseLmdbStore):
  42. ... path = '/base/store/path'
  43. ... dbi_flags = ('db1', 'db2')
  44. ...
  45. >>> ms = MyStore()
  46. >>> # "with" wraps the operation in a transaction.
  47. >>> with ms.cur(index='db1', write=True):
  48. ... cur.put(b'key1', b'val1')
  49. True
  50. """
  51. dbi_labels = []
  52. dbi_flags = {}
  53. """
  54. Configuration of databases in the environment.
  55. This is an OderedDict whose keys are the database labels and whose values
  56. are LMDB flags for creating and opening the databases as per
  57. `http://www.lmdb.tech/doc/group__mdb.html#gac08cad5b096925642ca359a6d6f0562a`_
  58. .
  59. If the environment has only one database, do not override this value (i.e.
  60. leave it to ``None``).
  61. :rtype: dict or None
  62. """
  63. env_flags = 0
  64. """
  65. LMDB environment flags.
  66. See `mdb_env_open
  67. <http://www.lmdb.tech/doc/group__mdb.html#ga32a193c6bf4d7d5c5d579e71f22e9340>`_
  68. """
  69. env_perms = 0o640
  70. """
  71. The UNIX permissions to set on created environment files and semaphores.
  72. See `mdb_env_open
  73. <http://www.lmdb.tech/doc/group__mdb.html#ga32a193c6bf4d7d5c5d579e71f22e9340>`_
  74. """
  75. options = {}
  76. """
  77. LMDB environment option overrides. Setting this is not required.
  78. See `LMDB documentation
  79. <http://lmdb.readthedocs.io/en/release/#environment-class`>_ for details
  80. on available options.
  81. Default values are available for the following options:
  82. - ``map_size``: 1 Gib
  83. - ``max_dbs``: dependent on the number of DBs defined in
  84. :py:meth:``dbi_flags``. Only override if necessary.
  85. - ``max_spare_txns``: dependent on the number of threads, if accessed via
  86. WSGI, or ``1`` otherwise. Only override if necessary.
  87. :rtype: dict
  88. """
  89. readers_mult = 4
  90. """
  91. Number to multiply WSGI workers by to set the numer of LMDB reader slots.
  92. """
  93. ### INIT & TEARDOWN ###
  94. def __init__(self, env_path, open_env=True, create=True):
  95. """
  96. Initialize DB environment and databases.
  97. :param str env_path: The file path of the store.
  98. :param bool open: Whether to open the store immediately. If ``False``
  99. the store can be manually opened with :py:meth:`opn_env`.
  100. :param bool create: Whether the file and directory structure should
  101. be created if the store is opened immediately.
  102. """
  103. self._open = False
  104. self.is_txn_open = False
  105. self.env_path = env_path
  106. if open_env:
  107. self.open_env(create)
  108. #logger.info('Init DB with path: {}'.format(self.env_path))
  109. def __dealloc__(self):
  110. self.close_env()
  111. @property
  112. def is_open(self):
  113. return self._open
  114. @property
  115. def readers(self):
  116. return self._readers
  117. def open_env(self, create):
  118. """
  119. Open, and optionally create, store environment.
  120. """
  121. if self.is_open:
  122. logger.warning('Environment already open.')
  123. return
  124. logger.debug('Opening environment at {}.'.format(self.env_path))
  125. if create:
  126. #logger.info('Creating db env at {}'.format(self.env_path))
  127. parent_path = (
  128. path.dirname(self.env_path)
  129. if lmdb.MDB_NOSUBDIR & self.flags
  130. else self.env_path)
  131. if not path.exists(parent_path):
  132. #logger.info(
  133. # 'Creating store directory at {}'.format(parent_path))
  134. try:
  135. makedirs(parent_path, mode=0o750, exist_ok=True)
  136. except Exception as e:
  137. raise LmdbError(
  138. 'Could not create store at {}. Error: {}'.format(
  139. self.env_path, e))
  140. # Create environment handle.
  141. _check(
  142. lmdb.mdb_env_create(&self.dbenv),
  143. 'Error creating DB environment handle: {}')
  144. logger.debug('Created DBenv @ {:x}'.format(<unsigned long>self.dbenv))
  145. # Set map size.
  146. _check(
  147. lmdb.mdb_env_set_mapsize(self.dbenv, self.options.get(
  148. 'map_size', 1024 ** 3)),
  149. 'Error setting map size: {}')
  150. # Set max databases.
  151. max_dbs = self.options.get('max_dbs', len(self.dbi_labels))
  152. _check(
  153. lmdb.mdb_env_set_maxdbs(self.dbenv, max_dbs),
  154. 'Error setting max. databases: {}')
  155. # Set max readers.
  156. self._readers = self.options.get(
  157. 'max_spare_txns', wsgi.workers * self.readers_mult)
  158. _check(
  159. lmdb.mdb_env_set_maxreaders(self.dbenv, self._readers),
  160. 'Error setting max. readers: {}')
  161. logger.debug('Max. readers: {}'.format(self._readers))
  162. # Clear stale readers.
  163. self._clear_stale_readers()
  164. # Open DB environment.
  165. logger.debug('DBenv address: {:x}'.format(<unsigned long>self.dbenv))
  166. _check(
  167. lmdb.mdb_env_open(
  168. self.dbenv, self.env_path.encode(),
  169. self.env_flags, self.env_perms),
  170. f'Error opening the database environment: {self.env_path}.')
  171. self._init_dbis(create)
  172. self._open = True
  173. cdef void _clear_stale_readers(self) except *:
  174. """
  175. Clear stale readers.
  176. """
  177. cdef int stale_readers
  178. _check(lmdb.mdb_reader_check(self.dbenv, &stale_readers))
  179. if stale_readers > 0:
  180. logger.debug('Cleared {} stale readers.'.format(stale_readers))
  181. cdef void _init_dbis(self, create=True) except *:
  182. """
  183. Initialize databases and cursors.
  184. """
  185. cdef:
  186. size_t i
  187. lmdb.MDB_txn *txn
  188. lmdb.MDB_dbi dbi
  189. # At least one slot (for environments without a database)
  190. self.dbis = <lmdb.MDB_dbi *>PyMem_Malloc(
  191. max(len(self.dbi_labels), 1) * sizeof(lmdb.MDB_dbi))
  192. if not self.dbis:
  193. raise MemoryError()
  194. # DBIs seem to start from 2. We want to map cursor pointers in the
  195. # array to DBIs, so we need an extra slot.
  196. self.curs = <lmdb.MDB_cursor **>PyMem_Malloc(
  197. (len(self.dbi_labels) + 2) * sizeof(lmdb.MDB_cursor*))
  198. if not self.curs:
  199. raise MemoryError()
  200. create_flag = lmdb.MDB_CREATE if create is True else 0
  201. txn_flags = 0 if create else lmdb.MDB_RDONLY
  202. rc = lmdb.mdb_txn_begin(self.dbenv, NULL, txn_flags, &txn)
  203. logger.info(f'Creating DBs.')
  204. try:
  205. if len(self.dbi_labels):
  206. for i, dblabel in enumerate(self.dbi_labels):
  207. flags = self.dbi_flags.get(dblabel, 0) | create_flag
  208. _check(lmdb.mdb_dbi_open(
  209. txn, dblabel.encode(), flags, self.dbis + i))
  210. dbi = self.dbis[i]
  211. logger.debug(f'Created DB {dblabel}: {dbi}')
  212. # Open and close cursor to initialize the memory slot.
  213. _check(lmdb.mdb_cursor_open(
  214. txn, dbi, self.curs + dbi))
  215. #lmdb.mdb_cursor_close(self.curs[dbi])
  216. else:
  217. _check(lmdb.mdb_dbi_open(txn, NULL, 0, self.dbis))
  218. _check(lmdb.mdb_cursor_open(txn, self.dbis[0], self.curs))
  219. #lmdb.mdb_cursor_close(self.curs[self.dbis[0]])
  220. _check(lmdb.mdb_txn_commit(txn))
  221. except:
  222. lmdb.mdb_txn_abort(txn)
  223. raise
  224. cpdef void close_env(self, bint commit_pending_transaction=False) except *:
  225. logger.debug('Cleaning up store env.')
  226. if self.is_open:
  227. logger.debug('Closing store env.')
  228. if self.is_txn_open is True:
  229. if commit_pending_transaction:
  230. self._txn_commit()
  231. else:
  232. self._txn_abort()
  233. self._clear_stale_readers()
  234. PyMem_Free(self.dbis)
  235. PyMem_Free(self.curs)
  236. lmdb.mdb_env_close(self.dbenv)
  237. self._open = False
  238. cpdef void destroy(self, _path='') except *:
  239. """
  240. Destroy the store.
  241. https://www.youtube.com/watch?v=lIVq7FJnPwg
  242. :param str _path: unused. Left for RDFLib API compatibility. (actually
  243. quite dangerous if it were used: it could turn into a
  244. general-purpose recursive file and folder delete method!)
  245. """
  246. if path.exists(self.env_path):
  247. if lmdb.MDB_NOSUBDIR & self.flags:
  248. try:
  249. os.unlink(self.env_path)
  250. except FileNotFoundError:
  251. pass
  252. try:
  253. os.unlink(self.env_path + '-lock')
  254. except FileNotFoundError:
  255. pass
  256. else:
  257. rmtree(self.env_path)
  258. ### PYTHON-ACCESSIBLE METHODS ###
  259. @contextmanager
  260. def txn_ctx(self, write=False):
  261. """
  262. Transaction context manager.
  263. :param bool write: Whether a write transaction is to be opened.
  264. :rtype: lmdb.Transaction
  265. """
  266. if not self.is_open:
  267. raise LmdbError('Store is not open.')
  268. if self.is_txn_open:
  269. logger.debug(
  270. 'Transaction is already active. Not opening another one.')
  271. #logger.debug('before yield')
  272. yield
  273. #logger.debug('after yield')
  274. else:
  275. #logger.debug('Beginning {} transaction.'.format(
  276. # 'RW' if write else 'RO'))
  277. try:
  278. self._txn_begin(write=write)
  279. self.is_txn_rw = write
  280. #logger.debug('In txn_ctx, before yield')
  281. yield
  282. #logger.debug('In txn_ctx, after yield')
  283. self._txn_commit()
  284. #logger.debug('after _txn_commit')
  285. except:
  286. self._txn_abort()
  287. raise
  288. def begin(self, write=False):
  289. """
  290. Begin a transaction manually if not already in a txn context.
  291. The :py:meth:`txn_ctx` context manager should be used whenever
  292. possible rather than this method.
  293. """
  294. if not self.is_open:
  295. raise RuntimeError('Store must be opened first.')
  296. #logger.debug('Beginning a {} transaction.'.format(
  297. # 'read/write' if write else 'read-only'))
  298. self._txn_begin(write=write)
  299. def commit(self):
  300. """Commit main transaction."""
  301. #logger.debug('Committing transaction.')
  302. self._txn_commit()
  303. def abort(self):
  304. """Abort main transaction."""
  305. #logger.debug('Rolling back transaction.')
  306. self._txn_abort()
  307. def rollback(self):
  308. """Alias for :py:meth:`abort`"""
  309. self.abort()
  310. def key_exists(self, key, dblabel='', new_txn=True):
  311. """
  312. Return whether a key exists in a database (Python-facing method).
  313. Wrap in a new transaction. Only use this if a transaction has not been
  314. opened.
  315. """
  316. if new_txn is True:
  317. with self.txn_ctx():
  318. return self._key_exists(
  319. key, len(key), dblabel=dblabel.encode())
  320. else:
  321. return self._key_exists(key, len(key), dblabel=dblabel.encode())
  322. cdef inline bint _key_exists(
  323. self, unsigned char *key, unsigned char klen,
  324. unsigned char *dblabel=b'') except -1:
  325. """
  326. Return whether a key exists in a database.
  327. To be used within an existing transaction.
  328. """
  329. cdef lmdb.MDB_val key_v, data_v
  330. key_v.mv_data = key
  331. key_v.mv_size = klen
  332. #logger.debug(
  333. # 'Checking if key {} with size {} exists...'.format(key, klen))
  334. try:
  335. _check(lmdb.mdb_get(
  336. self.txn, self.get_dbi(dblabel), &key_v, &data_v))
  337. except KeyNotFoundError:
  338. #logger.debug('...no.')
  339. return False
  340. #logger.debug('...yes.')
  341. return True
  342. def put(self, key, data, dblabel='', flags=0):
  343. """
  344. Put one key/value pair (Python-facing method).
  345. """
  346. self._put(
  347. key, len(key), data, len(data), dblabel=dblabel.encode(),
  348. txn=self.txn, flags=flags)
  349. cdef void _put(
  350. self, unsigned char *key, size_t key_size, unsigned char *data,
  351. size_t data_size, unsigned char *dblabel='',
  352. lmdb.MDB_txn *txn=NULL, unsigned int flags=0) except *:
  353. """
  354. Put one key/value pair.
  355. """
  356. if txn is NULL:
  357. txn = self.txn
  358. key_v.mv_data = key
  359. key_v.mv_size = key_size
  360. data_v.mv_data = data
  361. data_v.mv_size = data_size
  362. #logger.debug('Putting: {}, {} into DB {}'.format(key[: key_size],
  363. # data[: data_size], dblabel))
  364. rc = lmdb.mdb_put(txn, self.get_dbi(dblabel), &key_v, &data_v, flags)
  365. _check(rc, 'Error putting data: {}, {}'.format(
  366. key[: key_size], data[: data_size]))
  367. cpdef bytes get_data(self, key, dblabel=''):
  368. """
  369. Get a single value (non-dup) for a key (Python-facing method).
  370. """
  371. cdef lmdb.MDB_val rv
  372. try:
  373. self._get_data(key, len(key), &rv, dblabel=dblabel.encode())
  374. return (<unsigned char *>rv.mv_data)[: rv.mv_size]
  375. except KeyNotFoundError:
  376. return None
  377. cdef void _get_data(
  378. self, unsigned char *key, size_t klen, lmdb.MDB_val *rv,
  379. unsigned char *dblabel='') except *:
  380. """
  381. Get a single value (non-dup) for a key.
  382. """
  383. cdef:
  384. unsigned char *ret
  385. key_v.mv_data = key
  386. key_v.mv_size = len(key)
  387. _check(
  388. lmdb.mdb_get(self.txn, self.get_dbi(dblabel), &key_v, rv),
  389. 'Error getting data for key \'{}\'.'.format(key.decode()))
  390. def delete(self, key, dblabel=''):
  391. """
  392. Delete one single value by key. Python-facing method.
  393. """
  394. self._delete(key, len(key), dblabel.encode())
  395. cdef void _delete(
  396. self, unsigned char *key, size_t klen,
  397. unsigned char *dblabel=b'') except *:
  398. """
  399. Delete one single value by key from a non-dup database.
  400. TODO Allow deleting duplicate keys.
  401. """
  402. key_v.mv_data = key
  403. key_v.mv_size = klen
  404. try:
  405. _check(lmdb.mdb_del(self.txn, self.get_dbi(dblabel), &key_v, NULL))
  406. except KeyNotFoundError:
  407. pass
  408. cpdef dict stats(self):
  409. """Gather statistics about the database."""
  410. return self._stats()
  411. cdef dict _stats(self):
  412. """
  413. Gather statistics about the database.
  414. Cython-only, non-transaction-aware method.
  415. """
  416. cdef:
  417. lmdb.MDB_stat stat
  418. size_t entries
  419. lmdb.mdb_env_stat(self.dbenv, &stat)
  420. env_stats = <dict>stat
  421. db_stats = {}
  422. for i, dblabel in enumerate(self.dbi_labels):
  423. _check(
  424. lmdb.mdb_stat(self.txn, self.dbis[i], &stat),
  425. 'Error getting datbase stats: {}')
  426. entries = stat.ms_entries
  427. db_stats[dblabel.encode()] = <dict>stat
  428. return {
  429. 'env_stats': env_stats,
  430. 'env_size': os.stat(self.env_path).st_size,
  431. 'db_stats': {
  432. db_label: db_stats[db_label.encode()]
  433. for db_label in self.dbi_labels
  434. },
  435. }
  436. # UNFINISHED
  437. #cdef int _reader_list_callback(self, const unsigned char *msg, void *ctx):
  438. # """
  439. # Callback for reader info function.
  440. # Example from py-lmdb:
  441. # static int env_readers_callback(const char *msg, void *str_)
  442. # {
  443. # PyObject **str = str_;
  444. # PyObject *s = PyUnicode_FromString(msg);
  445. # PyObject *new;
  446. # if(! s) {
  447. # return -1;
  448. # }
  449. # new = PyUnicode_Concat(*str, s);
  450. # Py_CLEAR(*str);
  451. # *str = new;
  452. # if(! new) {
  453. # return -1;
  454. # }
  455. # return 0;
  456. # }
  457. # """
  458. # cdef:
  459. # unicode str = ctx[0].decode('utf-8')
  460. # unicode s = msg.decode('utf-8')
  461. # if not len(s):
  462. # return -1
  463. # str += s
  464. # logger.info('message: {}'.format(msg))
  465. # if not len(str):
  466. # return -1
  467. # ctx = &str
  468. #cpdef str reader_list(self):
  469. # """
  470. # Information about the reader lock table.
  471. # """
  472. # cdef unsigned char *ctx
  473. # lmdb.mdb_reader_list(self.dbenv, <lmdb.MDB_msg_func *>self._reader_list_callback, &ctx)
  474. # logger.info('Reader info: {}'.format(ctx))
  475. # return (ctx).decode('ascii')
  476. ### CYTHON METHODS ###
  477. cdef void _txn_begin(self, write=True, lmdb.MDB_txn *parent=NULL) except *:
  478. if not self.is_open:
  479. raise LmdbError('Store is not open.')
  480. cdef:
  481. unsigned int flags
  482. flags = 0 if write else lmdb.MDB_RDONLY
  483. logger.debug('Opening {} transaction in PID {}, thread {}'.format(
  484. 'RW' if write else 'RO',
  485. multiprocessing.current_process().pid,
  486. threading.currentThread().getName()))
  487. #logger.debug('Readers: {}'.format(self.reader_list()))
  488. rc = lmdb.mdb_txn_begin(self.dbenv, parent, flags, &self.txn)
  489. _check(rc, 'Error opening transaction.')
  490. logger.debug('Opened transaction @ {:x}'.format(<unsigned long>self.txn))
  491. self.is_txn_open = True
  492. self.is_txn_rw = write
  493. logger.debug('txn is open: {}'.format(self.is_txn_open))
  494. cdef void _txn_commit(self) except *:
  495. txid = '{:x}'.format(<unsigned long>self.txn)
  496. try:
  497. _check(lmdb.mdb_txn_commit(self.txn))
  498. logger.debug('Transaction @ {} committed.'.format(txid))
  499. self.is_txn_open = False
  500. self.is_txn_rw = False
  501. except:
  502. self._txn_abort()
  503. raise
  504. cdef void _txn_abort(self) except *:
  505. txid = '{:x}'.format(<unsigned long>self.txn)
  506. lmdb.mdb_txn_abort(self.txn)
  507. self.is_txn_open = False
  508. self.is_txn_rw = False
  509. logger.info('Transaction @ {} aborted.'.format(txid))
  510. cpdef int txn_id(self):
  511. return self._txn_id()
  512. cdef size_t _txn_id(self) except -1:
  513. return lmdb.mdb_txn_id(self.txn)
  514. cdef lmdb.MDB_dbi get_dbi(
  515. self, unsigned char *dblabel=NULL, lmdb.MDB_txn *txn=NULL):
  516. """
  517. Return a DB handle by database name.
  518. """
  519. cdef size_t dbidx
  520. if txn is NULL:
  521. txn = self.txn
  522. if dblabel is NULL:
  523. logger.debug('Getting DBI without label.')
  524. dbidx = (
  525. 0 if dblabel is NULL
  526. else self.dbi_labels.index(dblabel.decode()))
  527. #logger.debug(
  528. # f'Got DBI {self.dbis[dbidx]} with label {dblabel} '
  529. # f'and index #{dbidx}')
  530. return self.dbis[dbidx]
  531. cdef lmdb.MDB_cursor *_cur_open(
  532. self, unsigned char *dblabel=NULL, lmdb.MDB_txn *txn=NULL) except *:
  533. cdef:
  534. lmdb.MDB_dbi dbi
  535. if txn is NULL:
  536. txn = self.txn
  537. dbi = self.get_dbi(dblabel, txn=txn)
  538. logger.debug(f'Opening cursor for DB {dblabel} (DBI {dbi})...')
  539. #try:
  540. # # FIXME Either reuse the cursor, if it works, or remove this code.
  541. # _check(lmdb.mdb_cursor_renew(txn, self.curs[dbi]))
  542. # logger.debug(f'Repurposed existing cursor for DBI {dbi}.')
  543. #except LmdbError as e:
  544. # _check(
  545. # lmdb.mdb_cursor_open(txn, dbi, self.curs + dbi),
  546. # f'Error opening cursor: {dblabel}')
  547. # logger.debug(f'Created brand new cursor for DBI {dbi}.')
  548. _check(
  549. lmdb.mdb_cursor_open(txn, dbi, self.curs + dbi),
  550. f'Error opening cursor: {dblabel}')
  551. logger.debug('...opened @ {:x}.'.format(<unsigned long>self.curs[dbi]))
  552. return self.curs[dbi]
  553. cdef void _cur_close(self, lmdb.MDB_cursor *cur) except *:
  554. """Close a cursor."""
  555. #logger.info('Closing cursor @ {:x} for DBI {}...'.format(
  556. # <unsigned long>cur, lmdb.mdb_cursor_dbi(cur) ))
  557. lmdb.mdb_cursor_close(cur)
  558. #logger.info('...closed.')