Provided by: libpmemkv-dev_1.5.0-4build1_amd64 bug

NAME

       pmemkv_iterator - Iterator API for libpmemkv

       This API is EXPERIMENTAL and might change.

SYNOPSIS

              #include <libpmemkv.h>

              int pmemkv_iterator_new(pmemkv_db *db, pmemkv_iterator **it);
              int pmemkv_write_iterator_new(pmemkv_db *db, pmemkv_write_iterator **it);

              void pmemkv_iterator_delete(pmemkv_iterator *it);
              void pmemkv_write_iterator_delete(pmemkv_write_iterator *it);

              int pmemkv_iterator_seek(pmemkv_iterator *it, const char *k, size_t kb);
              int pmemkv_iterator_seek_lower(pmemkv_iterator *it, const char *k, size_t kb);
              int pmemkv_iterator_seek_lower_eq(pmemkv_iterator *it, const char *k, size_t kb);
              int pmemkv_iterator_seek_higher(pmemkv_iterator *it, const char *k, size_t kb);
              int pmemkv_iterator_seek_higher_eq(pmemkv_iterator *it, const char *k, size_t kb);

              int pmemkv_iterator_seek_to_first(pmemkv_iterator *it);
              int pmemkv_iterator_seek_to_last(pmemkv_iterator *it);

              int pmemkv_iterator_is_next(pmemkv_iterator *it);
              int pmemkv_iterator_next(pmemkv_iterator *it);
              int pmemkv_iterator_prev(pmemkv_iterator *it);

              int pmemkv_iterator_key(pmemkv_iterator *it, const char **k, size_t *kb);

              int pmemkv_iterator_read_range(pmemkv_iterator *it, size_t pos, size_t n,
                                  const char **data, size_t *rb);
              int pmemkv_write_iterator_write_range(pmemkv_write_iterator *it, size_t pos, size_t n,
                                  char **data, size_t *wb);

              int pmemkv_write_iterator_commit(pmemkv_write_iterator *it);
              void pmemkv_write_iterator_abort(pmemkv_write_iterator *it);

       For general description of pmemkv and available engines see libpmemkv(7).  For description of pmemkv core
       API see libpmemkv(3).

DESCRIPTION

       Iterators provide methods to iterate over records in db.

       Both  iterator  types  (pmemkv_iterator  (read) and pmemkv_write_iterator) allow reading record’s key and
       value.   To  use  pmemkv_write_iterator  as  a  pmemkv_iterator  you  need  to  get  its  member   “iter”
       (write_it->iter).

       Example of calling pmemkv_iterator_seek_to_first() with both iterator types.

              /* read_it is already created, by pmemkv_iterator */
              int status = pmemkv_iterator_seek_to_first(read_it);

              /* write_it is already created, by pmemkv_write_iterator */
              int status = pmemkv_iterator_seek_to_first(write_it->iter);

       A pmemkv_write_iterator additionally can modify record’s value transactionally.

       Some of the functions are not guaranteed to be implemented by all engines.  If an engine does not support
       a certain function, it will return PMEMKV_STATUS_NOT_SUPPORTED.

       Holding simultaneously in the same thread more than one iterator is undefined behavior.

       int pmemkv_iterator_new(pmemkv_db *db, pmemkv_iterator **it);
              Creates a new pmemkv_iterator instance and stores a pointer to it in *it.

       int pmemkv_write_iterator_new(pmemkv_db *db, pmemkv_write_iterator **it);
              Creates a new pmemkv_write_iterator instance and stores a pointer to it in *it.

       void pmemkv_iterator_delete(pmemkv_iterator *it);
              Deletes pmemkv_iterator.

       void pmemkv_write_iterator_delete(pmemkv_write_iterator *it);
              Deletes pmemkv_write_iterator

       int pmemkv_iterator_seek(pmemkv_iterator *it, const char *k, size_t kb);
              Changes  iterator  position to the record with given key k of length kb.  If the record is present
              and no errors occurred, returns PMEMKV_STATUS_OK.  If  the  record  does  not  exist,  PMEMKV_STA‐
              TUS_NOT_FOUND  is  returned  and  the  iterator  position  is undefined.  It internally aborts all
              changes made to an element previously pointed by the iterator.

       int pmemkv_iterator_seek_lower(pmemkv_iterator *it, const char *k, size_t kb);
              Changes iterator position to the record with key lower than given key k  of  length  kb.   If  the
              record is present and no errors occurred, returns PMEMKV_STATUS_OK.  If the record does not exist,
              PMEMKV_STATUS_NOT_FOUND  is returned and the iterator position is undefined.  It internally aborts
              all changes made to an element previously pointed by the iterator.

       int pmemkv_iterator_seek_lower_eq(pmemkv_iterator *it, const char *k, size_t kb);
              Changes iterator position to the record with key equal or lower than given key k of length kb.  If
              the record is present and no errors occurred, returns PMEMKV_STATUS_OK.  If the  record  does  not
              exist,  PMEMKV_STATUS_NOT_FOUND is returned and the iterator position is undefined.  It internally
              aborts all changes made to an element previously pointed by the iterator.

       int pmemkv_iterator_seek_higher(pmemkv_iterator *it, const char *k, size_t kb);
              Changes iterator position to the record with key higher than given key k of  length  kb.   If  the
              record is present and no errors occurred, returns PMEMKV_STATUS_OK.  If the record does not exist,
              PMEMKV_STATUS_NOT_FOUND  is returned and the iterator position is undefined.  It internally aborts
              all changes made to an element previously pointed by the iterator.

       int pmemkv_iterator_seek_higher_eq(pmemkv_iterator *it, const char *k, size_t kb);
              Changes iterator position to the record with key equal or higher than given key k  of  length  kb.
              If the record is present and no errors occurred, returns PMEMKV_STATUS_OK.  If the record does not
              exist,  PMEMKV_STATUS_NOT_FOUND is returned and the iterator position is undefined.  It internally
              aborts all changes made to an element previously pointed by the iterator.

       int pmemkv_iterator_seek_to_first(pmemkv_iterator *it);
              Changes iterator position to the first record.  If db isn’t empty, and no errors occurred, returns
              PMEMKV_STATUS_OK.  If db is empty, PMEMKV_STATUS_NOT_FOUND is returned and the  iterator  position
              is undefined.  It internally aborts all changes made to an element previously pointed by the iter‐
              ator.

       int pmemkv_iterator_seek_to_last(pmemkv_iterator *it);
              Changes  iterator position to the last record.  If db isn’t empty, and no errors occurred, returns
              PMEMKV_STATUS_OK.  If db is empty, PMEMKV_STATUS_NOT_FOUND is returned and the  iterator  position
              is undefined.  It internally aborts all changes made to an element previously pointed by the iter‐
              ator.

       int pmemkv_iterator_is_next(pmemkv_iterator *it);
              Checks if there is a next record available.  If true is returned, it is guaranteed that pmemkv_it‐
              erator_next(it)  will  return  PMEMKV_STATUS_OK, otherwise iterator is already on the last element
              and pmemkv_iterator_next(it) will return PMEMKV_STATUS_NOT_FOUND.

       int pmemkv_iterator_next(pmemkv_iterator *it);
              Changes iterator position to the next record.  If the  next  record  exists,  returns  PMEMKV_STA‐
              TUS_OK,  otherwise PMEMKV_STATUS_NOT_FOUND is returned and the iterator position is undefined.  It
              internally aborts all changes made to an element previously pointed by the iterator.

       int pmemkv_iterator_prev(pmemkv_iterator *it);
              Changes iterator position to  the  previous  record.   If  the  previous  record  exists,  returns
              PMEMKV_STATUS_OK, otherwise PMEMKV_STATUS_NOT_FOUND is returned and the iterator position is unde‐
              fined.  It internally aborts all changes made to an element previously pointed by the iterator.

       int pmemkv_iterator_key(pmemkv_iterator *it, const char **k, size_t *kb);
              Assigns  record’s  key’s  address to k and key’s length to kb.  If the iterator is on an undefined
              position, calling this method is undefined behaviour.

       int pmemkv_iterator_read_range(pmemkv_iterator *it, size_t pos, size_t n, const char **data, size_t *rb);
              Allows getting record’s value’s range which can be only read.  You can  request  for  either  full
              value  or  only value’s subrange (n elements starting from pos).  Assigns pointer to the beginning
              of the requested range to data, and number of elements in range to rb.  If n is bigger than length
              of a value it’s automatically shrunk.  If the iterator is on an undefined position,  calling  this
              method is undefined behaviour.

       int pmemkv_write_iterator_write_range(pmemkv_write_iterator *it, size_t pos, size_t n, char **data,
       size_t *wb);
              Allows getting record’s value’s range which can be modified.  You can request for either full val‐
              ue  or  only value’s subrange (n elements starting from pos).  Assigns pointer to the beginning of
              the requested range to data, and number of elements in range to wb.  If n is bigger than length of
              a value it’s automatically shrunk.  Changes made on a requested range  are  not  persistent  until
              pmemkv_write_iterator_commit()  is  called.   If the iterator is on an undefined position, calling
              this method is undefined behaviour.

       int pmemkv_write_iterator_commit(pmemkv_write_iterator *it);
              Commits modifications made on the current record.  Calling this method is the  only  way  to  save
              modifications  made  by  the  iterator on the current record.  You need to call this method before
              changing the iterator position, otherwise modifications will be automatically aborted.

       void pmemkv_write_iterator_abort(pmemkv_write_iterator *it);
              Aborts uncommitted modifications made on the current record.

   ERRORS
       Each function, except for pmemkv_iterator_delete(), pmemkv_write_iterator_delete() and pmemkv_write_iter‐
       ator_abort(), returns one of the pmemkv status codes.  To check possible options see libpmemkv(3).

EXAMPLE

       The following example is taken from examples/pmemkv_iterator_c directory.

       Usage of basic iterator functions to iterate over all records and modify one of them.

              #include "libpmemkv.h"

              #include <assert.h>
              #include <stdio.h>
              #include <stdlib.h>
              #include <string.h>

              #define ASSERT(expr)                                                                     \
                  do {                                                                             \
                      if (!(expr))                                                             \
                          puts(pmemkv_errormsg());                                         \
                      assert(expr);                                                            \
                  } while (0)

              #define LOG(msg) puts(msg)

              static const uint64_t SIZE = 1024UL * 1024UL * 1024UL;

              int main(int argc, char *argv[])
              {
                  if (argc < 2) {
                      fprintf(stderr, "Usage: %s file\n", argv[0]);
                      exit(1);
                  }

                  const size_t n_elements = 10;
                  char buffer[64];

                  /* See libpmemkv_config(3) for more detailed example of config creation */
                  LOG("Creating config");
                  pmemkv_config *cfg = pmemkv_config_new();
                  ASSERT(cfg != NULL);

                  int s = pmemkv_config_put_path(cfg, argv[1]);
                  ASSERT(s == PMEMKV_STATUS_OK);
                  s = pmemkv_config_put_size(cfg, SIZE);
                  ASSERT(s == PMEMKV_STATUS_OK);
                  s = pmemkv_config_put_create_if_missing(cfg, true);
                  ASSERT(s == PMEMKV_STATUS_OK);

                  LOG("Opening pmemkv database with 'radix' engine");
                  pmemkv_db *db = NULL;
                  s = pmemkv_open("radix", cfg, &db);
                  ASSERT(s == PMEMKV_STATUS_OK);
                  ASSERT(db != NULL);

                  LOG("Putting new keys");
                  for (size_t i = 0; i < n_elements; ++i) {
                      char key[10];
                      const char *value = "value";
                      sprintf(key, "key%zu", i);
                      s = pmemkv_put(db, key, strlen(key), value, strlen(value));
                      ASSERT(s == PMEMKV_STATUS_OK);
                  }

                  /* get a new read iterator */
                  pmemkv_iterator *it;
                  s = pmemkv_iterator_new(db, &it);
                  ASSERT(s == PMEMKV_STATUS_OK);

                  LOG("Iterate from first to last element");
                  s = pmemkv_iterator_seek_to_first(it);
                  ASSERT(s == PMEMKV_STATUS_OK);

                  size_t element_number = 0;
                  do {
                      const char *str;
                      size_t cnt;
                      /* read a key */
                      s = pmemkv_iterator_key(it, &str, &cnt);
                      ASSERT(s == PMEMKV_STATUS_OK);
                      sprintf(buffer, "Key %zu = %s", element_number++, str);
                      LOG(buffer);
                  } while (pmemkv_iterator_next(it) != PMEMKV_STATUS_NOT_FOUND);

                  /* iterator must be deleted manually */
                  pmemkv_iterator_delete(it);

                  /* get a new write_iterator */
                  pmemkv_write_iterator *w_it;
                  s = pmemkv_write_iterator_new(db, &w_it);
                  ASSERT(s == PMEMKV_STATUS_OK);

                  /* if you want to get a pmemkv_iterator (read iterator) from a
                   * pmemkv_write_iterator, you should do: write_it->iter */
                  s = pmemkv_iterator_seek_to_last(w_it->iter);
                  ASSERT(s == PMEMKV_STATUS_OK);

                  /* get a write range, to modify last element's value */
                  char *data;
                  size_t cnt;
                  s = pmemkv_write_iterator_write_range(w_it, 0, 5, &data, &cnt);
                  ASSERT(s == PMEMKV_STATUS_OK);

                  for (size_t i = 0; i < cnt; ++i)
                      data[i] = 'x';

                  /* commit changes */
                  s = pmemkv_write_iterator_commit(w_it);
                  ASSERT(s == PMEMKV_STATUS_OK);

                  /* get a read range, to read modified value */
                  const char *str;
                  s = pmemkv_iterator_read_range(w_it->iter, 0, 5, &str, &cnt);
                  ASSERT(s == PMEMKV_STATUS_OK);

                  /* verify a modified value */
                  ASSERT(strcmp(str, "xxxxx") == 0);
                  sprintf(buffer, "Modified value = %s", str);
                  LOG(buffer);

                  /* iterator must be deleted manually */
                  pmemkv_write_iterator_delete(w_it);

                  LOG("Closing database");
                  pmemkv_close(db);

                  return 0;
              }

SEE ALSO

       libpmemkv(7), libpmemkv(3) and <https://pmem.io>

PMEMKV - pmemkv version 1.5.0                      2022-03-09                                 PMEMKV_ITERATOR(3)