package cc.fascinated.service; import cc.fascinated.common.PlayerUtils; import cc.fascinated.common.Tuple; import cc.fascinated.common.UUIDUtils; import cc.fascinated.config.Config; 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).getUniqueId(); } Optional cachedPlayer = playerCacheRepository.findById(uuid); if (cachedPlayer.isPresent() && Config.INSTANCE.isProduction()) { // 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 UUIDUtils.removeDashes(uuid), // Trimmed UUID mojangProfile.getName(), // Player Name skinAndCape.getLeft(), // Skin skinAndCape.getRight(), // Cape mojangProfile.getProperties(), // Raw properties 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 */ public CachedPlayerName usernameToUuid(String username) { log.info("Getting UUID from username: {}", username); Optional cachedPlayerName = playerNameCacheRepository.findById(username.toUpperCase()); if (cachedPlayerName.isPresent() && Config.INSTANCE.isProduction()) { return cachedPlayerName.get(); } 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.getUuid()); CachedPlayerName player = new CachedPlayerName(username, uuid); playerNameCacheRepository.save(player); log.info("Got UUID from username: {} -> {}", username, uuid); return player; } 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 * @param renderOverlay whether to render the overlay * @return the skin part */ public CachedPlayerSkinPart getSkinPart(Player player, Skin.Parts part, boolean renderOverlay, int size) { if (size > 512) { log.info("Size {} is too large, setting to 512", size); size = 512; } log.info("Getting skin part {} for player: {}", part.getName(), player.getUniqueId()); String key = "%s-%s-%s".formatted(player.getUniqueId(), part.getName(), size); Optional cache = playerSkinPartCacheRepository.findById(key); // The skin part is cached if (cache.isPresent() && Config.INSTANCE.isProduction()) { log.info("Skin part {} for player {} is cached", part.getName(), player.getUniqueId()); return cache.get(); } long before = System.currentTimeMillis(); byte[] skinPartBytes = part.getSkinRenderer().renderPart(player.getSkin(), part.getName(), renderOverlay, size); log.info("Took {}ms to render skin part {} for player: {}", System.currentTimeMillis() - before, part.getName(), player.getUniqueId()); CachedPlayerSkinPart skinPart = new CachedPlayerSkinPart( key, skinPartBytes ); log.info("Fetched skin part {} for player: {}", part.getName(), player.getUniqueId()); playerSkinPartCacheRepository.save(skinPart); return skinPart; } }