forked from MinecraftUtilities/Backend
initial commit
This commit is contained in:
35
src/main/java/cc/fascinated/Main.java
Normal file
35
src/main/java/cc/fascinated/Main.java
Normal file
@ -0,0 +1,35 @@
|
||||
package cc.fascinated;
|
||||
|
||||
import com.google.gson.Gson;
|
||||
import lombok.Getter;
|
||||
import lombok.SneakyThrows;
|
||||
import lombok.extern.log4j.Log4j2;
|
||||
import org.springframework.boot.SpringApplication;
|
||||
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
||||
|
||||
import java.io.File;
|
||||
import java.nio.file.Files;
|
||||
import java.nio.file.StandardCopyOption;
|
||||
import java.util.Objects;
|
||||
|
||||
@SpringBootApplication @Log4j2
|
||||
public class Main {
|
||||
|
||||
@Getter
|
||||
private static final Gson GSON = new Gson();
|
||||
|
||||
@SneakyThrows
|
||||
public static void main(String[] args) {
|
||||
File config = new File("application.yml");
|
||||
if (!config.exists()) { // Saving the default config if it doesn't exist locally
|
||||
Files.copy(Objects.requireNonNull(Main.class.getResourceAsStream("/application.yml")), config.toPath(), StandardCopyOption.REPLACE_EXISTING);
|
||||
log.info("Saved the default configuration to '{}', please re-launch the application", // Log the default config being saved
|
||||
config.getAbsolutePath()
|
||||
);
|
||||
return;
|
||||
}
|
||||
log.info("Found configuration at '{}'", config.getAbsolutePath()); // Log the found config
|
||||
|
||||
SpringApplication.run(Main.class, args);
|
||||
}
|
||||
}
|
@ -0,0 +1,29 @@
|
||||
package cc.fascinated.api.controller;
|
||||
|
||||
import cc.fascinated.player.PlayerManagerService;
|
||||
import cc.fascinated.player.impl.Player;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.http.MediaType;
|
||||
import org.springframework.http.ResponseEntity;
|
||||
import org.springframework.web.bind.annotation.*;
|
||||
|
||||
@RestController
|
||||
@RequestMapping(value = "/", produces = MediaType.APPLICATION_JSON_VALUE)
|
||||
public class PlayerController {
|
||||
|
||||
private final PlayerManagerService playerManagerService;
|
||||
|
||||
@Autowired
|
||||
public PlayerController(PlayerManagerService playerManagerService) {
|
||||
this.playerManagerService = playerManagerService;
|
||||
}
|
||||
|
||||
@GetMapping("/{id}") @ResponseBody
|
||||
public ResponseEntity<Player> getPlayer(@PathVariable String id) {
|
||||
Player player = playerManagerService.getPlayer(id);
|
||||
if (player == null) {
|
||||
return ResponseEntity.notFound().build();
|
||||
}
|
||||
return ResponseEntity.ok(player);
|
||||
}
|
||||
}
|
59
src/main/java/cc/fascinated/mojang/MojangAPIService.java
Normal file
59
src/main/java/cc/fascinated/mojang/MojangAPIService.java
Normal file
@ -0,0 +1,59 @@
|
||||
package cc.fascinated.mojang;
|
||||
|
||||
import cc.fascinated.Main;
|
||||
import cc.fascinated.mojang.types.MojangApiProfile;
|
||||
import cc.fascinated.mojang.types.MojangSessionServerProfile;
|
||||
import lombok.SneakyThrows;
|
||||
import org.springframework.beans.factory.annotation.Value;
|
||||
import org.springframework.stereotype.Service;
|
||||
|
||||
import java.net.URI;
|
||||
import java.net.http.HttpClient;
|
||||
import java.net.http.HttpRequest;
|
||||
import java.net.http.HttpResponse;
|
||||
|
||||
@Service
|
||||
public class MojangAPIService {
|
||||
|
||||
private static final HttpClient CLIENT = HttpClient.newHttpClient();
|
||||
|
||||
@Value("${mojang.session-server}")
|
||||
private String mojangSessionServerUrl;
|
||||
|
||||
@Value("${mojang.api}")
|
||||
private String mojangApiUrl;
|
||||
|
||||
/**
|
||||
* Gets the Session Server profile of the player with the given UUID.
|
||||
*
|
||||
* @param id the uuid or name of the player
|
||||
* @return the profile
|
||||
*/
|
||||
@SneakyThrows
|
||||
public MojangSessionServerProfile getSessionServerProfile(String id) {
|
||||
HttpRequest request = HttpRequest.newBuilder()
|
||||
.uri(new URI(mojangSessionServerUrl + "/session/minecraft/profile/" + id))
|
||||
.GET()
|
||||
.build();
|
||||
|
||||
HttpResponse<String> response = CLIENT.send(request, HttpResponse.BodyHandlers.ofString());
|
||||
return Main.getGSON().fromJson(response.body(), MojangSessionServerProfile.class);
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the Mojang API profile of the player with the given UUID.
|
||||
*
|
||||
* @param id the name of the player
|
||||
* @return the profile
|
||||
*/
|
||||
@SneakyThrows
|
||||
public MojangApiProfile getApiProfile(String id) {
|
||||
HttpRequest request = HttpRequest.newBuilder()
|
||||
.uri(new URI(mojangApiUrl + "/users/profiles/minecraft/" + id))
|
||||
.GET()
|
||||
.build();
|
||||
|
||||
HttpResponse<String> response = CLIENT.send(request, HttpResponse.BodyHandlers.ofString());
|
||||
return Main.getGSON().fromJson(response.body(), MojangApiProfile.class);
|
||||
}
|
||||
}
|
@ -0,0 +1,12 @@
|
||||
package cc.fascinated.mojang.types;
|
||||
|
||||
import lombok.Getter;
|
||||
|
||||
@Getter
|
||||
public class MojangApiProfile {
|
||||
|
||||
private String id;
|
||||
private String name;
|
||||
|
||||
public MojangApiProfile() {}
|
||||
}
|
@ -0,0 +1,42 @@
|
||||
package cc.fascinated.mojang.types;
|
||||
|
||||
import lombok.Getter;
|
||||
import lombok.ToString;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
@Getter @ToString
|
||||
public class MojangSessionServerProfile {
|
||||
|
||||
/**
|
||||
* The UUID of the player.
|
||||
*/
|
||||
private String id;
|
||||
|
||||
/**
|
||||
* The name of the player.
|
||||
*/
|
||||
private String name;
|
||||
|
||||
/**
|
||||
* The properties for the player.
|
||||
*/
|
||||
private final List<MojangSessionServerProfileProperties> properties = new ArrayList<>();
|
||||
|
||||
public MojangSessionServerProfile() {}
|
||||
|
||||
/**
|
||||
* Get the texture property for the player.
|
||||
*
|
||||
* @return the texture property
|
||||
*/
|
||||
public MojangSessionServerProfileProperties getTextureProperty() {
|
||||
for (MojangSessionServerProfileProperties property : properties) {
|
||||
if (property.getName().equals("textures")) {
|
||||
return property;
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
}
|
@ -0,0 +1,12 @@
|
||||
package cc.fascinated.mojang.types;
|
||||
|
||||
import lombok.Getter;
|
||||
import lombok.ToString;
|
||||
|
||||
@Getter @ToString
|
||||
public class MojangSessionServerProfileProperties {
|
||||
private String name;
|
||||
private String value;
|
||||
|
||||
public MojangSessionServerProfileProperties() {}
|
||||
}
|
72
src/main/java/cc/fascinated/player/PlayerManagerService.java
Normal file
72
src/main/java/cc/fascinated/player/PlayerManagerService.java
Normal file
@ -0,0 +1,72 @@
|
||||
package cc.fascinated.player;
|
||||
|
||||
import cc.fascinated.mojang.MojangAPIService;
|
||||
import cc.fascinated.mojang.types.MojangApiProfile;
|
||||
import cc.fascinated.mojang.types.MojangSessionServerProfile;
|
||||
import cc.fascinated.player.impl.Player;
|
||||
import cc.fascinated.util.UUIDUtils;
|
||||
import net.jodah.expiringmap.ExpirationPolicy;
|
||||
import net.jodah.expiringmap.ExpiringMap;
|
||||
import org.springframework.stereotype.Service;
|
||||
|
||||
import java.util.Map;
|
||||
import java.util.UUID;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
@Service
|
||||
public class PlayerManagerService {
|
||||
|
||||
/**
|
||||
* The cache of players.
|
||||
*/
|
||||
private final Map<UUID, Player> players = ExpiringMap.builder()
|
||||
.expiration(1, TimeUnit.HOURS)
|
||||
.expirationPolicy(ExpirationPolicy.CREATED)
|
||||
.build();
|
||||
|
||||
/**
|
||||
* The cache of player names to UUIDs.
|
||||
*/
|
||||
private final Map<String, UUID> playerNameToUUIDCache = ExpiringMap.builder()
|
||||
.expiration(1, TimeUnit.DAYS)
|
||||
.expirationPolicy(ExpirationPolicy.CREATED)
|
||||
.build();
|
||||
|
||||
private final MojangAPIService mojangAPIService;
|
||||
|
||||
public PlayerManagerService(MojangAPIService mojangAPIService) {
|
||||
this.mojangAPIService = mojangAPIService;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets a player by their UUID.
|
||||
*
|
||||
* @param id the uuid or name of the player
|
||||
* @return the player or null if the player does not exist
|
||||
*/
|
||||
public Player getPlayer(String id) {
|
||||
UUID uuid;
|
||||
if (id.length() == 32 || id.length() == 36) {
|
||||
uuid = UUID.fromString(id.length() == 32 ? UUIDUtils.addUUIDDashes(id) : id);
|
||||
} else {
|
||||
uuid = playerNameToUUIDCache.get(id.toUpperCase());
|
||||
}
|
||||
|
||||
if (players.containsKey(uuid)) {
|
||||
return players.get(uuid);
|
||||
}
|
||||
|
||||
MojangSessionServerProfile profile = uuid == null ? null : mojangAPIService.getSessionServerProfile(uuid.toString());
|
||||
if (profile == null) {
|
||||
MojangApiProfile apiProfile = mojangAPIService.getApiProfile(id);
|
||||
if (apiProfile == null) {
|
||||
return null;
|
||||
}
|
||||
profile = mojangAPIService.getSessionServerProfile(apiProfile.getId().length() == 32 ? UUIDUtils.addUUIDDashes(apiProfile.getId()) : apiProfile.getId());
|
||||
}
|
||||
Player player = new Player(profile);
|
||||
players.put(player.getUuid(), player);
|
||||
playerNameToUUIDCache.put(player.getName().toUpperCase(), player.getUuid());
|
||||
return player;
|
||||
}
|
||||
}
|
57
src/main/java/cc/fascinated/player/impl/Player.java
Normal file
57
src/main/java/cc/fascinated/player/impl/Player.java
Normal file
@ -0,0 +1,57 @@
|
||||
package cc.fascinated.player.impl;
|
||||
|
||||
import cc.fascinated.Main;
|
||||
import cc.fascinated.mojang.types.MojangSessionServerProfile;
|
||||
import cc.fascinated.mojang.types.MojangSessionServerProfileProperties;
|
||||
import cc.fascinated.util.UUIDUtils;
|
||||
import com.google.gson.JsonObject;
|
||||
import lombok.Getter;
|
||||
|
||||
import java.util.UUID;
|
||||
|
||||
@Getter
|
||||
public class Player {
|
||||
|
||||
/**
|
||||
* The UUID of the player
|
||||
*/
|
||||
private final UUID uuid;
|
||||
|
||||
/**
|
||||
* The name of the player
|
||||
*/
|
||||
private final String name;
|
||||
|
||||
/**
|
||||
* The skin of the player
|
||||
* <p>
|
||||
* This will be null if the player does not have a skin.
|
||||
* </p>
|
||||
*/
|
||||
private Skin skin;
|
||||
|
||||
public Player(MojangSessionServerProfile profile) {
|
||||
this.uuid = UUID.fromString(UUIDUtils.addUUIDDashes(profile.getId()));
|
||||
this.name = profile.getName();
|
||||
|
||||
MojangSessionServerProfileProperties textureProperty = profile.getTextureProperty();
|
||||
if (textureProperty == null) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Decode the texture property
|
||||
String decoded = new String(java.util.Base64.getDecoder().decode(textureProperty.getValue()));
|
||||
|
||||
// Parse the decoded JSON
|
||||
JsonObject json = Main.getGSON().fromJson(decoded, JsonObject.class);
|
||||
JsonObject textures = json.getAsJsonObject("textures");
|
||||
JsonObject skin = textures.getAsJsonObject("SKIN");
|
||||
JsonObject metadata = skin.get("metadata").getAsJsonObject();
|
||||
|
||||
String url = skin.get("url").getAsString();
|
||||
SkinType model = SkinType.fromString(metadata.get("model").getAsString());
|
||||
|
||||
this.skin = new Skin(url, model);
|
||||
}
|
||||
|
||||
}
|
18
src/main/java/cc/fascinated/player/impl/Skin.java
Normal file
18
src/main/java/cc/fascinated/player/impl/Skin.java
Normal file
@ -0,0 +1,18 @@
|
||||
package cc.fascinated.player.impl;
|
||||
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Getter;
|
||||
|
||||
@Getter @AllArgsConstructor
|
||||
public class Skin {
|
||||
|
||||
/**
|
||||
* The URL of the skin
|
||||
*/
|
||||
private final String url;
|
||||
|
||||
/**
|
||||
* The model of the skin
|
||||
*/
|
||||
private final SkinType model;
|
||||
}
|
22
src/main/java/cc/fascinated/player/impl/SkinType.java
Normal file
22
src/main/java/cc/fascinated/player/impl/SkinType.java
Normal file
@ -0,0 +1,22 @@
|
||||
package cc.fascinated.player.impl;
|
||||
|
||||
public enum SkinType {
|
||||
|
||||
DEFAULT,
|
||||
SLIM;
|
||||
|
||||
/**
|
||||
* Get the skin type from a string
|
||||
*
|
||||
* @param string the string
|
||||
* @return the skin type
|
||||
*/
|
||||
public static SkinType fromString(String string) {
|
||||
for (SkinType type : values()) {
|
||||
if (type.name().equalsIgnoreCase(string)) {
|
||||
return type;
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
}
|
19
src/main/java/cc/fascinated/util/UUIDUtils.java
Normal file
19
src/main/java/cc/fascinated/util/UUIDUtils.java
Normal file
@ -0,0 +1,19 @@
|
||||
package cc.fascinated.util;
|
||||
|
||||
public class UUIDUtils {
|
||||
|
||||
/**
|
||||
* Add dashes to a UUID.
|
||||
*
|
||||
* @param idNoDashes the UUID without dashes
|
||||
* @return the UUID with dashes
|
||||
*/
|
||||
public static String addUUIDDashes(String idNoDashes) {
|
||||
StringBuilder idBuff = new StringBuilder(idNoDashes);
|
||||
idBuff.insert(20, '-');
|
||||
idBuff.insert(16, '-');
|
||||
idBuff.insert(12, '-');
|
||||
idBuff.insert(8, '-');
|
||||
return idBuff.toString();
|
||||
}
|
||||
}
|
7
src/main/resources/application.yml
Normal file
7
src/main/resources/application.yml
Normal file
@ -0,0 +1,7 @@
|
||||
server:
|
||||
address: 0.0.0.0
|
||||
port: 7500
|
||||
|
||||
mojang:
|
||||
session-server: https://sessionserver.mojang.com
|
||||
api: https://api.mojang.com
|
Reference in New Issue
Block a user