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.
proxysql/test/tap/tests/reg_test_3765_ssl_pollout-t...

196 lines
6.3 KiB

/**
* @file reg_test_3765_ssl_pollout-t.cpp
* @brief This test opens multiple connections against ProxySQL with different client flags and checks that
* CPU usage by ProxySQL didn't increase significantly. Tested connections types are: normal, SSL, and
* compression.
* @details The goal of the test is to detect regressions or incompatibilities in the way ProxySQL polling
* operations interacts with OpenSSL library.
*/
#include <algorithm>
#include <cstring>
#include <vector>
#include <string>
#include <stdio.h>
#include <unistd.h>
#include <utility>
#include <poll.h>
#include <sys/epoll.h>
#include "mysql.h"
#include "tap.h"
#include "command_line.h"
#include "utils.h"
using std::pair;
using std::string;
using std::vector;
/**
* @brief TODO: Refactor this into utils, also used in another PR.
*/
int create_connections(const conn_opts_t& conn_opts, uint32_t cons_num, std::vector<MYSQL*>& proxy_conns) {
std::vector<MYSQL*> result {};
for (uint32_t i = 0; i < cons_num; i++) {
const char* host = conn_opts.host.c_str();
const char* user = conn_opts.user.c_str();
const char* pass = conn_opts.pass.c_str();
const int port = conn_opts.port;
MYSQL* proxysql = mysql_init(NULL);
if (conn_opts.client_flags & CLIENT_SSL) {
mysql_ssl_set(proxysql, NULL, NULL, NULL, NULL, NULL);
}
if (!mysql_real_connect(proxysql, host, user, pass, NULL, port, NULL, conn_opts.client_flags)) {
diag("File %s, line %d, Error: %s", __FILE__, __LINE__, mysql_error(proxysql));
return EXIT_FAILURE;
} else {
result.push_back(proxysql);
}
}
proxy_conns = result;
return EXIT_SUCCESS;
}
const uint32_t ADMIN_CONN_NUM = 100;
const uint32_t MYSQL_CONN_NUM = 100;
double MAX_IDLE_CPU_USAGE = (double) get_env_int("MAX_IDLE_CPU_USAGE", 10);
double MAX_INCREASE_CPU_USAGE = (double) get_env_int("TAP_MAX_INCREASE_CPU_USAGE", 2);
int get_idle_conns_cpu_usage(CommandLine& cl, uint64_t mode, double& no_conns_cpu, double& idle_conns_cpu) {
// get ProxySQL idle cpu usage
int idle_err = get_proxysql_cpu_usage(cl, no_conns_cpu);
if (idle_err) {
diag("Unable to get 'no_conns_cpu' usage.");
return idle_err;
}
conn_opts_t proxy_conns_opts { cl.host, cl.username, cl.password, cl.port, mode };
conn_opts_t admin_conns_opts { cl.admin_host, cl.admin_username, cl.admin_password, cl.admin_port, mode };
// Create 'N' admin and mysql connections without SSL
vector<MYSQL*> v_admin_conns {};
int admin_conns_res = create_connections(admin_conns_opts, ADMIN_CONN_NUM, v_admin_conns);
if (admin_conns_res != EXIT_SUCCESS) {
diag("File %s, line %d, Exiting...", __FILE__, __LINE__);
return EXIT_FAILURE;
}
vector<MYSQL*> v_proxy_conns {};
int mysql_conns_res = create_connections(proxy_conns_opts, MYSQL_CONN_NUM, v_proxy_conns);
if (mysql_conns_res != EXIT_SUCCESS) {
diag("File %s, line %d, Exiting...", __FILE__, __LINE__);
return EXIT_FAILURE;
}
int final_err = get_proxysql_cpu_usage(cl, idle_conns_cpu);
if (final_err) {
diag("Unable to get 'idle_conns_cpu' usage.");
return idle_err;
}
std::for_each(v_admin_conns.begin(), v_admin_conns.end(), [](MYSQL* conn) -> void { mysql_close(conn); });
std::for_each(v_proxy_conns.begin(), v_proxy_conns.end(), [](MYSQL* conn) -> void { mysql_close(conn); });
return EXIT_SUCCESS;
}
int main(int argc, char** argv) {
CommandLine cl;
if (cl.getEnv()) {
diag("Failed to get the required environmental variables.");
return exit_status();
}
plan(6);
// For ASAN builds we don't care about correctness in this measurement.
if (get_env_int("WITHASAN", 0)) {
MAX_IDLE_CPU_USAGE = 80;
}
double no_conns_cpu = 0;
double idle_conns_cpu = 0;
MYSQL* proxysql_admin = mysql_init(NULL);
if (!mysql_real_connect(proxysql_admin, cl.host, cl.admin_username, cl.admin_password, NULL, cl.admin_port, NULL, 0)) {
fprintf(stderr, "File %s, line %d, Error: %s\n", __FILE__, __LINE__, mysql_error(proxysql_admin));
return EXIT_FAILURE;
}
pair<int,vector<MYSQL*>> p_err_nodes_conns { disable_core_nodes_scheduler(cl, proxysql_admin) };
if (p_err_nodes_conns.first) { return EXIT_FAILURE; }
vector<MYSQL*>& nodes_conns { p_err_nodes_conns.second };
MYSQL_QUERY(proxysql_admin, "SET mysql-have_ssl=1");
MYSQL_QUERY(proxysql_admin, "LOAD MYSQL VARIABLES TO RUNTIME");
mysql_close(proxysql_admin);
diag("Testing regular connections...");
int ret_cpu_usage = get_idle_conns_cpu_usage(cl, 0, no_conns_cpu, idle_conns_cpu);
if (ret_cpu_usage != EXIT_SUCCESS) { return EXIT_FAILURE; }
ok(
no_conns_cpu < MAX_IDLE_CPU_USAGE,
"ProxySQL 'no clients' CPU usage should be below expected: (MAX_IDLE_CPU_USAGE: %%%lf, Act: %%%lf)",
MAX_IDLE_CPU_USAGE, no_conns_cpu
);
ok(
idle_conns_cpu < MAX_IDLE_CPU_USAGE + MAX_INCREASE_CPU_USAGE,
"ProxySQL 'with clients' CPU usage should be below expected:"
" (MAX_IDLE_CPU_USAGE + MAX_INCREASE_CPU_USAGE: %%%lf, Act: %%%lf)",
MAX_IDLE_CPU_USAGE + MAX_INCREASE_CPU_USAGE, idle_conns_cpu
);
diag("Testing SSL connections...");
ret_cpu_usage = get_idle_conns_cpu_usage(cl, CLIENT_SSL, no_conns_cpu, idle_conns_cpu);
if (ret_cpu_usage != EXIT_SUCCESS) { return EXIT_FAILURE; }
ok(
no_conns_cpu < MAX_IDLE_CPU_USAGE,
"ProxySQL 'no clients' CPU usage should be below expected: (Exp: %%%lf, Act: %%%lf)",
MAX_IDLE_CPU_USAGE, no_conns_cpu
);
ok(
idle_conns_cpu < MAX_IDLE_CPU_USAGE + MAX_INCREASE_CPU_USAGE,
"ProxySQL 'with SSL clients' CPU usage should be below expected:"
" (MAX_IDLE_CPU_USAGE + MAX_INCREASE_CPU_USAGE: %%%lf, Act: %%%lf)",
MAX_IDLE_CPU_USAGE + MAX_INCREASE_CPU_USAGE, idle_conns_cpu
);
diag("Testing SSL and compressed connections...");
ret_cpu_usage = get_idle_conns_cpu_usage(cl, CLIENT_SSL|CLIENT_COMPRESS, no_conns_cpu, idle_conns_cpu);
if (ret_cpu_usage != EXIT_SUCCESS) { return EXIT_FAILURE; }
ok(
no_conns_cpu < MAX_IDLE_CPU_USAGE,
"ProxySQL 'no clients' CPU usage should be below expected: (Exp: %%%lf, Act: %%%lf)",
MAX_IDLE_CPU_USAGE, no_conns_cpu
);
ok(
idle_conns_cpu < MAX_IDLE_CPU_USAGE + MAX_INCREASE_CPU_USAGE,
"ProxySQL 'with SSL|COMPRESS clients' CPU usage should be below expected: (Exp: %%%lf, Act: %%%lf)",
MAX_IDLE_CPU_USAGE + MAX_INCREASE_CPU_USAGE, idle_conns_cpu
);
// Recover cluster scheduler
for (MYSQL* myconn : nodes_conns) {
MYSQL_QUERY_T(myconn, "LOAD SCHEDULER FROM DISK");
MYSQL_QUERY_T(myconn, "LOAD SCHEDULER TO RUNTIME");
mysql_close(myconn);
}
return exit_status();
}