Stable
This commit is contained in:
commit
9a47b201ca
119
.gitignore
vendored
Normal file
119
.gitignore
vendored
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
# User-specific stuff
|
||||||
|
.idea/
|
||||||
|
|
||||||
|
*.iml
|
||||||
|
*.ipr
|
||||||
|
*.iws
|
||||||
|
|
||||||
|
# IntelliJ
|
||||||
|
out/
|
||||||
|
# mpeltonen/sbt-idea plugin
|
||||||
|
.idea_modules/
|
||||||
|
|
||||||
|
# JIRA plugin
|
||||||
|
atlassian-ide-plugin.xml
|
||||||
|
|
||||||
|
# Compiled class file
|
||||||
|
*.class
|
||||||
|
|
||||||
|
# Log file
|
||||||
|
*.log
|
||||||
|
|
||||||
|
# BlueJ files
|
||||||
|
*.ctxt
|
||||||
|
|
||||||
|
# Package Files #
|
||||||
|
*.jar
|
||||||
|
*.war
|
||||||
|
*.nar
|
||||||
|
*.ear
|
||||||
|
*.zip
|
||||||
|
*.tar.gz
|
||||||
|
*.rar
|
||||||
|
|
||||||
|
# virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml
|
||||||
|
hs_err_pid*
|
||||||
|
|
||||||
|
*~
|
||||||
|
|
||||||
|
# temporary files which can be created if a process still has a handle open of a deleted file
|
||||||
|
.fuse_hidden*
|
||||||
|
|
||||||
|
# KDE directory preferences
|
||||||
|
.directory
|
||||||
|
|
||||||
|
# Linux trash folder which might appear on any partition or disk
|
||||||
|
.Trash-*
|
||||||
|
|
||||||
|
# .nfs files are created when an open file is removed but is still being accessed
|
||||||
|
.nfs*
|
||||||
|
|
||||||
|
# General
|
||||||
|
.DS_Store
|
||||||
|
.AppleDouble
|
||||||
|
.LSOverride
|
||||||
|
|
||||||
|
# Icon must end with two \r
|
||||||
|
Icon
|
||||||
|
|
||||||
|
# Thumbnails
|
||||||
|
._*
|
||||||
|
|
||||||
|
# Files that might appear in the root of a volume
|
||||||
|
.DocumentRevisions-V100
|
||||||
|
.fseventsd
|
||||||
|
.Spotlight-V100
|
||||||
|
.TemporaryItems
|
||||||
|
.Trashes
|
||||||
|
.VolumeIcon.icns
|
||||||
|
.com.apple.timemachine.donotpresent
|
||||||
|
|
||||||
|
# Directories potentially created on remote AFP share
|
||||||
|
.AppleDB
|
||||||
|
.AppleDesktop
|
||||||
|
Network Trash Folder
|
||||||
|
Temporary Items
|
||||||
|
.apdisk
|
||||||
|
|
||||||
|
# Windows thumbnail cache files
|
||||||
|
Thumbs.db
|
||||||
|
Thumbs.db:encryptable
|
||||||
|
ehthumbs.db
|
||||||
|
ehthumbs_vista.db
|
||||||
|
|
||||||
|
# Dump file
|
||||||
|
*.stackdump
|
||||||
|
|
||||||
|
# Folder config file
|
||||||
|
[Dd]esktop.ini
|
||||||
|
|
||||||
|
# Recycle Bin used on file shares
|
||||||
|
$RECYCLE.BIN/
|
||||||
|
|
||||||
|
# Windows Installer files
|
||||||
|
*.cab
|
||||||
|
*.msi
|
||||||
|
*.msix
|
||||||
|
*.msm
|
||||||
|
*.msp
|
||||||
|
|
||||||
|
# Windows shortcuts
|
||||||
|
*.lnk
|
||||||
|
|
||||||
|
.gradle
|
||||||
|
build/
|
||||||
|
|
||||||
|
# Ignore Gradle GUI config
|
||||||
|
gradle-app.setting
|
||||||
|
|
||||||
|
# Cache of project
|
||||||
|
.gradletasknamecache
|
||||||
|
|
||||||
|
**/build/
|
||||||
|
|
||||||
|
# Common working directory
|
||||||
|
run/
|
||||||
|
runs/
|
||||||
|
|
||||||
|
# Avoid ignoring Gradle wrapper jar file (.jar files are usually ignored)
|
||||||
|
!gradle-wrapper.jar
|
||||||
21
LICENSE.txt
Normal file
21
LICENSE.txt
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
The MIT License (MIT)
|
||||||
|
|
||||||
|
Copyright (c) 2025 Verox
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in
|
||||||
|
all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
THE SOFTWARE.
|
||||||
92
build.gradle
Normal file
92
build.gradle
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
plugins {
|
||||||
|
id 'fabric-loom' version '1.11-SNAPSHOT'
|
||||||
|
id 'maven-publish'
|
||||||
|
}
|
||||||
|
|
||||||
|
version = project.mod_version
|
||||||
|
group = project.maven_group
|
||||||
|
|
||||||
|
base {
|
||||||
|
archivesName = project.archives_base_name
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
// Add repositories to retrieve artifacts from in here.
|
||||||
|
// You should only use this when depending on other mods because
|
||||||
|
// Loom adds the essential maven repositories to download Minecraft and libraries from automatically.
|
||||||
|
// See https://docs.gradle.org/current/userguide/declaring_repositories.html
|
||||||
|
// for more information about repositories.
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
// To change the versions see the gradle.properties file
|
||||||
|
minecraft "com.mojang:minecraft:${project.minecraft_version}"
|
||||||
|
mappings "net.fabricmc:yarn:${project.yarn_mappings}:v2"
|
||||||
|
modImplementation "net.fabricmc:fabric-loader:${project.loader_version}"
|
||||||
|
|
||||||
|
modImplementation "net.fabricmc.fabric-api:fabric-api:${project.fabric_version}"
|
||||||
|
|
||||||
|
implementation("com.electronwill.night-config:core:3.8.2")
|
||||||
|
implementation("com.electronwill.night-config:json:3.8.2")
|
||||||
|
}
|
||||||
|
|
||||||
|
processResources {
|
||||||
|
inputs.property "version", project.version
|
||||||
|
inputs.property "minecraft_version", project.minecraft_version
|
||||||
|
inputs.property "loader_version", project.loader_version
|
||||||
|
filteringCharset "UTF-8"
|
||||||
|
|
||||||
|
filesMatching("fabric.mod.json") {
|
||||||
|
expand "version": project.version,
|
||||||
|
"minecraft_version": project.minecraft_version,
|
||||||
|
"loader_version": project.loader_version
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def targetJavaVersion = 21
|
||||||
|
tasks.withType(JavaCompile).configureEach {
|
||||||
|
// ensure that the encoding is set to UTF-8, no matter what the system default is
|
||||||
|
// this fixes some edge cases with special characters not displaying correctly
|
||||||
|
// see http://yodaconditions.net/blog/fix-for-java-file-encoding-problems-with-gradle.html
|
||||||
|
// If Javadoc is generated, this must be specified in that task too.
|
||||||
|
it.options.encoding = "UTF-8"
|
||||||
|
if (targetJavaVersion >= 10 || JavaVersion.current().isJava10Compatible()) {
|
||||||
|
it.options.release.set(targetJavaVersion)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
java {
|
||||||
|
def javaVersion = JavaVersion.toVersion(targetJavaVersion)
|
||||||
|
if (JavaVersion.current() < javaVersion) {
|
||||||
|
toolchain.languageVersion = JavaLanguageVersion.of(targetJavaVersion)
|
||||||
|
}
|
||||||
|
// Loom will automatically attach sourcesJar to a RemapSourcesJar task and to the "build" task
|
||||||
|
// if it is present.
|
||||||
|
// If you remove this line, sources will not be generated.
|
||||||
|
withSourcesJar()
|
||||||
|
}
|
||||||
|
|
||||||
|
jar {
|
||||||
|
from("LICENSE") {
|
||||||
|
rename { "${it}_${project.archivesBaseName}" }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// configure the maven publication
|
||||||
|
publishing {
|
||||||
|
publications {
|
||||||
|
create("mavenJava", MavenPublication) {
|
||||||
|
artifactId = project.archives_base_name
|
||||||
|
from components.java
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// See https://docs.gradle.org/current/userguide/publishing_maven.html for information on how to set up publishing.
|
||||||
|
repositories {
|
||||||
|
// Add repositories to publish to here.
|
||||||
|
// Notice: This block does NOT have the same function as the block in the top level.
|
||||||
|
// The repositories here will be used for publishing your artifact, not for
|
||||||
|
// retrieving dependencies.
|
||||||
|
}
|
||||||
|
}
|
||||||
14
gradle.properties
Normal file
14
gradle.properties
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# Done to increase the memory available to gradle.
|
||||||
|
org.gradle.jvmargs=-Xmx1G
|
||||||
|
# Fabric Properties
|
||||||
|
# check these on https://modmuss50.me/fabric.html
|
||||||
|
minecraft_version=1.21.5
|
||||||
|
yarn_mappings=1.21.5+build.1
|
||||||
|
loader_version=0.16.14
|
||||||
|
# Mod Properties
|
||||||
|
mod_version=1.0.0
|
||||||
|
maven_group=com.cimeyclust
|
||||||
|
archives_base_name=EZCheat
|
||||||
|
# Dependencies
|
||||||
|
# check this on https://modmuss50.me/fabric.html
|
||||||
|
fabric_version=0.128.1+1.21.5
|
||||||
1
gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
1
gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
distributionUrl=https\://services.gradle.org/distributions/gradle-8.14.1-bin.zip
|
||||||
9
settings.gradle
Normal file
9
settings.gradle
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
pluginManagement {
|
||||||
|
repositories {
|
||||||
|
maven {
|
||||||
|
name = 'Fabric'
|
||||||
|
url = 'https://maven.fabricmc.net/'
|
||||||
|
}
|
||||||
|
gradlePluginPortal()
|
||||||
|
}
|
||||||
|
}
|
||||||
32
src/main/java/com/cimeyclust/ezcheat/Config.java
Normal file
32
src/main/java/com/cimeyclust/ezcheat/Config.java
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
package com.cimeyclust.ezcheat;
|
||||||
|
|
||||||
|
import com.electronwill.nightconfig.core.file.CommentedFileConfig;
|
||||||
|
import com.electronwill.nightconfig.core.file.FileConfig;
|
||||||
|
import com.electronwill.nightconfig.core.io.WritingMode;
|
||||||
|
import net.fabricmc.loader.api.FabricLoader;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import java.nio.file.Path;
|
||||||
|
|
||||||
|
public class Config {
|
||||||
|
private static final Logger LOGGER = LoggerFactory.getLogger("ezcheat");
|
||||||
|
private static final Path CONFIG_PATH =
|
||||||
|
FabricLoader.getInstance().getConfigDir().resolve("ezcheat.json");
|
||||||
|
|
||||||
|
public static String MOD_ENDPOINT_URL = "https://modlist.ugasmp.com/api/mods";
|
||||||
|
|
||||||
|
public static void load() {
|
||||||
|
var cfg = FileConfig.builder(CONFIG_PATH)
|
||||||
|
.sync()
|
||||||
|
.autosave()
|
||||||
|
.writingMode(WritingMode.REPLACE)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
cfg.load();
|
||||||
|
MOD_ENDPOINT_URL = cfg.getOrElse("modEndpointUrl",
|
||||||
|
"https://modlist.ugasmp.com/api/mods");
|
||||||
|
cfg.save();
|
||||||
|
LOGGER.info("Loaded config: modEndpointUrl={}", MOD_ENDPOINT_URL);
|
||||||
|
}
|
||||||
|
}
|
||||||
135
src/main/java/com/cimeyclust/ezcheat/Ezcheat.java
Normal file
135
src/main/java/com/cimeyclust/ezcheat/Ezcheat.java
Normal file
@ -0,0 +1,135 @@
|
|||||||
|
package com.cimeyclust.ezcheat;
|
||||||
|
|
||||||
|
import com.google.gson.Gson;
|
||||||
|
import com.google.gson.reflect.TypeToken;
|
||||||
|
import net.fabricmc.api.ModInitializer;
|
||||||
|
import net.fabricmc.fabric.api.client.networking.v1.ClientPlayNetworking;
|
||||||
|
import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents;
|
||||||
|
import net.fabricmc.fabric.api.networking.v1.*;
|
||||||
|
import net.fabricmc.loader.api.FabricLoader;
|
||||||
|
import net.minecraft.client.network.ClientPlayerEntity;
|
||||||
|
import net.minecraft.network.PacketByteBuf;
|
||||||
|
import net.minecraft.server.MinecraftServer;
|
||||||
|
import net.minecraft.server.network.ServerPlayNetworkHandler;
|
||||||
|
import net.minecraft.server.network.ServerPlayerEntity;
|
||||||
|
import net.minecraft.text.Text;
|
||||||
|
import net.minecraft.util.Identifier;
|
||||||
|
import org.apache.commons.codec.digest.DigestUtils;
|
||||||
|
import org.apache.logging.log4j.core.jmx.Server;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
|
import java.io.Reader;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.net.HttpURLConnection;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.util.*;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
|
||||||
|
public class Ezcheat implements ModInitializer {
|
||||||
|
public static final String MODID = "ezcheat";
|
||||||
|
public static final Logger LOGGER = LoggerFactory.getLogger(MODID);
|
||||||
|
private static final Gson GSON = new Gson();
|
||||||
|
private static final Map<UUID, Long> pendingChecks = new ConcurrentHashMap<>();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInitialize() {
|
||||||
|
ServerPlayConnectionEvents.JOIN.register(this::onPlayerJoin);
|
||||||
|
ServerTickEvents.END_SERVER_TICK.register(this::onServerTick);
|
||||||
|
|
||||||
|
PayloadTypeRegistry.playC2S().register(EzcheatPayload.ID, EzcheatPayload.CODEC);
|
||||||
|
|
||||||
|
ServerPlayNetworking.registerGlobalReceiver(EzcheatPayload.ID, (payload, context) -> {
|
||||||
|
context.server().execute(() -> {
|
||||||
|
ServerPlayNetworkHandler handler = context.player().networkHandler;
|
||||||
|
MinecraftServer server = handler.player.getServer();
|
||||||
|
UUID uuid = handler.player.getUuid();
|
||||||
|
|
||||||
|
// check if we have a pending check for this player
|
||||||
|
if (pendingChecks.containsKey(uuid)) {
|
||||||
|
pendingChecks.remove(uuid);
|
||||||
|
verifyPlayerMods(handler.player, payload.hashes());
|
||||||
|
} else {
|
||||||
|
LOGGER.warn("Received mod hashes from {} without a pending check", handler.player.getName().getString());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// load config
|
||||||
|
Config.load();
|
||||||
|
LOGGER.info("Ezcheat initialized; endpoint = {}", Config.MOD_ENDPOINT_URL);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void onPlayerJoin(ServerPlayNetworkHandler handler, PacketSender sender, MinecraftServer server) {
|
||||||
|
if (server.isSingleplayer()) {
|
||||||
|
LOGGER.debug("Skipping mod check for singleplayer");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
UUID uuid = handler.player.getUuid();
|
||||||
|
pendingChecks.put(uuid, System.currentTimeMillis());
|
||||||
|
LOGGER.debug("Pending mod check for {}", handler.player.getName().getString());
|
||||||
|
}
|
||||||
|
|
||||||
|
private void onServerTick(MinecraftServer server) {
|
||||||
|
long now = System.currentTimeMillis();
|
||||||
|
pendingChecks.entrySet().removeIf(entry -> {
|
||||||
|
if (now - entry.getValue() > 10_000) {
|
||||||
|
ServerPlayerEntity p = server.getPlayerManager().getPlayer(entry.getKey());
|
||||||
|
if (p != null) {
|
||||||
|
p.networkHandler.disconnect(Text.literal("You must have Ezcheat installed (If you do, maybe there was an issue with your connection. Please try to join again in that case)."));
|
||||||
|
LOGGER.info("Kicked {} for not sending mod hashes in time", p.getName().getString());
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private String mapToString(Map<String, String> map) {
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
for (Map.Entry<String, String> entry : map.entrySet()) {
|
||||||
|
sb.append(entry.getKey()).append("=").append(entry.getValue()).append(", ");
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void verifyPlayerMods(ServerPlayerEntity player, Map<String, String> hashes) {
|
||||||
|
LOGGER.info("Verifying mod hashes: {}", mapToString(hashes));
|
||||||
|
try {
|
||||||
|
// fetch allowed list
|
||||||
|
HttpURLConnection conn = (HttpURLConnection) new URL(Config.MOD_ENDPOINT_URL).openConnection();
|
||||||
|
conn.setRequestMethod("GET");
|
||||||
|
conn.setDoInput(true);
|
||||||
|
|
||||||
|
try (InputStream is = conn.getInputStream();
|
||||||
|
Reader rdr = new InputStreamReader(is)) {
|
||||||
|
Type mapType = new TypeToken<Map<String, String>>() {
|
||||||
|
}.getType();
|
||||||
|
Map<String, String> allowed = GSON.fromJson(rdr, mapType);
|
||||||
|
|
||||||
|
List<String> bad = new ArrayList<>();
|
||||||
|
for (var e : hashes.entrySet()) {
|
||||||
|
String modid = e.getKey(), hash = e.getValue();
|
||||||
|
if (!allowed.containsValue(hash)) {
|
||||||
|
bad.add(modid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!bad.isEmpty()) {
|
||||||
|
String msg = "Unallowed mods: " + String.join(", ", bad);
|
||||||
|
player.networkHandler.disconnect(Text.literal(msg));
|
||||||
|
LOGGER.info("Kicked {} for unallowed mods: {}", player.getName().getString(), bad);
|
||||||
|
} else {
|
||||||
|
LOGGER.info("{} passed mod check", player.getName().getString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
LOGGER.error("Failed mod verification for {}: {}", player.getName().getString(), e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
42
src/main/java/com/cimeyclust/ezcheat/EzcheatPayload.java
Normal file
42
src/main/java/com/cimeyclust/ezcheat/EzcheatPayload.java
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
package com.cimeyclust.ezcheat;
|
||||||
|
|
||||||
|
import net.minecraft.network.RegistryByteBuf;
|
||||||
|
import net.minecraft.network.codec.PacketCodec;
|
||||||
|
import net.minecraft.network.packet.CustomPayload;
|
||||||
|
import net.minecraft.util.Identifier;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public record EzcheatPayload(Map<String, String> hashes) implements CustomPayload {
|
||||||
|
public static final CustomPayload.Id<EzcheatPayload> ID =
|
||||||
|
new CustomPayload.Id<>(Identifier.of(Ezcheat.MODID, "mod_hashes"));
|
||||||
|
|
||||||
|
public static final PacketCodec<RegistryByteBuf, EzcheatPayload> CODEC = new PacketCodec<>() {
|
||||||
|
@Override
|
||||||
|
public void encode(RegistryByteBuf buf, EzcheatPayload value) {
|
||||||
|
Map<String, String> hashes = value.hashes();
|
||||||
|
buf.writeVarInt(hashes.size());
|
||||||
|
for (Map.Entry<String, String> entry : hashes.entrySet()) {
|
||||||
|
buf.writeString(entry.getKey());
|
||||||
|
buf.writeString(entry.getValue());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public EzcheatPayload decode(RegistryByteBuf buf) {
|
||||||
|
int size = buf.readVarInt();
|
||||||
|
Map<String, String> hashes = new HashMap<>();
|
||||||
|
for (int i = 0; i < size; i++) {
|
||||||
|
String key = buf.readString();
|
||||||
|
String value = buf.readString();
|
||||||
|
hashes.put(key, value);
|
||||||
|
}
|
||||||
|
return new EzcheatPayload(hashes);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public CustomPayload.Id<? extends CustomPayload> getId() {
|
||||||
|
return ID;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,68 @@
|
|||||||
|
package com.cimeyclust.ezcheat.client;
|
||||||
|
|
||||||
|
import com.cimeyclust.ezcheat.Ezcheat;
|
||||||
|
import com.cimeyclust.ezcheat.EzcheatPayload;
|
||||||
|
import net.fabricmc.api.ClientModInitializer;
|
||||||
|
import net.fabricmc.fabric.api.client.networking.v1.ClientPlayConnectionEvents;
|
||||||
|
import net.fabricmc.fabric.api.client.networking.v1.ClientPlayNetworking;
|
||||||
|
import net.fabricmc.fabric.api.networking.v1.PayloadTypeRegistry;
|
||||||
|
import net.fabricmc.loader.api.FabricLoader;
|
||||||
|
import org.apache.commons.codec.digest.DigestUtils;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public class EzcheatClient implements ClientModInitializer {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInitializeClient() {
|
||||||
|
ClientPlayConnectionEvents.JOIN.register((handler, sender, client) -> {
|
||||||
|
if (!client.isInSingleplayer()) {
|
||||||
|
this.sendHashes();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Called client‑side to compute & send hashes */
|
||||||
|
private void sendHashes() {
|
||||||
|
// gather mod files
|
||||||
|
Path modsDir = FabricLoader.getInstance()
|
||||||
|
.getGameDir()
|
||||||
|
.resolve("mods")
|
||||||
|
.toAbsolutePath();
|
||||||
|
|
||||||
|
Map<String, String> hashes = new HashMap<>();
|
||||||
|
for (var mod : FabricLoader.getInstance().getAllMods()) {
|
||||||
|
try {
|
||||||
|
Path path = null;
|
||||||
|
var paths = mod.getOrigin().getPaths();
|
||||||
|
for (Path p : paths) {
|
||||||
|
if (Files.isRegularFile(p)) {
|
||||||
|
path = p;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (path == null) {
|
||||||
|
Ezcheat.LOGGER.debug("Skipping mod {}: No valid path", mod.getMetadata().getId());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (path.startsWith(modsDir)) {
|
||||||
|
try (InputStream in = Files.newInputStream(path)) {
|
||||||
|
hashes.put(mod.getMetadata().getId(), DigestUtils.sha256Hex(in));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
Ezcheat.LOGGER.warn("Could not hash {}: {}", mod.getMetadata().getId(), e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
EzcheatPayload payload = new EzcheatPayload(hashes);
|
||||||
|
ClientPlayNetworking.send(payload);
|
||||||
|
}
|
||||||
|
}
|
||||||
25
src/main/resources/fabric.mod.json
Normal file
25
src/main/resources/fabric.mod.json
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
{
|
||||||
|
"schemaVersion": 1,
|
||||||
|
"id": "ezcheat",
|
||||||
|
"version": "${version}",
|
||||||
|
"name": "EZCheat",
|
||||||
|
"description": "An easy AntiCheat based on Mod-Whitelists",
|
||||||
|
"authors": [],
|
||||||
|
"contact": {},
|
||||||
|
"license": "MIT",
|
||||||
|
"icon": "assets/ezcheat/icon.png",
|
||||||
|
"environment": "*",
|
||||||
|
"entrypoints": {
|
||||||
|
"client": [
|
||||||
|
"com.cimeyclust.ezcheat.client.EzcheatClient"
|
||||||
|
],
|
||||||
|
"main": [
|
||||||
|
"com.cimeyclust.ezcheat.Ezcheat"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"depends": {
|
||||||
|
"fabricloader": ">=${loader_version}",
|
||||||
|
"fabric": "*",
|
||||||
|
"minecraft": "${minecraft_version}"
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
x
Reference in New Issue
Block a user