2024-06-24 13:56:01 +01:00
|
|
|
package cc.fascinated.bat.service;
|
|
|
|
|
2024-07-01 01:12:32 +01:00
|
|
|
import cc.fascinated.bat.common.TimerUtils;
|
2024-07-01 01:33:52 +01:00
|
|
|
import cc.fascinated.bat.event.EventListener;
|
2024-07-01 19:41:13 +01:00
|
|
|
import cc.fascinated.bat.model.BatGuild;
|
2024-06-25 11:55:26 +01:00
|
|
|
import cc.fascinated.bat.model.BatUser;
|
2024-07-01 01:12:32 +01:00
|
|
|
import com.mongodb.client.model.Filters;
|
2024-06-27 13:00:45 +01:00
|
|
|
import lombok.Getter;
|
2024-06-24 13:56:01 +01:00
|
|
|
import lombok.NonNull;
|
|
|
|
import lombok.extern.log4j.Log4j2;
|
2024-07-03 22:59:05 +01:00
|
|
|
import net.dv8tion.jda.api.entities.User;
|
2024-07-01 19:41:13 +01:00
|
|
|
import net.dv8tion.jda.api.events.guild.member.GuildMemberJoinEvent;
|
2024-07-01 19:40:32 +01:00
|
|
|
import net.dv8tion.jda.api.events.user.update.UserUpdateGlobalNameEvent;
|
2024-07-06 21:14:20 +01:00
|
|
|
import net.jodah.expiringmap.ExpirationPolicy;
|
2024-07-06 00:34:53 +01:00
|
|
|
import net.jodah.expiringmap.ExpiringMap;
|
2024-07-01 01:12:32 +01:00
|
|
|
import org.bson.Document;
|
2024-06-24 13:56:01 +01:00
|
|
|
import org.springframework.beans.factory.annotation.Autowired;
|
|
|
|
import org.springframework.context.annotation.DependsOn;
|
|
|
|
import org.springframework.stereotype.Service;
|
|
|
|
|
2024-07-01 01:12:32 +01:00
|
|
|
import java.util.HashMap;
|
2024-06-27 13:00:45 +01:00
|
|
|
import java.util.Map;
|
2024-06-27 23:51:40 +01:00
|
|
|
import java.util.concurrent.TimeUnit;
|
2024-06-24 13:56:01 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @author Fascinated (fascinated7)
|
|
|
|
*/
|
2024-06-28 03:01:21 +01:00
|
|
|
@Service
|
2024-07-01 01:12:32 +01:00
|
|
|
@Log4j2(topic = "User Service")
|
2024-06-28 03:01:21 +01:00
|
|
|
@Getter
|
2024-07-01 01:12:32 +01:00
|
|
|
@DependsOn({"discordService", "mongoService"})
|
2024-07-01 01:33:52 +01:00
|
|
|
public class UserService implements EventListener {
|
2024-07-01 01:12:32 +01:00
|
|
|
private static final long SAVE_INTERVAL = TimeUnit.MINUTES.toMillis(5);
|
2024-06-27 13:00:45 +01:00
|
|
|
|
2024-06-24 13:56:01 +01:00
|
|
|
/**
|
2024-07-01 01:12:32 +01:00
|
|
|
* The cached users
|
2024-06-24 13:56:01 +01:00
|
|
|
*/
|
2024-07-06 00:34:53 +01:00
|
|
|
private final Map<String, BatUser> users = ExpiringMap.builder()
|
|
|
|
.expiration(1, TimeUnit.HOURS)
|
2024-07-06 21:14:20 +01:00
|
|
|
.expirationPolicy(ExpirationPolicy.ACCESSED) // Expire after not being accessed for 1 hour
|
2024-07-06 00:34:53 +01:00
|
|
|
.expirationListener((key, cachedUser) -> {
|
|
|
|
BatUser user = (BatUser) cachedUser;
|
|
|
|
log.info("Removed user \"{}\" - \"{}\" from cache", user.getName(), user.getId());
|
|
|
|
user.save();
|
|
|
|
})
|
|
|
|
.build();
|
2024-06-24 13:56:01 +01:00
|
|
|
|
|
|
|
@Autowired
|
2024-07-01 01:12:32 +01:00
|
|
|
public UserService() {
|
|
|
|
TimerUtils.scheduleRepeating(() -> {
|
|
|
|
long before = System.currentTimeMillis();
|
|
|
|
for (BatUser user : users.values()) {
|
|
|
|
user.save();
|
|
|
|
}
|
|
|
|
log.info("Saved {} users in {}ms", users.size(), System.currentTimeMillis() - before);
|
|
|
|
}, SAVE_INTERVAL, SAVE_INTERVAL);
|
2024-06-24 13:56:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-07-01 01:12:32 +01:00
|
|
|
* Gets a user by its ID
|
2024-06-24 13:56:01 +01:00
|
|
|
*
|
|
|
|
* @param id The ID of the user
|
|
|
|
* @return The user
|
|
|
|
*/
|
2024-07-04 03:33:06 +01:00
|
|
|
public BatUser getUser(@NonNull String id, User user) {
|
2024-07-01 01:12:32 +01:00
|
|
|
long before = System.currentTimeMillis();
|
|
|
|
// User is cached
|
2024-06-27 13:00:45 +01:00
|
|
|
if (users.containsKey(id)) {
|
|
|
|
return users.get(id);
|
|
|
|
}
|
2024-07-01 01:12:32 +01:00
|
|
|
// User is not cached
|
|
|
|
Document document = MongoService.INSTANCE.getUsersCollection().find(Filters.eq("_id", id)).first();
|
|
|
|
if (document != null) {
|
2024-07-04 03:33:06 +01:00
|
|
|
BatUser batUser = new BatUser(id, user, document);
|
2024-07-03 22:59:05 +01:00
|
|
|
users.put(id, batUser);
|
|
|
|
return batUser;
|
2024-06-24 13:56:01 +01:00
|
|
|
}
|
2024-07-01 01:12:32 +01:00
|
|
|
// New user
|
2024-07-04 03:33:06 +01:00
|
|
|
BatUser batUser = new BatUser(id, user, new Document());
|
2024-07-03 22:59:05 +01:00
|
|
|
users.put(id, batUser);
|
2024-07-04 06:47:19 +01:00
|
|
|
log.info("Created user \"{}\" - \"{}\" in {}ms", batUser.getName(), batUser.getId(), System.currentTimeMillis() - before);
|
2024-07-03 22:59:05 +01:00
|
|
|
return batUser;
|
2024-06-24 13:56:01 +01:00
|
|
|
}
|
2024-07-01 01:33:52 +01:00
|
|
|
|
2024-07-04 03:33:06 +01:00
|
|
|
/**
|
|
|
|
* Gets a user by its ID
|
|
|
|
*
|
|
|
|
* @param id The ID of the user
|
|
|
|
* @return The user
|
|
|
|
*/
|
|
|
|
public BatUser getUser(@NonNull String id) {
|
2024-07-05 22:58:32 +01:00
|
|
|
User user = DiscordService.JDA.retrieveUserById(id).complete();
|
|
|
|
if (user == null || user.isBot()) {
|
2024-07-04 03:33:06 +01:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return getUser(id, null);
|
|
|
|
}
|
|
|
|
|
2024-07-01 01:33:52 +01:00
|
|
|
@Override
|
2024-07-04 02:25:26 +01:00
|
|
|
public void onShutdown() {
|
2024-07-01 01:33:52 +01:00
|
|
|
log.info("Saving all users before shutdown...");
|
|
|
|
for (BatUser user : users.values()) {
|
|
|
|
user.save();
|
|
|
|
}
|
|
|
|
log.info("Saved all users.");
|
|
|
|
}
|
2024-07-01 19:40:32 +01:00
|
|
|
|
2024-07-01 19:41:13 +01:00
|
|
|
@Override
|
|
|
|
public void onGuildMemberJoin(@NonNull BatGuild guild, @NonNull BatUser user, @NonNull GuildMemberJoinEvent event) {
|
2024-07-04 02:25:26 +01:00
|
|
|
user.setGlobalName(event.getUser().getGlobalName()); // Ensure the user's name is up-to-date
|
2024-07-01 19:41:13 +01:00
|
|
|
}
|
|
|
|
|
2024-07-01 19:40:32 +01:00
|
|
|
@Override
|
|
|
|
public void onUserUpdateGlobalName(@NonNull BatUser user, String oldName, String newName, @NonNull UserUpdateGlobalNameEvent event) {
|
|
|
|
user.setGlobalName(newName);
|
|
|
|
}
|
2024-06-24 13:56:01 +01:00
|
|
|
}
|