xapi/mysqlx.cc (1,591 lines of code) (raw):

/* * Copyright (c) 2016, 2024, Oracle and/or its affiliates. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License, version 2.0, as * published by the Free Software Foundation. * * This program is designed to work with certain software (including * but not limited to OpenSSL) that is licensed under separate terms, as * designated in a particular file or component or in included license * documentation. The authors of MySQL hereby grant you an additional * permission to link the program and your derivative works with the * separately licensed software that they have either included with * the program or referenced in the documentation. * * Without limiting anything contained in the foregoing, this file, * which is part of Connector/C++, is also subject to the * Universal FOSS Exception, version 1.0, a copy of which can be found at * https://oss.oracle.com/licenses/universal-foss-exception. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * See the GNU General Public License, version 2.0, for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include <mysqlx/common.h> #include <mysqlx/xapi.h> #include "mysqlx_cc_internal.h" #include <stdlib.h> #include <string.h> #include <iostream> #include <stdexcept> using namespace mysqlx; using common::Value; using common::throw_error; #define SAFE_EXCEPTION_BEGIN(HANDLE, ERR) try { \ if (HANDLE == NULL) return ERR; #define SAFE_EXCEPTION_END(HANDLE, ERR) } \ catch(const cdk::Error &cdkerr) \ { \ HANDLE->set_diagnostic(cdkerr.what(), cdkerr.code().value()); \ return ERR; \ } \ catch(const Mysqlx_exception &mysqlx_ex) \ { \ HANDLE->set_diagnostic(mysqlx_ex); \ return ERR; \ } \ catch (const std::exception &ex) \ { \ HANDLE->set_diagnostic(ex.what(), 0); \ return ERR; \ } \ catch(...) \ { \ HANDLE->set_diagnostic("Unknown error!", MYSQLX_ERR_UNKNOWN); \ return ERR; \ } #define SAFE_EXCEPTION_SILENT_END(ERR) } catch(...) { return ERR; } #define HANDLE_SESSION_ERROR(OBJ,CODE,MSG) \ if (error) \ *error = new mysqlx_dyn_error_struct(MSG, CODE); \ if (OBJ) { delete OBJ; OBJ = NULL; } \ #define HANDLE_EXCEPTIONS(OBJ) \ catch(const cdk::Error &e) \ { HANDLE_SESSION_ERROR(OBJ, e.code().value(), e.what()); } \ catch(const Mysqlx_exception &e) \ { HANDLE_SESSION_ERROR(OBJ, e.code(), e.message().c_str()); } \ catch(const std::exception &e) \ { HANDLE_SESSION_ERROR(OBJ, 0, e.what()); } \ catch(...) \ { HANDLE_SESSION_ERROR(OBJ, 0, "Unknown error"); } #define PARAM_NULL_EMPTY_CHECK(PARAM, HANDLE, ERR_MSG, ERR) if (!PARAM || !(*PARAM)) \ { \ HANDLE->set_diagnostic(ERR_MSG, 0); \ return ERR; \ } #define PARAM_NULL_CHECK(PARAM, HANDLE, ERR_MSG, ERR) if (!PARAM) \ { \ HANDLE->set_diagnostic(ERR_MSG, 0); \ return ERR; \ } #define OUT_BUF_CHECK(PARAM, HANDLE, ERR_MSG, ERR) if (!PARAM) \ { \ HANDLE->set_diagnostic(ERR_MSG, 0); \ return ERR; \ } static mysqlx_client_struct * _get_client(const char *conn_str, const char *client_opt, mysqlx_error_t **error) { mysqlx_client_struct *cli = NULL; try { cli = new mysqlx_client_struct(conn_str, client_opt); } HANDLE_EXCEPTIONS(cli) return cli; } static mysqlx_client_struct * _get_client(mysqlx_session_options_t *opt, mysqlx_error_t **error) { mysqlx_client_struct *cli = NULL; try { if(!opt) { throw cdk::Error(0, "Client options structure not initialized"); } cli = new mysqlx_client_struct(opt); } HANDLE_EXCEPTIONS(cli) return cli; } static mysqlx_session_struct * _get_session(const char *host, unsigned short port, const char *user, const char *password, const char *database, const char *conn_str, mysqlx_error_t **error) { mysqlx_session_struct *sess = NULL; try { if(!conn_str) { std::string pwd(password ? password : ""); std::string db(database ? database : ""); // TODO: the default user has to be determined in a more flexible way sess = new mysqlx_session_struct(host ? host : "localhost", port, user ? user : "root", password ? &pwd : NULL, database ? &db : NULL); } else { sess = new mysqlx_session_struct(conn_str); } if (!sess->is_valid()) { const cdk::Error *err = sess->get_cdk_error(); if (err) throw *err; } } HANDLE_EXCEPTIONS(sess) return sess; } static mysqlx_session_struct * _get_session_opt(mysqlx_session_options_t *opt, mysqlx_error_t **error) { mysqlx_session_struct *sess = NULL; try { if(!opt) { throw cdk::Error(0, "Session options structure not initialized"); } sess = new mysqlx_session_struct(opt); if (!sess->is_valid()) { const cdk::Error *err = sess->get_cdk_error(); if (err) throw *err; } } HANDLE_EXCEPTIONS(sess) return sess; } /* Get client using the connection string */ mysqlx_client_struct * STDCALL mysqlx_get_client_from_url(const char *conn_string, const char *client_opts, mysqlx_error_t **error) { return _get_client(conn_string, client_opts, error); } /* Get client using options */ mysqlx_client_struct * STDCALL mysqlx_get_client_from_options(mysqlx_session_options_t *opt, mysqlx_error_t **error) { return _get_client(opt, error); } /* Get session from client pool */ mysqlx_session_t * STDCALL mysqlx_get_session_from_client(mysqlx_client_t *cli, mysqlx_error_t **error) { mysqlx_session_t *sess = nullptr; try { if (cli) { sess = new mysqlx_session_t(cli); } } HANDLE_EXCEPTIONS(sess) return sess; } /* Close client pool */ void mysqlx_client_close(mysqlx_client_t *cli) { try { delete cli; } catch (...) { // Ignore errors that might happen during client/session destruction/close. } } /* Establish the X session using string options provided as function parameters */ mysqlx_session_struct * STDCALL mysqlx_get_session(const char *host, int port, const char *user, const char *password, const char *database, mysqlx_error_t **error) { return _get_session(host, port, user, password, database, NULL, error); } /* Establish the X session using the connection string */ mysqlx_session_struct * STDCALL mysqlx_get_session_from_url(const char *conn_string, mysqlx_error_t **error) { return _get_session(NULL, 0, NULL, NULL, NULL, conn_string, error); } /* Establish the X session using the options structure */ mysqlx_session_struct * STDCALL mysqlx_get_session_from_options(mysqlx_session_options_t *opt, mysqlx_error_t **error) { return _get_session_opt(opt, error); } /* Execute a plain SQL query (supports parameters and placeholders) PARAMETERS: sess - pointer to the current session query - SQL query length - length of the query RETURN: statement handler containing the results and/or error NULL is not supposed to be returned even in case of error. It is very unlikely for this function to end with an error because it does not do any parsing, parameter checking etc. */ mysqlx_stmt_struct * STDCALL mysqlx_sql_new(mysqlx_session_struct *sess, const char *query, uint32_t length) { SAFE_EXCEPTION_BEGIN(sess, NULL) return sess->sql_query(query, length); SAFE_EXCEPTION_END(sess, NULL) } /* Function for binding values for parametrized queries. PARAMETERS: stmt - pointer to CRUD ... - variadic arguments that follow in sequences like type1, value1, type2, value2, ..., type_n, value_n, PARAM_END (PARAM_END marks the end of parameters list) RETURN: RESULT_OK - on success RESULT_ERROR - on error NOTE: Each new call resets the binds set by the previous call to mysqlx_stmt_struct::sql_bind() */ int mysqlx_stmt_bind(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) int res = RESULT_OK; va_list args; va_start(args, stmt); /* Processing of SQL parameters and parameters for other statement operations is different. Therefore, use two different bind functions depending on the operation type. */ if (stmt->op_type() == OP_SQL) res = stmt->sql_bind(args); else res = stmt->param_bind(args); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int mysqlx_set_insert_row(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) int res = RESULT_OK; va_list args; va_start(args, stmt); // Just row data, no columns in args res = stmt->add_row(false, args); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int mysqlx_set_insert_columns(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) int res = RESULT_OK; va_list args; va_start(args, stmt); res = stmt->add_columns(args); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int STDCALL mysqlx_set_add_document(mysqlx_stmt_struct *stmt, const char *json_doc) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(json_doc, stmt, MYSQLX_ERROR_EMPTY_JSON, RESULT_ERROR) return stmt->add_document(json_doc); SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } mysqlx_stmt_struct * STDCALL mysqlx_table_select_new(mysqlx_table_struct *table) { SAFE_EXCEPTION_BEGIN(table, NULL) return table->get_session().new_stmt<OP_SELECT>(*table); SAFE_EXCEPTION_END(table, NULL) } mysqlx_stmt_struct * STDCALL mysqlx_table_insert_new(mysqlx_table_struct *table) { SAFE_EXCEPTION_BEGIN(table, NULL) return table->get_session().new_stmt<OP_INSERT>(*table); SAFE_EXCEPTION_END(table, NULL) } mysqlx_stmt_struct * STDCALL mysqlx_table_update_new(mysqlx_table_struct *table) { SAFE_EXCEPTION_BEGIN(table, NULL) return table->get_session().new_stmt<OP_UPDATE>(*table); SAFE_EXCEPTION_END(table, NULL) } mysqlx_stmt_struct * STDCALL mysqlx_table_delete_new(mysqlx_table_struct *table) { SAFE_EXCEPTION_BEGIN(table, NULL) return table->get_session().new_stmt<OP_DELETE>(*table); SAFE_EXCEPTION_END(table, NULL) } mysqlx_stmt_struct * STDCALL mysqlx_collection_add_new(mysqlx_collection_struct *collection) { SAFE_EXCEPTION_BEGIN(collection, NULL) return collection->get_session().new_stmt<OP_ADD>(*collection); SAFE_EXCEPTION_END(collection, NULL) } mysqlx_stmt_struct * STDCALL mysqlx_collection_modify_new(mysqlx_collection_struct *collection) { SAFE_EXCEPTION_BEGIN(collection, NULL) return collection->get_session().new_stmt<OP_MODIFY>(*collection); SAFE_EXCEPTION_END(collection, NULL) } mysqlx_stmt_struct * STDCALL mysqlx_collection_remove_new(mysqlx_collection_struct *collection) { SAFE_EXCEPTION_BEGIN(collection, NULL) return collection->get_session().new_stmt<OP_REMOVE>(*collection); SAFE_EXCEPTION_END(collection, NULL) } mysqlx_stmt_struct * STDCALL mysqlx_collection_find_new(mysqlx_collection_struct *collection) { SAFE_EXCEPTION_BEGIN(collection, NULL) return collection->get_session().new_stmt<OP_FIND>(*collection); SAFE_EXCEPTION_END(collection, NULL) } int STDCALL mysqlx_set_where(mysqlx_stmt_struct *stmt, const char *where_expr) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) return stmt->set_where(where_expr); SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int STDCALL mysqlx_set_having(mysqlx_stmt_struct *stmt, const char *having_expr) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) return stmt->set_having(having_expr); SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int STDCALL mysqlx_set_group_by(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) va_list args; int res = RESULT_OK; va_start(args, stmt); res = stmt->add_group_by(args); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int STDCALL mysqlx_set_limit_and_offset(mysqlx_stmt_struct *stmt, uint64_t row_count, uint64_t offset) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) return stmt->set_limit(row_count, offset); SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int mysqlx_set_row_locking(mysqlx_stmt_t *stmt, int locking, int contention) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) stmt->set_row_locking((mysqlx_row_locking_t)locking, (mysqlx_lock_contention_t)contention); return RESULT_OK; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } /* Set ORDER BY clause for statement operation Operations supported by this function: SELECT, FIND, UPDATE, MODIFY, DELETE, REMOVE Calling it for INSERT or ADD will result in an error PARAMETERS: stmt - pointer to statement structure ... - variable parameters list consisting of (expression, direction) pairs terminated by PARAM_END: expr_1, direction_1, ..., expr_n, direction_n, PARAM_END (PARAM_END marks the end of parameters list) Each expression computes value used to sort the rows/documents in ascending or descending order, as determined by direction constant (list the direction enum names). RETURN: RESULT_OK - on success RESULT_ERROR - on error NOTE: this function is not supposed to be used directly. For SELECT operation the user code should use mysqlx_set_select_xxxx() macros that map the corresponding mysqlx_set_xxxx() functions. This way the unsupported operations will not be used. Eeach call of this function replaces previously set ORDER BY */ int STDCALL mysqlx_set_order_by(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) va_list args; int res = RESULT_OK; va_start(args, stmt); res = stmt->add_order_by(args); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int STDCALL mysqlx_set_items(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) va_list args; int res = RESULT_OK; va_start(args, stmt); res = stmt->add_projections(args); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int STDCALL mysqlx_set_find_projection( mysqlx_stmt_struct *stmt, const char *proj ) { /* The call to mysqlx_set_items will take care of exceptions and operations validity */ return mysqlx_set_items(stmt, proj, PARAM_END); } /* Execute a statement. PARAMETERS: stmt - pointer to statement structure RETURN: A MYSQL_RESULT handle that can be used to access results of the operation. Returned handle is valid until the CRUD handle is freed (when session is closed or explicitly with mysqlx_free()) or until another call to mysqlx_execute() on the same statement handle is made. It is also possible to close a RESULT hanlde and free all resources used by it earlier with mysqlx_result_free() call. On error NULL is returned. */ mysqlx_result_struct * STDCALL mysqlx_execute(mysqlx_stmt_struct *stmt) { SAFE_EXCEPTION_BEGIN(stmt, NULL) if (!stmt || !stmt->session_valid() || stmt->get_error()) return NULL; return stmt->exec(); SAFE_EXCEPTION_END(stmt, NULL) } int STDCALL mysqlx_set_update_values(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) va_list args; int res = RESULT_OK; va_start(args, stmt); res = stmt->add_table_update_values(args); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int STDCALL mysqlx_set_modify_set(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) va_list args; int res = RESULT_OK; va_start(args, stmt); res = stmt->add_coll_modify_values(args, MODIFY_SET); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int STDCALL mysqlx_set_modify_unset(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) va_list args; int res = RESULT_OK; va_start(args, stmt); res = stmt->add_coll_modify_values(args, MODIFY_UNSET); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int STDCALL mysqlx_set_modify_array_insert(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) va_list args; int res = RESULT_OK; va_start(args, stmt); res = stmt->add_coll_modify_values(args, MODIFY_ARRAY_INSERT); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int STDCALL mysqlx_set_modify_array_append(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) va_list args; int res = RESULT_OK; va_start(args, stmt); res = stmt->add_coll_modify_values(args, MODIFY_ARRAY_APPEND); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } int STDCALL mysqlx_set_modify_array_delete(mysqlx_stmt_struct *stmt, ...) { SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) va_list args; int res = RESULT_OK; va_start(args, stmt); res = stmt->add_coll_modify_values(args, MODIFY_ARRAY_DELETE); va_end(args); return res; SAFE_EXCEPTION_END(stmt, RESULT_ERROR) } static int _mysqlx_set_modify_patch(mysqlx_stmt_struct *stmt, ...) { int res = RESULT_OK; va_list args; SAFE_EXCEPTION_BEGIN(stmt, RESULT_ERROR) va_start(args, stmt); res = stmt->add_coll_modify_values(args, MODIFY_MERGE_PATCH); va_end(args); SAFE_EXCEPTION_END(stmt, RESULT_ERROR) return res; } int STDCALL mysqlx_set_modify_patch(mysqlx_stmt_struct *stmt, const char *patch_spec) { return _mysqlx_set_modify_patch(stmt, patch_spec); } /* Fetch one row from the result and advance to the next row PARAMETERS: res - pointer to the result structure RETURN: pointer to mysqlx_row_struct or NULL if no more rows left */ mysqlx_row_struct * STDCALL mysqlx_row_fetch_one(mysqlx_result_struct *res) { SAFE_EXCEPTION_BEGIN(res, NULL) return res->read_row(); SAFE_EXCEPTION_END(res, NULL) } //mysqlx_doc_struct * STDCALL mysqlx_doc_fetch_one(mysqlx_result_struct *res) //{ // SAFE_EXCEPTION_BEGIN(res, NULL) // return res->read_doc(); // SAFE_EXCEPTION_END(res, NULL) //} const char * STDCALL mysqlx_json_fetch_one(mysqlx_result_struct *res, size_t *length) { SAFE_EXCEPTION_BEGIN(res, NULL) return res->read_json(length); SAFE_EXCEPTION_END(res, NULL) } int STDCALL _store_result(mysqlx_result_struct *result, size_t *num, bool no_data_error) { SAFE_EXCEPTION_BEGIN(result, RESULT_ERROR) if (no_data_error && !result->has_data()) throw Mysqlx_exception("Attempt to store data for result without a data set"); cdk::row_count_t row_num = result->count(); if (num) *num = row_num; return RESULT_OK; SAFE_EXCEPTION_END(result, RESULT_ERROR) } int STDCALL mysqlx_store_result(mysqlx_result_struct *result, size_t *num) { return _store_result(result, num, true); } int STDCALL mysqlx_get_count(mysqlx_result_struct *result, size_t *num) { return _store_result(result, num, false); } /* Accessing row fields ------------------------------------------------------------------------- */ using impl::common::bytes; #define CHECK_COLUMN_RANGE(COL, ROW) if (COL >= ROW->col_count()) \ { \ ROW->set_diagnostic(MYSQLX_ERROR_INDEX_OUT_OF_RANGE_MSG, \ MYSQLX_ERROR_INDEX_OUT_OF_RANGE); \ return RESULT_ERROR; \ } int STDCALL mysqlx_get_bytes( mysqlx_row_struct* row, uint32_t col, uint64_t offset, void *buf, size_t *buf_len ) { SAFE_EXCEPTION_BEGIN(row, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(buf_len, row, MYSQLX_ERROR_OUTPUT_BUFFER_ZERO, RESULT_ERROR) OUT_BUF_CHECK(buf, row, MYSQLX_ERROR_OUTPUT_BUFFER_NULL, RESULT_ERROR) CHECK_COLUMN_RANGE(col, row) bytes b = row->get_bytes(col); int rc = RESULT_OK; if (b.size() == 0) return RESULT_NULL; if (offset >= b.size()) { // Report 0 bytes written in the buffer and do nothing else *buf_len = 0; return RESULT_OK; } if (b.size() - offset < *buf_len) *buf_len = b.size() - (size_t)offset; else rc = RESULT_MORE_DATA; memcpy(buf, b.begin() + offset, *buf_len); return rc; SAFE_EXCEPTION_END(row, RESULT_ERROR) } int STDCALL mysqlx_get_uint(mysqlx_row_struct* row, uint32_t col, uint64_t *val) { SAFE_EXCEPTION_BEGIN(row, RESULT_ERROR) OUT_BUF_CHECK(val, row, MYSQLX_ERROR_OUTPUT_BUFFER_NULL, RESULT_ERROR) CHECK_COLUMN_RANGE(col, row) Value &v = row->get(col); if (v.is_null()) return RESULT_NULL; *val = v.get_uint(); return RESULT_OK; SAFE_EXCEPTION_END(row, RESULT_ERROR) } int STDCALL mysqlx_get_sint(mysqlx_row_struct* row, uint32_t col, int64_t *val) { SAFE_EXCEPTION_BEGIN(row, RESULT_ERROR) OUT_BUF_CHECK(val, row, MYSQLX_ERROR_OUTPUT_BUFFER_NULL, RESULT_ERROR) CHECK_COLUMN_RANGE(col, row) Value &v = row->get(col); if (v.is_null()) return RESULT_NULL; *val = v.get_sint(); return RESULT_OK; SAFE_EXCEPTION_END(row, RESULT_ERROR) } int STDCALL mysqlx_get_float(mysqlx_row_struct* row, uint32_t col, float *val) { SAFE_EXCEPTION_BEGIN(row, RESULT_ERROR) OUT_BUF_CHECK(val, row, MYSQLX_ERROR_OUTPUT_BUFFER_NULL, RESULT_ERROR) CHECK_COLUMN_RANGE(col, row) Value &v = row->get(col); if (v.is_null()) return RESULT_NULL; if (Value::FLOAT == v.get_type()) { *val = v.get_float(); } else { double vd = v.get_double(); if ( vd > std::numeric_limits<float>::max() || vd < std::numeric_limits<float>::lowest() ) throw Mysqlx_exception("Numeric overflow"); *val = static_cast<float>(vd); } return RESULT_OK; SAFE_EXCEPTION_END(row, RESULT_ERROR) } int STDCALL mysqlx_get_double(mysqlx_row_struct* row, uint32_t col, double *val) { SAFE_EXCEPTION_BEGIN(row, RESULT_ERROR) OUT_BUF_CHECK(val, row, MYSQLX_ERROR_OUTPUT_BUFFER_NULL, RESULT_ERROR) CHECK_COLUMN_RANGE(col, row) Value &v = row->get(col); if (v.is_null()) return RESULT_NULL; *val = v.get_double(); return RESULT_OK; SAFE_EXCEPTION_END(row, RESULT_ERROR) } /* Get the number of columns in the result PARAMETERS: res - pointer to the result structure RETURN: the number of columns */ uint32_t STDCALL mysqlx_column_get_count(mysqlx_result_struct *res) { SAFE_EXCEPTION_BEGIN(res, 0) return res->get_col_count(); SAFE_EXCEPTION_END(res, 0) } /* Get column name PARAMETERS: res - pointer to the result structure pos - zero-based column number RETURN: character string containing column name */ const char * STDCALL mysqlx_column_get_name( mysqlx_result_struct *res, uint32_t pos ) { SAFE_EXCEPTION_BEGIN(res, NULL) return res->get_column(pos).m_label.c_str(); SAFE_EXCEPTION_END(res, NULL) } /* Get column original name PARAMETERS: res - pointer to the result structure pos - zero-based column number RETURN: character string containing column original name */ const char * STDCALL mysqlx_column_get_original_name( mysqlx_result_struct *res, uint32_t pos ) { SAFE_EXCEPTION_BEGIN(res, NULL) return res->get_column(pos).m_name.c_str(); SAFE_EXCEPTION_END(res, NULL) } /* Get column table name PARAMETERS: res - pointer to the result structure pos - zero-based column number RETURN: character string containing column table name */ const char * STDCALL mysqlx_column_get_table( mysqlx_result_struct *res, uint32_t pos ) { SAFE_EXCEPTION_BEGIN(res, NULL) return res->get_column(pos).m_table_label.c_str(); SAFE_EXCEPTION_END(res, NULL) } /* Get column original table name PARAMETERS: res - pointer to the result structure pos - zero-based column number RETURN: character string containing column original table name */ const char * STDCALL mysqlx_column_get_original_table( mysqlx_result_struct *res, uint32_t pos ) { SAFE_EXCEPTION_BEGIN(res, NULL) return res->get_column(pos).m_table_name.c_str(); SAFE_EXCEPTION_END(res, NULL) } /* Get column schema name PARAMETERS: res - pointer to the result structure pos - zero-based column number RETURN: character string containing column schema name */ const char * STDCALL mysqlx_column_get_schema( mysqlx_result_struct *res, uint32_t pos ) { SAFE_EXCEPTION_BEGIN(res, NULL) return res->get_column(pos).m_schema_name.c_str(); SAFE_EXCEPTION_END(res, NULL) } /* Get column catalog name PARAMETERS: res - pointer to the result structure pos - zero-based column number RETURN: character string containing column catalog name */ const char * STDCALL mysqlx_column_get_catalog( mysqlx_result_struct *res, uint32_t pos ) { SAFE_EXCEPTION_BEGIN(res, NULL) return res->get_column(pos).m_catalog.c_str(); SAFE_EXCEPTION_END(res, NULL) } /* Get column type identifier PARAMETERS: res - pointer to the result structure pos - zero-based column number RETURN: 16-bit unsigned int number with the column type identifier */ uint16_t STDCALL mysqlx_column_get_type(mysqlx_result_struct *res, uint32_t pos) { SAFE_EXCEPTION_BEGIN(res, MYSQLX_TYPE_UNDEFINED) return get_type(res->get_column(pos)); SAFE_EXCEPTION_END(res, MYSQLX_TYPE_UNDEFINED) } /* Get column collation number PARAMETERS: res - pointer to the result structure pos - zero-based column number RETURN: 16-bit unsigned int number with the column collation number */ uint16_t STDCALL mysqlx_column_get_collation( mysqlx_result_struct *res, uint32_t pos ) { SAFE_EXCEPTION_BEGIN(res, MYSQLX_COLLATION_UNDEFINED) return res->get_column(pos).m_collation; SAFE_EXCEPTION_END(res, MYSQLX_COLLATION_UNDEFINED) } /* Get column length PARAMETERS: res - pointer to the result structure pos - zero-based column number RETURN: 32-bit unsigned int number indicating the maximum data length */ uint32_t STDCALL mysqlx_column_get_length( mysqlx_result_struct *res, uint32_t pos ) { SAFE_EXCEPTION_BEGIN(res, 0) return res->get_column(pos).m_length; SAFE_EXCEPTION_END(res, 0) } /* Get column precision PARAMETERS: res - pointer to the result structure pos - zero-based column number RETURN: 16-bit unsigned int number of digits after the decimal point */ uint16_t STDCALL mysqlx_column_get_precision( mysqlx_result_struct *res, uint32_t pos ) { SAFE_EXCEPTION_BEGIN(res, 0) return res->get_column(pos).m_decimals; SAFE_EXCEPTION_END(res, 0) } /* Get number of rows affected by the last operation PARAMETERS: res - pointer to the result structure RETURN: 64-bit unsigned int number containing the number of affected rows */ uint64_t STDCALL mysqlx_get_affected_count(mysqlx_result_struct *res) { SAFE_EXCEPTION_BEGIN(res, 0) return res->get_affected_rows(); SAFE_EXCEPTION_END(res, 0) } /* Free the objects such as statement, session options, error and result. */ void STDCALL mysqlx_free(void *objs) { if (objs) { Mysqlx_diag_base *obj = (Mysqlx_diag_base*)objs; if (typeid(*obj) == typeid(mysqlx_stmt_struct)) { mysqlx_stmt_struct *stmt = (mysqlx_stmt_struct*)obj; stmt->get_session().rm_stmt(stmt); } else if (typeid(*obj) == typeid(mysqlx_session_options_struct)) { mysqlx_free_options((mysqlx_session_options_struct*)obj); } else if (typeid(*obj) == typeid(mysqlx_result_struct)) { mysqlx_result_free((mysqlx_result_struct*)obj); } else if (typeid(*obj) == typeid(mysqlx_dyn_error_struct) || typeid(*obj) == typeid(mysqlx_collection_options_struct)) { delete obj; } } } int STDCALL mysqlx_next_result(mysqlx_result_struct *res) { SAFE_EXCEPTION_BEGIN(res, RESULT_ERROR) return res->next_result() ? RESULT_OK : RESULT_NULL; SAFE_EXCEPTION_END(res, RESULT_ERROR) } /* Free the result explicitly, otherwise it will be done automatically when statement handler is destroyed */ void STDCALL mysqlx_result_free(mysqlx_result_struct *res) { if (res && res->m_stmt) { auto *stmt = res->m_stmt; try { stmt->rm_result(res); } catch (...) {} } } /* Closing the session. This function must be called by the user to prevent memory leaks. */ void STDCALL mysqlx_session_close(mysqlx_session_struct *sess) { if (sess) { try { delete sess; } catch (...) { // Ignore errors that might happen during session destruction. } } } int STDCALL mysqlx_schema_create(mysqlx_session_struct *sess, const char *schema) { SAFE_EXCEPTION_BEGIN(sess, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(schema, sess, MYSQLX_ERROR_MISSING_SCHEMA_NAME_MSG, RESULT_ERROR) sess->create_schema(schema); return RESULT_OK; SAFE_EXCEPTION_END(sess, RESULT_ERROR) } int STDCALL mysqlx_schema_drop(mysqlx_session_struct *sess, const char *schema) { SAFE_EXCEPTION_BEGIN(sess, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(schema, sess, MYSQLX_ERROR_MISSING_SCHEMA_NAME_MSG, RESULT_ERROR) sess->drop_schema(schema); return RESULT_OK; SAFE_EXCEPTION_END(sess, RESULT_ERROR) } int STDCALL mysqlx_collection_create(mysqlx_schema_struct *schema, const char *collection) { SAFE_EXCEPTION_BEGIN(schema, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(collection, schema, MYSQLX_ERROR_MISSING_COLLECTION_NAME_MSG, RESULT_ERROR) schema->create_collection(collection, true); return RESULT_OK; SAFE_EXCEPTION_END(schema, RESULT_ERROR) } mysqlx_collection_options_t * STDCALL mysqlx_collection_options_new() { return new mysqlx_collection_options_struct(); } template<mysqlx_collection_opt_enum OPT> void set_collection_opt(mysqlx_collection_options_t&, va_list&) { throw Mysqlx_exception("Unexpected collection option"); } template<> void set_collection_opt<MYSQLX_OPT_COLLECTION_REUSE>( mysqlx_collection_options_t &options, va_list &args) { if(options.m_usage.test(mysqlx_collection_options_t::REUSE)) throw Mysqlx_exception("Option reuse already set."); options.m_usage.set(mysqlx_collection_options_t::REUSE); options.m_reuse = va_arg(args, unsigned int); } template<> void set_collection_opt<MYSQLX_OPT_COLLECTION_VALIDATION>( mysqlx_collection_options_t &options, va_list &args) { if(options.m_usage.test(mysqlx_collection_options_t::VALIDATION) || options.m_usage.test(mysqlx_collection_options_t::VALIDATION_LEVEL) || options.m_usage.test(mysqlx_collection_options_t::VALIDATION_SCHEMA)) throw Mysqlx_exception("Collection validation already set."); options.m_usage.set(mysqlx_collection_options_t::VALIDATION); options.m_validation = va_arg(args, char*); } template <mysqlx_collection_validation_opt_enum OPT> void set_collection_validation_opt(mysqlx_collection_options_t&, va_list&) { throw Mysqlx_exception("Unexpected collection validation option"); } template <> void set_collection_validation_opt<MYSQLX_OPT_COLLECTION_VALIDATION_LEVEL>( mysqlx_collection_options_t &options, va_list &args) { if(options.m_usage.test(mysqlx_collection_options_t::VALIDATION) || options.m_usage.test(mysqlx_collection_options_t::VALIDATION_LEVEL)) throw Mysqlx_exception("Validation level already set."); options.m_usage.set(mysqlx_collection_options_t::VALIDATION_LEVEL); #define SCHEMA_VALIDATION_LEVEL_CASE(X,Y)\ case MYSQLX_OPT_COLLECTION_VALIDATION_LEVEL_##X:\ options.m_validation_level = #X;\ break; switch (va_arg(args, int)) { COLLECTION_VALIDATION_LEVEL(SCHEMA_VALIDATION_LEVEL_CASE) } } template<> void set_collection_validation_opt<MYSQLX_OPT_COLLECTION_VALIDATION_SCHEMA>( mysqlx_collection_options_t &options, va_list &args) { if(options.m_usage.test(mysqlx_collection_options_t::VALIDATION) || options.m_usage.test(mysqlx_collection_options_t::VALIDATION_SCHEMA)) throw Mysqlx_exception("Validation schema already set."); options.m_usage.set(mysqlx_collection_options_t::VALIDATION_SCHEMA); options.m_validation_schema = va_arg(args, const char*); } int STDCALL mysqlx_collection_options_set(mysqlx_collection_options_t * options,...) { SAFE_EXCEPTION_BEGIN(options, RESULT_ERROR) va_list args; mysqlx_collection_options_t tmp_options(*options); int type; try { va_start(args, options); while (0 != (type = va_arg(args, int))) { switch (type) { #define COLLECTION_OPTIONS_OPTION_SET(X,Y)\ case MYSQLX_OPT_COLLECTION_##X:\ set_collection_opt<MYSQLX_OPT_COLLECTION_##X>(tmp_options, args); \ break; COLLECTION_OPTIONS_OPTION(COLLECTION_OPTIONS_OPTION_SET) #define COLLECTION_VALIDATION_OPTION_SET(X,Y)\ case MYSQLX_OPT_COLLECTION_VALIDATION_##X:\ set_collection_validation_opt<MYSQLX_OPT_COLLECTION_VALIDATION_##X>(tmp_options, args);\ break; COLLECTION_VALIDATION_OPTION(COLLECTION_VALIDATION_OPTION_SET) default: throw Mysqlx_exception("Unrecognized option"); } } va_end(args); } catch (...) { va_end(args); throw; } *options = tmp_options; return RESULT_OK; SAFE_EXCEPTION_END(options, RESULT_ERROR) } int STDCALL mysqlx_collection_create_with_options(mysqlx_schema_t *schema, const char *collection, mysqlx_collection_options_t *options) { SAFE_EXCEPTION_BEGIN(schema, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(collection, schema, MYSQLX_ERROR_MISSING_COLLECTION_NAME_MSG, RESULT_ERROR) PARAM_NULL_CHECK(options, schema, MYSQLX_ERROR_MISSING_COLLECTION_OPT_MSG, RESULT_ERROR) if( options->m_usage.test(mysqlx_collection_options_t::VALIDATION) ) { schema->create_collection(collection, options->m_reuse, options->m_validation); } else { schema->create_collection(collection, options->m_reuse, options->m_validation_level, options->m_validation_schema); } return RESULT_OK; SAFE_EXCEPTION_END(schema, RESULT_ERROR) } int STDCALL mysqlx_collection_create_with_json_options(mysqlx_schema_t *schema, const char *collection, const char* json_options) { SAFE_EXCEPTION_BEGIN(schema, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(collection, schema, MYSQLX_ERROR_MISSING_COLLECTION_NAME_MSG, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(json_options, schema, MYSQLX_ERROR_MISSING_COLLECTION_NAME_MSG, RESULT_ERROR) schema->create_collection(collection, std::string(json_options)); return RESULT_OK; SAFE_EXCEPTION_END(schema, RESULT_ERROR) } int STDCALL mysqlx_collection_modify_with_options(mysqlx_schema_t *schema, const char *collection, mysqlx_collection_options_t *options) { SAFE_EXCEPTION_BEGIN(schema, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(collection, schema, MYSQLX_ERROR_MISSING_COLLECTION_NAME_MSG, RESULT_ERROR) PARAM_NULL_CHECK(options, schema, MYSQLX_ERROR_MISSING_COLLECTION_OPT_MSG, RESULT_ERROR) if(options->m_reuse) throw_error("Can't use OPT_COLLECTION_REUSE mysqlx_collection_modify_with_options"); if(options->m_validation.empty()) { schema->modify_collection(collection, options->m_validation_level, options->m_validation_schema); } else { schema->modify_collection(collection, options->m_validation, true); } return RESULT_OK; SAFE_EXCEPTION_END(schema, RESULT_ERROR) } int STDCALL mysqlx_collection_modify_with_json_options(mysqlx_schema_t *schema, const char* collection, const char* json_options) { SAFE_EXCEPTION_BEGIN(schema, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(collection, schema, MYSQLX_ERROR_MISSING_COLLECTION_NAME_MSG, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(json_options, schema, MYSQLX_ERROR_MISSING_COLLECTION_NAME_MSG, RESULT_ERROR) schema->modify_collection(collection, std::string(json_options)); return RESULT_OK; SAFE_EXCEPTION_END(schema, RESULT_ERROR) } int STDCALL mysqlx_collection_drop(mysqlx_schema_struct *schema, const char *collection) { SAFE_EXCEPTION_BEGIN(schema, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(collection, schema, MYSQLX_ERROR_MISSING_COLLECTION_NAME_MSG, RESULT_ERROR) schema->drop_collection(collection); return RESULT_OK; SAFE_EXCEPTION_END(schema, RESULT_ERROR) } /* STMT will be unavailable outside the function, set session error to the upper level object and return */ #define SET_ERROR_FROM_STMT(OBJ, STMT, R) do { \ mysqlx_error_struct *err = STMT->get_error(); \ if (err) \ OBJ->set_diagnostic(err->message(), err->error_num()); \ else \ OBJ->set_diagnostic("Unknown error!", 0); \ return R; \ } while (0) mysqlx_result_struct * STDCALL mysqlx_sql(mysqlx_session_struct *sess, const char *query, size_t query_len) { SAFE_EXCEPTION_BEGIN(sess, NULL) mysqlx_stmt_struct *stmt = sess->sql_query(query, query_len); mysqlx_result_struct *res = mysqlx_execute(stmt); if (res == NULL) SET_ERROR_FROM_STMT(sess, stmt, NULL); return res; SAFE_EXCEPTION_END(sess, NULL) } mysqlx_result_struct * STDCALL mysqlx_sql_param(mysqlx_session_struct *sess, const char *query, size_t query_len, ...) { SAFE_EXCEPTION_BEGIN(sess, NULL) int rc = RESULT_OK; mysqlx_stmt_struct *stmt; if ((stmt = sess->sql_query(query, query_len)) == NULL) return NULL; va_list args; va_start(args, query_len); rc = stmt->sql_bind(args); va_end(args); if (rc != RESULT_OK) SET_ERROR_FROM_STMT(sess, stmt, NULL); mysqlx_result_struct *res = mysqlx_execute(stmt); if (res == NULL) SET_ERROR_FROM_STMT(sess, stmt, NULL); return res; SAFE_EXCEPTION_END(sess, NULL) } mysqlx_result_struct * STDCALL mysqlx_table_select(mysqlx_table_struct *table, const char *criteria) { SAFE_EXCEPTION_BEGIN(table, NULL) mysqlx_stmt_struct *stmt; if ((stmt = mysqlx_table_select_new(table)) == NULL) return NULL; if (RESULT_OK != stmt->set_where(criteria)) SET_ERROR_FROM_STMT(table, stmt, NULL); mysqlx_result_struct *res = mysqlx_execute(stmt); if (res == NULL) SET_ERROR_FROM_STMT(table, stmt, NULL); return res; SAFE_EXCEPTION_END(table, NULL) } mysqlx_result_struct * STDCALL mysqlx_table_select_limit(mysqlx_table_struct *table, const char *criteria, uint64_t row_count, uint64_t offset, ...) { SAFE_EXCEPTION_BEGIN(table, NULL) mysqlx_stmt_struct *stmt; int rc = RESULT_OK; if ((stmt = mysqlx_table_select_new(table)) == NULL) return NULL; if (RESULT_OK != stmt->set_where(criteria)) SET_ERROR_FROM_STMT(table, stmt, NULL); if (RESULT_OK != stmt->set_limit(row_count, offset)) SET_ERROR_FROM_STMT(table, stmt, NULL); va_list args; va_start(args, offset); rc= stmt->add_order_by(args); va_end(args); if (rc != RESULT_OK) SET_ERROR_FROM_STMT(table, stmt, NULL); mysqlx_result_struct *res = mysqlx_execute(stmt); if (res == NULL) SET_ERROR_FROM_STMT(table, stmt, NULL); return res; SAFE_EXCEPTION_END(table, NULL) } mysqlx_result_struct * STDCALL mysqlx_table_insert(mysqlx_table_struct *table, ...) { SAFE_EXCEPTION_BEGIN(table, NULL) mysqlx_stmt_struct *stmt; int rc = RESULT_OK; if ((stmt = mysqlx_table_insert_new(table)) == NULL) return NULL; va_list args; va_start(args, table); /* Parameters are triplets: <column name, value type, value> */ rc= stmt->add_row(true, args); va_end(args); if (rc != RESULT_OK) SET_ERROR_FROM_STMT(table, stmt, NULL); mysqlx_result_struct *res = mysqlx_execute(stmt); if (res == NULL) SET_ERROR_FROM_STMT(table, stmt, NULL); return res; SAFE_EXCEPTION_END(table, NULL) } mysqlx_result_struct * STDCALL mysqlx_table_update(mysqlx_table_struct *table, const char *criteria, ...) { SAFE_EXCEPTION_BEGIN(table, NULL) mysqlx_stmt_struct *stmt; int rc = RESULT_OK; if ((stmt = mysqlx_table_update_new(table)) == NULL) return NULL; if (RESULT_OK != stmt->set_where(criteria)) SET_ERROR_FROM_STMT(table, stmt, NULL); va_list args; va_start(args, criteria); /* Parameters are triplets: <column name, value type, value> */ rc= stmt->add_table_update_values(args); va_end(args); if (rc != RESULT_OK) SET_ERROR_FROM_STMT(table, stmt, NULL); mysqlx_result_struct *res = mysqlx_execute(stmt); if (res == NULL) SET_ERROR_FROM_STMT(table, stmt, NULL); return res; SAFE_EXCEPTION_END(table, NULL) } mysqlx_result_struct * STDCALL mysqlx_table_delete(mysqlx_table_struct *table, const char *criteria) { SAFE_EXCEPTION_BEGIN(table, NULL) mysqlx_stmt_struct *stmt; if ((stmt = mysqlx_table_delete_new(table)) == NULL) return NULL; if (RESULT_OK != stmt->set_where(criteria)) SET_ERROR_FROM_STMT(table, stmt, NULL); mysqlx_result_struct *res = mysqlx_execute(stmt); if (res == NULL) SET_ERROR_FROM_STMT(table, stmt, NULL); return res; SAFE_EXCEPTION_END(table, NULL) } int STDCALL mysqlx_table_count(mysqlx_table_t *table, uint64_t *count) { SAFE_EXCEPTION_BEGIN(table, RESULT_ERROR) PARAM_NULL_CHECK(count, table, MYSQLX_ERROR_OUTPUT_VARIABLE_NULL, RESULT_ERROR); *count = table->count(); return RESULT_OK; SAFE_EXCEPTION_END(table, RESULT_ERROR) } int STDCALL mysqlx_collection_count(mysqlx_collection_t *collection, uint64_t *count) { SAFE_EXCEPTION_BEGIN(collection, RESULT_ERROR) PARAM_NULL_CHECK(count, collection, MYSQLX_ERROR_OUTPUT_VARIABLE_NULL, RESULT_ERROR); *count = collection->count(); return RESULT_OK; SAFE_EXCEPTION_END(collection, RESULT_ERROR) } mysqlx_result_struct * STDCALL mysqlx_collection_find(mysqlx_collection_struct *collection, const char *criteria) { SAFE_EXCEPTION_BEGIN(collection, NULL) if (!criteria) criteria = "true"; mysqlx_stmt_struct *stmt; if ((stmt = mysqlx_collection_find_new(collection)) == NULL) return NULL; if (RESULT_OK != stmt->set_where(criteria)) SET_ERROR_FROM_STMT(collection, stmt, NULL); mysqlx_result_struct *res = mysqlx_execute(stmt); if (res == NULL) SET_ERROR_FROM_STMT(collection, stmt, NULL); return res; SAFE_EXCEPTION_END(collection, NULL) } mysqlx_result_struct * STDCALL mysqlx_collection_add(mysqlx_collection_struct *collection, ...) { SAFE_EXCEPTION_BEGIN(collection, NULL) mysqlx_stmt_struct *stmt; int rc = RESULT_OK; if ((stmt = mysqlx_collection_add_new(collection)) == NULL) return NULL; va_list args; va_start(args, collection); rc= stmt->add_multiple_documents(args); va_end(args); if (rc != RESULT_OK) SET_ERROR_FROM_STMT(collection, stmt, NULL); mysqlx_result_struct *res = mysqlx_execute(stmt); if (res == NULL) SET_ERROR_FROM_STMT(collection, stmt, NULL); return res; SAFE_EXCEPTION_END(collection, NULL) } mysqlx_result_struct * STDCALL _mysqlx_collection_modify_exec(mysqlx_collection_struct *collection, const char *criteria, mysqlx_modify_op modify_op, va_list &args) { SAFE_EXCEPTION_BEGIN(collection, NULL) mysqlx_stmt_struct *stmt; int rc = RESULT_OK; if ((stmt = mysqlx_collection_modify_new(collection)) == NULL) return NULL; if (!criteria) criteria = "true"; if (RESULT_OK != stmt->set_where(criteria)) SET_ERROR_FROM_STMT(collection, stmt, NULL); rc= stmt->add_coll_modify_values(args, modify_op); if (rc != RESULT_OK) SET_ERROR_FROM_STMT(collection, stmt, NULL); mysqlx_result_struct *res = mysqlx_execute(stmt); if (res == NULL) SET_ERROR_FROM_STMT(collection, stmt, NULL); return res; SAFE_EXCEPTION_END(collection, NULL) } mysqlx_result_struct * STDCALL mysqlx_collection_modify_set(mysqlx_collection_struct *collection, const char *criteria, ...) { mysqlx_result_struct *res; va_list args; SAFE_EXCEPTION_BEGIN(collection, NULL) va_start(args, criteria); res = _mysqlx_collection_modify_exec(collection, criteria, MODIFY_SET, args); va_end(args); return res; SAFE_EXCEPTION_END(collection, NULL) } mysqlx_result_struct * STDCALL mysqlx_collection_modify_unset(mysqlx_collection_struct *collection, const char *criteria, ...) { mysqlx_result_struct *res; va_list args; SAFE_EXCEPTION_BEGIN(collection, NULL) va_start(args, criteria); res = _mysqlx_collection_modify_exec(collection, criteria, MODIFY_UNSET, args); va_end(args); return res; SAFE_EXCEPTION_END(collection, NULL) } static mysqlx_result_t * _mysqlx_collection_modify_patch(mysqlx_collection_t *collection, const char *criteria, ...) { mysqlx_result_t *res; va_list args; SAFE_EXCEPTION_BEGIN(collection, NULL) va_start(args, criteria); res = _mysqlx_collection_modify_exec(collection, criteria, MODIFY_MERGE_PATCH, args); va_end(args); return res; SAFE_EXCEPTION_END(collection, NULL) } mysqlx_result_t * STDCALL mysqlx_collection_modify_patch(mysqlx_collection_t *collection, const char *criteria, const char *patch_spec) { return _mysqlx_collection_modify_patch(collection, criteria, patch_spec); } mysqlx_result_struct * STDCALL mysqlx_collection_remove(mysqlx_collection_struct *collection, const char*criteria) { SAFE_EXCEPTION_BEGIN(collection, NULL) mysqlx_stmt_struct *stmt; if ((stmt = mysqlx_collection_remove_new(collection)) == NULL) return NULL; if (RESULT_OK != stmt->set_where(criteria)) SET_ERROR_FROM_STMT(collection, stmt, NULL); mysqlx_result_struct *res = mysqlx_execute(stmt); if (res == NULL) SET_ERROR_FROM_STMT(collection, stmt, NULL); return res; SAFE_EXCEPTION_END(collection, NULL) } mysqlx_result_struct * STDCALL mysqlx_get_tables(mysqlx_schema_struct *schema, const char *table_pattern, int show_views) { SAFE_EXCEPTION_BEGIN(schema, NULL) return schema->get_tables(table_pattern, 0 != show_views); SAFE_EXCEPTION_END(schema, NULL) } mysqlx_result_struct * STDCALL mysqlx_get_collections(mysqlx_schema_struct *schema, const char *col_pattern) { SAFE_EXCEPTION_BEGIN(schema, NULL) return schema->get_collections(col_pattern); SAFE_EXCEPTION_END(schema, NULL) } mysqlx_result_struct * STDCALL mysqlx_get_schemas(mysqlx_session_struct *sess, const char *schema_pattern) { SAFE_EXCEPTION_BEGIN(sess, NULL) return sess->get_schemas(schema_pattern); SAFE_EXCEPTION_END(sess, NULL) } unsigned int STDCALL mysqlx_result_warning_count(mysqlx_result_struct *result) { SAFE_EXCEPTION_BEGIN(result, 0) return (unsigned int)result->get_warning_count(); SAFE_EXCEPTION_END(result, 0) } mysqlx_error_struct * STDCALL mysqlx_result_next_warning(mysqlx_result_struct *result) { SAFE_EXCEPTION_BEGIN(result, 0) return result->get_next_warning(); SAFE_EXCEPTION_END(result, 0) } uint64_t STDCALL mysqlx_get_auto_increment_value(mysqlx_result_struct *res) { SAFE_EXCEPTION_BEGIN(res, 0) return res->get_auto_increment(); SAFE_EXCEPTION_END(res, 0) } int STDCALL mysqlx_transaction_begin(mysqlx_session_struct *sess) { SAFE_EXCEPTION_BEGIN(sess, RESULT_ERROR) sess->transaction_begin(); return RESULT_OK; SAFE_EXCEPTION_END(sess, RESULT_ERROR) } int STDCALL mysqlx_transaction_commit(mysqlx_session_struct *sess) { SAFE_EXCEPTION_BEGIN(sess, RESULT_ERROR) sess->transaction_commit(); return RESULT_OK; SAFE_EXCEPTION_END(sess, RESULT_ERROR) } int STDCALL mysqlx_transaction_rollback(mysqlx_session_struct *sess) { SAFE_EXCEPTION_BEGIN(sess, RESULT_ERROR) sess->transaction_rollback(nullptr); return RESULT_OK; SAFE_EXCEPTION_END(sess, RESULT_ERROR) } const char* STDCALL mysqlx_savepoint_set(mysqlx_session_t *sess, const char *name) { SAFE_EXCEPTION_BEGIN(sess, NULL) return sess->savepoint_set(name); SAFE_EXCEPTION_END(sess, NULL) } int STDCALL mysqlx_savepoint_release(mysqlx_session_t *sess, const char *name) { SAFE_EXCEPTION_BEGIN(sess, RESULT_ERROR) sess->savepoint_remove(name); return RESULT_OK; SAFE_EXCEPTION_END(sess, RESULT_ERROR) } int STDCALL mysqlx_rollback_to(mysqlx_session_t *sess, const char *name) { SAFE_EXCEPTION_BEGIN(sess, RESULT_ERROR) if (name == NULL || strlen(name) == 0) { sess->set_diagnostic("Invalid save point name", 0); return RESULT_ERROR; } sess->transaction_rollback(name); return RESULT_OK; SAFE_EXCEPTION_END(sess, RESULT_ERROR) } const char * STDCALL mysqlx_fetch_generated_id(mysqlx_result_struct *result) { SAFE_EXCEPTION_BEGIN(result, NULL) return result->get_next_generated_id(); SAFE_EXCEPTION_END(result, NULL) } int STDCALL mysqlx_session_valid(mysqlx_session_struct *sess) { SAFE_EXCEPTION_BEGIN(sess, 0) return sess->is_valid(); SAFE_EXCEPTION_END(sess, 0) } mysqlx_session_options_t * STDCALL mysqlx_session_options_new() { return new mysqlx_session_options_struct(); } void STDCALL mysqlx_free_options(mysqlx_session_options_t *opt) { if (opt) delete opt; } /* NULL as value of a string option means remove this option from settings. But empty string is not valid. */ template<mysqlx_opt_type_enum OPT> void check_option(const char *val) { if (val && !*val) throw Mysqlx_exception("Invalid empty string as value of option "); } /* On MYSQLX_OPT_COMPRESSION_ALGORITHMS it is allowed to have empty strings */ template<> void check_option<MYSQLX_OPT_COMPRESSION_ALGORITHMS>(const char *) { } /* In case of HOST and SOCKET settings, which accumulate, it is not possible to remove them by passing NULL. TODO: Consider if these checks are not better done in Settings_impl */ template<> void check_option<MYSQLX_OPT_HOST>(const char *val) { if (!val || !*val) throw Mysqlx_exception(MYSQLX_ERROR_MISSING_HOST_NAME); } template<> void check_option<MYSQLX_OPT_SOCKET>(const char *val) { if (!val || !*val) throw Mysqlx_exception(MYSQLX_ERROR_MISSING_SOCKET_NAME); } /* Since USER setting is compulsory, we do not allow removing it, one can only overwrite with a new value. */ template<> void check_option<MYSQLX_OPT_USER>(const char *val) { if (!val || !*val) throw Mysqlx_exception("Empty user name"); } /* It is OK to give empty string as a password. */ template<> void check_option<MYSQLX_OPT_PWD>(const char *) {} int STDCALL mysqlx_session_option_set(mysqlx_session_options_struct *opt, ...) { // Clear diagnostic information opt->Mysqlx_diag::clear(); SAFE_EXCEPTION_BEGIN(opt, RESULT_ERROR) va_list args; int type; uint64_t uint_data = 0; const char *char_data = NULL; using Option = mysqlx_session_options_struct::Session_option_impl; using COption = mysqlx_session_options_struct::Client_option_impl; mysqlx_session_options_struct::Setter set(*opt); try { va_start(args, opt); while (0 != (type = va_arg(args, int))) { switch (type) { #define OPT_SET_str(X,N) \ case N: \ { char_data = va_arg(args, char*); \ check_option<MYSQLX_OPT_##X>(char_data); \ if (!char_data) \ set.key_val(Option::X)->scalar()->null(); \ else \ set.key_val(Option::X)->scalar()->str(char_data); }; \ break; #define OPT_SET_num(X,N) \ case N: \ { \ uint_data = va_arg(args, unsigned); \ set.key_val(Option::X)->scalar()->num(uint_data); }; \ break; #define OPT_SET_any(X,N) OPT_SET_num(X,N) #define OPT_SET_bool(X,N) \ case N: \ { \ uint_data = va_arg(args, unsigned); \ set.key_val(Option::X)->scalar()->num(uint_data); }; \ break; SESSION_OPTION_LIST(OPT_SET) #define CLIENT_OPT_SET_bool(X,N) \ case -N: \ { uint_data = va_arg(args, int); \ set.key_val(COption::X)->scalar()->num(uint_data); }; \ break; #define CLIENT_OPT_SET_num(X,N) \ case -N: \ { uint_data = va_arg(args, uint64_t); \ set.key_val(COption::X)->scalar()->num(uint_data); }; \ break; CLIENT_OPTION_LIST(CLIENT_OPT_SET) default: throw Mysqlx_exception("Unrecognized option"); } } va_end(args); } catch (...) { va_end(args); throw; } set.commit(); return RESULT_OK; SAFE_EXCEPTION_END(opt, RESULT_ERROR) } #define CHECK_OUTPUT_BUF(V, T) V = va_arg(args, T); \ if (V == NULL) \ { \ opt->set_diagnostic(MYSQLX_ERROR_OUTPUT_BUFFER_NULL, 0); \ rc = RESULT_ERROR; \ break; \ } /* TODO: This function needs to be able to return information about hosts and corresponding parameters in the muliple host configurations. */ int STDCALL mysqlx_session_option_get( mysqlx_session_options_struct *opt, int type, ... ) { using Option = mysqlx_session_options_struct::Session_option_impl; SAFE_EXCEPTION_BEGIN(opt, RESULT_ERROR) if (!opt->has_option(mysqlx_opt_type_enum(type))) { opt->set_diagnostic("Option ... is not set", 0); return RESULT_ERROR; } int rc = RESULT_OK; unsigned int *uint_data = 0; char *char_data = NULL; va_list args; va_start(args, type); // Note: assumes the same enum values as used in Settings_impl::Option switch(type) { #define OPT_GET_str(X,N) \ case N: \ CHECK_OUTPUT_BUF(char_data, char*) \ strcpy(char_data, opt->get(Option::X).get_string().c_str()); \ break; #define OPT_GET_num(X,N) \ case N: \ { CHECK_OUTPUT_BUF(uint_data, unsigned int*) \ auto val = opt->get(Option::X).get_uint(); \ ASSERT_NUM_LIMITS(unsigned, val); \ *uint_data = (unsigned)val; }; break; #define OPT_GET_any(X,N) OPT_GET_num(X,N) #define OPT_GET_bool(X,N) OPT_GET_num(X,N) SESSION_OPTION_LIST(OPT_GET) default: opt->set_diagnostic("Invalid option value", 0); rc = RESULT_ERROR; } va_end(args); return rc; SAFE_EXCEPTION_END(opt, RESULT_ERROR) } mysqlx_schema_struct * STDCALL mysqlx_get_schema(mysqlx_session_struct *sess, const char *schema_name, unsigned int check) { SAFE_EXCEPTION_BEGIN(sess, NULL) PARAM_NULL_EMPTY_CHECK(schema_name, sess, MYSQLX_ERROR_MISSING_SCHEMA_NAME_MSG, NULL) return sess->get_schema(schema_name, check > 0); SAFE_EXCEPTION_END(sess, NULL) } mysqlx_collection_struct * STDCALL mysqlx_get_collection(mysqlx_schema_struct *schema, const char *col_name, unsigned int check) { SAFE_EXCEPTION_BEGIN(schema, NULL) PARAM_NULL_EMPTY_CHECK(col_name, schema, MYSQLX_ERROR_MISSING_COLLECTION_NAME_MSG, NULL) return schema->get_collection(col_name, check > 0); SAFE_EXCEPTION_END(schema, NULL) } mysqlx_table_struct * STDCALL mysqlx_get_table(mysqlx_schema_struct *schema, const char *tab_name, unsigned int check) { SAFE_EXCEPTION_BEGIN(schema, NULL) PARAM_NULL_EMPTY_CHECK(tab_name, schema, MYSQLX_ERROR_MISSING_TABLE_NAME_MSG, NULL) return schema->get_table(tab_name, check > 0); SAFE_EXCEPTION_END(schema, NULL) } mysqlx_error_struct * STDCALL mysqlx_error(void *obj) { Mysqlx_diag_base *diag = (Mysqlx_diag_base*)obj; SAFE_EXCEPTION_BEGIN(diag, NULL) return diag->get_error(); SAFE_EXCEPTION_SILENT_END(NULL) } const char * STDCALL mysqlx_error_message(void *obj) { if (!obj) return nullptr; mysqlx_error_struct *error = mysqlx_error(obj); if (error) { const char *c = error->message(); return c; } return nullptr; } unsigned int STDCALL mysqlx_error_num(void *obj) { mysqlx_error_struct *error = mysqlx_error(obj); if (error) return error->error_num(); return 0; } int STDCALL mysqlx_collection_create_index( mysqlx_collection_struct *coll, const char *name, const char *idx_json ) { SAFE_EXCEPTION_BEGIN(coll, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(name, coll, MYSQLX_ERROR_MISSING_COLLECTION_NAME_MSG, RESULT_ERROR) coll->create_index(name, idx_json); return RESULT_OK; SAFE_EXCEPTION_END(coll, RESULT_ERROR) } int STDCALL mysqlx_collection_drop_index( mysqlx_collection_struct *coll, const char *name ) { SAFE_EXCEPTION_BEGIN(coll, RESULT_ERROR) PARAM_NULL_EMPTY_CHECK(name, coll, MYSQLX_ERROR_MISSING_COLLECTION_NAME_MSG, RESULT_ERROR) coll->drop_index(name); return RESULT_OK; SAFE_EXCEPTION_END(coll, RESULT_ERROR) } #ifdef _WIN32 BOOL WINAPI DllMain( _In_ HINSTANCE, _In_ DWORD, _In_ LPVOID ) { return true; } #endif // TODO: add implementations for other mysqlx_xxxxxx functions