2
0
mirror of https://github.com/openvswitch/ovs synced 2025-08-30 22:05:19 +00:00
Files
ovs/Documentation/topics/idl-compound-indexes.rst
Ben Pfaff d14e007c1f ovsdb-idl: Redesign use of indexes.
The design of the compound index feature in the C OVSDB IDL was unusual.
Indexes were generally referenced only by name rather than by pointer, and
could be obtained only from the top-level ovsdb_idl object.  To iterate or
otherwise search an index required explicitly creating a special
ovsdb_idl_cursor object, which at least seemed somewhat heavy-weight given
that it required a string lookup in a table of indexes.

This commit redesigns the compound index interface.  It discards the use of
names for indexes, instead having clients pass in a pointer to the index
object itself.  It simplifies how indexes are created, gets rid of the need
for explicit cursor objects, and updates all of the users to the new
interface.

The underlying reason for this commit is to make it easier in
ovn-controller to keep track of the dependencies for a given function, by
making the indexes explicit arguments to any function that needs to use
them.

Signed-off-by: Ben Pfaff <blp@ovn.org>
Acked-by: Han Zhou <hzhou8@ebay.com>
2018-06-12 08:24:08 -07:00

278 lines
11 KiB
ReStructuredText

..
Licensed under the Apache License, Version 2.0 (the "License"); you may
not use this file except in compliance with the License. You may obtain
a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
License for the specific language governing permissions and limitations
under the License.
Convention for heading levels in Open vSwitch documentation:
======= Heading 0 (reserved for the title in a document)
------- Heading 1
~~~~~~~ Heading 2
+++++++ Heading 3
''''''' Heading 4
Avoid deeper levels because they do not render well.
======================
C IDL Compound Indexes
======================
Introduction
------------
This document describes the design and usage of the C IDL Compound
Indexes feature, which allows OVSDB client applications to efficiently
search table contents using arbitrary sets of column values in a generic
way.
This feature is implemented entirely in the client IDL, requiring no changes
to the OVSDB Server, OVSDB Protocol (OVSDB RFC (RFC 7047)) or additional
interaction with the OVSDB server.
Please note that in this document, the term "index" refers to the common
database term defined as "a data structure that facilitates data
retrieval". Unless stated otherwise, the definition for index from the
OVSDB RFC (RFC 7047) is not used.
Typical Use Cases
-----------------
Fast lookups
~~~~~~~~~~~~
Depending on the topology, the route table of a network device could
manage thousands of routes. Commands such as "show ip route <*specific
route*>" would need to do a sequential lookup of the routing table to
find the specific route. With an index created, the lookup time could be
faster.
This same scenario could be applied to other features such as Access
List rules and even interfaces lists.
Lexicographic order
~~~~~~~~~~~~~~~~~~~
There are a number of cases in which retrieving data in a particular
lexicographic order is needed. For example, SNMP. When an administrator
or even a NMS would like to retrieve data from a specific device, it's
possible that they will request data from full tables instead of just
specific values. Also, they would like to have this information displayed
in lexicographic order. This operation could be done by the SNMP daemon or
by the CLI, but it would be better if the database could provide the
data ready for consumption. Also, duplicate efforts by different
processes will be avoided. Another use case for requesting data in
lexicographic order is for user interfaces (web or CLI) where it would
be better and quicker if the DB sends the data sorted instead of letting
each process to sort the data by itself.
Implementation Design
---------------------
This feature maintains a collection of indexes per table. The application
can create any number of indexes per table.
An index can be defined over any number of columns, and supports the
following options:
- Add a column with type string, boolean, uuid, integer or real (using
default comparators).
- Select ordering direction of a column (ascending or descending, must
be selected when creating the index).
- Use a custom ordering comparator (eg: treat a string column like a IP,
or sort by the value of the "config" key in a map column).
Indexes can be searched for matches based on the key. They can also
be iterated across a range of keys or in full.
For lookups, the user needs to provide a key to be used for locating the
specific rows that meet his criteria. This key could be an IP address, a MAC
address, an ACL rule, etc. If several rows match the query then the user can
easily iterate over all of the matches.
For accessing data in lexicographic order, the user can use the ranged
iterators, which use "from" and "to" values to define a range.
The indexes maintain a pointer to the row in the local replica, avoiding
the need to make additional copies of the data and thereby minimizing any
additional memory and CPU overhead for their maintenance. It is intended
that creating and maintaining indexes should be very cheap.
Another potential issue is the time needed to create the data structure
and the time needed to add/remove elements. The indexes are always
synchronized with the replica. For this reason is VERY IMPORTANT that
the comparison functions (built-in and user provided) are FAST.
Skiplists are used as the primary data structure for the implementation of
indexes. Indexes therefore have an expected ``O(log(n))`` cost when
inserting, deleting or modifying a row, ``O(log(n))`` when retrieving
a row by key, and O(1) when retrieving the first or next row.
Indexes are maintained incrementally in the replica as notifications of
database changes are received from the OVSDB server, as shown in the
following diagram.
::
+---------------------------------------------------------+
| |
+-------------+Client changes to data IDL |
| | |
+---v---+ | |
| OVSDB +--------->OVSDB Notification |
+-------+ | + |
| | +------------+ |
| | | | |
| | | Insert Row +----> Insert row to indexes |
| | | | ^ |
| +-> | Modify Row +-------------------+ |
| | | v |
| | Delete Row +----> Delete row from indexes |
| | | |
| +----+-------+ |
| | |
| +-> IDL Replica |
| |
+---------------------------------------------------------+
C IDL API
---------
Index Creation
~~~~~~~~~~~~~~
Each index must be created with the function ``ovsdb_idl_index_create()`` or
one of the simpler convenience functions ``ovsdb_idl_index_create1()`` or
``ovsdb_idl_index_create2()``. All indexes must be created before the first
call to ovsdb\_idl\_run().
Index Creation Example
^^^^^^^^^^^^^^^^^^^^^^
::
/* Define a custom comparator for the column "stringField" in table
* "Test". (Note that custom comparison functions are not often
* necessary.)
*/
int stringField_comparator(const void *a, const void *b)
{
struct ovsrec_test *AAA, *BBB;
AAA = (struct ovsrec_test *)a;
BBB = (struct ovsrec_test *)b;
return strcmp(AAA->stringField, BBB->stringField);
}
void init_idl(struct ovsdb_idl **, char *remote)
{
/* Add the columns to the IDL */
*idl = ovsdb_idl_create(remote, &ovsrec_idl_class, false, true);
ovsdb_idl_add_table(*idl, &ovsrec_table_test);
ovsdb_idl_add_column(*idl, &ovsrec_test_col_stringField);
ovsdb_idl_add_column(*idl, &ovsrec_test_col_numericField);
ovsdb_idl_add_column(*idl, &ovsrec_test_col_enumField);
ovsdb_idl_add_column(*idl, &ovsrec_test_col_boolField);
struct ovsdb_idl_index_column columns[] = {
{ .column = &ovsrec_test_col_stringField,
.comparer = stringField_comparator },
{ .column = &ovsrec_test_col_numericField,
.order = OVSDB_INDEX_DESC },
};
struct ovsdb_idl_index *index = ovsdb_idl_create_index(
*idl, columns, ARRAY_SIZE(columns));
...
}
Index Usage
-----------
Iterators
~~~~~~~~~
The recommended way to do queries is using a "ranged foreach", an "equal
foreach" or a "full foreach" over an index. The mechanism works as
follows:
1. Create index row objects with index columns set to desired search key
values (one is needed for equality iterators, two for range iterators,
a search key is not needed for the full index iterator).
2. Pass the index, an iteration variable, and the index row object to the
iterator.
3. Use the values within iterator loop.
The library implements three different iterators: a range iterator, an
equality iterator and a full index iterator. The range iterator
receives two values and iterates over all rows with values that are
within that range (inclusive of the two values defining the range). The
equality iterator iterates over all rows that exactly match the value
passed. The full index iterator iterates over all rows in the index, in
an order determined by the comparison function and configured direction
(ascending or descending).
Note that indexes are *sorted by the "concatenation" of the values in
all indexed columns*, so the ranged iterator returns all the values
between "from.col1 from.col2 ... from.coln" and "to.col1 to.col2 ...
to.coln", *NOT the rows with a value in column 1 between from.col1 and
to.col1, and so on*.
The iterators are macros specific to each table. An example of the use of
these iterators follows:
::
/*
* Equality iterator; iterates over all the records equal to "value".
*/
struct ovsrec_test *target = ovsrec_test_index_init_row(index);
ovsrec_test_index_set_stringField(target, "hello world");
struct ovsrec_test *record;
OVSREC_TEST_FOR_EACH_EQUAL (record, target, index) {
/* Can return zero, one or more records */
assert(strcmp(record->stringField, "hello world") == 0);
printf("Found one record with %s", record->stringField);
}
ovsrec_test_index_destroy_row(target);
/*
* Range iterator; iterates over all records between two values
* (inclusive).
*/
struct ovsrec_test *from = ovsrec_test_index_init_row(index);
struct ovsrec_test *to = ovsrec_test_index_init_row(index);
ovsrec_test_index_set_stringField(from, "aaa");
ovsrec_test_index_set_stringField(to, "mmm");
OVSREC_TEST_FOR_EACH_RANGE (record, from, to, index) {
/* Can return zero, one or more records */
assert(strcmp("aaa", record->stringField) <= 0);
assert(strcmp(record->stringField, "mmm") <= 0);
printf("Found one record with %s", record->stringField);
}
ovsrec_test_index_destroy_row(from);
ovsrec_test_index_destroy_row(to);
/*
* Index iterator; iterates over all nodes in the index, in order
* determined by comparison function and configured order (ascending
* or descending).
*/
OVSREC_TEST_FOR_EACH_BYINDEX (record, index) {
/* Can return zero, one or more records */
printf("Found one record with %s", record->stringField);
}
General Index Access
~~~~~~~~~~~~~~~~~~~~
While the currently defined iterators are suitable for many use cases, it is
also possible to create custom iterators using the more general API on which
the existing iterators have been built. See ``ovsdb-idl.h`` for the details.