HyCodeYourTale
classpublicPriority 1

WorldGenReloadCommand

com.hypixel.hytale.server.core.command.commands.world.worldgen.WorldGenReloadCommand

extends AbstractAsyncWorldCommand

0

Methods

0

Public Methods

0

Fields

1

Constructors

Constants

AtomicBooleanIS_RUNNING= new AtomicBoolean(false)
MessageMESSAGE_COMMANDS_WORLD_GEN_BENCHMARK_ABORT= Message.translation("server.commands.worldgen.reload.abort")
MessageMESSAGE_COMMANDS_WORLD_GEN_RELOAD_ALREADY_IN_PROGRESS= Message.translation("server.commands.worldgen.reload.alreadyInProgress")
MessageMESSAGE_COMMANDS_WORLD_GEN_RELOAD_CHUNK_SAVING_DISABLED= Message.translation( "server.commands.worldgen.reload.chunkSavingDisabled" )
MessageMESSAGE_COMMANDS_WORLD_GEN_RELOAD_CHUNK_SAVING_ENABLED= Message.translation( "server.commands.worldgen.reload.chunkSavingEnabled" )
MessageMESSAGE_COMMANDS_WORLD_GEN_RELOAD_COMPLETE= Message.translation("server.commands.worldgen.reload.complete")
MessageMESSAGE_COMMANDS_WORLD_GEN_RELOAD_DELETING_CHUNKS= Message.translation("server.commands.worldgen.reload.deletingChunks")
MessageMESSAGE_COMMANDS_WORLD_GEN_RELOAD_REGENERATING_LOADED_CHUNKS= Message.translation( "server.commands.worldgen.reload.regeneratingLoadedChunks" )
MessageMESSAGE_COMMANDS_WORLD_GEN_RELOAD_STARTED= Message.translation("server.commands.worldgen.reload.started")
MessageMKESSAGE_COMMANDS_WORLD_GEN_RELOAD_CHUNK_SAVING_ENABLED= Message.translation( "server.commands.worldgen.reload.chunkSavingEnabled" )

Constructors

public
WorldGenReloadCommand()

Inheritance

Parent
Current
Interface
Child

Use mouse wheel to zoom, drag to pan. Click nodes to navigate.

Related Classes

Source Code

package com.hypixel.hytale.server.core.command.commands.world.worldgen;

import com.hypixel.hytale.component.Store;
import com.hypixel.hytale.logger.HytaleLogger;
import com.hypixel.hytale.logger.sentry.SkipSentryException;
import com.hypixel.hytale.math.util.ChunkUtil;
import com.hypixel.hytale.math.util.MathUtil;
import com.hypixel.hytale.server.core.Message;
import com.hypixel.hytale.server.core.command.system.CommandContext;
import com.hypixel.hytale.server.core.command.system.arguments.system.FlagArg;
import com.hypixel.hytale.server.core.command.system.basecommands.AbstractAsyncWorldCommand;
import com.hypixel.hytale.server.core.universe.world.World;
import com.hypixel.hytale.server.core.universe.world.WorldConfig;
import com.hypixel.hytale.server.core.universe.world.storage.ChunkStore;
import com.hypixel.hytale.server.core.universe.world.storage.IChunkSaver;
import com.hypixel.hytale.server.core.universe.world.storage.component.ChunkSavingSystems;
import com.hypixel.hytale.server.core.universe.world.worldgen.IWorldGen;
import com.hypixel.hytale.server.core.universe.world.worldgen.WorldGenLoadException;
import com.hypixel.hytale.server.core.universe.world.worldmap.IWorldMap;
import com.hypixel.hytale.sneakythrow.SneakyThrow;
import it.unimi.dsi.fastutil.longs.LongIterator;
import it.unimi.dsi.fastutil.longs.LongSet;
import it.unimi.dsi.fastutil.objects.ObjectArrayList;
import java.io.IOException;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.logging.Level;
import javax.annotation.Nonnull;

public class WorldGenReloadCommand extends AbstractAsyncWorldCommand {
   private static final AtomicBoolean IS_RUNNING = new AtomicBoolean(false);
   @Nonnull
   private static final Message MESSAGE_COMMANDS_WORLD_GEN_RELOAD_STARTED = Message.translation("server.commands.worldgen.reload.started");
   @Nonnull
   private static final Message MESSAGE_COMMANDS_WORLD_GEN_RELOAD_COMPLETE = Message.translation("server.commands.worldgen.reload.complete");
   @Nonnull
   private static final Message MESSAGE_COMMANDS_WORLD_GEN_RELOAD_CHUNK_SAVING_DISABLED = Message.translation(
      "server.commands.worldgen.reload.chunkSavingDisabled"
   );
   @Nonnull
   private static final Message MESSAGE_COMMANDS_WORLD_GEN_RELOAD_DELETING_CHUNKS = Message.translation("server.commands.worldgen.reload.deletingChunks");
   @Nonnull
   private static final Message MKESSAGE_COMMANDS_WORLD_GEN_RELOAD_CHUNK_SAVING_ENABLED = Message.translation(
      "server.commands.worldgen.reload.chunkSavingEnabled"
   );
   @Nonnull
   private static final Message MESSAGE_COMMANDS_WORLD_GEN_RELOAD_REGENERATING_LOADED_CHUNKS = Message.translation(
      "server.commands.worldgen.reload.regeneratingLoadedChunks"
   );
   @Nonnull
   private static final Message MESSAGE_COMMANDS_WORLD_GEN_RELOAD_CHUNK_SAVING_ENABLED = Message.translation(
      "server.commands.worldgen.reload.chunkSavingEnabled"
   );
   @Nonnull
   private static final Message MESSAGE_COMMANDS_WORLD_GEN_RELOAD_ALREADY_IN_PROGRESS = Message.translation("server.commands.worldgen.reload.alreadyInProgress");
   @Nonnull
   public static final Message MESSAGE_COMMANDS_WORLD_GEN_BENCHMARK_ABORT = Message.translation("server.commands.worldgen.reload.abort");
   @Nonnull
   private final FlagArg clearArg = this.withFlagArg("clear", "server.commands.worldgen.reload.clear.desc");

   public WorldGenReloadCommand() {
      super("reload", "server.commands.worldgen.reload.desc");
   }

   @Nonnull
   @Override
   protected CompletableFuture<Void> executeAsync(@Nonnull CommandContext context, @Nonnull World world) {
      if (IS_RUNNING.get()) {
         context.sendMessage(MESSAGE_COMMANDS_WORLD_GEN_RELOAD_ALREADY_IN_PROGRESS);
         return CompletableFuture.completedFuture(null);
      } else {
         context.sendMessage(MESSAGE_COMMANDS_WORLD_GEN_RELOAD_STARTED);
         WorldConfig worldConfig = world.getWorldConfig();
         ChunkStore chunkComponentStore = world.getChunkStore();
         if (IS_RUNNING.getAndSet(true)) {
            context.sendMessage(MESSAGE_COMMANDS_WORLD_GEN_BENCHMARK_ABORT);
            return CompletableFuture.completedFuture(null);
         } else {
            CompletableFuture worldMap;
            try {
               IWorldGen worldGen = worldConfig.getWorldGenProvider().getGenerator();
               chunkComponentStore.setGenerator(worldGen);
               worldConfig.setDefaultSpawnProvider(worldGen);
               worldConfig.markChanged();
               IWorldMap worldMapx = worldConfig.getWorldMapProvider().getGenerator(world);
               world.getWorldMapManager().setGenerator(worldMapx);
               context.sendMessage(MESSAGE_COMMANDS_WORLD_GEN_RELOAD_COMPLETE);
               return this.clearArg.provided(context) ? clearChunks(context, world) : CompletableFuture.completedFuture(null);
            } catch (WorldGenLoadException var11) {
               context.sendMessage(Message.translation("server.commands.worldgen.reload.failed").param("error", var11.getTraceMessage("\n")));
               ((HytaleLogger.Api)HytaleLogger.getLogger().at(Level.SEVERE).withCause(new SkipSentryException(var11))).log("Failed to load WorldGen!");
               return CompletableFuture.completedFuture(null);
            } catch (Exception var12) {
               context.sendMessage(Message.translation("server.commands.worldgen.reload.failed").param("error", var12.getMessage()));
               ((HytaleLogger.Api)HytaleLogger.getLogger().at(Level.SEVERE).withCause(var12)).log("Exception when trying to load WorldGen!");
               worldMap = CompletableFuture.completedFuture(null);
            } finally {
               IS_RUNNING.set(false);
            }

            return worldMap;
         }
      }
   }

   @Nonnull
   private static CompletableFuture<Void> clearChunks(@Nonnull CommandContext context, @Nonnull World world) {
      ChunkStore chunkComponentStore = world.getChunkStore();
      Store<ChunkStore> componentStore = chunkComponentStore.getStore();
      ChunkSavingSystems.Data data = componentStore.getResource(ChunkStore.SAVE_RESOURCE);
      data.isSaving = false;
      data.clearSaveQueue();
      context.sendMessage(MESSAGE_COMMANDS_WORLD_GEN_RELOAD_CHUNK_SAVING_DISABLED);
      context.sendMessage(MESSAGE_COMMANDS_WORLD_GEN_RELOAD_DELETING_CHUNKS);
      IChunkSaver saver = chunkComponentStore.getSaver();
      if (saver == null) {
         context.sendMessage(MKESSAGE_COMMANDS_WORLD_GEN_RELOAD_CHUNK_SAVING_ENABLED);
         return CompletableFuture.completedFuture(null);
      } else {
         return CompletableFuture.supplyAsync(SneakyThrow.sneakySupplier(() -> {
               try {
                  return saver.getIndexes();
               } catch (IOException var3x) {
                  ((HytaleLogger.Api)HytaleLogger.getLogger().at(Level.SEVERE).withCause(var3x)).log("Failed to get chunk indexes for clearing!");
                  context.sendMessage(Message.translation("server.commands.worldgen.reload.failed").param("error", var3x.getMessage()));
                  throw SneakyThrow.sneakyThrow(var3x);
               }
            }), world)
            .thenComposeAsync(
               indexes -> {
                  AtomicInteger counter = new AtomicInteger();
                  double total = (double)indexes.size();
                  ObjectArrayList<CompletableFuture<Void>> futures = new ObjectArrayList();
                  LongIterator iterator = indexes.iterator();

                  while (iterator.hasNext()) {
                     long index = iterator.nextLong();
                     int x = ChunkUtil.xOfChunkIndex(index);
                     int z = ChunkUtil.zOfChunkIndex(index);
                     futures.add(
                        saver.removeHolder(x, z)
                           .thenRun(
                              () -> {
                                 int i = counter.getAndIncrement();
                                 if (i > 0 && i % 64 == 0) {
                                    world.execute(
                                       () -> context.sendMessage(
                                             Message.translation("server.commands.worldgen.reload.deletingChunksProgress")
                                                .param("progress", MathUtil.round((double)(i * 100) / total, 2))
                                          )
                                    );
                                 }
                              }
                           )
                     );
                  }

                  return CompletableFuture.allOf((CompletableFuture<?>[])futures.toArray(CompletableFuture[]::new));
               },
               world
            )
            .thenComposeAsync(aVoid -> {
               context.sendMessage(MESSAGE_COMMANDS_WORLD_GEN_RELOAD_REGENERATING_LOADED_CHUNKS);
               LongSet chunkIndexes = chunkComponentStore.getChunkIndexes();
               ObjectArrayList<CompletableFuture<?>> regenerateFutures = new ObjectArrayList();
               LongIterator chunkIterator = chunkIndexes.iterator();

               while (chunkIterator.hasNext()) {
                  long index = chunkIterator.nextLong();
                  regenerateFutures.add(chunkComponentStore.getChunkReferenceAsync(index, 9));
               }

               return CompletableFuture.allOf((CompletableFuture<?>[])regenerateFutures.toArray(CompletableFuture[]::new));
            }, world)
            .thenRunAsync(() -> {
               Store<ChunkStore> chunkStore = chunkComponentStore.getStore();
               ChunkSavingSystems.Data saveData = chunkStore.getResource(ChunkStore.SAVE_RESOURCE);
               saveData.isSaving = true;
               context.sendMessage(MESSAGE_COMMANDS_WORLD_GEN_RELOAD_CHUNK_SAVING_ENABLED);
            }, world);
      }
   }
}