package cc.fascinated.service; import cc.fascinated.common.PlayerUtils; import cc.fascinated.common.Tuple; import cc.fascinated.common.UUIDUtils; import cc.fascinated.exception.impl.MojangAPIRateLimitException; import cc.fascinated.exception.impl.RateLimitException; import cc.fascinated.exception.impl.ResourceNotFoundException; import cc.fascinated.model.cache.CachedPlayer; import cc.fascinated.model.cache.CachedPlayerName; import cc.fascinated.model.cache.CachedPlayerSkinPart; import cc.fascinated.model.mojang.MojangProfile; import cc.fascinated.model.mojang.MojangUsernameToUuid; import cc.fascinated.model.player.Cape; import cc.fascinated.model.player.Player; import cc.fascinated.model.player.Skin; import cc.fascinated.repository.PlayerCacheRepository; import cc.fascinated.repository.PlayerNameCacheRepository; import cc.fascinated.repository.PlayerSkinPartCacheRepository; import lombok.extern.log4j.Log4j2; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import java.util.Optional; import java.util.UUID; @Service @Log4j2 public class PlayerService { private final MojangService mojangAPIService; private final PlayerCacheRepository playerCacheRepository; private final PlayerNameCacheRepository playerNameCacheRepository; private final PlayerSkinPartCacheRepository playerSkinPartCacheRepository; @Autowired public PlayerService(MojangService mojangAPIService, PlayerCacheRepository playerCacheRepository, PlayerNameCacheRepository playerNameCacheRepository, PlayerSkinPartCacheRepository playerSkinPartCacheRepository) { this.mojangAPIService = mojangAPIService; this.playerCacheRepository = playerCacheRepository; this.playerNameCacheRepository = playerNameCacheRepository; this.playerSkinPartCacheRepository = playerSkinPartCacheRepository; } /** * Get a player from the cache or * from the Mojang API. * * @param id the id of the player * @return the player */ public CachedPlayer getPlayer(String id) { String originalId = id; id = id.toUpperCase(); // Convert the id to uppercase to prevent case sensitivity log.info("Getting player: {}", originalId); UUID uuid = PlayerUtils.getUuidFromString(originalId); if (uuid == null) { // If the id is not a valid uuid, get the uuid from the username uuid = usernameToUuid(originalId); } Optional cachedPlayer = playerCacheRepository.findById(uuid); if (cachedPlayer.isPresent()) { // Return the cached player if it exists log.info("Player {} is cached", originalId); return cachedPlayer.get(); } try { log.info("Getting player profile from Mojang: {}", originalId); MojangProfile mojangProfile = mojangAPIService.getProfile(uuid.toString()); // Get the player profile from Mojang log.info("Got player profile from Mojang: {}", originalId); Tuple skinAndCape = mojangProfile.getSkinAndCape(); CachedPlayer player = new CachedPlayer( uuid, // Player UUID mojangProfile.getName(), // Player Name skinAndCape.getLeft(), // Skin skinAndCape.getRight(), // Cape System.currentTimeMillis() // Cache time ); playerCacheRepository.save(player); player.setCached(-1); // Indicate that the player is not cached return player; } catch (RateLimitException exception) { throw new MojangAPIRateLimitException(); } } /** * Gets the player's uuid from their username. * * @param username the username of the player * @return the uuid of the player */ private UUID usernameToUuid(String username) { log.info("Getting UUID from username: {}", username); Optional cachedPlayerName = playerNameCacheRepository.findById(username.toUpperCase()); if (cachedPlayerName.isPresent()) { return cachedPlayerName.get().getUniqueId(); } try { MojangUsernameToUuid mojangUsernameToUuid = mojangAPIService.getUuidFromUsername(username); if (mojangUsernameToUuid == null) { log.info("Player with username '{}' not found", username); throw new ResourceNotFoundException("Player with username '%s' not found".formatted(username)); } UUID uuid = UUIDUtils.addDashes(mojangUsernameToUuid.getId()); playerNameCacheRepository.save(new CachedPlayerName(username, uuid)); log.info("Got UUID from username: {} -> {}", username, uuid); return uuid; } catch (RateLimitException exception) { throw new MojangAPIRateLimitException(); } } /** * Gets a skin part from the player's skin. * * @param player the player * @param part the part of the skin * @return the skin part */ public CachedPlayerSkinPart getSkinPart(Player player, Skin.Parts part, int size) { log.info("Getting skin part: {} for player: {}", part.getName(), player.getUuid()); String key = "%s-%s-%s".formatted(player.getUuid(), part.getName(), size); Optional cache = playerSkinPartCacheRepository.findById(key); // The skin part is cached if (cache.isPresent()) { log.info("Skin part {} for player {} is cached", part.getName(), player.getUuid()); return cache.get(); } byte[] skinPartBytes = PlayerUtils.getSkinPartBytes(player.getSkin(), part, size); CachedPlayerSkinPart skinPart = new CachedPlayerSkinPart( key, skinPartBytes ); log.info("Fetched skin part: {} for player: {}", part.getName(), player.getUuid()); playerSkinPartCacheRepository.save(skinPart); return skinPart; } }