// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2023 // // 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/telegram/files/FileId.h" #include "td/telegram/files/FileSourceId.h" #include "td/telegram/FullMessageId.h" #include "td/telegram/MessageEntity.h" #include "td/telegram/StoryFullId.h" #include "td/telegram/StoryId.h" #include "td/telegram/StoryInteractionInfo.h" #include "td/telegram/td_api.h" #include "td/telegram/UserId.h" #include "td/telegram/UserPrivacySettingRule.h" #include "td/actor/actor.h" #include "td/utils/common.h" #include "td/utils/Promise.h" #include "td/utils/WaitFreeHashMap.h" #include "td/utils/WaitFreeHashSet.h" #include namespace td { struct BinlogEvent; class StoryContent; class Td; class StoryManager final : public Actor { struct Story { int32 date_ = 0; int32 expire_date_ = 0; bool is_pinned_ = false; bool is_public_ = false; bool is_for_close_friends_ = false; mutable bool is_update_sent_ = false; // whether the story is known to the app StoryInteractionInfo interaction_info_; UserPrivacySettingRules privacy_rules_; unique_ptr content_; FormattedText caption_; mutable int64 edit_generation_ = 0; }; struct BeingEditedStory { unique_ptr content_; FormattedText caption_; bool edit_caption_ = false; vector> promises_; }; struct PendingStory { DialogId dialog_id_; StoryId story_id_; uint64 log_event_id_ = 0; uint32 send_story_num_ = 0; int64 random_id_ = 0; bool was_reuploaded_ = false; unique_ptr story_; PendingStory(DialogId dialog_id, StoryId story_id, uint64 log_event_id, uint32 send_story_num, int64 random_id, unique_ptr &&story); }; struct PendingStoryViews { FlatHashSet story_ids_; bool has_query_ = false; }; struct ActiveStories { StoryId max_read_story_id_; vector story_ids_; }; public: StoryManager(Td *td, ActorShared<> parent); StoryManager(const StoryManager &) = delete; StoryManager &operator=(const StoryManager &) = delete; StoryManager(StoryManager &&) = delete; StoryManager &operator=(StoryManager &&) = delete; ~StoryManager() final; void get_story(DialogId owner_dialog_id, StoryId story_id, Promise> &&promise); void send_story(td_api::object_ptr &&input_story_content, td_api::object_ptr &&input_caption, td_api::object_ptr &&rules, int32 active_period, bool is_pinned, Promise> &&promise); void on_send_story_file_part_missing(unique_ptr &&pending_story, int bad_part); void edit_story(StoryId story_id, td_api::object_ptr &&input_story_content, td_api::object_ptr &&input_caption, Promise &&promise); void set_story_privacy_rules(StoryId story_id, td_api::object_ptr &&rules, Promise &&promise); void toggle_story_is_pinned(StoryId story_id, bool is_pinned, Promise &&promise); void delete_story(StoryId story_id, Promise &&promise); void toggle_dialog_stories_hidden(DialogId dialog_id, bool are_hidden, Promise &&promise); void get_dialog_pinned_stories(DialogId owner_dialog_id, StoryId from_story_id, int32 limit, Promise> &&promise); void get_story_archive(StoryId from_story_id, int32 limit, Promise> &&promise); void get_dialog_expiring_stories(DialogId owner_dialog_id, Promise> &&promise); void open_story(DialogId owner_dialog_id, StoryId story_id, Promise &&promise); StoryId on_get_story(DialogId owner_dialog_id, telegram_api::object_ptr &&story_item_ptr); std::pair> on_get_stories(DialogId owner_dialog_id, vector &&expected_story_ids, telegram_api::object_ptr &&stories); bool have_story(StoryFullId story_full_id) const; bool have_story_force(StoryFullId story_full_id) const; bool is_inaccessible_story(StoryFullId story_full_id) const; int32 get_story_duration(StoryFullId story_full_id) const; void register_story(StoryFullId story_full_id, FullMessageId full_message_id, const char *source); void unregister_story(StoryFullId story_full_id, FullMessageId full_message_id, const char *source); td_api::object_ptr get_story_object(StoryFullId story_full_id) const; td_api::object_ptr get_stories_object(int32 total_count, const vector &story_full_ids) const; FileSourceId get_story_file_source_id(StoryFullId story_full_id); telegram_api::object_ptr get_input_media(StoryFullId story_full_id) const; void reload_story(StoryFullId story_full_id, Promise &&promise); void on_binlog_events(vector &&events); private: class UploadMediaCallback; class SendStoryQuery; class EditStoryQuery; class DeleteStoryOnServerLogEvent; void tear_down() final; bool is_story_owned(DialogId owner_dialog_id) const; bool is_active_story(StoryFullId story_full_id) const; bool is_active_story(const Story *story) const; const Story *get_story(StoryFullId story_full_id) const; Story *get_story_editable(StoryFullId story_full_id); void on_story_changed(StoryFullId story_full_id, const Story *story, bool is_changed, bool need_save_to_database); td_api::object_ptr get_story_info_object(StoryFullId story_full_id) const; td_api::object_ptr get_story_info_object(StoryFullId story_full_id, const Story *story) const; td_api::object_ptr get_story_object(StoryFullId story_full_id, const Story *story) const; td_api::object_ptr get_active_stories_object(DialogId owner_dialog_id) const; StoryId on_get_story(DialogId owner_dialog_id, telegram_api::object_ptr &&story_item); StoryId on_get_skipped_story(DialogId owner_dialog_id, telegram_api::object_ptr &&story_item); StoryId on_get_deleted_story(DialogId owner_dialog_id, telegram_api::object_ptr &&story_item); void on_delete_story(DialogId owner_dialog_id, StoryId story_id); DialogId on_get_user_stories(DialogId owner_dialog_id, telegram_api::object_ptr &&user_stories); void on_get_dialog_pinned_stories(DialogId owner_dialog_id, telegram_api::object_ptr &&stories, Promise> &&promise); void on_get_story_archive(telegram_api::object_ptr &&stories, Promise> &&promise); void on_get_dialog_expiring_stories(DialogId owner_dialog_id, telegram_api::object_ptr &&stories, Promise> &&promise); vector get_story_file_ids(const Story *story) const; static uint64 save_delete_story_on_server_log_event(DialogId dialog_id, StoryId story_id); void delete_story_on_server(DialogId dialog_id, StoryId story_id, uint64 log_event_id, Promise &&promise); void delete_story_files(const Story *story) const; void change_story_files(StoryFullId story_full_id, const Story *story, const vector &old_file_ids); void do_get_story(StoryFullId story_full_id, Result &&result, Promise> &&promise); void do_send_story(unique_ptr &&pending_story, vector bad_parts); void on_upload_story(FileId file_id, telegram_api::object_ptr input_file); void on_upload_story_error(FileId file_id, Status status); void do_edit_story(FileId file_id, unique_ptr &&pending_story, telegram_api::object_ptr input_file); void on_story_edited(FileId file_id, unique_ptr pending_story, Result result); void on_toggle_story_is_pinned(StoryId story_id, bool is_pinned, Promise &&promise); void on_update_active_stories(DialogId owner_dialog_id, StoryId max_read_story_id, vector &&story_ids); void send_update_active_stories(DialogId owner_dialog_id); void increment_story_views(DialogId owner_dialog_id, PendingStoryViews &story_views); void on_increment_story_views(DialogId owner_dialog_id); std::shared_ptr upload_media_callback_; WaitFreeHashMap story_full_id_to_file_source_id_; WaitFreeHashMap, StoryFullIdHash> stories_; WaitFreeHashSet inaccessible_story_full_ids_; WaitFreeHashSet deleted_story_full_ids_; WaitFreeHashMap, StoryFullIdHash> story_messages_; WaitFreeHashMap, DialogIdHash> active_stories_; FlatHashMap, StoryFullIdHash> being_edited_stories_; FlatHashMap pending_story_views_; uint32 send_story_count_ = 0; FlatHashMap, FileIdHash> being_uploaded_files_; Td *td_; ActorShared<> parent_; }; } // namespace td