mirror of https://github.com/sysown/proxysql
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
186 lines
6.8 KiB
186 lines
6.8 KiB
/**
|
|
* @file test_ffto_mysql_errors-t.cpp
|
|
* @brief FFTO E2E TAP test -- MySQL error recording in stats_mysql_errors.
|
|
*
|
|
* Validates that errors occurring during MySQL fast-forward sessions
|
|
* are properly recorded in stats_mysql_errors with correct errno
|
|
* and error message.
|
|
*
|
|
* @par Test scenarios
|
|
* 1. Syntax error -> errno 1064 recorded
|
|
* 2. Table not found -> errno 1146 recorded
|
|
* 3. Recovery: successful query after error
|
|
*/
|
|
|
|
#include <string>
|
|
#include <stdio.h>
|
|
#include <cstring>
|
|
#include <unistd.h>
|
|
#include "mysql.h"
|
|
#include "tap.h"
|
|
#include "command_line.h"
|
|
#include "utils.h"
|
|
|
|
static constexpr int kPlannedTests = 7;
|
|
|
|
#define FAIL_AND_SKIP_REMAINING(cleanup_label, fmt, ...) \
|
|
do { \
|
|
diag(fmt, ##__VA_ARGS__); \
|
|
int remaining = kPlannedTests - tests_last(); \
|
|
if (remaining > 0) { \
|
|
skip(remaining, "Skipping remaining assertions after setup failure"); \
|
|
} \
|
|
goto cleanup_label; \
|
|
} while (0)
|
|
|
|
/**
|
|
* @brief Check if a specific errno appears in stats_mysql_errors.
|
|
* Polls up to 2 seconds. Returns true if found.
|
|
*/
|
|
static bool verify_mysql_error(MYSQL* admin, int expected_errno, const char* expected_msg_substr) {
|
|
char query[1024];
|
|
snprintf(query, sizeof(query),
|
|
"SELECT err_no, last_error FROM stats_mysql_errors WHERE err_no = %d",
|
|
expected_errno);
|
|
|
|
for (int attempt = 0; attempt < 20; attempt++) {
|
|
if (mysql_query(admin, query) != 0) { usleep(100000); continue; }
|
|
MYSQL_RES* res = mysql_store_result(admin);
|
|
if (!res) { usleep(100000); continue; }
|
|
MYSQL_ROW row = mysql_fetch_row(res);
|
|
if (row) {
|
|
bool msg_ok = (expected_msg_substr == nullptr) ||
|
|
(row[1] && strstr(row[1], expected_msg_substr));
|
|
mysql_free_result(res);
|
|
return msg_ok;
|
|
}
|
|
mysql_free_result(res);
|
|
usleep(100000);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
int main(int argc, char** argv) {
|
|
CommandLine cl;
|
|
if (cl.getEnv()) { diag("Failed to get env vars."); return -1; }
|
|
|
|
diag("=== FFTO MySQL Error Recording Test ===");
|
|
plan(kPlannedTests);
|
|
|
|
MYSQL* admin = mysql_init(NULL);
|
|
MYSQL* conn = NULL;
|
|
|
|
if (!mysql_real_connect(admin, cl.host, cl.admin_username, cl.admin_password,
|
|
NULL, cl.admin_port, NULL, 0)) {
|
|
diag("Admin connection failed"); return -1;
|
|
}
|
|
|
|
// Configure FF mode
|
|
MYSQL_QUERY(admin, "UPDATE global_variables SET variable_value='true' "
|
|
"WHERE variable_name='mysql-ffto_enabled'");
|
|
MYSQL_QUERY(admin, "LOAD MYSQL VARIABLES TO RUNTIME");
|
|
{
|
|
char eu[256], ep[256];
|
|
mysql_real_escape_string(admin, eu, cl.mysql_username, strlen(cl.mysql_username));
|
|
mysql_real_escape_string(admin, ep, cl.mysql_password, strlen(cl.mysql_password));
|
|
char uq[1024];
|
|
snprintf(uq, sizeof(uq),
|
|
"INSERT OR REPLACE INTO mysql_users (username, password, fast_forward) "
|
|
"VALUES ('%s', '%s', 1)", eu, ep);
|
|
MYSQL_QUERY(admin, uq);
|
|
MYSQL_QUERY(admin, "LOAD MYSQL USERS TO RUNTIME");
|
|
}
|
|
|
|
// Reset error stats
|
|
MYSQL_QUERY(admin, "SELECT * FROM stats_mysql_errors_reset");
|
|
{ MYSQL_RES* r = mysql_store_result(admin); if (r) mysql_free_result(r); }
|
|
|
|
// Connect through ProxySQL in FF mode
|
|
conn = mysql_init(NULL);
|
|
if (!mysql_real_connect(conn, cl.host, cl.mysql_username, cl.mysql_password,
|
|
"information_schema", cl.mysql_port, NULL, 0)) {
|
|
FAIL_AND_SKIP_REMAINING(cleanup, "FF connection failed: %s", mysql_error(conn));
|
|
}
|
|
ok(conn != NULL, "Connected to MySQL via ProxySQL in FF mode");
|
|
|
|
/* Scenario 1: Syntax error -> errno 1064 */
|
|
diag("--- Scenario 1: syntax error ---");
|
|
mysql_query(conn, "SELEC BAD SYNTAX"); // intentional error
|
|
ok(verify_mysql_error(admin, 1064, "syntax"),
|
|
"Error 1064 recorded in stats_mysql_errors");
|
|
|
|
/* Scenario 2: Table not found -> errno 1146 */
|
|
diag("--- Scenario 2: table not found ---");
|
|
mysql_query(conn, "SELECT * FROM nonexistent_table_ffto_test");
|
|
ok(verify_mysql_error(admin, 1146, NULL),
|
|
"Error 1146 recorded in stats_mysql_errors");
|
|
|
|
/* Scenario 3: Recovery -- successful query after errors */
|
|
diag("--- Scenario 3: recovery after error ---");
|
|
{
|
|
MYSQL_QUERY(admin, "SELECT * FROM stats_mysql_query_digest_reset");
|
|
MYSQL_RES* r = mysql_store_result(admin); if (r) mysql_free_result(r);
|
|
}
|
|
if (mysql_query(conn, "SELECT 1") == 0) {
|
|
MYSQL_RES* r = mysql_store_result(conn);
|
|
if (r) mysql_free_result(r);
|
|
}
|
|
|
|
// Verify the successful query appears in digest stats
|
|
{
|
|
bool found = false;
|
|
for (int i = 0; i < 20; i++) {
|
|
if (mysql_query(admin, "SELECT count_star FROM stats_mysql_query_digest "
|
|
"WHERE digest_text LIKE '%SELECT %'") != 0) { usleep(100000); continue; }
|
|
MYSQL_RES* res = mysql_store_result(admin);
|
|
MYSQL_ROW row = res ? mysql_fetch_row(res) : NULL;
|
|
if (row && atoi(row[0]) > 0) found = true;
|
|
if (res) mysql_free_result(res);
|
|
if (found) break;
|
|
usleep(100000);
|
|
}
|
|
ok(found, "Recovery: SELECT 1 recorded in digest after errors");
|
|
}
|
|
|
|
/* Verify error stats have count > 0 */
|
|
{
|
|
bool has_errors = false;
|
|
if (mysql_query(admin, "SELECT count_star FROM stats_mysql_errors") == 0) {
|
|
MYSQL_RES* res = mysql_store_result(admin);
|
|
MYSQL_ROW row = res ? mysql_fetch_row(res) : NULL;
|
|
if (row && atoi(row[0]) > 0) has_errors = true;
|
|
if (res) mysql_free_result(res);
|
|
}
|
|
ok(has_errors, "stats_mysql_errors has entries from FF session");
|
|
}
|
|
|
|
/* Verify errors have non-empty last_error message */
|
|
{
|
|
bool has_msg = false;
|
|
if (mysql_query(admin, "SELECT last_error FROM stats_mysql_errors LIMIT 1") == 0) {
|
|
MYSQL_RES* res = mysql_store_result(admin);
|
|
MYSQL_ROW row = res ? mysql_fetch_row(res) : NULL;
|
|
if (row && row[0] && strlen(row[0]) > 0) has_msg = true;
|
|
if (res) mysql_free_result(res);
|
|
}
|
|
ok(has_msg, "stats_mysql_errors entries have non-empty error messages");
|
|
}
|
|
|
|
/* Verify errors have valid err_no */
|
|
{
|
|
bool has_errno = false;
|
|
if (mysql_query(admin, "SELECT err_no FROM stats_mysql_errors LIMIT 1") == 0) {
|
|
MYSQL_RES* res = mysql_store_result(admin);
|
|
MYSQL_ROW row = res ? mysql_fetch_row(res) : NULL;
|
|
if (row && atoi(row[0]) > 0) has_errno = true;
|
|
if (res) mysql_free_result(res);
|
|
}
|
|
ok(has_errno, "stats_mysql_errors entries have non-zero err_no");
|
|
}
|
|
|
|
cleanup:
|
|
if (conn) mysql_close(conn);
|
|
if (admin) mysql_close(admin);
|
|
return exit_status();
|
|
}
|