Database cursors
Applies to: ✅ Azure Data Explorer
A database cursor is a database-level object that lets you query a database multiple times. You get consistent results even if there are data-append
or data-retention
operations happening in parallel with the queries.
Database cursors are designed to address two important scenarios:
The ability to repeat the same query multiple times and get the same results, as long as the query indicates "same data set".
The ability to make an "exactly once" query. This query only "sees" the data that a previous query didn't see, because the data wasn't available then. The query lets you iterate, for example, through all the newly arrived data in a table without fear of processing the same record twice or skipping records by mistake.
The database cursor is represented in the query language as a scalar value of type
string
. The actual value should be considered opaque and there's no support for any operation other than to save its value or use the following cursor functions.
Cursor functions
Kusto provides three functions to help implement the two above scenarios:
cursor_current(): Use this function to retrieve the current value of the database cursor. You can use this value as an argument to the two other functions.
cursor_after(rhs:string): This special function can be used on table records that have the IngestionTime policy enabled. It returns a scalar value of type
bool
indicating whether the record'singestion_time()
database cursor value comes after therhs
database cursor value.cursor_before_or_at(rhs:string): This special function can be used on the table records that have the IngestionTime policy enabled. It returns a scalar value of type
bool
indicating whether the record'singestion_time()
database cursor value comes before or at therhs
database cursor value.
The two special functions (cursor_after
and cursor_before_or_at
) also have a side-effect: When they're used, Kusto emits the current value of the database cursor to the @ExtendedProperties
result set of the query. The property name for the cursor is Cursor
, and its value is a single string
.
For example:
{"Cursor" : "636040929866477946"}
Restrictions
Database cursors can only be used with tables for which the IngestionTime policy is enabled. Each record in such a table is associated with the value of the database cursor that was in effect when the record was ingested. As such, the ingestion_time() function can be used.
The database cursor object holds no meaningful value unless the database has at least one table that has an IngestionTime policy defined. This value is guaranteed to update, as-needed by the ingestion history, into such tables and the queries run, that reference such tables. It might, or might not, be updated in other cases.
The ingestion process first commits the data, so that it's available for querying, and only then assigns an actual cursor value to each record. Querying for data immediately after ingestion using a database cursor might not incorporate the last records added because the cursor value wasn't yet assigned. Also, retrieving the current database cursor value repeatedly might return the same value, even if ingestion was done in between, because only a cursor commit can update its value.
Querying a table based on database cursors is only guaranteed to "work" (providing exactly-once guarantees) if the records are ingested directly into that table. If you use extents commands, such as .move extents or .replace extents to move data into the table, or if you're using .rename table, then querying this table using database cursors isn't guaranteed to avoid missing any data. This is because the ingestion time of the records is assigned when initially ingested, and doesn't change during the move extents operation.
When the extents are moved into the target table, the assigned cursor value might already have been processed, and the next query by the database cursor will miss the new records.
Example: Processing records exactly once
For a table Employees
with schema [Name, Salary]
, to continuously process new records as they're ingested into the table, use the following process:
// [Once] Enable the IngestionTime policy on table Employees
.set table Employees policy ingestiontime true
// [Once] Get all the data that the Employees table currently holds
Employees | where cursor_after('')
// The query above will return the database cursor value in
// the @ExtendedProperties result set. Lets assume that it returns
// the value '636040929866477946'
// [Many] Get all the data that was added to the Employees table
// since the previous query was run using the previously-returned
// database cursor
Employees | where cursor_after('636040929866477946') // -> 636040929866477950
Employees | where cursor_after('636040929866477950') // -> 636040929866479999
Employees | where cursor_after('636040929866479999') // -> 636040939866479000