/* * This file is part of the AzerothCore Project. See AUTHORS file for Copyright information * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU Affero General Public License as published by the * Free Software Foundation; either version 3 of the License, or (at your * option) any later version. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for * more details. * * You should have received a copy of the GNU General Public License along * with this program. If not, see . */ #ifndef _TRANSACTION_H #define _TRANSACTION_H #include "DatabaseEnvFwd.h" #include "Define.h" #include "SQLOperation.h" #include "StringFormat.h" #include #include #include /*! Transactions, high level class. */ class AC_DATABASE_API TransactionBase { friend class TransactionTask; friend class MySQLConnection; template friend class DatabaseWorkerPool; public: TransactionBase() = default; virtual ~TransactionBase() { Cleanup(); } void Append(std::string_view sql); template void Append(std::string_view sql, Args&&... args) { Append(Acore::StringFormat(sql, std::forward(args)...)); } [[nodiscard]] std::size_t GetSize() const { return m_queries.size(); } protected: void AppendPreparedStatement(PreparedStatementBase* statement); void Cleanup(); std::vector m_queries; private: bool _cleanedUp{false}; }; template class Transaction : public TransactionBase { public: using TransactionBase::Append; void Append(PreparedStatement* statement) { AppendPreparedStatement(statement); } }; /*! Low level class*/ class AC_DATABASE_API TransactionTask : public SQLOperation { template friend class DatabaseWorkerPool; friend class DatabaseWorker; friend class TransactionCallback; public: TransactionTask(std::shared_ptr trans) : m_trans(std::move(trans)) { } ~TransactionTask() override = default; protected: bool Execute() override; int TryExecute(); void CleanupOnFailure(); std::shared_ptr m_trans; static std::mutex _deadlockLock; }; class AC_DATABASE_API TransactionWithResultTask : public TransactionTask { public: TransactionWithResultTask(std::shared_ptr trans) : TransactionTask(trans) { } TransactionFuture GetFuture() { return m_result.get_future(); } protected: bool Execute() override; TransactionPromise m_result; }; class AC_DATABASE_API TransactionCallback { public: TransactionCallback(TransactionFuture&& future) : m_future(std::move(future)) { } TransactionCallback(TransactionCallback&&) = default; TransactionCallback& operator=(TransactionCallback&&) = default; void AfterComplete(std::function callback) & { m_callback = std::move(callback); } bool InvokeIfReady(); TransactionFuture m_future; std::function m_callback; }; #endif