From 96d2543de60b8c28a8c0628a7d6dc06b0aa18fd0 Mon Sep 17 00:00:00 2001 From: Jake Potrebic Date: Mon, 23 Dec 2024 18:47:30 -0800 Subject: [PATCH 1/5] Add datapack registration lifecycle event --- build-data/paper.at | 2 + .../io/papermc/paper/datapack/Datapack.java | 63 +----- .../paper/datapack/DatapackRegistrar.java | 204 ++++++++++++++++++ .../paper/datapack/DiscoveredDatapack.java | 69 ++++++ .../event/types/LifecycleEvents.java | 10 + .../server/MinecraftServer.java.patch | 9 + .../server/commands/ReloadCommand.java.patch | 9 + .../repository/PackRepository.java.patch | 75 +++++++ .../repository/ServerPacksSource.java.patch | 12 +- .../papermc/paper/datapack/PaperDatapack.java | 61 +----- .../datapack/PaperDatapackRegistrar.java | 165 ++++++++++++++ .../datapack/PaperDiscoveredDatapack.java | 68 ++++++ .../paper/datapack/PluginPackSource.java | 25 +++ .../testplugin/TestPluginBootstrap.java | 17 ++ .../machine_maker/tags/entity_type/test.json | 6 + .../data/machine_maker/tags/item/test.json | 6 + .../src/main/resources/pack/tags/pack.mcmeta | 6 + 17 files changed, 697 insertions(+), 110 deletions(-) create mode 100644 paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java create mode 100644 paper-api/src/main/java/io/papermc/paper/datapack/DiscoveredDatapack.java create mode 100644 paper-server/patches/sources/net/minecraft/server/packs/repository/PackRepository.java.patch create mode 100644 paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java create mode 100644 paper-server/src/main/java/io/papermc/paper/datapack/PaperDiscoveredDatapack.java create mode 100644 paper-server/src/main/java/io/papermc/paper/datapack/PluginPackSource.java create mode 100644 test-plugin/src/main/resources/pack/data/machine_maker/tags/entity_type/test.json create mode 100644 test-plugin/src/main/resources/pack/data/machine_maker/tags/item/test.json create mode 100644 test-plugin/src/main/resources/pack/tags/pack.mcmeta diff --git a/build-data/paper.at b/build-data/paper.at index 8e1b527ed378..47ca2bb35903 100644 --- a/build-data/paper.at +++ b/build-data/paper.at @@ -110,6 +110,8 @@ public net.minecraft.server.network.ServerLoginPacketListenerImpl connection public net.minecraft.server.network.ServerLoginPacketListenerImpl state public net.minecraft.server.network.ServerLoginPacketListenerImpl$State public net.minecraft.server.packs.VanillaPackResourcesBuilder safeGetPath(Ljava/net/URI;)Ljava/nio/file/Path; +public net.minecraft.server.packs.repository.FolderRepositorySource$FolderPackDetector +public net.minecraft.server.packs.repository.FolderRepositorySource$FolderPackDetector (Lnet/minecraft/world/level/validation/DirectoryValidator;)V public net.minecraft.server.packs.repository.Pack resources public net.minecraft.server.players.PlayerList playerIo public net.minecraft.server.players.PlayerList players diff --git a/paper-api/src/main/java/io/papermc/paper/datapack/Datapack.java b/paper-api/src/main/java/io/papermc/paper/datapack/Datapack.java index 95039ec90f81..7c992578666e 100644 --- a/paper-api/src/main/java/io/papermc/paper/datapack/Datapack.java +++ b/paper-api/src/main/java/io/papermc/paper/datapack/Datapack.java @@ -1,10 +1,8 @@ package io.papermc.paper.datapack; -import java.util.Set; import net.kyori.adventure.text.Component; -import org.bukkit.FeatureFlag; +import org.jetbrains.annotations.ApiStatus; import org.jetbrains.annotations.Contract; -import org.jetbrains.annotations.Unmodifiable; import org.jspecify.annotations.NullMarked; /** @@ -12,50 +10,9 @@ * won't be updated as datapacks are updated. */ @NullMarked -public interface Datapack { +@ApiStatus.NonExtendable +public interface Datapack extends DiscoveredDatapack { - /** - * Gets the name/id of this datapack. - * - * @return the name of the pack - */ - @Contract(pure = true) - String getName(); - - /** - * Gets the title component of this datapack. - * - * @return the title - */ - Component getTitle(); - - /** - * Gets the description component of this datapack. - * - * @return the description - */ - Component getDescription(); - - /** - * Gets if this datapack is required to be enabled. - * - * @return true if the pack is required - */ - boolean isRequired(); - - /** - * Gets the compatibility status of this pack. - * - * @return the compatibility of the pack - */ - Compatibility getCompatibility(); - - /** - * Gets the set of required features for this datapack. - * - * @return the set of required features - */ - @Unmodifiable Set getRequiredFeatures(); /** * Gets the enabled state of this pack. @@ -74,13 +31,6 @@ public interface Datapack { */ void setEnabled(boolean enabled); - /** - * Gets the source for this datapack. - * - * @return the pack source - */ - DatapackSource getSource(); - /** * Computes the component vanilla Minecraft uses * to display this datapack. Includes the {@link #getSource()}, @@ -96,4 +46,11 @@ enum Compatibility { TOO_NEW, COMPATIBLE, } + + /** + * Position of the pack in the load order. + */ + enum Position { + TOP, BOTTOM + } } diff --git a/paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java b/paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java new file mode 100644 index 000000000000..456c26c9295a --- /dev/null +++ b/paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java @@ -0,0 +1,204 @@ +package io.papermc.paper.datapack; + +import io.papermc.paper.plugin.configuration.PluginMeta; +import io.papermc.paper.plugin.lifecycle.event.registrar.Registrar; +import java.io.IOException; +import java.net.URI; +import java.nio.file.Path; +import java.util.Map; +import java.util.function.Consumer; +import net.kyori.adventure.text.Component; +import org.checkerframework.checker.nullness.qual.NonNull; +import org.checkerframework.checker.nullness.qual.Nullable; +import org.jetbrains.annotations.ApiStatus; +import org.jetbrains.annotations.Contract; +import org.jetbrains.annotations.Unmodifiable; + +/** + * The registrar for datapacks. The event for this registrar + * is called anytime the game tries to discover datapacks at any of the + * configured locations. This means that if a datapack should stay available to the server, + * it must always be discovered whenever this event fires. + *

An example of a plugin loading a datapack from within it's own jar is below

+ *
{@code
+ * public class YourPluginBootstrap implements PluginBootstrap {
+ *     @Override
+ *     public void bootstrap(BoostrapContext context) {
+ *         final LifecycleEventManager manager = context.getLifecycleManager();
+ *         manager.registerEventHandler(LifecycleEvents.DATAPACK_DISCOVERY, event -> {
+ *             DatapackRegistrar registrar = event.registrar();
+ *             try {
+ *                 final URI uri = Objects.requireNonNull(
+ *                     YourPluginBootstrap.class.getResource("/pack")
+ *                 ).toURI();
+ *                 registrar.discoverPack(uri, "packId");
+ *             } catch (final URISyntaxException | IOException e) {
+ *                 throw new RuntimeException(e);
+ *             }
+ *         });
+ *     }
+ * }
+ * }
+ * @see io.papermc.paper.plugin.lifecycle.event.types.LifecycleEvents#DATAPACK_DISCOVERY + */ +@ApiStatus.NonExtendable +@ApiStatus.Experimental +public interface DatapackRegistrar extends Registrar { + + /** + * Checks if a datapack with the specified name has been discovered. + * + * @param name the name of the pack + * @return true if the pack has been discovered + * @see Datapack#getName() + */ + @Contract(pure = true) + boolean hasPackDiscovered(@NonNull String name); + + /** + * Gets a discovered datapack by its name. + * + * @param name the name of the pack + * @return the datapack + * @throws java.util.NoSuchElementException if the pack is not discovered + * @see Datapack#getName() + */ + @Contract(pure = true) + @NonNull DiscoveredDatapack getDiscoveredPack(@NonNull String name); + + /** + * Removes a discovered datapack by its name. + * + * @param name the name of the pack + * @return true if the pack was removed + * @see Datapack#getName() + */ + @Contract(mutates = "this") + boolean removeDiscoveredPack(@NonNull String name); + + /** + * Gets all discovered datapacks. + * + * @return an unmodifiable map of discovered packs + */ + @Contract(pure = true) + @Unmodifiable @NonNull Map getDiscoveredPacks(); + + /** + * Discovers a datapack at the specified {@link URI} with the id. + *

Symlinks obey the {@code allowed_symlinks.txt} in the server root directory.

+ * + * @param uri the location of the pack + * @param id a unique id (will be combined with plugin for the datapacks name) + * @return the discovered datapack (or null if it failed) + * @throws IOException if any IO error occurs + */ + default @Nullable DiscoveredDatapack discoverPack(final @NonNull URI uri, final @NonNull String id) throws IOException { + return this.discoverPack(uri, id, c -> {}); + } + + /** + * Discovers a datapack at the specified {@link URI} with the id. + *

Symlinks obey the {@code allowed_symlinks.txt} in the server root directory.

+ * + * @param uri the location of the pack + * @param id a unique id (will be combined with plugin for the datapacks name) + * @param configurer a configurer for extra options + * @return the discovered datapack (or null if it failed) + * @throws IOException if any IO error occurs + */ + @Nullable DiscoveredDatapack discoverPack(@NonNull URI uri, @NonNull String id, @NonNull Consumer configurer) throws IOException; + + /** + * Discovers a datapack at the specified {@link Path} with the id. + *

Symlinks obey the {@code allowed_symlinks.txt} in the server root directory.

+ * + * @param path the location of the pack + * @param id a unique id (will be combined with plugin for the datapacks name) + * @return the discovered datapack (or null if it failed) + * @throws IOException if any IO error occurs + */ + default @Nullable DiscoveredDatapack discoverPack(final @NonNull Path path, final @NonNull String id) throws IOException { + return this.discoverPack(path, id, c -> {}); + } + + /** + * Discovers a datapack at the specified {@link Path} with the id. + *

Symlinks obey the {@code allowed_symlinks.txt} in the server root directory.

+ * + * @param path the location of the pack + * @param id a unique id (will be combined with plugin for the datapacks name) + * @param configurer a configurer for extra options + * @return the discovered datapack (or null if it failed) + * @throws IOException if any IO error occurs + */ + @Nullable DiscoveredDatapack discoverPack(@NonNull Path path, @NonNull String id, @NonNull Consumer configurer) throws IOException; + + /** + * Discovers a datapack at the specified {@link URI} with the id. + *

Symlinks obey the {@code allowed_symlinks.txt} in the server root directory.

+ * + * @param pluginMeta the plugin which will be the "owner" of this datapack + * @param uri the location of the pack + * @param id a unique id (will be combined with plugin for the datapacks name) + * @param configurer a configurer for extra options + * @return the discovered datapack (or null if it failed) + * @throws IOException if any IO error occurs + */ + @Nullable DiscoveredDatapack discoverPack(@NonNull PluginMeta pluginMeta, @NonNull URI uri, @NonNull String id, @NonNull Consumer configurer) throws IOException; + + /** + * Discovers a datapack at the specified {@link Path} with the id. + *

Symlinks obey the {@code allowed_symlinks.txt} in the server root directory.

+ * + * @param pluginMeta the plugin which will be the "owner" of this datapack + * @param path the location of the pack + * @param id a unique id (will be combined with plugin for the datapacks name) + * @param configurer a configurer for extra options + * @return the discovered datapack (or null if it failed) + * @throws IOException if any IO error occurs + */ + @Nullable DiscoveredDatapack discoverPack(@NonNull PluginMeta pluginMeta, @NonNull Path path, @NonNull String id, @NonNull Consumer configurer) throws IOException; + + /** + * Configures additional, optional, details about a datapack. + */ + @ApiStatus.NonExtendable + @ApiStatus.Experimental + interface Configurer { + + /** + * Changes the title of the datapack from the default which + * is just the "id" in the {@code registerPack} methods. + * + * @param title the new title + * @return the configurer for chaining + */ + @Contract(value = "_ -> this", mutates = "this") + @NonNull Configurer title(@NonNull Component title); + + /** + * Sets if this pack is required. Defaults to false. + * A required pack cannot be disabled once enabled. Marking + * a pack as required does not mean it will immediately be enabled + * upon discovery. It may be enabled if this event was fired + * due to a pending (re)load. + * + * @param required true to require the pack + * @return the configurer for chaining + */ + @Contract(value = "_ -> this", mutates = "this") + @NonNull Configurer required(boolean required); + + /** + * Configures the position in the + * load order of this datapack. + * + * @param fixed won't move around in the load order as packs are added/removed + * @param position try to insert at the top of the order or bottom + * @return the configurer for chaining + */ + @Contract(value = "_, _ -> this", mutates = "this") + @NonNull Configurer position(boolean fixed, Datapack.@NonNull Position position); + } +} diff --git a/paper-api/src/main/java/io/papermc/paper/datapack/DiscoveredDatapack.java b/paper-api/src/main/java/io/papermc/paper/datapack/DiscoveredDatapack.java new file mode 100644 index 000000000000..109b93009547 --- /dev/null +++ b/paper-api/src/main/java/io/papermc/paper/datapack/DiscoveredDatapack.java @@ -0,0 +1,69 @@ +package io.papermc.paper.datapack; + +import java.util.Set; +import net.kyori.adventure.text.Component; +import org.bukkit.FeatureFlag; +import org.jetbrains.annotations.ApiStatus; +import org.jetbrains.annotations.Contract; +import org.jetbrains.annotations.Unmodifiable; +import org.jspecify.annotations.NullMarked; + +/** + * This is a snapshot of a discovered datapack on the server. It + * won't be updated as datapacks are updated. + */ +@NullMarked +@ApiStatus.NonExtendable +public interface DiscoveredDatapack { + + /** + * Gets the name/id of this datapack. + * + * @return the name of the pack + */ + @Contract(pure = true) + String getName(); + + /** + * Gets the title component of this datapack. + * + * @return the title + */ + Component getTitle(); + + /** + * Gets the description component of this datapack. + * + * @return the description + */ + Component getDescription(); + + /** + * Gets if this datapack is required to be enabled. + * + * @return true if the pack is required + */ + boolean isRequired(); + + /** + * Gets the compatibility status of this pack. + * + * @return the compatibility of the pack + */ + Datapack.Compatibility getCompatibility(); + + /** + * Gets the set of required features for this datapack. + * + * @return the set of required features + */ + @Unmodifiable + Set getRequiredFeatures(); + + /** + * Gets the source for this datapack. + * + * @return the pack source + */ + DatapackSource getSource(); +} diff --git a/paper-api/src/main/java/io/papermc/paper/plugin/lifecycle/event/types/LifecycleEvents.java b/paper-api/src/main/java/io/papermc/paper/plugin/lifecycle/event/types/LifecycleEvents.java index b65c5fd5443c..8ff52773b0df 100644 --- a/paper-api/src/main/java/io/papermc/paper/plugin/lifecycle/event/types/LifecycleEvents.java +++ b/paper-api/src/main/java/io/papermc/paper/plugin/lifecycle/event/types/LifecycleEvents.java @@ -1,10 +1,12 @@ package io.papermc.paper.plugin.lifecycle.event.types; import io.papermc.paper.command.brigadier.Commands; +import io.papermc.paper.datapack.DatapackRegistrar; import io.papermc.paper.plugin.bootstrap.BootstrapContext; import io.papermc.paper.plugin.lifecycle.event.LifecycleEvent; import io.papermc.paper.plugin.lifecycle.event.LifecycleEventManager; import io.papermc.paper.plugin.lifecycle.event.LifecycleEventOwner; +import io.papermc.paper.plugin.lifecycle.event.registrar.RegistrarEvent; import io.papermc.paper.plugin.lifecycle.event.registrar.ReloadableRegistrarEvent; import org.bukkit.plugin.Plugin; import org.jetbrains.annotations.ApiStatus; @@ -32,6 +34,14 @@ public final class LifecycleEvents { */ public static final TagEventTypeProvider TAGS = LifecycleEventTypeProvider.provider().tagProvider(); + + /** + * This event is for informing the server about any available datapacks from other sources such as inside a plugin's jar. You + * can register a handler for this event only in {@link io.papermc.paper.plugin.bootstrap.PluginBootstrap#bootstrap(BootstrapContext)}. + * @see DatapackRegistrar an example of a datapack being discovered + */ + public static final LifecycleEventType.Prioritizable> DATAPACK_DISCOVERY = bootstrapPrioritized("datapack_discovery"); + // @ApiStatus.Internal static LifecycleEventType.Monitorable plugin(final String name) { diff --git a/paper-server/patches/sources/net/minecraft/server/MinecraftServer.java.patch b/paper-server/patches/sources/net/minecraft/server/MinecraftServer.java.patch index d8518cce120d..019f5d45a1f0 100644 --- a/paper-server/patches/sources/net/minecraft/server/MinecraftServer.java.patch +++ b/paper-server/patches/sources/net/minecraft/server/MinecraftServer.java.patch @@ -1232,6 +1232,15 @@ }, this ); +@@ -1550,7 +_,7 @@ + DataPackConfig dataPackConfig = initialDataConfig.dataPacks(); + FeatureFlagSet featureFlagSet = initMode ? FeatureFlagSet.of() : initialDataConfig.enabledFeatures(); + FeatureFlagSet featureFlagSet1 = initMode ? FeatureFlags.REGISTRY.allFlags() : initialDataConfig.enabledFeatures(); +- packRepository.reload(); ++ packRepository.reload(true); // Paper - will load resource packs + if (safeMode) { + return configureRepositoryWithSelection(packRepository, List.of("vanilla"), featureFlagSet, false); + } else { @@ -1652,10 +_,11 @@ if (this.isEnforceWhitelist()) { PlayerList playerList = commandSource.getServer().getPlayerList(); diff --git a/paper-server/patches/sources/net/minecraft/server/commands/ReloadCommand.java.patch b/paper-server/patches/sources/net/minecraft/server/commands/ReloadCommand.java.patch index 47fc313fa862..cfcc1b2b5427 100644 --- a/paper-server/patches/sources/net/minecraft/server/commands/ReloadCommand.java.patch +++ b/paper-server/patches/sources/net/minecraft/server/commands/ReloadCommand.java.patch @@ -9,6 +9,15 @@ LOGGER.warn("Failed to execute reload", throwable); source.sendFailure(Component.translatable("commands.reload.failure")); return null; +@@ -24,7 +_,7 @@ + } + + private static Collection discoverNewPacks(PackRepository packRepository, WorldData worldData, Collection selectedIds) { +- packRepository.reload(); ++ packRepository.reload(true); // Paper - will perform a full reload + Collection list = Lists.newArrayList(selectedIds); + Collection disabled = worldData.getDataConfiguration().dataPacks().getDisabled(); + @@ -36,6 +_,16 @@ return list; diff --git a/paper-server/patches/sources/net/minecraft/server/packs/repository/PackRepository.java.patch b/paper-server/patches/sources/net/minecraft/server/packs/repository/PackRepository.java.patch new file mode 100644 index 000000000000..d1064bb1e6a6 --- /dev/null +++ b/paper-server/patches/sources/net/minecraft/server/packs/repository/PackRepository.java.patch @@ -0,0 +1,75 @@ +--- a/net/minecraft/server/packs/repository/PackRepository.java ++++ b/net/minecraft/server/packs/repository/PackRepository.java +@@ -21,9 +_,13 @@ + private final Set sources; + private Map available = ImmutableMap.of(); + private List selected = ImmutableList.of(); ++ private final net.minecraft.world.level.validation.DirectoryValidator validator; // Paper - add validator + +- public PackRepository(RepositorySource... sources) { +- this.sources = ImmutableSet.copyOf(sources); ++ // Paper start - add validator ++ public PackRepository(net.minecraft.world.level.validation.DirectoryValidator validator, RepositorySource... providers) { ++ this.validator = validator; ++ // Paper end - add validator ++ this.sources = ImmutableSet.copyOf(providers); + } + + public static String displayPackList(Collection packs) { +@@ -31,9 +_,14 @@ + } + + public void reload() { ++ // Paper start - perform a full reload ++ this.reload(false); ++ } ++ public void reload(boolean addRequiredPacks) { ++ // Paper end + List list = this.selected.stream().map(Pack::getId).collect(ImmutableList.toImmutableList()); + this.available = this.discoverAvailable(); +- this.selected = this.rebuildSelected(list); ++ this.selected = this.rebuildSelected(list, addRequiredPacks); // Paper + } + + private Map discoverAvailable() { +@@ -43,16 +_,23 @@ + repositorySource.loadPacks(pack -> map.put(pack.getId(), pack)); + } + +- return ImmutableMap.copyOf(map); ++ // Paper start - custom plugin-loaded datapacks ++ final io.papermc.paper.datapack.PaperDatapackRegistrar registrar = new io.papermc.paper.datapack.PaperDatapackRegistrar(this.validator, map); ++ io.papermc.paper.plugin.lifecycle.event.LifecycleEventRunner.INSTANCE.callStaticRegistrarEvent(io.papermc.paper.plugin.lifecycle.event.types.LifecycleEvents.DATAPACK_DISCOVERY, ++ registrar, ++ io.papermc.paper.plugin.bootstrap.BootstrapContext.class ++ ); ++ return ImmutableMap.copyOf(registrar.discoveredPacks); ++ // Paper end - custom plugin-loaded datapacks + } + + public boolean isAbleToClearAnyPack() { +- List list = this.rebuildSelected(List.of()); ++ List list = this.rebuildSelected(List.of(), false); // Paper - add willReload boolean + return !this.selected.equals(list); + } + + public void setSelected(Collection ids) { +- this.selected = this.rebuildSelected(ids); ++ this.selected = this.rebuildSelected(ids, false); // Paper - add willReload boolean + } + + public boolean addPack(String id) { +@@ -79,11 +_,11 @@ + } + } + +- private List rebuildSelected(Collection ids) { ++ private List rebuildSelected(Collection ids, boolean addRequiredPacks) { // Paper - add addRequiredPacks boolean + List list = this.getAvailablePacks(ids).collect(Util.toMutableList()); + + for (Pack pack : this.available.values()) { +- if (pack.isRequired() && !list.contains(pack)) { ++ if (pack.isRequired() && !list.contains(pack) && addRequiredPacks) { // Paper - add addRequiredPacks boolean + pack.getDefaultPosition().insert(list, pack, Pack::selectionConfig, false); + } + } diff --git a/paper-server/patches/sources/net/minecraft/server/packs/repository/ServerPacksSource.java.patch b/paper-server/patches/sources/net/minecraft/server/packs/repository/ServerPacksSource.java.patch index dc3a6578a8da..ee3142e9b547 100644 --- a/paper-server/patches/sources/net/minecraft/server/packs/repository/ServerPacksSource.java.patch +++ b/paper-server/patches/sources/net/minecraft/server/packs/repository/ServerPacksSource.java.patch @@ -9,7 +9,7 @@ .applyDevelopmentConfig() .pushJarResources() .build(VANILLA_PACK_INFO); -@@ -68,7 +_,18 @@ +@@ -68,15 +_,26 @@ @Nullable @Override protected Pack createBuiltinPack(String id, Pack.ResourcesSupplier resources, Component title) { @@ -29,3 +29,13 @@ } public static PackRepository createPackRepository(Path folder, DirectoryValidator validator) { +- return new PackRepository(new ServerPacksSource(validator), new FolderRepositorySource(folder, PackType.SERVER_DATA, PackSource.WORLD, validator)); ++ return new PackRepository(validator, new ServerPacksSource(validator), new FolderRepositorySource(folder, PackType.SERVER_DATA, PackSource.WORLD, validator)); // Paper - add validator + } + + public static PackRepository createVanillaTrustedRepository() { +- return new PackRepository(new ServerPacksSource(new DirectoryValidator(path -> true))); ++ return new PackRepository(new DirectoryValidator(path -> true), new ServerPacksSource(new DirectoryValidator(path -> true))); // Paper - add validator + } + + public static PackRepository createPackRepository(LevelStorageSource.LevelStorageAccess level) { diff --git a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapack.java b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapack.java index 8bd8263b51fb..bbd709529c58 100644 --- a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapack.java +++ b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapack.java @@ -2,71 +2,25 @@ import io.papermc.paper.adventure.PaperAdventure; import io.papermc.paper.event.server.ServerResourcesReloadedEvent; -import io.papermc.paper.world.flag.PaperFeatureFlagProviderImpl; import java.util.ArrayList; import java.util.List; -import java.util.Map; -import java.util.Set; -import java.util.concurrent.ConcurrentHashMap; import net.kyori.adventure.text.Component; import net.minecraft.server.MinecraftServer; import net.minecraft.server.packs.repository.Pack; -import net.minecraft.server.packs.repository.PackSource; -import org.bukkit.FeatureFlag; -import org.checkerframework.checker.nullness.qual.NonNull; -import org.checkerframework.checker.nullness.qual.Nullable; -import org.checkerframework.framework.qual.DefaultQualifier; +import org.jspecify.annotations.NullMarked; -@DefaultQualifier(NonNull.class) -public class PaperDatapack implements Datapack { - - private static final Map PACK_SOURCES = new ConcurrentHashMap<>(); - static { - PACK_SOURCES.put(PackSource.DEFAULT, DatapackSource.DEFAULT); - PACK_SOURCES.put(PackSource.BUILT_IN, DatapackSource.BUILT_IN); - PACK_SOURCES.put(PackSource.FEATURE, DatapackSource.FEATURE); - PACK_SOURCES.put(PackSource.WORLD, DatapackSource.WORLD); - PACK_SOURCES.put(PackSource.SERVER, DatapackSource.SERVER); - } +@NullMarked +public class PaperDatapack extends PaperDiscoveredDatapack implements Datapack { private final Pack pack; private final boolean enabled; PaperDatapack(final Pack pack, final boolean enabled) { + super(pack); this.pack = pack; this.enabled = enabled; } - @Override - public String getName() { - return this.pack.getId(); - } - - @Override - public Component getTitle() { - return PaperAdventure.asAdventure(this.pack.getTitle()); - } - - @Override - public Component getDescription() { - return PaperAdventure.asAdventure(this.pack.getDescription()); - } - - @Override - public boolean isRequired() { - return this.pack.isRequired(); - } - - @Override - public Compatibility getCompatibility() { - return Datapack.Compatibility.valueOf(this.pack.getCompatibility().name()); - } - - @Override - public Set getRequiredFeatures() { - return PaperFeatureFlagProviderImpl.fromNms(this.pack.getRequestedFeatures()); - } - @Override public boolean isEnabled() { return this.enabled; @@ -76,7 +30,7 @@ public boolean isEnabled() { public void setEnabled(final boolean enabled) { final MinecraftServer server = MinecraftServer.getServer(); final List enabledPacks = new ArrayList<>(server.getPackRepository().getSelectedPacks()); - final @Nullable Pack packToChange = server.getPackRepository().getPack(this.getName()); + final Pack packToChange = server.getPackRepository().getPack(this.getName()); if (packToChange == null) { throw new IllegalStateException("Cannot toggle state of pack that doesn't exist: " + this.getName()); } @@ -91,11 +45,6 @@ public void setEnabled(final boolean enabled) { server.reloadResources(enabledPacks.stream().map(Pack::getId).toList(), ServerResourcesReloadedEvent.Cause.PLUGIN); } - @Override - public DatapackSource getSource() { - return PACK_SOURCES.computeIfAbsent(this.pack.location().source(), source -> new DatapackSourceImpl(source.toString())); - } - @Override public Component computeDisplayName() { return PaperAdventure.asAdventure(this.pack.getChatLink(this.enabled)); diff --git a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java new file mode 100644 index 000000000000..c8a854bd9036 --- /dev/null +++ b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java @@ -0,0 +1,165 @@ +package io.papermc.paper.datapack; + +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableMap; +import com.mojang.logging.LogUtils; +import io.papermc.paper.adventure.PaperAdventure; +import io.papermc.paper.plugin.bootstrap.BootstrapContext; +import io.papermc.paper.plugin.configuration.PluginMeta; +import io.papermc.paper.plugin.lifecycle.event.registrar.PaperRegistrar; +import java.io.IOException; +import java.net.URI; +import java.nio.file.Path; +import java.util.ArrayList; +import java.util.List; +import java.util.Map; +import java.util.NoSuchElementException; +import java.util.Optional; +import java.util.function.Consumer; +import net.kyori.adventure.text.Component; +import net.minecraft.server.packs.PackLocationInfo; +import net.minecraft.server.packs.PackSelectionConfig; +import net.minecraft.server.packs.PackType; +import net.minecraft.server.packs.VanillaPackResourcesBuilder; +import net.minecraft.server.packs.repository.FolderRepositorySource; +import net.minecraft.server.packs.repository.Pack; +import net.minecraft.server.packs.repository.PackDetector; +import net.minecraft.world.level.validation.ContentValidationException; +import net.minecraft.world.level.validation.DirectoryValidator; +import net.minecraft.world.level.validation.ForbiddenSymlinkInfo; +import org.jetbrains.annotations.Unmodifiable; +import org.jspecify.annotations.NullMarked; +import org.jspecify.annotations.Nullable; +import org.slf4j.Logger; + +@NullMarked +public class PaperDatapackRegistrar implements PaperRegistrar, DatapackRegistrar { + + private static final Logger LOGGER = LogUtils.getClassLogger(); + + private final PackDetector detector; + public final Map discoveredPacks; + private @Nullable BootstrapContext owner; + + public PaperDatapackRegistrar(final DirectoryValidator symlinkValidator, final Map discoveredPacks) { + this.detector = new FolderRepositorySource.FolderPackDetector(symlinkValidator); + this.discoveredPacks = discoveredPacks; + } + + @Override + public void setCurrentContext(final @Nullable BootstrapContext owner) { + this.owner = owner; + } + + @Override + public boolean hasPackDiscovered(final String name) { + return this.discoveredPacks.containsKey(name); + } + + @Override + public DiscoveredDatapack getDiscoveredPack(final String name) { + if (!this.hasPackDiscovered(name)) { + throw new NoSuchElementException("No pack with id " + name + " was discovered"); + } + return new PaperDiscoveredDatapack(this.discoveredPacks.get(name)); + } + + @Override + public boolean removeDiscoveredPack(final String name) { + return this.discoveredPacks.remove(name) != null; + } + + @Override + public @Unmodifiable Map getDiscoveredPacks() { + final ImmutableMap.Builder builder = ImmutableMap.builderWithExpectedSize(this.discoveredPacks.size()); + for (final Map.Entry entry : this.discoveredPacks.entrySet()) { + builder.put(entry.getKey(), new PaperDiscoveredDatapack(entry.getValue())); + } + return builder.build(); + } + + @Override + public @Nullable DiscoveredDatapack discoverPack(final URI uri, final String id, final Consumer configurer) throws IOException { + Preconditions.checkState(this.owner != null, "Cannot register a datapack without specifying a PluginMeta yet"); + return this.discoverPack(this.owner.getPluginMeta(), uri, id, configurer); + } + + @Override + public @Nullable DiscoveredDatapack discoverPack(final Path path, final String id, final Consumer configurer) throws IOException { + Preconditions.checkState(this.owner != null, "Cannot register a datapack without specifying a PluginMeta yet"); + return this.discoverPack(this.owner.getPluginMeta(), path, id, configurer); + } + + @Override + public @Nullable DiscoveredDatapack discoverPack(final PluginMeta pluginMeta, final URI uri, final String id, final Consumer configurer) throws IOException { + return this.discoverPack(pluginMeta, VanillaPackResourcesBuilder.safeGetPath(uri), id, configurer); + } + + @Override + public @Nullable DiscoveredDatapack discoverPack(final PluginMeta pluginMeta, final Path path, final String id, final Consumer configurer) throws IOException { + final List badLinks = new ArrayList<>(); + final Pack.ResourcesSupplier resourcesSupplier = this.detector.detectPackResources(path, badLinks); + if (!badLinks.isEmpty()) { + LOGGER.warn("Ignoring potential pack entry: {}", ContentValidationException.getMessage(path, badLinks)); + } else if (resourcesSupplier != null) { + final String packId = pluginMeta.getName() + "/" + id; + final ConfigurerImpl configurerImpl = new ConfigurerImpl(Component.text(packId)); + configurer.accept(configurerImpl); + final PackLocationInfo locInfo = new PackLocationInfo(packId, + PaperAdventure.asVanilla(configurerImpl.title), + PluginPackSource.INSTANCE, + Optional.empty() + ); + final Pack pack = Pack.readMetaAndCreate(locInfo, + resourcesSupplier, + PackType.SERVER_DATA, + new PackSelectionConfig( + configurerImpl.required, + configurerImpl.position, + configurerImpl.fixedPosition + )); + if (pack != null) { + this.discoveredPacks.put(packId, pack); + return new PaperDiscoveredDatapack(pack); + } + return null; + } else { + LOGGER.info("Found non-pack entry '{}', ignoring", path); + } + return null; + } + + static final class ConfigurerImpl implements Configurer { + + private Component title; + private boolean required = false; + private boolean fixedPosition = false; + private Pack.Position position = Pack.Position.TOP; + + ConfigurerImpl(final Component title) { + this.title = title; + } + + @Override + public Configurer title(final Component title) { + this.title = title; + return this; + } + + @Override + public Configurer required(final boolean required) { + this.required = required; + return this; + } + + @Override + public Configurer position(final boolean fixed, final Datapack.Position position) { + this.fixedPosition = fixed; + this.position = switch (position) { + case TOP -> Pack.Position.TOP; + case BOTTOM -> Pack.Position.BOTTOM; + }; + return this; + } + } +} diff --git a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDiscoveredDatapack.java b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDiscoveredDatapack.java new file mode 100644 index 000000000000..7267a016bdcf --- /dev/null +++ b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDiscoveredDatapack.java @@ -0,0 +1,68 @@ +package io.papermc.paper.datapack; + +import com.google.common.collect.ImmutableMap; +import io.papermc.paper.adventure.PaperAdventure; +import io.papermc.paper.world.flag.PaperFeatureFlagProviderImpl; +import java.util.Map; +import java.util.Set; +import net.kyori.adventure.text.Component; +import net.minecraft.server.packs.repository.Pack; +import net.minecraft.server.packs.repository.PackSource; +import org.bukkit.FeatureFlag; +import org.jspecify.annotations.NullMarked; + +@NullMarked +public class PaperDiscoveredDatapack implements DiscoveredDatapack { + + private static final Map PACK_SOURCES; + static { + PACK_SOURCES = ImmutableMap.builder() + .put(PackSource.DEFAULT, DatapackSource.DEFAULT) + .put(PackSource.BUILT_IN, DatapackSource.BUILT_IN) + .put(PackSource.FEATURE, DatapackSource.FEATURE) + .put(PackSource.WORLD, DatapackSource.WORLD) + .put(PackSource.SERVER, DatapackSource.SERVER) + .buildOrThrow(); + } + + private final Pack pack; + + PaperDiscoveredDatapack(final Pack pack) { + this.pack = pack; + } + + @Override + public String getName() { + return this.pack.getId(); + } + + @Override + public Component getTitle() { + return PaperAdventure.asAdventure(this.pack.getTitle()); + } + + @Override + public Component getDescription() { + return PaperAdventure.asAdventure(this.pack.getDescription()); + } + + @Override + public boolean isRequired() { + return this.pack.isRequired(); + } + + @Override + public Datapack.Compatibility getCompatibility() { + return Datapack.Compatibility.valueOf(this.pack.getCompatibility().name()); + } + + @Override + public Set getRequiredFeatures() { + return PaperFeatureFlagProviderImpl.fromNms(this.pack.getRequestedFeatures()); + } + + @Override + public DatapackSource getSource() { + return PACK_SOURCES.computeIfAbsent(this.pack.location().source(), source -> new DatapackSourceImpl(source.toString())); + } +} diff --git a/paper-server/src/main/java/io/papermc/paper/datapack/PluginPackSource.java b/paper-server/src/main/java/io/papermc/paper/datapack/PluginPackSource.java new file mode 100644 index 000000000000..79a92a2a4047 --- /dev/null +++ b/paper-server/src/main/java/io/papermc/paper/datapack/PluginPackSource.java @@ -0,0 +1,25 @@ +package io.papermc.paper.datapack; + +import net.minecraft.ChatFormatting; +import net.minecraft.network.chat.Component; +import net.minecraft.server.packs.repository.PackSource; +import org.jspecify.annotations.NullMarked; + +@NullMarked +final class PluginPackSource implements PackSource { + + static final PackSource INSTANCE = new PluginPackSource(); + + private PluginPackSource() { + } + + @Override + public Component decorate(final Component packDisplayName) { + return Component.translatable("pack.nameAndSource", packDisplayName, "plugin").withStyle(ChatFormatting.GRAY); + } + + @Override + public boolean shouldAddAutomatically() { + return true; + } +} diff --git a/test-plugin/src/main/java/io/papermc/testplugin/TestPluginBootstrap.java b/test-plugin/src/main/java/io/papermc/testplugin/TestPluginBootstrap.java index fe2b287b257e..bfb7009bcbe5 100644 --- a/test-plugin/src/main/java/io/papermc/testplugin/TestPluginBootstrap.java +++ b/test-plugin/src/main/java/io/papermc/testplugin/TestPluginBootstrap.java @@ -1,7 +1,14 @@ package io.papermc.testplugin; +import io.papermc.paper.datapack.DatapackRegistrar; import io.papermc.paper.plugin.bootstrap.BootstrapContext; import io.papermc.paper.plugin.bootstrap.PluginBootstrap; +import io.papermc.paper.plugin.lifecycle.event.LifecycleEventManager; +import io.papermc.paper.plugin.lifecycle.event.types.LifecycleEvents; +import java.io.IOException; +import java.net.URI; +import java.net.URISyntaxException; +import java.util.Objects; import org.jetbrains.annotations.NotNull; public class TestPluginBootstrap implements PluginBootstrap { @@ -9,6 +16,16 @@ public class TestPluginBootstrap implements PluginBootstrap { @Override public void bootstrap(@NotNull BootstrapContext context) { // io.papermc.testplugin.brigtests.Registration.registerViaBootstrap(context); + final LifecycleEventManager manager = context.getLifecycleManager(); + manager.registerEventHandler(LifecycleEvents.DATAPACK_DISCOVERY, event -> { + final DatapackRegistrar registrar = event.registrar(); + try { + final URI uri = Objects.requireNonNull(TestPluginBootstrap.class.getResource("/pack")).toURI(); + registrar.discoverPack(uri, "test"); + } catch (final URISyntaxException | IOException e) { + throw new RuntimeException(e); + } + }); } } diff --git a/test-plugin/src/main/resources/pack/data/machine_maker/tags/entity_type/test.json b/test-plugin/src/main/resources/pack/data/machine_maker/tags/entity_type/test.json new file mode 100644 index 000000000000..36722a9a395a --- /dev/null +++ b/test-plugin/src/main/resources/pack/data/machine_maker/tags/entity_type/test.json @@ -0,0 +1,6 @@ +{ + "values": [ + "minecraft:zombie", + "minecraft:spider" + ] +} diff --git a/test-plugin/src/main/resources/pack/data/machine_maker/tags/item/test.json b/test-plugin/src/main/resources/pack/data/machine_maker/tags/item/test.json new file mode 100644 index 000000000000..2a659578fa88 --- /dev/null +++ b/test-plugin/src/main/resources/pack/data/machine_maker/tags/item/test.json @@ -0,0 +1,6 @@ +{ + "values": [ + "minecraft:stone_bricks", + "minecraft:string" + ] +} diff --git a/test-plugin/src/main/resources/pack/tags/pack.mcmeta b/test-plugin/src/main/resources/pack/tags/pack.mcmeta new file mode 100644 index 000000000000..8a117a64fc7c --- /dev/null +++ b/test-plugin/src/main/resources/pack/tags/pack.mcmeta @@ -0,0 +1,6 @@ +{ + "pack":{ + "pack_format": 41, + "description": "Test datapack for tags" + } +} From 956208c3cc231e69f443dc7b1724b435fef0755f Mon Sep 17 00:00:00 2001 From: Jake Potrebic Date: Fri, 27 Dec 2024 16:22:59 -0800 Subject: [PATCH 2/5] fixes --- .../papermc/paper/datapack/PaperDatapackRegistrar.java | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java index c8a854bd9036..cb1abc878ff2 100644 --- a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java +++ b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java @@ -75,18 +75,18 @@ public boolean removeDiscoveredPack(final String name) { for (final Map.Entry entry : this.discoveredPacks.entrySet()) { builder.put(entry.getKey(), new PaperDiscoveredDatapack(entry.getValue())); } - return builder.build(); + return builder.buildOrThrow(); } @Override public @Nullable DiscoveredDatapack discoverPack(final URI uri, final String id, final Consumer configurer) throws IOException { - Preconditions.checkState(this.owner != null, "Cannot register a datapack without specifying a PluginMeta yet"); + Preconditions.checkState(this.owner != null, "Discovering packs is not supported outside of lifecycle events"); return this.discoverPack(this.owner.getPluginMeta(), uri, id, configurer); } @Override public @Nullable DiscoveredDatapack discoverPack(final Path path, final String id, final Consumer configurer) throws IOException { - Preconditions.checkState(this.owner != null, "Cannot register a datapack without specifying a PluginMeta yet"); + Preconditions.checkState(this.owner != null, "Discovering packs is not supported outside of lifecycle events"); return this.discoverPack(this.owner.getPluginMeta(), path, id, configurer); } @@ -97,10 +97,12 @@ public boolean removeDiscoveredPack(final String name) { @Override public @Nullable DiscoveredDatapack discoverPack(final PluginMeta pluginMeta, final Path path, final String id, final Consumer configurer) throws IOException { + Preconditions.checkState(this.owner != null, "Discovering packs is not supported outside of lifecycle events"); final List badLinks = new ArrayList<>(); final Pack.ResourcesSupplier resourcesSupplier = this.detector.detectPackResources(path, badLinks); if (!badLinks.isEmpty()) { LOGGER.warn("Ignoring potential pack entry: {}", ContentValidationException.getMessage(path, badLinks)); + return null; } else if (resourcesSupplier != null) { final String packId = pluginMeta.getName() + "/" + id; final ConfigurerImpl configurerImpl = new ConfigurerImpl(Component.text(packId)); @@ -125,8 +127,8 @@ public boolean removeDiscoveredPack(final String name) { return null; } else { LOGGER.info("Found non-pack entry '{}', ignoring", path); + return null; } - return null; } static final class ConfigurerImpl implements Configurer { From 54423b6cced279ae7c81f715b57d5150705b6543 Mon Sep 17 00:00:00 2001 From: Bjarne Koll Date: Sat, 28 Dec 2024 21:35:42 +0100 Subject: [PATCH 3/5] Rebase --- .../io/papermc/paper/datapack/Datapack.java | 1 - .../paper/datapack/DatapackRegistrar.java | 11 +++----- .../paper/datapack/DiscoveredDatapack.java | 4 ++- .../server/MinecraftServer.java.patch | 27 ++++++++++++++++--- .../repository/PackRepository.java.patch | 19 ++++++------- .../datapack/PaperDatapackRegistrar.java | 4 +-- .../testplugin/TestPluginBootstrap.java | 17 ------------ .../machine_maker/tags/entity_type/test.json | 6 ----- .../data/machine_maker/tags/item/test.json | 6 ----- .../src/main/resources/pack/tags/pack.mcmeta | 6 ----- 10 files changed, 43 insertions(+), 58 deletions(-) delete mode 100644 test-plugin/src/main/resources/pack/data/machine_maker/tags/entity_type/test.json delete mode 100644 test-plugin/src/main/resources/pack/data/machine_maker/tags/item/test.json delete mode 100644 test-plugin/src/main/resources/pack/tags/pack.mcmeta diff --git a/paper-api/src/main/java/io/papermc/paper/datapack/Datapack.java b/paper-api/src/main/java/io/papermc/paper/datapack/Datapack.java index 7c992578666e..fff0cdb98f1b 100644 --- a/paper-api/src/main/java/io/papermc/paper/datapack/Datapack.java +++ b/paper-api/src/main/java/io/papermc/paper/datapack/Datapack.java @@ -13,7 +13,6 @@ @ApiStatus.NonExtendable public interface Datapack extends DiscoveredDatapack { - /** * Gets the enabled state of this pack. * diff --git a/paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java b/paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java index 456c26c9295a..18bbbff1d946 100644 --- a/paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java +++ b/paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java @@ -178,17 +178,14 @@ interface Configurer { @NonNull Configurer title(@NonNull Component title); /** - * Sets if this pack is required. Defaults to false. - * A required pack cannot be disabled once enabled. Marking - * a pack as required does not mean it will immediately be enabled - * upon discovery. It may be enabled if this event was fired - * due to a pending (re)load. + * Sets whether this pack is going to be automatically enabled on server starts even if previously disabled. + * Defaults to false. * - * @param required true to require the pack + * @param autoEnableOnServerStart true to ensure the pack is enabled on server starts. * @return the configurer for chaining */ @Contract(value = "_ -> this", mutates = "this") - @NonNull Configurer required(boolean required); + @NonNull Configurer autoEnableOnServerStart(boolean autoEnableOnServerStart); /** * Configures the position in the diff --git a/paper-api/src/main/java/io/papermc/paper/datapack/DiscoveredDatapack.java b/paper-api/src/main/java/io/papermc/paper/datapack/DiscoveredDatapack.java index 109b93009547..2756ca0ddb24 100644 --- a/paper-api/src/main/java/io/papermc/paper/datapack/DiscoveredDatapack.java +++ b/paper-api/src/main/java/io/papermc/paper/datapack/DiscoveredDatapack.java @@ -39,7 +39,9 @@ public interface DiscoveredDatapack { Component getDescription(); /** - * Gets if this datapack is required to be enabled. + * Gets if this datapack is required. + *

+ * A "required" datapack will always be enabled on server startup, even if previously disabled. * * @return true if the pack is required */ diff --git a/paper-server/patches/sources/net/minecraft/server/MinecraftServer.java.patch b/paper-server/patches/sources/net/minecraft/server/MinecraftServer.java.patch index 019f5d45a1f0..a56079b3cd82 100644 --- a/paper-server/patches/sources/net/minecraft/server/MinecraftServer.java.patch +++ b/paper-server/patches/sources/net/minecraft/server/MinecraftServer.java.patch @@ -1193,15 +1193,18 @@ return ReloadableServerResources.loadResources( closeableResourceManager, this.registries, -@@ -1520,6 +_,7 @@ +@@ -1520,20 +_,39 @@ ) .thenAcceptAsync( reloadableResources -> { + io.papermc.paper.command.brigadier.PaperBrigadier.moveBukkitCommands(this.resources.managers().getCommands(), reloadableResources.managers().commands); // Paper this.resources.close(); this.resources = reloadableResources; - this.packRepository.setSelected(selectedIds); -@@ -1529,11 +_,29 @@ +- this.packRepository.setSelected(selectedIds); ++ this.packRepository.setSelected(selectedIds, false); // Paper - add pendingReload flag to determine required pack loading - false as this is *after* a reload (see above) + WorldDataConfiguration worldDataConfiguration = new WorldDataConfiguration( + getSelectedPacks(this.packRepository, true), this.worldData.enabledFeatures() + ); this.worldData.setDataConfiguration(worldDataConfiguration); this.resources.managers.updateStaticRegistryTags(); this.resources.managers.getRecipeManager().finalizeRecipeLoading(this.worldData.enabledFeatures()); @@ -1241,6 +1244,24 @@ if (safeMode) { return configureRepositoryWithSelection(packRepository, List.of("vanilla"), featureFlagSet, false); } else { +@@ -1605,7 +_,7 @@ + private static WorldDataConfiguration configureRepositoryWithSelection( + PackRepository packRepository, Collection selectedPacks, FeatureFlagSet enabledFeatures, boolean safeMode + ) { +- packRepository.setSelected(selectedPacks); ++ packRepository.setSelected(selectedPacks, true); // Paper - add pendingReload flag to determine required pack loading - before the initial server load + enableForcedFeaturePacks(packRepository, enabledFeatures); + DataPackConfig selectedPacks1 = getSelectedPacks(packRepository, safeMode); + FeatureFlagSet featureFlagSet = packRepository.getRequestedFeatureFlags().join(enabledFeatures); +@@ -1637,7 +_,7 @@ + } + } + +- packRepository.setSelected(set); ++ packRepository.setSelected(set, true); // Paper - add pendingReload flag to determine required pack loading - before the initial server start + } + } + @@ -1652,10 +_,11 @@ if (this.isEnforceWhitelist()) { PlayerList playerList = commandSource.getServer().getPlayerList(); diff --git a/paper-server/patches/sources/net/minecraft/server/packs/repository/PackRepository.java.patch b/paper-server/patches/sources/net/minecraft/server/packs/repository/PackRepository.java.patch index d1064bb1e6a6..8a7d513c121f 100644 --- a/paper-server/patches/sources/net/minecraft/server/packs/repository/PackRepository.java.patch +++ b/paper-server/patches/sources/net/minecraft/server/packs/repository/PackRepository.java.patch @@ -20,15 +20,15 @@ } public void reload() { -+ // Paper start - perform a full reload ++ // Paper start - add pendingReload flag to determine required pack loading + this.reload(false); + } -+ public void reload(boolean addRequiredPacks) { -+ // Paper end ++ public void reload(final boolean pendingReload) { ++ // Paper end - add pendingReload flag to determine required pack loading List list = this.selected.stream().map(Pack::getId).collect(ImmutableList.toImmutableList()); this.available = this.discoverAvailable(); - this.selected = this.rebuildSelected(list); -+ this.selected = this.rebuildSelected(list, addRequiredPacks); // Paper ++ this.selected = this.rebuildSelected(list, pendingReload); // Paper - add pendingReload flag to determine required pack loading } private Map discoverAvailable() { @@ -49,13 +49,14 @@ public boolean isAbleToClearAnyPack() { - List list = this.rebuildSelected(List.of()); -+ List list = this.rebuildSelected(List.of(), false); // Paper - add willReload boolean ++ List list = this.rebuildSelected(List.of(), false); // Paper - add pendingReload flag to determine required pack loading return !this.selected.equals(list); } - public void setSelected(Collection ids) { +- public void setSelected(Collection ids) { - this.selected = this.rebuildSelected(ids); -+ this.selected = this.rebuildSelected(ids, false); // Paper - add willReload boolean ++ public void setSelected(Collection ids, final boolean pendingReload) { // Paper - add pendingReload flag to determine required pack loading ++ this.selected = this.rebuildSelected(ids, pendingReload); // Paper - add pendingReload flag to determine required pack loading } public boolean addPack(String id) { @@ -64,12 +65,12 @@ } - private List rebuildSelected(Collection ids) { -+ private List rebuildSelected(Collection ids, boolean addRequiredPacks) { // Paper - add addRequiredPacks boolean ++ private List rebuildSelected(Collection ids, boolean pendingReload) { // Paper - add pendingReload flag to determine required pack loading List list = this.getAvailablePacks(ids).collect(Util.toMutableList()); for (Pack pack : this.available.values()) { - if (pack.isRequired() && !list.contains(pack)) { -+ if (pack.isRequired() && !list.contains(pack) && addRequiredPacks) { // Paper - add addRequiredPacks boolean ++ if (pack.isRequired() && !list.contains(pack) && pendingReload) { // Paper - add pendingReload flag to determine required pack loading pack.getDefaultPosition().insert(list, pack, Pack::selectionConfig, false); } } diff --git a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java index cb1abc878ff2..adde6b2dcee5 100644 --- a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java +++ b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java @@ -149,8 +149,8 @@ public Configurer title(final Component title) { } @Override - public Configurer required(final boolean required) { - this.required = required; + public Configurer autoEnableOnServerStart(final boolean autoEnableOnServerStart) { + this.required = autoEnableOnServerStart; return this; } diff --git a/test-plugin/src/main/java/io/papermc/testplugin/TestPluginBootstrap.java b/test-plugin/src/main/java/io/papermc/testplugin/TestPluginBootstrap.java index bfb7009bcbe5..fe2b287b257e 100644 --- a/test-plugin/src/main/java/io/papermc/testplugin/TestPluginBootstrap.java +++ b/test-plugin/src/main/java/io/papermc/testplugin/TestPluginBootstrap.java @@ -1,14 +1,7 @@ package io.papermc.testplugin; -import io.papermc.paper.datapack.DatapackRegistrar; import io.papermc.paper.plugin.bootstrap.BootstrapContext; import io.papermc.paper.plugin.bootstrap.PluginBootstrap; -import io.papermc.paper.plugin.lifecycle.event.LifecycleEventManager; -import io.papermc.paper.plugin.lifecycle.event.types.LifecycleEvents; -import java.io.IOException; -import java.net.URI; -import java.net.URISyntaxException; -import java.util.Objects; import org.jetbrains.annotations.NotNull; public class TestPluginBootstrap implements PluginBootstrap { @@ -16,16 +9,6 @@ public class TestPluginBootstrap implements PluginBootstrap { @Override public void bootstrap(@NotNull BootstrapContext context) { // io.papermc.testplugin.brigtests.Registration.registerViaBootstrap(context); - final LifecycleEventManager manager = context.getLifecycleManager(); - manager.registerEventHandler(LifecycleEvents.DATAPACK_DISCOVERY, event -> { - final DatapackRegistrar registrar = event.registrar(); - try { - final URI uri = Objects.requireNonNull(TestPluginBootstrap.class.getResource("/pack")).toURI(); - registrar.discoverPack(uri, "test"); - } catch (final URISyntaxException | IOException e) { - throw new RuntimeException(e); - } - }); } } diff --git a/test-plugin/src/main/resources/pack/data/machine_maker/tags/entity_type/test.json b/test-plugin/src/main/resources/pack/data/machine_maker/tags/entity_type/test.json deleted file mode 100644 index 36722a9a395a..000000000000 --- a/test-plugin/src/main/resources/pack/data/machine_maker/tags/entity_type/test.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "values": [ - "minecraft:zombie", - "minecraft:spider" - ] -} diff --git a/test-plugin/src/main/resources/pack/data/machine_maker/tags/item/test.json b/test-plugin/src/main/resources/pack/data/machine_maker/tags/item/test.json deleted file mode 100644 index 2a659578fa88..000000000000 --- a/test-plugin/src/main/resources/pack/data/machine_maker/tags/item/test.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "values": [ - "minecraft:stone_bricks", - "minecraft:string" - ] -} diff --git a/test-plugin/src/main/resources/pack/tags/pack.mcmeta b/test-plugin/src/main/resources/pack/tags/pack.mcmeta deleted file mode 100644 index 8a117a64fc7c..000000000000 --- a/test-plugin/src/main/resources/pack/tags/pack.mcmeta +++ /dev/null @@ -1,6 +0,0 @@ -{ - "pack":{ - "pack_format": 41, - "description": "Test datapack for tags" - } -} From 0336f2a02a4b83b8902536f677682bd41a8ddc98 Mon Sep 17 00:00:00 2001 From: Bjarne Koll Date: Sat, 28 Dec 2024 21:43:06 +0100 Subject: [PATCH 4/5] Add plugin source to API --- .../main/java/io/papermc/paper/datapack/DatapackSource.java | 1 + .../io/papermc/paper/datapack/PaperDatapackRegistrar.java | 6 +++--- .../io/papermc/paper/datapack/PaperDiscoveredDatapack.java | 1 + 3 files changed, 5 insertions(+), 3 deletions(-) diff --git a/paper-api/src/main/java/io/papermc/paper/datapack/DatapackSource.java b/paper-api/src/main/java/io/papermc/paper/datapack/DatapackSource.java index b9b81cd974c2..1739d598f965 100644 --- a/paper-api/src/main/java/io/papermc/paper/datapack/DatapackSource.java +++ b/paper-api/src/main/java/io/papermc/paper/datapack/DatapackSource.java @@ -13,6 +13,7 @@ public sealed interface DatapackSource permits DatapackSourceImpl { DatapackSource FEATURE = create("feature"); DatapackSource WORLD = create("world"); DatapackSource SERVER = create("server"); + DatapackSource PLUGIN = create("plugin"); private static DatapackSource create(final String name) { return new DatapackSourceImpl(name); diff --git a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java index adde6b2dcee5..f5479c8b268b 100644 --- a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java +++ b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDatapackRegistrar.java @@ -116,7 +116,7 @@ public boolean removeDiscoveredPack(final String name) { resourcesSupplier, PackType.SERVER_DATA, new PackSelectionConfig( - configurerImpl.required, + configurerImpl.autoEnableOnServerStart, configurerImpl.position, configurerImpl.fixedPosition )); @@ -134,7 +134,7 @@ public boolean removeDiscoveredPack(final String name) { static final class ConfigurerImpl implements Configurer { private Component title; - private boolean required = false; + private boolean autoEnableOnServerStart = false; private boolean fixedPosition = false; private Pack.Position position = Pack.Position.TOP; @@ -150,7 +150,7 @@ public Configurer title(final Component title) { @Override public Configurer autoEnableOnServerStart(final boolean autoEnableOnServerStart) { - this.required = autoEnableOnServerStart; + this.autoEnableOnServerStart = autoEnableOnServerStart; return this; } diff --git a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDiscoveredDatapack.java b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDiscoveredDatapack.java index 7267a016bdcf..a9c7f117deb4 100644 --- a/paper-server/src/main/java/io/papermc/paper/datapack/PaperDiscoveredDatapack.java +++ b/paper-server/src/main/java/io/papermc/paper/datapack/PaperDiscoveredDatapack.java @@ -22,6 +22,7 @@ public class PaperDiscoveredDatapack implements DiscoveredDatapack { .put(PackSource.FEATURE, DatapackSource.FEATURE) .put(PackSource.WORLD, DatapackSource.WORLD) .put(PackSource.SERVER, DatapackSource.SERVER) + .put(PluginPackSource.INSTANCE, DatapackSource.PLUGIN) .buildOrThrow(); } From 1ca3fdbccee97128d7ee11e9701e2fd300a712c6 Mon Sep 17 00:00:00 2001 From: Bjarne Koll Date: Sat, 28 Dec 2024 21:47:21 +0100 Subject: [PATCH 5/5] jspecify --- .../paper/datapack/DatapackRegistrar.java | 31 ++++++++++--------- 1 file changed, 16 insertions(+), 15 deletions(-) diff --git a/paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java b/paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java index 18bbbff1d946..b009df66dbb9 100644 --- a/paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java +++ b/paper-api/src/main/java/io/papermc/paper/datapack/DatapackRegistrar.java @@ -8,11 +8,11 @@ import java.util.Map; import java.util.function.Consumer; import net.kyori.adventure.text.Component; -import org.checkerframework.checker.nullness.qual.NonNull; -import org.checkerframework.checker.nullness.qual.Nullable; import org.jetbrains.annotations.ApiStatus; import org.jetbrains.annotations.Contract; import org.jetbrains.annotations.Unmodifiable; +import org.jspecify.annotations.NullMarked; +import org.jspecify.annotations.Nullable; /** * The registrar for datapacks. The event for this registrar @@ -43,6 +43,7 @@ */ @ApiStatus.NonExtendable @ApiStatus.Experimental +@NullMarked public interface DatapackRegistrar extends Registrar { /** @@ -53,7 +54,7 @@ public interface DatapackRegistrar extends Registrar { * @see Datapack#getName() */ @Contract(pure = true) - boolean hasPackDiscovered(@NonNull String name); + boolean hasPackDiscovered(String name); /** * Gets a discovered datapack by its name. @@ -64,7 +65,7 @@ public interface DatapackRegistrar extends Registrar { * @see Datapack#getName() */ @Contract(pure = true) - @NonNull DiscoveredDatapack getDiscoveredPack(@NonNull String name); + DiscoveredDatapack getDiscoveredPack(String name); /** * Removes a discovered datapack by its name. @@ -74,7 +75,7 @@ public interface DatapackRegistrar extends Registrar { * @see Datapack#getName() */ @Contract(mutates = "this") - boolean removeDiscoveredPack(@NonNull String name); + boolean removeDiscoveredPack(String name); /** * Gets all discovered datapacks. @@ -82,7 +83,7 @@ public interface DatapackRegistrar extends Registrar { * @return an unmodifiable map of discovered packs */ @Contract(pure = true) - @Unmodifiable @NonNull Map getDiscoveredPacks(); + @Unmodifiable Map getDiscoveredPacks(); /** * Discovers a datapack at the specified {@link URI} with the id. @@ -93,7 +94,7 @@ public interface DatapackRegistrar extends Registrar { * @return the discovered datapack (or null if it failed) * @throws IOException if any IO error occurs */ - default @Nullable DiscoveredDatapack discoverPack(final @NonNull URI uri, final @NonNull String id) throws IOException { + default @Nullable DiscoveredDatapack discoverPack(final URI uri, final String id) throws IOException { return this.discoverPack(uri, id, c -> {}); } @@ -107,7 +108,7 @@ public interface DatapackRegistrar extends Registrar { * @return the discovered datapack (or null if it failed) * @throws IOException if any IO error occurs */ - @Nullable DiscoveredDatapack discoverPack(@NonNull URI uri, @NonNull String id, @NonNull Consumer configurer) throws IOException; + @Nullable DiscoveredDatapack discoverPack(URI uri, String id, Consumer configurer) throws IOException; /** * Discovers a datapack at the specified {@link Path} with the id. @@ -118,7 +119,7 @@ public interface DatapackRegistrar extends Registrar { * @return the discovered datapack (or null if it failed) * @throws IOException if any IO error occurs */ - default @Nullable DiscoveredDatapack discoverPack(final @NonNull Path path, final @NonNull String id) throws IOException { + default @Nullable DiscoveredDatapack discoverPack(final Path path, final String id) throws IOException { return this.discoverPack(path, id, c -> {}); } @@ -132,7 +133,7 @@ public interface DatapackRegistrar extends Registrar { * @return the discovered datapack (or null if it failed) * @throws IOException if any IO error occurs */ - @Nullable DiscoveredDatapack discoverPack(@NonNull Path path, @NonNull String id, @NonNull Consumer configurer) throws IOException; + @Nullable DiscoveredDatapack discoverPack(Path path, String id, Consumer configurer) throws IOException; /** * Discovers a datapack at the specified {@link URI} with the id. @@ -145,7 +146,7 @@ public interface DatapackRegistrar extends Registrar { * @return the discovered datapack (or null if it failed) * @throws IOException if any IO error occurs */ - @Nullable DiscoveredDatapack discoverPack(@NonNull PluginMeta pluginMeta, @NonNull URI uri, @NonNull String id, @NonNull Consumer configurer) throws IOException; + @Nullable DiscoveredDatapack discoverPack(PluginMeta pluginMeta, URI uri, String id, Consumer configurer) throws IOException; /** * Discovers a datapack at the specified {@link Path} with the id. @@ -158,7 +159,7 @@ public interface DatapackRegistrar extends Registrar { * @return the discovered datapack (or null if it failed) * @throws IOException if any IO error occurs */ - @Nullable DiscoveredDatapack discoverPack(@NonNull PluginMeta pluginMeta, @NonNull Path path, @NonNull String id, @NonNull Consumer configurer) throws IOException; + @Nullable DiscoveredDatapack discoverPack(PluginMeta pluginMeta, Path path, String id, Consumer configurer) throws IOException; /** * Configures additional, optional, details about a datapack. @@ -175,7 +176,7 @@ interface Configurer { * @return the configurer for chaining */ @Contract(value = "_ -> this", mutates = "this") - @NonNull Configurer title(@NonNull Component title); + Configurer title(Component title); /** * Sets whether this pack is going to be automatically enabled on server starts even if previously disabled. @@ -185,7 +186,7 @@ interface Configurer { * @return the configurer for chaining */ @Contract(value = "_ -> this", mutates = "this") - @NonNull Configurer autoEnableOnServerStart(boolean autoEnableOnServerStart); + Configurer autoEnableOnServerStart(boolean autoEnableOnServerStart); /** * Configures the position in the @@ -196,6 +197,6 @@ interface Configurer { * @return the configurer for chaining */ @Contract(value = "_, _ -> this", mutates = "this") - @NonNull Configurer position(boolean fixed, Datapack.@NonNull Position position); + Configurer position(boolean fixed, Datapack.Position position); } }