minecraft-src/net/minecraft/server/ReloadableServerResources.java
2025-07-04 01:41:11 +03:00

164 lines
6.5 KiB
Java

package net.minecraft.server;
import com.mojang.logging.LogUtils;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Executor;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import net.minecraft.commands.CommandBuildContext;
import net.minecraft.commands.Commands;
import net.minecraft.core.Holder;
import net.minecraft.core.HolderLookup;
import net.minecraft.core.LayeredRegistryAccess;
import net.minecraft.core.Registry;
import net.minecraft.core.RegistryAccess;
import net.minecraft.resources.ResourceKey;
import net.minecraft.resources.ResourceLocation;
import net.minecraft.server.packs.resources.PreparableReloadListener;
import net.minecraft.server.packs.resources.ResourceManager;
import net.minecraft.server.packs.resources.SimpleReloadInstance;
import net.minecraft.tags.TagKey;
import net.minecraft.tags.TagManager;
import net.minecraft.util.Unit;
import net.minecraft.world.flag.FeatureFlagSet;
import net.minecraft.world.item.crafting.RecipeManager;
import net.minecraft.world.level.block.Blocks;
import net.minecraft.world.level.block.entity.AbstractFurnaceBlockEntity;
import org.slf4j.Logger;
public class ReloadableServerResources {
private static final Logger LOGGER = LogUtils.getLogger();
private static final CompletableFuture<Unit> DATA_RELOAD_INITIAL_TASK = CompletableFuture.completedFuture(Unit.INSTANCE);
private final ReloadableServerRegistries.Holder fullRegistryHolder;
private final ReloadableServerResources.ConfigurableRegistryLookup registryLookup;
private final Commands commands;
private final RecipeManager recipes;
private final TagManager tagManager;
private final ServerAdvancementManager advancements;
private final ServerFunctionLibrary functionLibrary;
private ReloadableServerResources(
RegistryAccess.Frozen registryAccess, FeatureFlagSet enabledFeatures, Commands.CommandSelection commandSelection, int functionCompilationLevel
) {
this.fullRegistryHolder = new ReloadableServerRegistries.Holder(registryAccess);
this.registryLookup = new ReloadableServerResources.ConfigurableRegistryLookup(registryAccess);
this.registryLookup.missingTagAccessPolicy(ReloadableServerResources.MissingTagAccessPolicy.CREATE_NEW);
this.recipes = new RecipeManager(this.registryLookup);
this.tagManager = new TagManager(registryAccess);
this.commands = new Commands(commandSelection, CommandBuildContext.simple(this.registryLookup, enabledFeatures));
this.advancements = new ServerAdvancementManager(this.registryLookup);
this.functionLibrary = new ServerFunctionLibrary(functionCompilationLevel, this.commands.getDispatcher());
}
public ServerFunctionLibrary getFunctionLibrary() {
return this.functionLibrary;
}
public ReloadableServerRegistries.Holder fullRegistries() {
return this.fullRegistryHolder;
}
public RecipeManager getRecipeManager() {
return this.recipes;
}
public Commands getCommands() {
return this.commands;
}
public ServerAdvancementManager getAdvancements() {
return this.advancements;
}
public List<PreparableReloadListener> listeners() {
return List.of(this.tagManager, this.recipes, this.functionLibrary, this.advancements);
}
public static CompletableFuture<ReloadableServerResources> loadResources(
ResourceManager resourceManager,
LayeredRegistryAccess<RegistryLayer> registries,
FeatureFlagSet enabledFeatures,
Commands.CommandSelection commandSelection,
int functionCompilationLevel,
Executor backgroundExecutor,
Executor gameExecutor
) {
return ReloadableServerRegistries.reload(registries, resourceManager, backgroundExecutor)
.thenCompose(
layeredRegistryAccess -> {
ReloadableServerResources reloadableServerResources = new ReloadableServerResources(
layeredRegistryAccess.compositeAccess(), enabledFeatures, commandSelection, functionCompilationLevel
);
return SimpleReloadInstance.create(
resourceManager, reloadableServerResources.listeners(), backgroundExecutor, gameExecutor, DATA_RELOAD_INITIAL_TASK, LOGGER.isDebugEnabled()
)
.done()
.whenComplete(
(object, throwable) -> reloadableServerResources.registryLookup.missingTagAccessPolicy(ReloadableServerResources.MissingTagAccessPolicy.FAIL)
)
.thenApply(object -> reloadableServerResources);
}
);
}
public void updateRegistryTags() {
this.tagManager.getResult().forEach(loadResult -> updateRegistryTags(this.fullRegistryHolder.get(), loadResult));
AbstractFurnaceBlockEntity.invalidateCache();
Blocks.rebuildCache();
}
private static <T> void updateRegistryTags(RegistryAccess registryAccess, TagManager.LoadResult<T> loadResult) {
ResourceKey<? extends Registry<T>> resourceKey = loadResult.key();
Map<TagKey<T>, List<Holder<T>>> map = (Map<TagKey<T>, List<Holder<T>>>)loadResult.tags()
.entrySet()
.stream()
.collect(
Collectors.toUnmodifiableMap(entry -> TagKey.create(resourceKey, (ResourceLocation)entry.getKey()), entry -> List.copyOf((Collection)entry.getValue()))
);
registryAccess.registryOrThrow(resourceKey).bindTags(map);
}
static class ConfigurableRegistryLookup implements HolderLookup.Provider {
private final RegistryAccess registryAccess;
ReloadableServerResources.MissingTagAccessPolicy missingTagAccessPolicy = ReloadableServerResources.MissingTagAccessPolicy.FAIL;
ConfigurableRegistryLookup(RegistryAccess registryAccess) {
this.registryAccess = registryAccess;
}
public void missingTagAccessPolicy(ReloadableServerResources.MissingTagAccessPolicy missingTagAccessPolicy) {
this.missingTagAccessPolicy = missingTagAccessPolicy;
}
@Override
public Stream<ResourceKey<? extends Registry<?>>> listRegistries() {
return this.registryAccess.listRegistries();
}
@Override
public <T> Optional<HolderLookup.RegistryLookup<T>> lookup(ResourceKey<? extends Registry<? extends T>> registryKey) {
return this.registryAccess.registry(registryKey).map(registry -> this.createDispatchedLookup(registry.asLookup(), registry.asTagAddingLookup()));
}
private <T> HolderLookup.RegistryLookup<T> createDispatchedLookup(HolderLookup.RegistryLookup<T> lookup, HolderLookup.RegistryLookup<T> tagAddingLookup) {
return new HolderLookup.RegistryLookup.Delegate<T>() {
@Override
public HolderLookup.RegistryLookup<T> parent() {
return switch (ConfigurableRegistryLookup.this.missingTagAccessPolicy) {
case CREATE_NEW -> tagAddingLookup;
case FAIL -> lookup;
};
}
};
}
}
static enum MissingTagAccessPolicy {
CREATE_NEW,
FAIL;
}
}