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.
153 lines
5.5 KiB
153 lines
5.5 KiB
/**
|
|
* @file test_ffto_pgsql_error_stats-t.cpp
|
|
* @brief FFTO E2E TAP test -- PostgreSQL error recording in stats_pgsql_errors.
|
|
*
|
|
* Validates that errors occurring during PostgreSQL fast-forward sessions
|
|
* are properly recorded in stats_pgsql_errors with correct SQLSTATE
|
|
* and error message.
|
|
*
|
|
* @par Test scenarios
|
|
* 1. Syntax error -> sqlstate 42601 recorded
|
|
* 2. Undefined table -> sqlstate 42P01 recorded
|
|
* 3. Error entries have non-empty messages and sqlstate
|
|
*/
|
|
|
|
#include <string>
|
|
#include <stdio.h>
|
|
#include <cstring>
|
|
#include <unistd.h>
|
|
#include "libpq-fe.h"
|
|
#include "mysql.h"
|
|
#include "tap.h"
|
|
#include "command_line.h"
|
|
#include "utils.h"
|
|
|
|
static constexpr int kPlannedTests = 5;
|
|
|
|
#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)
|
|
|
|
static bool verify_pgsql_error(MYSQL* admin, const char* expected_sqlstate) {
|
|
char query[1024];
|
|
snprintf(query, sizeof(query),
|
|
"SELECT sqlstate, last_error FROM stats_pgsql_errors WHERE sqlstate = '%s'",
|
|
expected_sqlstate);
|
|
|
|
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) {
|
|
mysql_free_result(res);
|
|
return true;
|
|
}
|
|
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 PostgreSQL Error Stats Test ===");
|
|
plan(kPlannedTests);
|
|
|
|
MYSQL* admin = mysql_init(NULL);
|
|
PGconn* 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;
|
|
}
|
|
|
|
MYSQL_QUERY(admin, "UPDATE global_variables SET variable_value='true' "
|
|
"WHERE variable_name='pgsql-ffto_enabled'");
|
|
MYSQL_QUERY(admin, "LOAD PGSQL VARIABLES TO RUNTIME");
|
|
|
|
{
|
|
char eu[256], ep[256];
|
|
mysql_real_escape_string(admin, eu, cl.pgsql_root_username, strlen(cl.pgsql_root_username));
|
|
mysql_real_escape_string(admin, ep, cl.pgsql_root_password, strlen(cl.pgsql_root_password));
|
|
char uq[1024];
|
|
snprintf(uq, sizeof(uq),
|
|
"INSERT OR REPLACE INTO pgsql_users (username, password, fast_forward) "
|
|
"VALUES ('%s', '%s', 1)", eu, ep);
|
|
MYSQL_QUERY(admin, uq);
|
|
MYSQL_QUERY(admin, "LOAD PGSQL USERS TO RUNTIME");
|
|
}
|
|
{
|
|
char sq[1024];
|
|
snprintf(sq, sizeof(sq),
|
|
"INSERT OR REPLACE INTO pgsql_servers (hostgroup_id, hostname, port) "
|
|
"VALUES (0, '%s', %d)", cl.pgsql_server_host, cl.pgsql_server_port);
|
|
MYSQL_QUERY(admin, sq);
|
|
MYSQL_QUERY(admin, "LOAD PGSQL SERVERS TO RUNTIME");
|
|
}
|
|
|
|
// Reset error stats
|
|
MYSQL_QUERY(admin, "SELECT * FROM stats_pgsql_errors_reset");
|
|
{ MYSQL_RES* r = mysql_store_result(admin); if (r) mysql_free_result(r); }
|
|
|
|
{
|
|
char ci[1024];
|
|
snprintf(ci, sizeof(ci), "host=%s port=%d user=%s password=%s dbname=postgres sslmode=disable",
|
|
cl.pgsql_host, cl.pgsql_port, cl.pgsql_root_username, cl.pgsql_root_password);
|
|
conn = PQconnectdb(ci);
|
|
}
|
|
if (PQstatus(conn) != CONNECTION_OK) {
|
|
FAIL_AND_SKIP_REMAINING(cleanup, "PG connection failed: %s", PQerrorMessage(conn));
|
|
}
|
|
ok(conn != NULL, "Connected to PostgreSQL via ProxySQL in FF mode");
|
|
|
|
/* Scenario 1: Syntax error -> SQLSTATE 42601 */
|
|
diag("--- Scenario 1: syntax error ---");
|
|
{ PGresult* r = PQexec(conn, "SELEC BAD"); if (r) PQclear(r); }
|
|
ok(verify_pgsql_error(admin, "42601"),
|
|
"SQLSTATE 42601 recorded in stats_pgsql_errors");
|
|
|
|
/* Scenario 2: Undefined table -> SQLSTATE 42P01 */
|
|
diag("--- Scenario 2: undefined table ---");
|
|
{ PGresult* r = PQexec(conn, "SELECT * FROM nonexistent_table_ffto_test"); if (r) PQclear(r); }
|
|
ok(verify_pgsql_error(admin, "42P01"),
|
|
"SQLSTATE 42P01 recorded in stats_pgsql_errors");
|
|
|
|
/* Verify entries have non-empty messages */
|
|
{
|
|
bool has_msg = false;
|
|
if (mysql_query(admin, "SELECT last_error FROM stats_pgsql_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_pgsql_errors entries have non-empty error messages");
|
|
}
|
|
|
|
/* Verify entries have non-empty sqlstate */
|
|
{
|
|
bool has_sqlstate = false;
|
|
if (mysql_query(admin, "SELECT sqlstate FROM stats_pgsql_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_sqlstate = true;
|
|
if (res) mysql_free_result(res);
|
|
}
|
|
ok(has_sqlstate, "stats_pgsql_errors entries have non-empty sqlstate");
|
|
}
|
|
|
|
cleanup:
|
|
if (conn) PQfinish(conn);
|
|
if (admin) mysql_close(admin);
|
|
return exit_status();
|
|
}
|