use std::{borrow::Cow, collections::HashSet};
use crate::{
CacheableModels, CacheableSticker, GuildResource, InMemoryCache, UpdateCache,
config::ResourceType,
};
use twilight_model::{
channel::message::Sticker,
gateway::payload::incoming::GuildStickersUpdate,
id::{Id, marker::GuildMarker},
};
impl<CacheModels: CacheableModels> InMemoryCache<CacheModels> {
pub(crate) fn cache_stickers(&self, guild_id: Id<GuildMarker>, stickers: Vec<Sticker>) {
if let Some(mut guild_stickers) = self.guild_stickers.get_mut(&guild_id) {
let incoming_sticker_ids = stickers
.iter()
.map(|sticker| sticker.id)
.collect::<HashSet<_>>();
guild_stickers.retain(|sticker_id| {
let retain = incoming_sticker_ids.contains(sticker_id);
if !retain {
self.stickers.remove(sticker_id);
}
retain
});
}
for sticker in stickers {
self.cache_sticker(guild_id, sticker);
}
}
pub(crate) fn cache_sticker(&self, guild_id: Id<GuildMarker>, sticker: Sticker) {
if let Some(cached_sticker) = self.stickers.get(&sticker.id)
&& cached_sticker.value == sticker
{
return;
}
if let Some(user) = sticker.user.clone() {
self.cache_user(Cow::Owned(user), Some(guild_id));
}
let sticker_id = sticker.id;
let cached = CacheModels::Sticker::from(sticker);
self.stickers.insert(
cached.id(),
GuildResource {
guild_id,
value: cached,
},
);
self.guild_stickers
.entry(guild_id)
.or_default()
.insert(sticker_id);
}
}
impl<CacheModels: CacheableModels> UpdateCache<CacheModels> for GuildStickersUpdate {
fn update(&self, cache: &InMemoryCache<CacheModels>) {
if !cache.wants(ResourceType::STICKER) {
return;
}
cache.cache_stickers(self.guild_id, self.stickers.clone());
}
}
#[cfg(test)]
mod tests {
use crate::{DefaultCacheModels, InMemoryCache, test};
use twilight_model::id::{
Id,
marker::{GuildMarker, StickerMarker},
};
const GUILD_ID: Id<GuildMarker> = Id::new(1);
const STICKER_ONE_ID: Id<StickerMarker> = Id::new(2);
const STICKER_TWO_ID: Id<StickerMarker> = Id::new(3);
fn cache_with_stickers() -> InMemoryCache<DefaultCacheModels> {
let cache = test::cache();
let one = test::sticker(STICKER_ONE_ID, GUILD_ID);
let two = test::sticker(STICKER_TWO_ID, GUILD_ID);
cache.cache_stickers(GUILD_ID, Vec::from([one, two]));
cache
}
#[test]
fn cache_stickers() {
let cache = cache_with_stickers();
assert_eq!(cache.stickers.len(), 2);
let one = test::sticker(STICKER_ONE_ID, GUILD_ID);
let two = test::sticker(STICKER_TWO_ID, GUILD_ID);
assert!(
cache
.stickers
.get(&STICKER_ONE_ID)
.is_some_and(|r| r.id == STICKER_ONE_ID)
);
assert!(
cache
.stickers
.get(&STICKER_TWO_ID)
.is_some_and(|r| r.id == STICKER_TWO_ID)
);
let guild_stickers = cache
.guild_stickers
.get(&GUILD_ID)
.expect("cache has stickers for guild");
assert_eq!(guild_stickers.len(), 2);
assert!(guild_stickers.contains(&one.id));
assert!(guild_stickers.contains(&two.id));
}
#[test]
fn cache_stickers_removal() {
let cache = cache_with_stickers();
let one = test::sticker(STICKER_ONE_ID, GUILD_ID);
cache.cache_stickers(GUILD_ID, Vec::from([one]));
assert_eq!(cache.stickers.len(), 1);
assert!(
cache
.stickers
.get(&STICKER_ONE_ID)
.is_some_and(|r| r.id == STICKER_ONE_ID)
);
let guild_stickers = cache
.guild_stickers
.get(&GUILD_ID)
.expect("cache has stickers for guild");
assert_eq!(guild_stickers.len(), 1);
assert!(guild_stickers.contains(&STICKER_ONE_ID));
}
}