twilight_cache_inmemory/
traits.rs

1//! Traits for implementing a [`InMemoryCache`] with custom structs.
2//!
3//! By default, the cache uses widely compatible default types that contain almost all
4//! fields that are present in the Discord API. Fields that are never used by the user
5//! will result in excess memory usage that will especially matter to big bots with a
6//! lot of cached data.
7//!
8//! The traits in this module allow creating custom cached representations of Discord
9//! API models compatible with the [`InMemoryCache`]. They may be mixed with the default
10//! types provided by twilight, which also implement these traits.
11//!
12//! However, as Discord extends its API models with new fields or changes the types,
13//! the trait definitions may change in minor crate releases to allow twilight to keep
14//! up with upstream API changes. Since not all fields are required for caching logic,
15//! this is not very likely to happen on a regular basis, but should be considered when
16//! deciding to opt for writing custom types.
17//!
18//! Many traits require getters for certain types, which means they are used for caching
19//! logic. However, users generally won't have to store all the fields. It is possible
20//! to return `None` or empty arrays on most of the methods if the data that is accessed
21//! is not stored in the custom implementation.
22//!
23//! [`InMemoryCache`]: crate::InMemoryCache
24
25use crate::model::member::ComputedInteractionMember;
26use std::fmt::Debug;
27use twilight_model::{
28    application::interaction::InteractionMember,
29    channel::{
30        message::{Reaction, Sticker},
31        Channel, ChannelType, Message, StageInstance,
32    },
33    gateway::{
34        payload::incoming::{GuildUpdate, MemberUpdate},
35        presence::Presence,
36    },
37    guild::{
38        scheduled_event::GuildScheduledEvent, Emoji, Guild, GuildIntegration, Member,
39        PartialMember, Role,
40    },
41    id::{
42        marker::{
43            ChannelMarker, GuildMarker, RoleMarker, ScheduledEventMarker, StickerMarker, UserMarker,
44        },
45        Id,
46    },
47    user::{CurrentUser, User},
48    util::{ImageHash, Timestamp},
49    voice::VoiceState,
50};
51#[cfg(feature = "permission-calculator")]
52use twilight_model::{channel::permission_overwrite::PermissionOverwrite, guild::Permissions};
53
54/// Super-trait for the generic cached representations of Discord API models.
55pub trait CacheableModels: Clone + Debug {
56    /// The cached [`Channel`] model representation.
57    type Channel: CacheableChannel;
58    /// The cached [`CurrentUser`] model representation.
59    type CurrentUser: CacheableCurrentUser;
60    /// The cached [`Emoji`] model representation.
61    type Emoji: CacheableEmoji;
62    /// The cached [`Guild`] model representation.
63    type Guild: CacheableGuild;
64    /// The cached [`GuildIntegration`] model representation.
65    type GuildIntegration: CacheableGuildIntegration;
66    /// The cached [`GuildScheduledEvent` model representation.
67    type GuildScheduledEvent: CacheableGuildScheduledEvent;
68    /// The cached [`Member`] model representation.
69    type Member: CacheableMember;
70    /// The cached [`Message`] model representation.
71    type Message: CacheableMessage;
72    /// The cached [`Presence`] model representation.
73    type Presence: CacheablePresence;
74    /// The cached [`Role`] model representation.
75    type Role: CacheableRole;
76    /// The cached [`StageInstance`] model representation.
77    type StageInstance: CacheableStageInstance;
78    /// The cached [`Sticker`] model representation.
79    type Sticker: CacheableSticker;
80    /// The cached [`User`] model representation.
81    type User: CacheableUser;
82    /// The cached [`VoiceState`] model representation.
83    type VoiceState: CacheableVoiceState;
84}
85
86/// Trait for a generic cached representation of a [`Member`].
87pub trait CacheableMember:
88    From<Member>
89    + From<ComputedInteractionMember>
90    + From<(Id<UserMarker>, PartialMember)>
91    + PartialEq<Member>
92    + PartialEq<PartialMember>
93    + PartialEq<InteractionMember>
94    + PartialEq<Self>
95    + Clone
96    + Debug
97{
98    /// Roles of this member.
99    fn roles(&self) -> &[Id<RoleMarker>];
100
101    /// Timestamp until which this member's communication is disabled.
102    #[cfg(feature = "permission-calculator")]
103    fn communication_disabled_until(&self) -> Option<Timestamp>;
104
105    /// Avatar of this member.
106    fn avatar(&self) -> Option<ImageHash>;
107
108    /// Whether this member is deafened.
109    fn deaf(&self) -> Option<bool>;
110
111    /// Whether this member is muted.
112    fn mute(&self) -> Option<bool>;
113
114    /// Update the cached data with a [`MemberUpdate`] event.
115    fn update_with_member_update(&mut self, member_update: &MemberUpdate);
116}
117
118/// Trait for a generic cached representation of a [`Role`].
119pub trait CacheableRole: From<Role> + PartialEq<Role> + PartialEq<Self> + Clone + Debug {
120    /// Role's position in the guild roles.
121    fn position(&self) -> i64;
122
123    /// ID of the role.
124    fn id(&self) -> Id<RoleMarker>;
125
126    /// Permissions granted to members with the role.
127    #[cfg(feature = "permission-calculator")]
128    fn permissions(&self) -> Permissions;
129}
130
131impl CacheableRole for Role {
132    fn position(&self) -> i64 {
133        self.position
134    }
135
136    fn id(&self) -> Id<RoleMarker> {
137        self.id
138    }
139
140    #[cfg(feature = "permission-calculator")]
141    fn permissions(&self) -> Permissions {
142        self.permissions
143    }
144}
145
146/// Trait for a generic cached representation of a [`Channel`].
147pub trait CacheableChannel:
148    From<Channel> + PartialEq<Channel> + PartialEq<Self> + Clone + Debug
149{
150    /// ID of the guild this channel belongs to.
151    fn guild_id(&self) -> Option<Id<GuildMarker>>;
152
153    /// Type of the channel.
154    fn kind(&self) -> ChannelType;
155
156    /// ID of the parent channel if this is a thread.
157    #[cfg(feature = "permission-calculator")]
158    fn parent_id(&self) -> Option<Id<ChannelMarker>>;
159
160    /// ID of the channel.
161    fn id(&self) -> Id<ChannelMarker>;
162
163    /// Permission overwrites for the channel.
164    #[cfg(feature = "permission-calculator")]
165    fn permission_overwrites(&self) -> Option<&[PermissionOverwrite]>;
166
167    /// Set the last pin timestamp to a new timestamp.
168    fn set_last_pin_timestamp(&mut self, timestamp: Option<Timestamp>);
169}
170
171impl CacheableChannel for Channel {
172    fn guild_id(&self) -> Option<Id<GuildMarker>> {
173        self.guild_id
174    }
175
176    fn kind(&self) -> ChannelType {
177        self.kind
178    }
179
180    #[cfg(feature = "permission-calculator")]
181    fn parent_id(&self) -> Option<Id<ChannelMarker>> {
182        self.parent_id
183    }
184
185    fn id(&self) -> Id<ChannelMarker> {
186        self.id
187    }
188
189    #[cfg(feature = "permission-calculator")]
190    fn permission_overwrites(&self) -> Option<&[PermissionOverwrite]> {
191        self.permission_overwrites.as_deref()
192    }
193
194    fn set_last_pin_timestamp(&mut self, timestamp: Option<Timestamp>) {
195        self.last_pin_timestamp = timestamp;
196    }
197}
198
199/// Trait for a generic cached representation of a [`Guild`].
200pub trait CacheableGuild: From<Guild> + PartialEq<Guild> + PartialEq<Self> + Clone + Debug {
201    /// ID of the guild.
202    fn id(&self) -> Id<GuildMarker>;
203
204    /// ID of the guild's owner.
205    #[cfg(feature = "permission-calculator")]
206    fn owner_id(&self) -> Id<UserMarker>;
207
208    /// Set the guild's unavailable flag.
209    fn set_unavailable(&mut self, unavailable: Option<bool>);
210
211    /// Update the cached data with a [`GuildUpdate`] event. Fields containing other
212    /// cached structures such as channels are cleared prior.
213    fn update_with_guild_update(&mut self, guild_update: &GuildUpdate);
214
215    /// Increase the guild member count.
216    fn increase_member_count(&mut self, amount: u64);
217
218    /// Decrease the guild member count.
219    fn decrease_member_count(&mut self, amount: u64);
220}
221
222/// Trait for a generic cached representation of a [`VoiceState`].
223pub trait CacheableVoiceState:
224    From<(Id<ChannelMarker>, Id<GuildMarker>, VoiceState)>
225    + PartialEq<VoiceState>
226    + PartialEq<Self>
227    + Clone
228    + Debug
229{
230    /// ID of the channel this voice state belongs to.
231    fn channel_id(&self) -> Id<ChannelMarker>;
232}
233
234/// Trait for a generic cached representation of a [`Message`].
235pub trait CacheableMessage:
236    From<Message> + PartialEq<Message> + PartialEq<Self> + Clone + Debug
237{
238    /// Reactions added to this message.
239    fn reactions(&self) -> &[Reaction];
240
241    /// Mutable getter for reactions added to this message.
242    fn reactions_mut(&mut self) -> &mut [Reaction];
243
244    /// Retain all reactions to this message matching a predicate, removing non-matching ones.
245    fn retain_reactions(&mut self, f: impl FnMut(&Reaction) -> bool);
246
247    /// Clear all reactions to this message.
248    fn clear_reactions(&mut self);
249
250    /// Add a reaction to this message.
251    fn add_reaction(&mut self, reaction: Reaction);
252
253    /// Remove a reaction from this message.
254    fn remove_reaction(&mut self, idx: usize);
255}
256
257/// Trait for a generic cached representation of a [`CurrentUser`].
258pub trait CacheableCurrentUser:
259    From<CurrentUser> + PartialEq<CurrentUser> + PartialEq<Self> + Clone + Debug
260{
261    /// ID of the user.
262    fn id(&self) -> Id<UserMarker>;
263}
264
265impl CacheableCurrentUser for CurrentUser {
266    fn id(&self) -> Id<UserMarker> {
267        self.id
268    }
269}
270
271/// Trait for a generic cached representation of a [`Sticker`].
272pub trait CacheableSticker:
273    From<Sticker> + PartialEq<Sticker> + PartialEq<Self> + Clone + Debug
274{
275    /// ID of the sticker.
276    fn id(&self) -> Id<StickerMarker>;
277}
278
279/// Trait for a generic cached representation of a [`Emoji`].
280pub trait CacheableEmoji: From<Emoji> + PartialEq<Emoji> + PartialEq<Self> + Clone + Debug {}
281
282/// Trait for a generic cached representation of a [`GuildIntegration`].
283pub trait CacheableGuildIntegration:
284    From<GuildIntegration> + PartialEq<GuildIntegration> + PartialEq<Self> + Clone + Debug
285{
286}
287
288impl CacheableGuildIntegration for GuildIntegration {}
289
290/// Trait for a generic cached representation of a [`Presence`].
291pub trait CacheablePresence:
292    From<Presence> + PartialEq<Presence> + PartialEq<Self> + Clone + Debug
293{
294}
295
296/// Trait for a generic cached representation of a [`StageInstance`].
297pub trait CacheableStageInstance:
298    From<StageInstance> + PartialEq<StageInstance> + PartialEq<Self> + Clone + Debug
299{
300}
301
302impl CacheableStageInstance for StageInstance {}
303
304/// Trait for a generic cached representation of a [`User`].
305pub trait CacheableUser: From<User> + PartialEq<User> + PartialEq<Self> + Clone + Debug {}
306
307impl CacheableUser for User {}
308
309/// Trait for a generic cached representation of a [`GuildScheduledEvent`].
310pub trait CacheableGuildScheduledEvent:
311    From<GuildScheduledEvent> + PartialEq<GuildScheduledEvent> + PartialEq<Self> + Clone + Debug
312{
313    /// Add a user to an event.
314    fn add_user(
315        &mut self,
316        guild_id: Id<GuildMarker>,
317        event_id: Id<ScheduledEventMarker>,
318        user_id: Id<UserMarker>,
319    );
320
321    /// Remove a user from an event.
322    fn remove_user(
323        &mut self,
324        guild_id: Id<GuildMarker>,
325        event_id: Id<ScheduledEventMarker>,
326        user_id: Id<UserMarker>,
327    );
328}
329
330impl CacheableGuildScheduledEvent for GuildScheduledEvent {
331    fn add_user(
332        &mut self,
333        _guild_id: Id<GuildMarker>,
334        _event_id: Id<ScheduledEventMarker>,
335        _user_id: Id<UserMarker>,
336    ) {
337        self.user_count = self.user_count.map(|count| count.saturating_add(1));
338    }
339
340    fn remove_user(
341        &mut self,
342        _guild_id: Id<GuildMarker>,
343        _event_id: Id<ScheduledEventMarker>,
344        _user_id: Id<UserMarker>,
345    ) {
346        self.user_count = self.user_count.map(|count| count.saturating_sub(1));
347    }
348}