// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // #pragma once #include "td/telegram/DialogId.h" #include "td/utils/common.h" #include "td/utils/logging.h" #include "td/utils/misc.h" #include "td/utils/StringBuilder.h" #include "td/utils/tl_helpers.h" #include <functional> #include <limits> #include <type_traits> namespace td { class ServerMessageId { int32 id = 0; public: ServerMessageId() = default; explicit ServerMessageId(int32 message_id) : id(message_id) { } template <class T, typename = std::enable_if_t<std::is_convertible<T, int32>::value>> ServerMessageId(T message_id) = delete; bool is_valid() const { return id > 0; } int32 get() const { return id; } bool operator==(const ServerMessageId &other) const { return id == other.id; } bool operator!=(const ServerMessageId &other) const { return id != other.id; } template <class StorerT> void store(StorerT &storer) const { storer.store_int(id); } template <class ParserT> void parse(ParserT &parser) { id = parser.fetch_int(); } }; enum class MessageType : int32 { None, Server, Local, YetUnsent }; class MessageId { int64 id = 0; public: static constexpr int32 SERVER_ID_SHIFT = 20; static constexpr int32 TYPE_MASK = (1 << 3) - 1; static constexpr int32 FULL_TYPE_MASK = (1 << SERVER_ID_SHIFT) - 1; static constexpr int32 TYPE_YET_UNSENT = 1; static constexpr int32 TYPE_LOCAL = 2; MessageId() = default; explicit MessageId(ServerMessageId server_message_id) : id(static_cast<int64>(server_message_id.get()) << SERVER_ID_SHIFT) { } explicit constexpr MessageId(int64 message_id) : id(message_id) { } template <class T, typename = std::enable_if_t<std::is_convertible<T, int64>::value>> MessageId(T message_id) = delete; static constexpr MessageId min() { return MessageId(static_cast<int64>(MessageId::TYPE_LOCAL)); } static constexpr MessageId max() { return MessageId(static_cast<int64>(std::numeric_limits<int32>::max()) << SERVER_ID_SHIFT); } bool is_valid() const { if (id <= 0 || id > max().get()) { return false; } if ((id & FULL_TYPE_MASK) == 0) { return true; } int32 type = (id & TYPE_MASK); return type == TYPE_YET_UNSENT || type == TYPE_LOCAL; } int64 get() const { return id; } MessageType get_type() const { if (id <= 0 || id > max().get()) { return MessageType::None; } if ((id & FULL_TYPE_MASK) == 0) { return MessageType::Server; } switch (id & TYPE_MASK) { case TYPE_YET_UNSENT: return MessageType::YetUnsent; case TYPE_LOCAL: return MessageType::Local; default: return MessageType::None; } } bool is_yet_unsent() const { CHECK(is_valid()); return (id & TYPE_MASK) == TYPE_YET_UNSENT; } bool is_local() const { CHECK(is_valid()); return (id & TYPE_MASK) == TYPE_LOCAL; } bool is_server() const { CHECK(is_valid()); return (id & FULL_TYPE_MASK) == 0; } ServerMessageId get_server_message_id() const { CHECK(id == 0 || is_server()); return ServerMessageId(narrow_cast<int32>(id >> SERVER_ID_SHIFT)); } // returns greatest server message id not bigger than this message id MessageId get_prev_server_message_id() const { return MessageId(id & ~FULL_TYPE_MASK); } // returns smallest server message id not less than this message id MessageId get_next_server_message_id() const { return MessageId((id + FULL_TYPE_MASK) & ~FULL_TYPE_MASK); } bool operator==(const MessageId &other) const { return id == other.id; } bool operator!=(const MessageId &other) const { return id != other.id; } template <class StorerT> void store(StorerT &storer) const { storer.store_long(id); } template <class ParserT> void parse(ParserT &parser) { id = parser.fetch_long(); } }; struct MessageIdHash { std::size_t operator()(MessageId message_id) const { return std::hash<int64>()(message_id.get()); } }; inline StringBuilder &operator<<(StringBuilder &string_builder, MessageId message_id) { if (!message_id.is_valid()) { return string_builder << "invalid message " << message_id.get(); } if (message_id.is_server()) { return string_builder << "server message " << (message_id.get() >> MessageId::SERVER_ID_SHIFT); } if (message_id.is_local()) { return string_builder << "local message " << (message_id.get() >> MessageId::SERVER_ID_SHIFT) << '.' << (message_id.get() & MessageId::FULL_TYPE_MASK); } if (message_id.is_yet_unsent()) { return string_builder << "yet unsent message " << (message_id.get() >> MessageId::SERVER_ID_SHIFT) << '.' << (message_id.get() & MessageId::FULL_TYPE_MASK); } return string_builder << "bugged message " << message_id.get(); } struct FullMessageId { private: DialogId dialog_id; MessageId message_id; public: FullMessageId() : dialog_id(), message_id() { } FullMessageId(DialogId dialog_id, MessageId message_id) : dialog_id(dialog_id), message_id(message_id) { } bool operator==(const FullMessageId &other) const { return dialog_id == other.dialog_id && message_id == other.message_id; } bool operator!=(const FullMessageId &other) const { return !(*this == other); } DialogId get_dialog_id() const { return dialog_id; } MessageId get_message_id() const { return message_id; } template <class StorerT> void store(StorerT &storer) const { using ::td::store; store(dialog_id, storer); store(message_id, storer); } template <class ParserT> void parse(ParserT &parser) { using ::td::parse; parse(dialog_id, parser); parse(message_id, parser); } }; struct FullMessageIdHash { std::size_t operator()(FullMessageId full_message_id) const { return DialogIdHash()(full_message_id.get_dialog_id()) * 2023654985u + MessageIdHash()(full_message_id.get_message_id()); } }; inline StringBuilder &operator<<(StringBuilder &string_builder, FullMessageId full_message_id) { return string_builder << full_message_id.get_message_id() << " in " << full_message_id.get_dialog_id(); } } // namespace td