Merge pull request #2882 from sysown/v2.1.0-2862

Port PR 2829 to 2.1.0 #2862
pull/2944/head
René Cannaò 6 years ago committed by GitHub
commit 3d74ef4828
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -18,6 +18,7 @@ using json = nlohmann::json;
#define STATUS_MYSQL_CONNECTION_SQL_LOG_BIN0 0x00000100
#define STATUS_MYSQL_CONNECTION_FOUND_ROWS 0x00000200
#define STATUS_MYSQL_CONNECTION_NO_BACKSLASH_ESCAPES 0x00000400
#define STATUS_MYSQL_CONNECTION_HAS_SAVEPOINT 0x00000800
class Variable {
public:
@ -143,6 +144,7 @@ class MySQL_Connection {
void set_status_no_backslash_escapes(bool);
void set_status_prepared_statement(bool);
void set_status_user_variable(bool);
void set_status_has_savepoint(bool);
void set_status_no_multiplex(bool);
void set_status_sql_log_bin0(bool);
void set_status_found_rows(bool);
@ -154,6 +156,7 @@ class MySQL_Connection {
bool get_status_no_backslash_escapes();
bool get_status_prepared_statement();
bool get_status_user_variable();
bool get_status_has_savepoint();
bool get_status_no_multiplex();
bool get_status_sql_log_bin0();
bool get_status_found_rows();

@ -351,12 +351,14 @@ enum MYSQL_COM_QUERY_command {
MYSQL_COM_QUERY_OPTIMIZE,
MYSQL_COM_QUERY_PREPARE,
MYSQL_COM_QUERY_PURGE,
MYSQL_COM_QUERY_RELEASE_SAVEPOINT,
MYSQL_COM_QUERY_RENAME_TABLE,
MYSQL_COM_QUERY_RESET_MASTER,
MYSQL_COM_QUERY_RESET_SLAVE,
MYSQL_COM_QUERY_REPLACE,
MYSQL_COM_QUERY_REVOKE,
MYSQL_COM_QUERY_ROLLBACK,
MYSQL_COM_QUERY_ROLLBACK_SAVEPOINT,
MYSQL_COM_QUERY_SAVEPOINT,
MYSQL_COM_QUERY_SELECT,
MYSQL_COM_QUERY_SELECT_FOR_UPDATE,

@ -1019,6 +1019,7 @@ void MySQL_Session::generate_proxysql_internal_session_json(json &j) {
j["backends"][i]["conn"]["no_backslash_escapes"] = _myconn->options.no_backslash_escapes;
j["backends"][i]["conn"]["status"]["get_lock"] = _myconn->get_status_get_lock();
j["backends"][i]["conn"]["status"]["lock_tables"] = _myconn->get_status_lock_tables();
j["backends"][i]["conn"]["status"]["has_savepoint"] = _myconn->get_status_has_savepoint();
j["backends"][i]["conn"]["status"]["temporary_table"] = _myconn->get_status_temporary_table();
j["backends"][i]["conn"]["status"]["user_variable"] = _myconn->get_status_user_variable();
j["backends"][i]["conn"]["status"]["found_rows"] = _myconn->get_status_found_rows();

@ -480,12 +480,14 @@ Query_Processor::Query_Processor() {
commands_counters_desc[MYSQL_COM_QUERY_OPTIMIZE]=(char *)"OPTIMIZE";
commands_counters_desc[MYSQL_COM_QUERY_PREPARE]=(char *)"PREPARE";
commands_counters_desc[MYSQL_COM_QUERY_PURGE]=(char *)"PURGE";
commands_counters_desc[MYSQL_COM_QUERY_RELEASE_SAVEPOINT]=(char *)"RELEASE_SAVEPOINT";
commands_counters_desc[MYSQL_COM_QUERY_RENAME_TABLE]=(char *)"RENAME_TABLE";
commands_counters_desc[MYSQL_COM_QUERY_RESET_MASTER]=(char *)"RESET_MASTER";
commands_counters_desc[MYSQL_COM_QUERY_RESET_SLAVE]=(char *)"RESET_SLAVE";
commands_counters_desc[MYSQL_COM_QUERY_REPLACE]=(char *)"REPLACE";
commands_counters_desc[MYSQL_COM_QUERY_REVOKE]=(char *)"REVOKE";
commands_counters_desc[MYSQL_COM_QUERY_ROLLBACK]=(char *)"ROLLBACK";
commands_counters_desc[MYSQL_COM_QUERY_ROLLBACK_SAVEPOINT]=(char *)"ROLLBACK_SAVEPOINT";
commands_counters_desc[MYSQL_COM_QUERY_SAVEPOINT]=(char *)"SAVEPOINT";
commands_counters_desc[MYSQL_COM_QUERY_SELECT]=(char *)"SELECT";
commands_counters_desc[MYSQL_COM_QUERY_SELECT_FOR_UPDATE]=(char *)"SELECT_FOR_UPDATE";
@ -2281,6 +2283,14 @@ __remove_paranthesis:
break;
case 'r':
case 'R':
if (!strcasecmp("RELEASE",token)) { // RELEASE
token=(char *)tokenize(&tok);
if (token==NULL) break;
if (!strcasecmp("SAVEPOINT",token)) {
ret=MYSQL_COM_QUERY_RELEASE_SAVEPOINT;
break;
}
}
if (!strcasecmp("RENAME",token)) { // RENAME
token=(char *)tokenize(&tok);
if (token==NULL) break;
@ -2311,7 +2321,20 @@ __remove_paranthesis:
break;
}
if (!strcasecmp("ROLLBACK",token)) { // ROLLBACK
ret=MYSQL_COM_QUERY_ROLLBACK;
token=(char *)tokenize(&tok);
if (token==NULL) {
ret=MYSQL_COM_QUERY_ROLLBACK;
break;
} else {
if (!strcasecmp("TO",token)) {
token=(char *)tokenize(&tok);
if (token==NULL) break;
if (!strcasecmp("SAVEPOINT",token)) {
ret=MYSQL_COM_QUERY_ROLLBACK_SAVEPOINT;
break;
}
}
}
break;
}
break;

@ -10,6 +10,8 @@
#include "query_processor.h"
#include "MySQL_Variables.h"
#include <atomic>
extern const MARIADB_CHARSET_INFO * proxysql_find_charset_nr(unsigned int nr);
MARIADB_CHARSET_INFO * proxysql_find_charset_name(const char *name);
@ -508,6 +510,14 @@ void MySQL_Connection::set_status_user_variable(bool v) {
}
}
void MySQL_Connection::set_status_has_savepoint(bool v) {
if (v) {
status_flags |= STATUS_MYSQL_CONNECTION_HAS_SAVEPOINT;
} else {
status_flags &= ~STATUS_MYSQL_CONNECTION_HAS_SAVEPOINT;
}
}
void MySQL_Connection::set_status_prepared_statement(bool v) {
if (v) {
status_flags |= STATUS_MYSQL_CONNECTION_PREPARED_STATEMENT;
@ -547,6 +557,10 @@ bool MySQL_Connection::get_status_user_variable() {
return status_flags & STATUS_MYSQL_CONNECTION_USER_VARIABLE;
}
bool MySQL_Connection::get_status_has_savepoint() {
return status_flags & STATUS_MYSQL_CONNECTION_HAS_SAVEPOINT;
}
bool MySQL_Connection::get_status_get_lock() {
return status_flags & STATUS_MYSQL_CONNECTION_GET_LOCK;
}
@ -1947,6 +1961,12 @@ bool MySQL_Connection::IsActiveTransaction() {
ret = true;
}
}
if (ret == false) {
if (get_status_has_savepoint()) {
// there are savepoints
ret = true;
}
}
}
return ret;
}
@ -1979,7 +1999,7 @@ bool MySQL_Connection::MultiplexDisabled() {
// status_flags stores information about the status of the connection
// can be used to determine if multiplexing can be enabled or not
bool ret=false;
if (status_flags & (STATUS_MYSQL_CONNECTION_TRANSACTION|STATUS_MYSQL_CONNECTION_USER_VARIABLE|STATUS_MYSQL_CONNECTION_PREPARED_STATEMENT|STATUS_MYSQL_CONNECTION_LOCK_TABLES|STATUS_MYSQL_CONNECTION_TEMPORARY_TABLE|STATUS_MYSQL_CONNECTION_GET_LOCK|STATUS_MYSQL_CONNECTION_NO_MULTIPLEX|STATUS_MYSQL_CONNECTION_SQL_LOG_BIN0|STATUS_MYSQL_CONNECTION_FOUND_ROWS|STATUS_MYSQL_CONNECTION_NO_BACKSLASH_ESCAPES) ) {
if (status_flags & (STATUS_MYSQL_CONNECTION_TRANSACTION|STATUS_MYSQL_CONNECTION_USER_VARIABLE|STATUS_MYSQL_CONNECTION_PREPARED_STATEMENT|STATUS_MYSQL_CONNECTION_LOCK_TABLES|STATUS_MYSQL_CONNECTION_TEMPORARY_TABLE|STATUS_MYSQL_CONNECTION_GET_LOCK|STATUS_MYSQL_CONNECTION_NO_MULTIPLEX|STATUS_MYSQL_CONNECTION_SQL_LOG_BIN0|STATUS_MYSQL_CONNECTION_FOUND_ROWS|STATUS_MYSQL_CONNECTION_NO_BACKSLASH_ESCAPES|STATUS_MYSQL_CONNECTION_HAS_SAVEPOINT) ) {
ret=true;
}
if (auto_increment_delay_token) return true;
@ -2164,6 +2184,35 @@ void MySQL_Connection::ProcessQueryAndSetStatusFlags(char *query_digest_text) {
set_status_found_rows(true);
}
}
if (get_status_has_savepoint()==false) {
if (mysql) {
if (
(mysql->server_status & SERVER_STATUS_IN_TRANS)
||
((mysql->server_status & SERVER_STATUS_AUTOCOMMIT) == 0)
) {
if (!strncasecmp(query_digest_text,"SAVEPOINT ", strlen("SAVEPOINT "))) {
set_status_has_savepoint(true);
}
}
}
} else {
if (
(
// make sure we don't have a transaction running
// checking just for COMMIT and ROLLBACK is not enough, because `SET autocommit=1` can commit too
(mysql->server_status & SERVER_STATUS_AUTOCOMMIT)
&&
( (mysql->server_status & SERVER_STATUS_IN_TRANS) == 0 )
)
||
!strcasecmp(query_digest_text,"COMMIT")
||
!strcasecmp(query_digest_text,"ROLLBACK")
) {
set_status_has_savepoint(false);
}
}
}
void MySQL_Connection::optimize() {

@ -0,0 +1,401 @@
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <mysql.h>
#include <string.h>
#include <string>
#include <time.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <pthread.h>
#include <sstream>
#include <iostream>
#include <fstream>
#include <mutex>
#include <mutex>
#include <atomic>
#include "tap.h"
#include "utils.h"
#include "command_line.h"
unsigned long long monotonic_time() {
struct timespec ts;
//clock_gettime(CLOCK_MONOTONIC_COARSE, &ts); // this is faster, but not precise
clock_gettime(CLOCK_MONOTONIC, &ts);
return (((unsigned long long) ts.tv_sec) * 1000000) + (ts.tv_nsec / 1000);
}
struct cpu_timer
{
cpu_timer() {
begin = monotonic_time();
}
~cpu_timer()
{
unsigned long long end = monotonic_time();
std::cerr << double( end - begin ) / 1000000 << " secs.\n" ;
begin=end-begin;
};
unsigned long long begin;
};
unsigned int num_threads=1;
int count=0;
char *username=NULL;
char *password=NULL;
char *host=(char *)"localhost";
int port=3306;
char *schema=(char *)"information_schema";
int silent = 0;
int sysbench = 0;
int local=0;
int transactions=0;
int uniquequeries=0;
int histograms=-1;
unsigned int g_passed=0;
unsigned int g_failed=0;
std::atomic<int> cnt_transactions;
std::atomic<int> cnt_SELECT_outside_transactions;
unsigned int status_connections = 0;
unsigned int connect_phase_completed = 0;
unsigned int query_phase_completed = 0;
__thread int g_seed;
std::mutex mtx_;
inline int fastrand() {
g_seed = (214013*g_seed+2531011);
return (g_seed>>16)&0x7FFF;
}
void gen_random(char *s, const int len) {
static const char alphanum[] =
"ABCDEFGHIJKLMNOPQRSTUVWXYZ"
"abcdefghijklmnopqrstuvwxyz";
for (int i = 0; i < len; ++i) {
s[i] = alphanum[fastrand() % (sizeof(alphanum) - 1)];
}
s[len] = 0;
}
void * my_conn_thread(void *arg) {
g_seed = time(NULL) ^ getpid() ^ pthread_self();
unsigned int select_OK=0;
unsigned int select_ERR=0;
int i, j;
MYSQL **mysqlconns=(MYSQL **)malloc(sizeof(MYSQL *)*count);
if (mysqlconns==NULL) {
exit(EXIT_FAILURE);
}
for (i=0; i<count; i++) {
MYSQL *mysql=mysql_init(NULL);
if (mysql==NULL) {
exit(EXIT_FAILURE);
}
MYSQL *rc=mysql_real_connect(mysql, host, username, password, schema, (local ? 0 : port) , NULL, 0);
if (rc==NULL) {
if (silent==0) {
fprintf(stderr,"%s\n", mysql_error(mysql));
}
exit(EXIT_FAILURE);
}
mysqlconns[i]=mysql;
__sync_add_and_fetch(&status_connections,1);
}
__sync_fetch_and_add(&connect_phase_completed,1);
while(__sync_fetch_and_add(&connect_phase_completed,0) != num_threads) {
}
MYSQL *mysql=NULL;
std::string sel1 = "SELECT * FROM test.test_savepoint LIMIT 1";
for (j=0; j<transactions; j++) {
int fr = fastrand();
int r1=fr%count;
bool explicit_transaction = false;
mysql=mysqlconns[r1];
int sleepDelay;
for (int i=0; i<fr%3; i++) {
std::string q = "SET autocommit=" + std::to_string(fr%2);
if (mysql_query(mysql, q.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", q.c_str(), mysql_error(mysql));
exit(EXIT_FAILURE);
}
sleepDelay = fastrand()%100;
usleep(sleepDelay * 100);
}
{
std::string q;
if (fr%2) {
q = "START TRANSACTION";
explicit_transaction = true;
} else {
q = "SET AUTOCOMMIT=0";
}
if (mysql_query(mysql, q.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", q.c_str(), mysql_error(mysql));
exit(EXIT_FAILURE);
}
sleepDelay = fastrand()%100;
usleep(sleepDelay * 100);
}
if (fr%3 == 0) {
if (mysql_query(mysql, sel1.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", sel1.c_str(), mysql_error(mysql));
} else {
MYSQL_RES *result = mysql_store_result(mysql);
mysql_free_result(result);
select_OK++;
if (explicit_transaction == false) {
cnt_SELECT_outside_transactions++;
}
}
sleepDelay = fastrand()%100;
usleep(sleepDelay * 100);
}
char buf[16];
memset(buf,0,16);
gen_random(buf,14);
{
std::string q = "SAVEPOINT ";
q += buf;
if (mysql_query(mysql, q.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", q.c_str(), mysql_error(mysql));
exit(EXIT_FAILURE);
}
sleepDelay = fastrand()%100;
usleep(sleepDelay * 100);
}
for (int i=0; i<fr%3+1; i++) {
if (mysql_query(mysql, sel1.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", sel1.c_str(), mysql_error(mysql));
} else {
MYSQL_RES *result = mysql_store_result(mysql);
mysql_free_result(result);
select_OK++;
}
sleepDelay = fastrand()%100;
usleep(sleepDelay * 100);
}
int aa = fr%10;
if (aa < 3) {
std::string q;
q = "RELEASE SAVEPOINT ";
q += buf;
if (mysql_query(mysql, q.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", q.c_str(), mysql_error(mysql));
exit(EXIT_FAILURE);
}
sleepDelay = fastrand()%100;
usleep(sleepDelay * 100);
} else {
std::string q;
q = "ROLLBACK TO SAVEPOINT ";
q += buf;
if (mysql_query(mysql, q.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", q.c_str(), mysql_error(mysql));
exit(EXIT_FAILURE);
}
sleepDelay = fastrand()%100;
usleep(sleepDelay * 100);
if (aa > 7) {
q = "RELEASE SAVEPOINT ";
q += buf;
if (mysql_query(mysql, q.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", q.c_str(), mysql_error(mysql));
exit(EXIT_FAILURE);
}
sleepDelay = fastrand()%100;
usleep(sleepDelay * 100);
}
}
{
std::string q;
int f = fr%3;
if (f==0) {
q = "COMMIT";
} else {
q = "ROLLBACK";
/*
// FIXME: this code is currently commented because of another bug
if (explicit_transaction==false) {
if (f!=1) {
q = "SET AUTOCOMMIT=1";
}
}
*/
}
if (mysql_query(mysql, q.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", q.c_str(), mysql_error(mysql));
exit(EXIT_FAILURE);
}
cnt_transactions++;
sleepDelay = fastrand()%100;
usleep(sleepDelay * 100);
}
/*
// we do not log every single transaction, too verbose
bool testPassed = true;
{
std::lock_guard<std::mutex> lock(mtx_);
ok(testPassed, "mysql connection [%p], thread_id [%lu], transaction completed", mysql, mysql->thread_id);
}
*/
}
for (i=0; i<count; i++) {
MYSQL *mysql = mysqlconns[i];
mysql_close(mysql);
}
__sync_fetch_and_add(&query_phase_completed,1);
return NULL;
}
void print_commands_stats(MYSQL *mysqladmin) {
std::string q = "SELECT Command,Total_Time_us,Total_cnt FROM stats_mysql_commands_counters WHERE Command IN ('COMMIT','RELEASE_SAVEPOINT','ROLLBACK','ROLLBACK_SAVEPOINT','SELECT','SET','START_TRANSACTION') ORDER BY Command";
{
if (mysql_query(mysqladmin, q.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", q.c_str(), mysql_error(mysqladmin));
exit(EXIT_FAILURE);
}
MYSQL_RES *result = mysql_store_result(mysqladmin);
MYSQL_ROW row;
std::cerr << "Stats from stats_mysql_commands_counters" << std::endl;
while ((row = mysql_fetch_row(result)))
{
std::cerr << row[0] << " \tCount: " << row[2] << " \tTime: " << row[1] << std::endl;
}
mysql_free_result(result);
}
}
int main(int argc, char *argv[]) {
cnt_transactions = 0;
cnt_SELECT_outside_transactions = 0;
CommandLine cl;
if(cl.getEnv())
return exit_status();
username = cl.username;
password = cl.password;
host = cl.host;
port = cl.port;
{
MYSQL *mysql=mysql_init(NULL);
if (mysql==NULL) {
exit(EXIT_FAILURE);
}
MYSQL *rc=mysql_real_connect(mysql, host, username, password, schema, (local ? 0 : port) , NULL, 0);
if (rc==NULL) {
if (silent==0) {
fprintf(stderr,"%s\n", mysql_error(mysql));
}
exit(EXIT_FAILURE);
}
MYSQL_QUERY(mysql, "CREATE DATABASE IF NOT EXISTS test");
MYSQL_QUERY(mysql, "CREATE TABLE IF NOT EXISTS test.test_savepoint(id INT NOT NULL AUTO_INCREMENT PRIMARY KEY) ENGINE=INNODB");
MYSQL_QUERY(mysql, "DELETE FROM test.test_savepoint");
MYSQL_QUERY(mysql, "INSERT INTO test.test_savepoint VALUES (1), (2)");
mysql_close(mysql);
}
MYSQL* mysqladmin = mysql_init(NULL);
if (!mysqladmin)
return exit_status();
if (!mysql_real_connect(mysqladmin, 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(mysqladmin));
return exit_status();
}
MYSQL_QUERY(mysqladmin, "update global_variables set variable_value='false' where variable_name='mysql-enforce_autocommit_on_reads'");
MYSQL_QUERY(mysqladmin, "LOAD MYSQL VARIABLES TO RUNTIME");
MYSQL_QUERY(mysqladmin, "CREATE TABLE mysql_query_rules_948 AS SELECT * FROM mysql_query_rules");
MYSQL_QUERY(mysqladmin, "DELETE FROM mysql_query_rules");
MYSQL_QUERY(mysqladmin, "LOAD MYSQL QUERY RULES TO RUNTIME");
int MyHGM_myconnpoll_push = 0;
std::string q;
q = "SELECT * FROM stats_mysql_global WHERE variable_name LIKE 'MyHGM%'";
{
if (mysql_query(mysqladmin, q.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", q.c_str(), mysql_error(mysqladmin));
exit(EXIT_FAILURE);
}
MYSQL_RES *result = mysql_store_result(mysqladmin);
MYSQL_ROW row;
while ((row = mysql_fetch_row(result)))
{
if (strstr(row[0], "MyHGM_myconnpoll_push")) {
MyHGM_myconnpoll_push = atoi(row[1]);
}
}
mysql_free_result(result);
}
print_commands_stats(mysqladmin);
num_threads = 4;
transactions = 200;
count = 10;
// plan(transactions * num_threads + 1); // this was too verbose
plan(1);
if (strcmp(host,"localhost")==0) {
local = 1;
}
pthread_t *thi=(pthread_t *)malloc(sizeof(pthread_t)*num_threads);
if (thi==NULL)
return exit_status();
for (unsigned int i=0; i<num_threads; i++) {
if ( pthread_create(&thi[i], NULL, my_conn_thread , NULL) != 0 )
perror("Thread creation");
}
for (unsigned int i=0; i<num_threads; i++) {
pthread_join(thi[i], NULL);
}
{
if (mysql_query(mysqladmin, q.c_str())) {
fprintf(stderr,"Error running query: %s. Error: %s\n", q.c_str(), mysql_error(mysqladmin));
exit(EXIT_FAILURE);
}
MYSQL_RES *result = mysql_store_result(mysqladmin);
MYSQL_ROW row;
while ((row = mysql_fetch_row(result)))
{
if (strstr(row[0], "MyHGM_myconnpoll_push")) {
MyHGM_myconnpoll_push = atoi(row[1]) - MyHGM_myconnpoll_push;
}
}
mysql_free_result(result);
}
print_commands_stats(mysqladmin);
std::cerr << std::endl << "MyHGM_myconnpoll_push: " << MyHGM_myconnpoll_push << std::endl;
std::cerr << "cnt_SELECT_outside_transactions: " << cnt_SELECT_outside_transactions << std::endl;
std::cerr << "cnt_transactions: " << cnt_transactions << std::endl;
//ok((MyHGM_myconnpoll_push == cnt_transactions+cnt_SELECT_outside_transactions) , "Number of transactions [%d] , Queries outside transaction [%d] , total connections returned [%d]", cnt_transactions.load(std::memory_order_relaxed), cnt_SELECT_outside_transactions.load(std::memory_order_relaxed), MyHGM_myconnpoll_push);
// FIXME: until we fix the autocommit bug, we may have some minor mismatch
ok((MyHGM_myconnpoll_push <= cnt_transactions+cnt_SELECT_outside_transactions && MyHGM_myconnpoll_push >= cnt_transactions+cnt_SELECT_outside_transactions-10) , "Number of transactions [%d] , Queries outside transaction [%d] , total connections returned [%d]", cnt_transactions.load(std::memory_order_relaxed), cnt_SELECT_outside_transactions.load(std::memory_order_relaxed), MyHGM_myconnpoll_push);
MYSQL_QUERY(mysqladmin, "DELETE FROM mysql_query_rules");
MYSQL_QUERY(mysqladmin, "INSERT INTO mysql_query_rules SELECT * FROM mysql_query_rules_948");
MYSQL_QUERY(mysqladmin, "LOAD MYSQL QUERY RULES TO RUNTIME");
return exit_status();
}
Loading…
Cancel
Save