All checks were successful
Deploy App / docker (ubuntu-latest, 2.44.0, 17, 3.8.5) (push) Successful in 1m40s
263 lines
10 KiB
Java
263 lines
10 KiB
Java
package cc.fascinated.service;
|
|
|
|
import cc.fascinated.Main;
|
|
import cc.fascinated.common.EndpointStatus;
|
|
import cc.fascinated.common.ExpiringSet;
|
|
import cc.fascinated.common.WebRequest;
|
|
import cc.fascinated.config.Config;
|
|
import cc.fascinated.model.cache.CachedEndpointStatus;
|
|
import cc.fascinated.model.mojang.MojangProfile;
|
|
import cc.fascinated.model.mojang.MojangUsernameToUuid;
|
|
import cc.fascinated.repository.EndpointStatusRepository;
|
|
import com.google.common.base.Joiner;
|
|
import com.google.common.base.Splitter;
|
|
import com.google.common.collect.Lists;
|
|
import com.google.common.hash.Hashing;
|
|
import io.micrometer.common.lang.NonNull;
|
|
import lombok.Getter;
|
|
import lombok.SneakyThrows;
|
|
import lombok.extern.log4j.Log4j2;
|
|
import net.jodah.expiringmap.ExpirationPolicy;
|
|
import org.springframework.beans.factory.annotation.Autowired;
|
|
import org.springframework.http.HttpStatus;
|
|
import org.springframework.http.ResponseEntity;
|
|
import org.springframework.stereotype.Service;
|
|
|
|
import java.io.InputStream;
|
|
import java.net.URL;
|
|
import java.nio.charset.StandardCharsets;
|
|
import java.util.*;
|
|
import java.util.concurrent.CompletableFuture;
|
|
import java.util.concurrent.TimeUnit;
|
|
|
|
@Service @Log4j2 @Getter
|
|
public class MojangService {
|
|
|
|
/**
|
|
* The splitter and joiner for dots.
|
|
*/
|
|
private static final Splitter DOT_SPLITTER = Splitter.on('.');
|
|
private static final Joiner DOT_JOINER = Joiner.on('.');
|
|
|
|
/**
|
|
* The Mojang API endpoints.
|
|
*/
|
|
private static final String SESSION_SERVER_ENDPOINT = "https://sessionserver.mojang.com";
|
|
private static final String API_ENDPOINT = "https://api.mojang.com";
|
|
private static final String FETCH_BLOCKED_SERVERS = SESSION_SERVER_ENDPOINT + "/blockedservers";
|
|
|
|
/**
|
|
* The interval to fetch the blocked servers from Mojang.
|
|
*/
|
|
private static final long FETCH_BLOCKED_SERVERS_INTERVAL = TimeUnit.HOURS.toMillis(1L);
|
|
|
|
/**
|
|
* Information about the Mojang API endpoints.
|
|
*/
|
|
private static final String MOJANG_ENDPOINT_STATUS_KEY = "mojang";
|
|
private static final List<EndpointStatus> MOJANG_ENDPOINTS = List.of(
|
|
new EndpointStatus("https://textures.minecraft.net", List.of(HttpStatus.BAD_REQUEST)),
|
|
new EndpointStatus("https://session.minecraft.net", List.of(HttpStatus.NOT_FOUND)),
|
|
new EndpointStatus("https://libraries.minecraft.net", List.of(HttpStatus.NOT_FOUND)),
|
|
new EndpointStatus("https://assets.mojang.com", List.of(HttpStatus.NOT_FOUND)),
|
|
new EndpointStatus("https://api.minecraftservices.com", List.of(HttpStatus.FORBIDDEN)),
|
|
new EndpointStatus(API_ENDPOINT, List.of(HttpStatus.OK)),
|
|
new EndpointStatus(SESSION_SERVER_ENDPOINT, List.of(HttpStatus.FORBIDDEN))
|
|
);
|
|
|
|
@Autowired
|
|
private EndpointStatusRepository mojangEndpointStatusRepository;
|
|
|
|
/**
|
|
* A list of banned server hashes provided by Mojang.
|
|
* <p>
|
|
* This is periodically fetched from Mojang, see
|
|
* {@link #fetchBlockedServers()} for more info.
|
|
* </p>
|
|
*
|
|
* @see <a href="https://wiki.vg/Mojang_API#Blocked_Servers">Mojang API</a>
|
|
*/
|
|
private List<String> bannedServerHashes;
|
|
|
|
/**
|
|
* A cache of blocked server hostnames.
|
|
*
|
|
* @see #isServerHostnameBlocked(String) for more
|
|
*/
|
|
private final ExpiringSet<String> blockedServersCache = new ExpiringSet<>(ExpirationPolicy.CREATED, 10L, TimeUnit.MINUTES);
|
|
|
|
public MojangService() {
|
|
new Timer().scheduleAtFixedRate(new TimerTask() {
|
|
@Override
|
|
public void run() {
|
|
fetchBlockedServers();
|
|
}
|
|
}, 0L, FETCH_BLOCKED_SERVERS_INTERVAL);
|
|
}
|
|
|
|
/**
|
|
* Fetch a list of blocked servers from Mojang.
|
|
*/
|
|
@SneakyThrows
|
|
private void fetchBlockedServers() {
|
|
log.info("Fetching blocked servers from Mojang");
|
|
try (
|
|
InputStream inputStream = new URL(FETCH_BLOCKED_SERVERS).openStream();
|
|
Scanner scanner = new Scanner(inputStream, StandardCharsets.UTF_8).useDelimiter("\n");
|
|
) {
|
|
List<String> hashes = new ArrayList<>();
|
|
while (scanner.hasNext()) {
|
|
hashes.add(scanner.next());
|
|
}
|
|
bannedServerHashes = Collections.synchronizedList(hashes);
|
|
log.info("Fetched {} banned server hashes", bannedServerHashes.size());
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Check if the server with the
|
|
* given hostname is blocked by Mojang.
|
|
*
|
|
* @param hostname the server hostname to check
|
|
* @return whether the hostname is blocked
|
|
*/
|
|
public boolean isServerBlocked(@NonNull String hostname) {
|
|
// Remove trailing dots
|
|
while (hostname.charAt(hostname.length() - 1) == '.') {
|
|
hostname = hostname.substring(0, hostname.length() - 1);
|
|
}
|
|
// Is the hostname banned?
|
|
if (isServerHostnameBlocked(hostname)) {
|
|
return true;
|
|
}
|
|
List<String> splitDots = Lists.newArrayList(DOT_SPLITTER.split(hostname)); // Split the hostname by dots
|
|
boolean isIp = splitDots.size() == 4; // Is it an IP address?
|
|
if (isIp) {
|
|
for (String element : splitDots) {
|
|
try {
|
|
int part = Integer.parseInt(element);
|
|
if (part >= 0 && part <= 255) { // Ensure the part is within the valid range
|
|
continue;
|
|
}
|
|
} catch (NumberFormatException ignored) {
|
|
// Safely ignore, not a number
|
|
}
|
|
isIp = false;
|
|
break;
|
|
}
|
|
}
|
|
// Check if the hostname is blocked
|
|
if (!isIp && isServerHostnameBlocked("*." + hostname)) {
|
|
return true;
|
|
}
|
|
// Additional checks for the hostname
|
|
while (splitDots.size() > 1) {
|
|
splitDots.remove(isIp ? splitDots.size() - 1 : 0);
|
|
String starredPart = isIp ? DOT_JOINER.join(splitDots) + ".*" : "*." + DOT_JOINER.join(splitDots);
|
|
if (isServerHostnameBlocked(starredPart)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Check if the hash for the given
|
|
* hostname is in the blocked server list.
|
|
*
|
|
* @param hostname the hostname to check
|
|
* @return whether the hostname is blocked
|
|
*/
|
|
private boolean isServerHostnameBlocked(@NonNull String hostname) {
|
|
// Check the cache first for the hostname
|
|
if (blockedServersCache.contains(hostname)) {
|
|
return true;
|
|
}
|
|
String hashed = Hashing.sha1().hashBytes(hostname.toLowerCase().getBytes(StandardCharsets.ISO_8859_1)).toString();
|
|
boolean blocked = bannedServerHashes.contains(hashed); // Is the hostname blocked?
|
|
if (blocked) { // Cache the blocked hostname
|
|
blockedServersCache.add(hostname);
|
|
}
|
|
return blocked;
|
|
}
|
|
|
|
/**
|
|
* Gets the status of the Mojang APIs.
|
|
*
|
|
* @return the status
|
|
*/
|
|
public CachedEndpointStatus getMojangApiStatus() {
|
|
log.info("Getting Mojang API status");
|
|
Optional<CachedEndpointStatus> endpointStatus = mojangEndpointStatusRepository.findById(MOJANG_ENDPOINT_STATUS_KEY);
|
|
if (endpointStatus.isPresent() && Config.INSTANCE.isProduction()) {
|
|
log.info("Got cached Mojang API status");
|
|
return endpointStatus.get();
|
|
}
|
|
|
|
// Fetch the status of the Mojang API endpoints
|
|
List<CompletableFuture<CachedEndpointStatus.Status>> futures = new ArrayList<>();
|
|
for (EndpointStatus endpoint : MOJANG_ENDPOINTS) {
|
|
CompletableFuture<CachedEndpointStatus.Status> future = CompletableFuture.supplyAsync(() -> {
|
|
boolean online = false;
|
|
long start = System.currentTimeMillis();
|
|
ResponseEntity<?> response = WebRequest.head(endpoint.getEndpoint(), String.class);
|
|
if (endpoint.getAllowedStatuses().contains(response.getStatusCode())) {
|
|
online = true;
|
|
}
|
|
if (online && System.currentTimeMillis() - start > 500) { // If the response took longer than 500ms
|
|
return CachedEndpointStatus.Status.DEGRADED;
|
|
}
|
|
return online ? CachedEndpointStatus.Status.ONLINE : CachedEndpointStatus.Status.OFFLINE;
|
|
}, Main.EXECUTOR_POOL);
|
|
|
|
futures.add(future);
|
|
}
|
|
CompletableFuture<Void> allFutures = CompletableFuture.allOf(futures.toArray(new CompletableFuture[0]));
|
|
try {
|
|
allFutures.get(5, TimeUnit.SECONDS); // Wait for the futures to complete
|
|
} catch (Exception e) {
|
|
log.error("Timeout while fetching Mojang API status: {}", e.getMessage());
|
|
}
|
|
|
|
// Process the results
|
|
Map<String, CachedEndpointStatus.Status> endpoints = new HashMap<>();
|
|
for (int i = 0; i < MOJANG_ENDPOINTS.size(); i++) {
|
|
EndpointStatus endpoint = MOJANG_ENDPOINTS.get(i);
|
|
CachedEndpointStatus.Status status = futures.get(i).join();
|
|
endpoints.put(endpoint.getEndpoint(), status);
|
|
}
|
|
|
|
log.info("Fetched Mojang API status for {} endpoints", endpoints.size());
|
|
CachedEndpointStatus status = new CachedEndpointStatus(
|
|
MOJANG_ENDPOINT_STATUS_KEY,
|
|
endpoints,
|
|
System.currentTimeMillis()
|
|
);
|
|
mojangEndpointStatusRepository.save(status);
|
|
status.setCached(-1L); // Indicate that the status is not cached
|
|
return status;
|
|
}
|
|
|
|
/**
|
|
* Gets the Session Server profile of the
|
|
* player with the given UUID.
|
|
*
|
|
* @param id the uuid or name of the player
|
|
* @return the profile
|
|
*/
|
|
public MojangProfile getProfile(String id) {
|
|
return WebRequest.getAsEntity(SESSION_SERVER_ENDPOINT + "/session/minecraft/profile/" + id, MojangProfile.class);
|
|
}
|
|
|
|
/**
|
|
* Gets the UUID of the player using
|
|
* the name of the player.
|
|
*
|
|
* @param id the name of the player
|
|
* @return the profile
|
|
*/
|
|
public MojangUsernameToUuid getUuidFromUsername(String id) {
|
|
return WebRequest.getAsEntity(API_ENDPOINT + "/users/profiles/minecraft/" + id, MojangUsernameToUuid.class);
|
|
}
|
|
}
|