123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484 |
- #include <memory>
- #include <string>
- #include <utility>
- #include <cassert>
- #include <cstring>
- #include <mysql.h>
- #include <mysqld_error.h>
- #include <kopano/ECConfig.h>
- #include <kopano/ECLogger.h>
- #include <kopano/database.hpp>
- #include <kopano/stringutil.h>
- #define LOG_SQL_DEBUG(_msg, ...) \
- ec_log(EC_LOGLEVEL_DEBUG | EC_LOGLEVEL_SQL, _msg, ##__VA_ARGS__)
- DB_RESULT::~DB_RESULT(void)
- {
- if (m_res == nullptr)
- return;
- assert(m_db != nullptr);
- if (m_db == nullptr)
- return;
- m_db->FreeResult_internal(m_res);
- m_res = nullptr;
- }
- KDatabase::KDatabase(void)
- {
- memset(&m_lpMySQL, 0, sizeof(m_lpMySQL));
- }
- ECRESULT KDatabase::Connect(ECConfig *cfg, bool reconnect,
- unsigned int mysql_flags, unsigned int gcm)
- {
- const char *mysql_port = cfg->GetSetting("mysql_port");
- const char *mysql_socket = cfg->GetSetting("mysql_socket");
- DB_RESULT result;
- DB_ROW row = nullptr;
- std::string query;
- if (*mysql_socket == '\0')
- mysql_socket = nullptr;
- auto er = InitEngine(reconnect);
- if (er != erSuccess) {
- ec_log_crit("KDatabase::Connect(): InitEngine failed %d", er);
- goto exit;
- }
- if (mysql_real_connect(&m_lpMySQL, cfg->GetSetting("mysql_host"),
- cfg->GetSetting("mysql_user"), cfg->GetSetting("mysql_password"),
- cfg->GetSetting("mysql_database"),
- mysql_port ? atoi(mysql_port) : 0,
- mysql_socket, mysql_flags) == nullptr) {
- if (mysql_errno(&m_lpMySQL) == ER_BAD_DB_ERROR)
-
- er = KCERR_DATABASE_NOT_FOUND;
- else
- er = KCERR_DATABASE_ERROR;
- ec_log_err("KDatabase::Connect(): database access error %d, mysql error: %s",
- er, GetError());
- goto exit;
- }
-
- er = DoSelect("SHOW tables", &result);
- if (er != erSuccess) {
- ec_log_err("KDatabase::Connect(): \"SHOW tables\" failed %d", er);
- goto exit;
- }
- if (GetNumRows(result) == 0) {
- er = KCERR_DATABASE_NOT_FOUND;
- ec_log_err("KDatabase::Connect(): database missing %d", er);
- goto exit;
- }
- query = "SHOW variables LIKE 'max_allowed_packet'";
- er = DoSelect(query, &result);
- if (er != erSuccess) {
- ec_log_err("KDatabase::Connect(): max_allowed_packet retrieval failed %d", er);
- goto exit;
- }
- row = FetchRow(result);
-
- if (row == nullptr || row[0] == nullptr || row[1] == nullptr) {
- ec_log_warn("Unable to retrieve max_allowed_packet value. Assuming %d.", KC_DFL_MAX_PACKET_SIZE);
- m_ulMaxAllowedPacket = KC_DFL_MAX_PACKET_SIZE;
- } else {
- m_ulMaxAllowedPacket = atoui(row[1]);
- }
-
- if (m_ulMaxAllowedPacket < KC_DFL_MAX_PACKET_SIZE)
- ec_log_warn("max_allowed_packet is smaller than 16M (%d). You are advised to increase this value by adding max_allowed_packet=16M in the [mysqld] section of my.cnf.", m_ulMaxAllowedPacket);
- m_bConnected = true;
- if (mysql_set_character_set(&m_lpMySQL, "utf8")) {
- ec_log_err("Unable to set character set to \"utf8\"");
- er = KCERR_DATABASE_ERROR;
- goto exit;
- }
- query = "SET SESSION group_concat_max_len = " + stringify(gcm);
- if (Query(query) != 0) {
- ec_log_crit("KDatabase::Connect(): group_concat_max_len set fail: %s", GetError());
- er = KCERR_DATABASE_ERROR;
- goto exit;
- }
- exit:
- if (er != erSuccess)
- Close();
- return er;
- }
- ECRESULT KDatabase::CreateDatabase(ECConfig *cfg, bool reconnect)
- {
- const char *dbname = cfg->GetSetting("mysql_database");
- const char *mysql_port = cfg->GetSetting("mysql_port");
- const char *mysql_socket = cfg->GetSetting("mysql_socket");
- if (*mysql_socket == '\0')
- mysql_socket = nullptr;
-
- auto er = InitEngine(reconnect);
- if (er != erSuccess)
- return er;
-
-
-
-
- if (mysql_real_connect(&m_lpMySQL, cfg->GetSetting("mysql_host"),
- cfg->GetSetting("mysql_user"), cfg->GetSetting("mysql_password"),
- nullptr, mysql_port != nullptr ? atoi(mysql_port) : 0,
- mysql_socket, 0) == nullptr) {
- ec_log_err("Failed to connect to database: %s", GetError());
- return KCERR_DATABASE_ERROR;
- }
- if (dbname == nullptr) {
- ec_log_crit("Unable to create database: Unknown database");
- return KCERR_DATABASE_ERROR;
- }
- ec_log_notice("Create database %s", dbname);
- er = IsInnoDBSupported();
- if (er != erSuccess)
- return er;
- std::string query;
- query = "CREATE DATABASE IF NOT EXISTS `" +
- std::string(cfg->GetSetting("mysql_database")) + "`";
- if (Query(query) != erSuccess) {
- ec_log_err("Unable to create database: %s", GetError());
- return KCERR_DATABASE_ERROR;
- }
- query = "USE `" + std::string(cfg->GetSetting("mysql_database")) + "`";
- er = DoInsert(query);
- if (er != erSuccess)
- return er;
- auto tables = GetDatabaseDefs();
- for (size_t i = 0; tables[i].lpSQL != nullptr; ++i) {
- ec_log_info("Create table: %s", tables[i].lpComment);
- er = DoInsert(tables[i].lpSQL);
- if (er != erSuccess)
- return er;
- }
- ec_log_info("Database structure has been created");
- return erSuccess;
- }
- ECRESULT KDatabase::Close(void)
- {
-
- m_bConnected = false;
- if (m_bMysqlInitialize)
- mysql_close(&m_lpMySQL);
- m_bMysqlInitialize = false;
- return erSuccess;
- }
- ECRESULT KDatabase::DoDelete(const std::string &q, unsigned int *aff)
- {
- autolock alk(*this);
- return _Update(q, aff);
- }
- ECRESULT KDatabase::DoInsert(const std::string &q, unsigned int *idp,
- unsigned int *aff)
- {
- autolock alk(*this);
- auto er = _Update(q, aff);
- if (er == erSuccess && idp != nullptr)
- *idp = GetInsertId();
- return er;
- }
- ECRESULT KDatabase::DoSelect(const std::string &q, DB_RESULT *res_p,
- bool stream)
- {
- assert(q.length() != 0);
- autolock alk(*this);
- if (Query(q) != erSuccess) {
- ec_log_err("KDatabsae::DoSelect(): query failed: %s: %s", q.c_str(), GetError());
- return KCERR_DATABASE_ERROR;
- }
- ECRESULT er = erSuccess;
- DB_RESULT res;
- if (stream)
- res = DB_RESULT(this, mysql_use_result(&m_lpMySQL));
- else
- res = DB_RESULT(this, mysql_store_result(&m_lpMySQL));
- if (res == nullptr) {
- if (!m_bSuppressLockErrorLogging ||
- GetLastError() == DB_E_UNKNOWN)
- ec_log_err("SQL [%08lu] result failed: %s, Query: \"%s\"",
- m_lpMySQL.thread_id, mysql_error(&m_lpMySQL), q.c_str());
- er = KCERR_DATABASE_ERROR;
- }
- if (res_p != nullptr)
- *res_p = std::move(res);
- return er;
- }
- ECRESULT KDatabase::DoSequence(const std::string &seq, unsigned int count,
- unsigned long long *firstidp)
- {
- unsigned int aff = 0;
- autolock alk(*this);
-
- auto er = DoUpdate("UPDATE settings SET value=LAST_INSERT_ID(value+1)+" +
- stringify(count - 1) + " WHERE name = '" + seq + "'", &aff);
- if (er != erSuccess) {
- ec_log_err("KDatabase::DoSequence() UPDATE failed %d", er);
- return er;
- }
-
- if (aff == 0) {
- er = Query("INSERT INTO settings (name, value) VALUES('" +
- seq + "',LAST_INSERT_ID(1)+" + stringify(count - 1) + ")");
- if (er != erSuccess) {
- ec_log_crit("KDatabase::DoSequence() INSERT INTO failed %d", er);
- return er;
- }
- }
- *firstidp = mysql_insert_id(&m_lpMySQL);
- return er;
- }
- ECRESULT KDatabase::DoUpdate(const std::string &q, unsigned int *aff)
- {
- autolock alk(*this);
- return _Update(q, aff);
- }
- std::string KDatabase::Escape(const std::string &s)
- {
- auto size = s.length() * 2 + 1;
- std::unique_ptr<char[]> esc(new char[size]);
- memset(esc.get(), 0, size);
- mysql_real_escape_string(&m_lpMySQL, esc.get(), s.c_str(), s.length());
- return esc.get();
- }
- std::string KDatabase::EscapeBinary(const unsigned char *data, size_t len)
- {
- auto size = len * 2 + 1;
- std::unique_ptr<char[]> esc(new char[size]);
- memset(esc.get(), 0, size);
- mysql_real_escape_string(&m_lpMySQL, esc.get(), reinterpret_cast<const char *>(data), len);
- return "'" + std::string(esc.get()) + "'";
- }
- std::string KDatabase::EscapeBinary(const std::string &s)
- {
- return EscapeBinary(reinterpret_cast<const unsigned char *>(s.c_str()), s.size());
- }
- DB_ROW KDatabase::FetchRow(DB_RESULT &r)
- {
- return mysql_fetch_row(static_cast<MYSQL_RES *>(r.get()));
- }
- DB_LENGTHS KDatabase::FetchRowLengths(DB_RESULT &r)
- {
- return mysql_fetch_lengths(static_cast<MYSQL_RES *>(r.get()));
- }
- void KDatabase::FreeResult_internal(void *r)
- {
- assert(r != nullptr);
- if (r != nullptr)
- mysql_free_result(static_cast<MYSQL_RES *>(r));
- }
- unsigned int KDatabase::GetAffectedRows(void)
- {
- return mysql_affected_rows(&m_lpMySQL);
- }
- const char *KDatabase::GetError(void)
- {
- if (!m_bMysqlInitialize)
- return "MYSQL not initialized";
- return mysql_error(&m_lpMySQL);
- }
- unsigned int KDatabase::GetInsertId(void)
- {
- return mysql_insert_id(&m_lpMySQL);
- }
- DB_ERROR KDatabase::GetLastError(void)
- {
- switch (mysql_errno(&m_lpMySQL)) {
- case ER_LOCK_WAIT_TIMEOUT:
- return DB_E_LOCK_WAIT_TIMEOUT;
- case ER_LOCK_DEADLOCK:
- return DB_E_LOCK_DEADLOCK;
- default:
- return DB_E_UNKNOWN;
- }
- }
- unsigned int KDatabase::GetNumRows(const DB_RESULT &r) const
- {
- return mysql_num_rows(static_cast<MYSQL_RES *>(r.get()));
- }
- ECRESULT KDatabase::InitEngine(bool reconnect)
- {
- assert(!m_bMysqlInitialize);
- if (!m_bMysqlInitialize && mysql_init(&m_lpMySQL) == nullptr) {
- ec_log_crit("KDatabase::InitEngine() mysql_init failed");
- return KCERR_DATABASE_ERROR;
- }
- m_bMysqlInitialize = true;
- m_lpMySQL.reconnect = reconnect;
- return erSuccess;
- }
- ECRESULT KDatabase::IsInnoDBSupported(void)
- {
- DB_RESULT res;
- DB_ROW row = nullptr;
- auto er = DoSelect("SHOW ENGINES", &res);
- if (er != erSuccess) {
- ec_log_crit("Unable to query supported database engines. Error: %s", GetError());
- return er;
- }
- while ((row = FetchRow(res)) != nullptr) {
- if (strcasecmp(row[0], "InnoDB") != 0)
- continue;
- if (strcasecmp(row[1], "DISABLED") == 0) {
-
- ec_log_crit("INNODB engine is disabled. Please re-enable the INNODB engine. Check your MySQL log for more information or comment out skip-innodb in the mysql configuration file.");
- return KCERR_DATABASE_ERROR;
- } else if (strcasecmp(row[1], "YES") != 0 && strcasecmp(row[1], "DEFAULT") != 0) {
-
- ec_log_crit("INNODB engine is not supported. Please enable the INNODB engine in the mysql configuration file.");
- return KCERR_DATABASE_ERROR;
- }
- break;
- }
- if (row == nullptr) {
- ec_log_crit("Unable to find 'InnoDB' engine from the mysql server. Probably INNODB is not supported.");
- return KCERR_DATABASE_ERROR;
- }
- return erSuccess;
- }
- ECRESULT KDatabase::Query(const std::string &q)
- {
- LOG_SQL_DEBUG("SQL [%08lu]: \"%s;\"", m_lpMySQL.thread_id, q.c_str());
-
- auto err = mysql_real_query(&m_lpMySQL, q.c_str(), q.length());
- if (err == 0)
- return erSuccess;
-
- if (m_lpMySQL.reconnect)
- ec_log_err("%p: SQL Failed: %s, Query: \"%s\"",
- static_cast<void *>(&m_lpMySQL), mysql_error(&m_lpMySQL),
- q.c_str());
- return KCERR_DATABASE_ERROR;
- }
- ECRESULT KDatabase::_Update(const std::string &q, unsigned int *aff)
- {
- if (Query(q) != 0) {
- ec_log_err("KDatabase::_Update() query failed: %s: %s",
- q.c_str(), GetError());
- return KCERR_DATABASE_ERROR;
- }
- if (aff != nullptr)
- *aff = GetAffectedRows();
- return erSuccess;
- }
|