mirror of
https://github.com/WiIIiam278/HuskSync.git
synced 2025-12-19 14:59:21 +00:00
* Upgrade the Fabric version and rewrite the code. * Migrate the completed code of version 1.19.2. * fabric: some events. * Updated open source license to Apache 2.0. * Add Plan analyzer support. * Fix build. * `UnsupportedOperationException` * More fabric implementation work, update to v3's structure * Suppress compiler warnings * Add commands, adjust registration order * Inventory and ender chest data/serializers * Update license headers * Fixup shaded library relocations * Fix build * Potion effects & location serializers * Catch `Files.createDirectory(path);` in `#getDataFolder` * Update fabric.mod.json metadata, correct icon * Events for Fabric (#218) * Added apache commons pool2 dependency A NoClassDefFoundError would get thrown without this dependency. Relocation appears to not work very well either, so it has been excluded for now * Added in Item Pickup and Drop events and mixins * Update husksync.mixins.json * Switch drop item event to using Network Handler mixin * Implemented even more events - Interact block (place too) - Interact Entity - Use Item - Block Break - Player damage - Inventory Click (handles drops) - Player Commands * Re-implement the dropItem mixin * Set dropItem mixin as cancellable * deps: Include all bukkit runtime deps * fix/fabric: Supply AudienceProvider to `ConsoleUser` constructor * docs: credit Fabric porters :) * fix: Item deserialization now working * refactor: Remove inventory debug log * docs: Update `fabric.mod.json` * refactor: update with upstream changes * fix: dangling JD comment * fix: config file reference fixes * refactor: optimize imports, fix relocation * refactor: move tag references to common * refactor: use lombok for data / serializer methods * fix: bad annotating * refactor: adjust callback formatting * fabric: bump deps, refactor to match main branch * fabric: more serializer type work * feat: register more fabric data serializers also fixes a compile issue on bukkit, and refactors the JSON serializer to be in the common module * feat: implement remaining Fabric serializers * feat: add on-the-fly DFU for Fabric Now auto-upgrades item data to support version bumps. Also improved the schema a lil' bit. * feat: add missing mixins * feat: implement toKeep/toDrop option on Fabric * feat: apply stats on sync * build: append fabric MC version to file name * feat: add HuskSync API support for Fabric Also updates the docs * refactor: fixup a deprecation in the wrong spot * refactor: optimize fabric item serializing in-line with Bukkit * feat: implement viewer GUIs on Fabric * docs: Fabric is in Alpha for now --------- Co-authored-by: hanbings <hanbings@hanbings.io> Co-authored-by: Stampede <carterblowers01@gmail.com>
342 lines
12 KiB
Java
342 lines
12 KiB
Java
/*
|
|
* This file is part of HuskSync, licensed under the Apache License 2.0.
|
|
*
|
|
* Copyright (c) William278 <will27528@gmail.com>
|
|
* Copyright (c) contributors
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package net.william278.husksync;
|
|
|
|
import com.google.common.collect.Lists;
|
|
import com.google.common.collect.Maps;
|
|
import com.google.common.collect.Sets;
|
|
import com.google.gson.Gson;
|
|
import lombok.AccessLevel;
|
|
import lombok.Getter;
|
|
import lombok.NoArgsConstructor;
|
|
import lombok.Setter;
|
|
import net.fabricmc.api.DedicatedServerModInitializer;
|
|
import net.fabricmc.fabric.api.command.v2.CommandRegistrationCallback;
|
|
import net.fabricmc.fabric.api.event.lifecycle.v1.ServerLifecycleEvents;
|
|
import net.fabricmc.loader.api.FabricLoader;
|
|
import net.fabricmc.loader.api.ModContainer;
|
|
import net.kyori.adventure.platform.AudienceProvider;
|
|
import net.kyori.adventure.platform.fabric.FabricServerAudiences;
|
|
import net.minecraft.server.MinecraftServer;
|
|
import net.william278.desertwell.util.Version;
|
|
import net.william278.husksync.adapter.DataAdapter;
|
|
import net.william278.husksync.adapter.GsonAdapter;
|
|
import net.william278.husksync.adapter.SnappyGsonAdapter;
|
|
import net.william278.husksync.api.FabricHuskSyncAPI;
|
|
import net.william278.husksync.command.Command;
|
|
import net.william278.husksync.command.FabricCommand;
|
|
import net.william278.husksync.config.Locales;
|
|
import net.william278.husksync.config.Server;
|
|
import net.william278.husksync.config.Settings;
|
|
import net.william278.husksync.data.*;
|
|
import net.william278.husksync.database.Database;
|
|
import net.william278.husksync.database.MySqlDatabase;
|
|
import net.william278.husksync.event.FabricEventDispatcher;
|
|
import net.william278.husksync.hook.PlanHook;
|
|
import net.william278.husksync.listener.EventListener;
|
|
import net.william278.husksync.listener.FabricEventListener;
|
|
import net.william278.husksync.migrator.Migrator;
|
|
import net.william278.husksync.redis.RedisManager;
|
|
import net.william278.husksync.sync.DataSyncer;
|
|
import net.william278.husksync.user.ConsoleUser;
|
|
import net.william278.husksync.user.FabricUser;
|
|
import net.william278.husksync.user.OnlineUser;
|
|
import net.william278.husksync.util.FabricTask;
|
|
import net.william278.husksync.util.LegacyConverter;
|
|
import org.jetbrains.annotations.NotNull;
|
|
import org.jetbrains.annotations.Nullable;
|
|
import org.slf4j.Logger;
|
|
import org.slf4j.LoggerFactory;
|
|
import org.slf4j.spi.LoggingEventBuilder;
|
|
|
|
import java.io.IOException;
|
|
import java.io.InputStream;
|
|
import java.nio.file.Files;
|
|
import java.nio.file.Path;
|
|
import java.util.*;
|
|
import java.util.logging.Level;
|
|
import java.util.stream.Collectors;
|
|
|
|
@Getter
|
|
@NoArgsConstructor
|
|
public class FabricHuskSync implements DedicatedServerModInitializer, HuskSync, FabricTask.Supplier,
|
|
FabricEventDispatcher {
|
|
|
|
private static final String PLATFORM_TYPE_ID = "fabric";
|
|
|
|
private final TreeMap<Identifier, Serializer<? extends Data>> serializers = Maps.newTreeMap(
|
|
SerializerRegistry.DEPENDENCY_ORDER_COMPARATOR
|
|
);
|
|
private final Map<UUID, Map<Identifier, Data>> playerCustomDataStore = Maps.newConcurrentMap();
|
|
private final Map<String, Boolean> permissions = Maps.newHashMap();
|
|
private final List<Migrator> availableMigrators = Lists.newArrayList();
|
|
private final Set<UUID> lockedPlayers = Sets.newConcurrentHashSet();
|
|
|
|
private Logger logger;
|
|
private ModContainer mod;
|
|
private MinecraftServer minecraftServer;
|
|
private boolean disabling;
|
|
private Gson gson;
|
|
private AudienceProvider audiences;
|
|
private Database database;
|
|
private RedisManager redisManager;
|
|
private EventListener eventListener;
|
|
private DataAdapter dataAdapter;
|
|
@Setter
|
|
private DataSyncer dataSyncer;
|
|
@Setter
|
|
private Settings settings;
|
|
@Setter
|
|
private Locales locales;
|
|
@Setter
|
|
@Getter(AccessLevel.NONE)
|
|
private Server serverName;
|
|
|
|
@Override
|
|
public void onInitializeServer() {
|
|
// Get the logger and mod container
|
|
this.logger = LoggerFactory.getLogger("HuskSync");
|
|
this.mod = FabricLoader.getInstance().getModContainer("husksync").orElseThrow();
|
|
this.disabling = false;
|
|
this.gson = createGson();
|
|
|
|
// Load settings and locales
|
|
initialize("plugin config & locale files", (plugin) -> {
|
|
loadSettings();
|
|
loadLocales();
|
|
loadServer();
|
|
});
|
|
|
|
// Register commands
|
|
initialize("commands", (plugin) -> this.registerCommands());
|
|
|
|
// Load HuskSync after server startup
|
|
ServerLifecycleEvents.SERVER_STARTED.register(server -> {
|
|
this.minecraftServer = server;
|
|
this.onEnable();
|
|
});
|
|
|
|
// Unload HuskSync before server shutdown
|
|
ServerLifecycleEvents.SERVER_STOPPING.register(server -> this.onDisable());
|
|
}
|
|
|
|
private void onEnable() {
|
|
// Initial plugin setup
|
|
this.audiences = FabricServerAudiences.of(minecraftServer);
|
|
|
|
// Prepare data adapter
|
|
initialize("data adapter", (plugin) -> {
|
|
if (getSettings().getSynchronization().isCompressData()) {
|
|
this.dataAdapter = new SnappyGsonAdapter(this);
|
|
} else {
|
|
this.dataAdapter = new GsonAdapter(this);
|
|
}
|
|
});
|
|
|
|
initialize("data serializers", (plugin) -> {
|
|
// PERSISTENT_DATA is not registered / available on the Fabric platform
|
|
registerSerializer(Identifier.INVENTORY, new FabricSerializer.Inventory(this));
|
|
registerSerializer(Identifier.ENDER_CHEST, new FabricSerializer.EnderChest(this));
|
|
registerSerializer(Identifier.ADVANCEMENTS, new FabricSerializer.Advancements(this));
|
|
registerSerializer(Identifier.STATISTICS, new Serializer.Json<>(this, FabricData.Statistics.class)); // TODO APPLY
|
|
registerSerializer(Identifier.POTION_EFFECTS, new FabricSerializer.PotionEffects(this));
|
|
registerSerializer(Identifier.GAME_MODE, new Serializer.Json<>(this, FabricData.GameMode.class));
|
|
registerSerializer(Identifier.FLIGHT_STATUS, new Serializer.Json<>(this, FabricData.FlightStatus.class));
|
|
registerSerializer(Identifier.ATTRIBUTES, new Serializer.Json<>(this, FabricData.Attributes.class));
|
|
registerSerializer(Identifier.HEALTH, new Serializer.Json<>(this, FabricData.Health.class));
|
|
registerSerializer(Identifier.HUNGER, new Serializer.Json<>(this, FabricData.Hunger.class));
|
|
registerSerializer(Identifier.EXPERIENCE, new Serializer.Json<>(this, FabricData.Experience.class));
|
|
registerSerializer(Identifier.LOCATION, new Serializer.Json<>(this, FabricData.Location.class));
|
|
validateDependencies();
|
|
});
|
|
|
|
// Initialize the database
|
|
initialize(getSettings().getDatabase().getType().getDisplayName() + " database connection", (plugin) -> {
|
|
this.database = new MySqlDatabase(this);
|
|
this.database.initialize();
|
|
});
|
|
|
|
// Prepare redis connection
|
|
initialize("Redis server connection", (plugin) -> {
|
|
this.redisManager = new RedisManager(this);
|
|
this.redisManager.initialize();
|
|
});
|
|
|
|
// Prepare data syncer
|
|
initialize("data syncer", (plugin) -> {
|
|
dataSyncer = getSettings().getSynchronization().getMode().create(this);
|
|
dataSyncer.initialize();
|
|
});
|
|
|
|
// Register events
|
|
initialize("events", (plugin) -> this.eventListener = new FabricEventListener(this));
|
|
|
|
// Register plugin hooks
|
|
initialize("hooks", (plugin) -> {
|
|
if (isDependencyLoaded("Plan") && getSettings().isEnablePlanHook()) {
|
|
new PlanHook(this).hookIntoPlan();
|
|
}
|
|
});
|
|
|
|
// Register API
|
|
initialize("api", (plugin) -> {
|
|
FabricHuskSyncAPI.register(this);
|
|
});
|
|
|
|
// Check for updates
|
|
this.checkForUpdates();
|
|
}
|
|
|
|
private void onDisable() {
|
|
// Handle shutdown
|
|
this.disabling = true;
|
|
|
|
// Close the event listener / data syncer
|
|
if (this.dataSyncer != null) {
|
|
this.dataSyncer.terminate();
|
|
}
|
|
if (this.eventListener != null) {
|
|
this.eventListener.handlePluginDisable();
|
|
}
|
|
|
|
// Cancel tasks, close audiences
|
|
if (audiences != null) {
|
|
this.audiences.close();
|
|
}
|
|
this.cancelTasks();
|
|
|
|
// Complete shutdown
|
|
log(Level.INFO, "Successfully disabled HuskSync v" + getPluginVersion());
|
|
}
|
|
|
|
private void registerCommands() {
|
|
final List<Command> commands = FabricCommand.Type.getCommands(this);
|
|
CommandRegistrationCallback.EVENT.register((dispatcher, registry, environment) ->
|
|
commands.forEach(command -> new FabricCommand(command, this).register(dispatcher))
|
|
);
|
|
}
|
|
|
|
@NotNull
|
|
@Override
|
|
public String getServerName() {
|
|
return serverName.getName();
|
|
}
|
|
|
|
@Override
|
|
public boolean isDependencyLoaded(@NotNull String name) {
|
|
return FabricLoader.getInstance().isModLoaded(name);
|
|
}
|
|
|
|
@Override
|
|
@NotNull
|
|
public Set<OnlineUser> getOnlineUsers() {
|
|
return minecraftServer.getPlayerManager().getPlayerList()
|
|
.stream().map(user -> (OnlineUser) FabricUser.adapt(user, this))
|
|
.collect(Collectors.toSet());
|
|
}
|
|
|
|
@Override
|
|
@NotNull
|
|
public Optional<OnlineUser> getOnlineUser(@NotNull UUID uuid) {
|
|
return Optional.ofNullable(minecraftServer.getPlayerManager().getPlayer(uuid))
|
|
.map(user -> FabricUser.adapt(user, this));
|
|
}
|
|
|
|
@Override
|
|
@Nullable
|
|
public InputStream getResource(@NotNull String name) {
|
|
return this.mod.findPath(name)
|
|
.map(path -> {
|
|
try {
|
|
return Files.newInputStream(path);
|
|
} catch (IOException e) {
|
|
log(Level.WARNING, "Failed to load resource: " + name, e);
|
|
}
|
|
return null;
|
|
})
|
|
.orElse(this.getClass().getClassLoader().getResourceAsStream(name));
|
|
}
|
|
|
|
@Override
|
|
@NotNull
|
|
public Path getConfigDirectory() {
|
|
final Path path = FabricLoader.getInstance().getConfigDir().resolve("husksync");
|
|
if (!Files.isDirectory(path)) {
|
|
try {
|
|
Files.createDirectory(path);
|
|
} catch (IOException e) {
|
|
log(Level.SEVERE, "Failed to create config directory", e);
|
|
}
|
|
}
|
|
return path;
|
|
}
|
|
|
|
@Override
|
|
public void log(@NotNull Level level, @NotNull String message, @NotNull Throwable... throwable) {
|
|
LoggingEventBuilder logEvent = logger.makeLoggingEventBuilder(
|
|
switch (level.getName()) {
|
|
case "WARNING" -> org.slf4j.event.Level.WARN;
|
|
case "SEVERE" -> org.slf4j.event.Level.ERROR;
|
|
default -> org.slf4j.event.Level.INFO;
|
|
}
|
|
);
|
|
if (throwable.length >= 1) {
|
|
logEvent = logEvent.setCause(throwable[0]);
|
|
}
|
|
logEvent.log(message);
|
|
}
|
|
|
|
@NotNull
|
|
@Override
|
|
public ConsoleUser getConsole() {
|
|
return new ConsoleUser(audiences);
|
|
}
|
|
|
|
@Override
|
|
@NotNull
|
|
public Version getPluginVersion() {
|
|
return Version.fromString(mod.getMetadata().getVersion().getFriendlyString(), "-");
|
|
}
|
|
|
|
@Override
|
|
@NotNull
|
|
public Version getMinecraftVersion() {
|
|
return Version.fromString(minecraftServer.getVersion());
|
|
}
|
|
|
|
@NotNull
|
|
@Override
|
|
public String getPlatformType() {
|
|
return PLATFORM_TYPE_ID;
|
|
}
|
|
|
|
@Override
|
|
public Optional<LegacyConverter> getLegacyConverter() {
|
|
return Optional.empty();
|
|
}
|
|
|
|
@Override
|
|
@NotNull
|
|
public FabricHuskSync getPlugin() {
|
|
return this;
|
|
}
|
|
|
|
}
|