App in C++

    To interact with YDB, create an instance of the driver, client, and session:

    • The YDB driver lets the app and YDB interact at the transport layer. The driver must exist throughout the YDB access lifecycle and be initialized before creating a client or session.
    • The YDB client runs on top of the YDB driver and enables the handling of entities and transactions.
    • The YDB session contains information about executed transactions and prepared queries, and is part of the YDB client context.

    App code snippet for driver initialization:

    App code snippet for creating a client:

    C++ - 图2

    Creating tables

    Creating tables to be used in operations on a test app. This step results in the creation of DB tables of the series directory data model:

    • Series
    • Seasons
    • Episodes

    Once the tables are created, the method for getting information about data schema objects is called and the result of its execution is output.

    To create tables, use the CreateTable method:

    1. //! Creates sample tables with CrateTable API.
    2. ThrowOnError(client.RetryOperationSync([path](TSession session) {
    3. auto seriesDesc = TTableBuilder()
    4. .AddNullableColumn("series_id", EPrimitiveType::Uint64)
    5. .AddNullableColumn("title", EPrimitiveType::Utf8)
    6. .AddNullableColumn("series_info", EPrimitiveType::Utf8)
    7. .AddNullableColumn("release_date", EPrimitiveType::Uint64)
    8. .SetPrimaryKeyColumn("series_id")
    9. .Build();
    10. return session.CreateTable(JoinPath(path, "series"), std::move(seriesDesc)).GetValueSync();
    11. }));

    Use the describeTable method to output information about the table structure and make sure that it was properly created.

    1. TMaybe<TTableDescription> desc;
    2. ThrowOnError(client.RetryOperationSync([path, name, &desc](TSession session) {
    3. auto result = session.DescribeTable(JoinPath(path, name)).GetValueSync();
    4. if (result.IsSuccess()) {
    5. desc = result.GetTableDescription();
    6. }
    7. return result;
    8. }));
    9. Cout << "> Describe table: " << name << Endl;
    10. for (auto& column : desc->GetColumns()) {
    11. Cout << "Column, name: " << column.Name << ", type: " << FormatType(column.Type) << Endl;
    12. }

    C++ - 图4

    The given code snippet outputs the following text to the console at startup:

    1. > Describe table: series
    2. Column, name: series_id, type: Uint64?
    3. Column, name: title, type: Utf8?
    4. Column, name: series_info, type: Utf8?
    5. Column, name: release_date, type: Uint64?

    Adding data to the created tables using an UPSERT statement of . A data update request is sent within a single request to the server with transaction auto-commit mode enabled.

    Code snippet for inserting and updating data:

    1. //! Shows basic usage of mutating operations.
    2. static TStatus UpsertSimpleTransaction(TSession session, const TString& path) {
    3. auto query = Sprintf(R"(
    4. PRAGMA TablePathPrefix("%s");
    5. UPSERT INTO episodes (series_id, season_id, episode_id, title) VALUES
    6. (2, 6, 1, "TBD");
    7. )", path.c_str());
    8. return session.ExecuteDataQuery(query,
    9. TTxControl::BeginTx(TTxSettings::SerializableRW()).CommitTx()).GetValueSync();
    10. }

    C++ - 图6

    For more information about PRAGMA YQL, see the YQL documentation.

    Retrieving data with a Select

    Retrieving data using a SELECT statement in . Handling the retrieved data selection in the app.

    To execute YQL queries, use the ExecuteDataQuery method.
    The SDK lets you explicitly control the execution of transactions and configure the transaction execution mode using the TTxControl class.

    In the code snippet below, the transaction is started with the TTxControl::BeginTx method. With TTxSettings, set the SerializableRW transaction execution mode. When all the queries in the transaction are completed, the transaction is automatically completed by explicitly setting CommitTx(). The query described using the YQL syntax is passed to the ExecuteDataQuery method for execution.

    1. //! Shows basic usage of YDB data queries and transactions.
    2. static TStatus SelectSimpleTransaction(TSession session, const TString& path,
    3. TMaybe<TResultSet>& resultSet)
    4. {
    5. auto query = Sprintf(R"(
    6. PRAGMA TablePathPrefix("%s");
    7. SELECT series_id, title, DateTime::ToDate(DateTime::FromDays(release_date)) AS release_date
    8. FROM series
    9. )", path.c_str());
    10. auto txControl =
    11. // Begin new transaction with SerializableRW mode
    12. TTxControl::BeginTx(TTxSettings::SerializableRW())
    13. // Commit transaction at the end of the query
    14. .CommitTx();
    15. // Executes data query with specified transaction control settings.
    16. auto result = session.ExecuteDataQuery(query, txControl).GetValueSync();
    17. if (result.IsSuccess()) {
    18. // Index of result set corresponds to its order in YQL query
    19. resultSet = result.GetResultSet(0);
    20. }
    21. return result;
    22. }

    C++ - 图8

    The TResultSetParser class is used for processing query results.
    The code snippet below shows how to process query results using the parser object:

    1. TResultSetParser parser(*resultSet);
    2. if (parser.TryNextRow()) {
    3. Cout << "> SelectSimple:" << Endl << "Series"
    4. << ", Id: " << parser.ColumnParser("series_id").GetOptionalUint64()
    5. << ", Title: " << parser.ColumnParser("title").GetOptionalUtf8()
    6. << ", Release date: " << parser.ColumnParser("release_date").GetOptionalString()
    7. << Endl;
    8. }

    The given code snippet outputs the following text to the console at startup:

    1. > SelectSimple:
    2. series, Id: 1, title: IT Crowd, Release date: 2006-02-03

    C++ - 图10

    Querying data using parameters. This query execution option is preferable as it allows the server to reuse the query execution plan for subsequent calls and also protects from such vulnerabilities as SQL Injection.

    The code snippet shows the use of parameterized queries and the GetParamsBuilder to generate parameters and pass them to the ExecuteDataQuery method.

    1. //! Shows usage of parameters in data queries.
    2. static TStatus SelectWithParamsTransaction(TSession session, const TString& path,
    3. ui64 seriesId, ui64 seasonId, TMaybe<TResultSet>& resultSet)
    4. {
    5. auto query = Sprintf(R"(
    6. PRAGMA TablePathPrefix("%s");
    7. DECLARE $seriesId AS Uint64;
    8. DECLARE $seasonId AS Uint64;
    9. SELECT sa.title AS season_title, sr.title AS series_title
    10. FROM seasons AS sa
    11. INNER JOIN series AS sr
    12. ON sa.series_id = sr.series_id
    13. WHERE sa.series_id = $seriesId AND sa.season_id = $seasonId;
    14. )", path.c_str());
    15. // Type of parameter values should be exactly the same as in DECLARE statements.
    16. auto params = session.GetParamsBuilder()
    17. .AddParam("$seriesId")
    18. .Uint64(seriesId)
    19. .Build()
    20. .AddParam("$seasonId")
    21. .Uint64(seasonId)
    22. .Build()
    23. .Build();
    24. auto result = session.ExecuteDataQuery(
    25. query,
    26. TTxControl::BeginTx(TTxSettings::SerializableRW()).CommitTx(),
    27. std::move(params)).GetValueSync();
    28. if (result.IsSuccess()) {
    29. resultSet = result.GetResultSet(0);
    30. }
    31. return result;
    32. }

    The given code snippet outputs the following text to the console at startup:

    1. > SelectWithParams:
    2. Season, title: Season 3, series title: Silicon Valley
    3. Finished preparing query: PreparedSelectTransaction

    C++ - 图12

    Parameterized prepared queries

    Parameterized prepared queries are saved as templates where specially formatted names are replaced by relevant parameter values each time you execute the query. Use parameterized queries to improve performance by reducing how often queries that only differ in parameter values are compiled and recompiled. The prepared query is stored in the session context.

    The given code snippet outputs the following text to the console at startup:

    1. > PreparedSelect:
    2. Episode 7, title: To Build a Better Beta, Air date: Sun Jun 05, 2016

    C++ - 图14

    You can use the GetPreparedQuery method to check whether there is a prepared query in the session. If there’s no prepared query in the session context yet, you can prepare one using PrepareDataQuery and save it for use within the current session via AddPreparedQuery.

    Multiple commands are executed within a single multistep transaction. The client-side code can be run between query executions. Using a transaction ensures that select queries made in its context are consistent with each other.

    The first step is to prepare and execute the first query:

    1. //! Shows usage of transactions consisting of multiple data queries with client logic between them.
    2. static TStatus MultiStepTransaction(TSession session, const TString& path, ui64 seriesId, ui64 seasonId,
    3. TMaybe<TResultSet>& resultSet)
    4. {
    5. auto query1 = Sprintf(R"(
    6. PRAGMA TablePathPrefix("%s");
    7. DECLARE $seriesId AS Uint64;
    8. SELECT first_aired AS from_date FROM seasons
    9. WHERE series_id = $seriesId AND season_id = $seasonId;
    10. auto params1 = session.GetParamsBuilder()
    11. .AddParam("$seriesId")
    12. .Uint64(seriesId)
    13. .Build()
    14. .AddParam("$seasonId")
    15. .Uint64(seasonId)
    16. .Build()
    17. .Build();
    18. // Execute first query to get the required values to the client.
    19. // Transaction control settings don't set CommitTx flag to keep transaction active
    20. // after query execution.
    21. auto result = session.ExecuteDataQuery(
    22. query1,
    23. TTxControl::BeginTx(TTxSettings::SerializableRW()),
    24. std::move(params1)).GetValueSync();
    25. if (!result.IsSuccess()) {
    26. return result;
    27. }

    To continue working within the current transaction, you need to get the current transaction ID:

    1. // Get active transaction id
    2. auto tx = result.GetTransaction();
    3. TResultSetParser parser(result.GetResultSet(0));
    4. parser.TryNextRow();
    5. auto date = parser.ColumnParser("from_date").GetOptionalUint64();
    6. // Perform some client logic on returned values
    7. auto userFunc = [] (const TInstant fromDate) {
    8. return fromDate + TDuration::Days(15);
    9. };
    10. TInstant fromDate = TInstant::Days(*date);
    11. TInstant toDate = userFunc(fromDate);

    C++ - 图16

    The next step is to create the next query that uses the results of code execution on the client side:

    1. // Construct next query based on the results of client logic
    2. auto query2 = Sprintf(R"(
    3. PRAGMA TablePathPrefix("%s");
    4. DECLARE $seriesId AS Uint64;
    5. DECLARE $fromDate AS Uint64;
    6. DECLARE $toDate AS Uint64;
    7. SELECT season_id, episode_id, title, air_date FROM episodes
    8. WHERE series_id = $seriesId AND air_date >= $fromDate AND air_date <= $toDate;
    9. )", path.c_str());
    10. auto params2 = session.GetParamsBuilder()
    11. .AddParam("$seriesId")
    12. .Uint64(seriesId)
    13. .Build()
    14. .AddParam("$fromDate")
    15. .Uint64(fromDate.Days())
    16. .Build()
    17. .AddParam("$toDate")
    18. .Uint64(toDate.Days())
    19. .Build()
    20. .Build();
    21. // Execute second query.
    22. // Transaction control settings continues active transaction (tx) and
    23. // commits it at the end of second query execution.
    24. result = session.ExecuteDataQuery(
    25. query2,
    26. TTxControl::Tx(*tx).CommitTx(),
    27. std::move(params2)).GetValueSync();
    28. if (result.IsSuccess()) {
    29. resultSet = result.GetResultSet(0);
    30. }
    31. return result;
    32. }

    The given code snippets output the following text to the console at startup:

    1. > MultiStep:
    2. Episode 1, Season: 5, title: Grow Fast or Die Slow, Air date: Sun Mar 25, 2018
    3. Episode 3, Season: 5, title: Chief Operating Officer, Air date: Sun Apr 08, 2018

    C++ - 图18

    Managing transactions

    Transactions are managed through Begin and Commit calls.

    In most cases, instead of explicitly using Begin and Commit calls, it’s better to use transaction control parameters in execute calls. This helps you avoid unnecessary requests to YDB and run your queries more efficiently.

    Code snippet for BeginTransaction and calls: