HyCodeYourTale
classpublicPriority 3

PathWatcherThread

com.hypixel.hytale.server.core.asset.monitor.PathWatcherThread

implements Runnable

4

Methods

4

Public Methods

3

Fields

1

Constructors

Constants

booleanHAS_FILE_TREE_SUPPORT= SystemUtil.TYPE == SystemUtil.SystemType.WINDOWS
HytaleLoggerLOGGER= HytaleLogger.forEnclosingClass()

Constructors

public
PathWatcherThread(BiConsumer<Path, EventKind> consumer)

throws IOException

Methods

Public Methods (4)

public
void addPath(Path path)

throws IOException

publicfinal
void run()
@Override
public
void shutdown()
public
void start()

Fields

Private/Package Fields (3)

privateBiConsumer<Path, EventKind> consumer
privateWatchService service
privateThread thread

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.asset.monitor;

import com.hypixel.hytale.common.util.SystemUtil;
import com.hypixel.hytale.logger.HytaleLogger;
import com.hypixel.hytale.server.core.util.io.FileUtil;
import com.hypixel.hytale.sneakythrow.SneakyThrow;
import com.sun.nio.file.ExtendedWatchEventModifier;
import com.sun.nio.file.SensitivityWatchEventModifier;
import java.io.IOException;
import java.nio.file.FileSystems;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.StandardWatchEventKinds;
import java.nio.file.WatchEvent;
import java.nio.file.WatchKey;
import java.nio.file.WatchService;
import java.nio.file.WatchEvent.Kind;
import java.nio.file.WatchEvent.Modifier;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.BiConsumer;
import java.util.logging.Level;
import java.util.stream.Stream;
import javax.annotation.Nonnull;

public class PathWatcherThread implements Runnable {
   public static final HytaleLogger LOGGER = HytaleLogger.forEnclosingClass();
   public static final boolean HAS_FILE_TREE_SUPPORT = SystemUtil.TYPE == SystemUtil.SystemType.WINDOWS;
   public static final Kind<?>[] WATCH_EVENT_KINDS = new Kind[]{
      StandardWatchEventKinds.ENTRY_CREATE, StandardWatchEventKinds.ENTRY_DELETE, StandardWatchEventKinds.ENTRY_MODIFY
   };
   private final BiConsumer<Path, EventKind> consumer;
   @Nonnull
   private final Thread thread;
   private final WatchService service;
   private final Map<Path, WatchKey> registered = new ConcurrentHashMap<>();

   public PathWatcherThread(BiConsumer<Path, EventKind> consumer) throws IOException {
      this.consumer = consumer;
      this.thread = new Thread(this, "PathWatcher");
      this.thread.setDaemon(true);
      this.service = FileSystems.getDefault().newWatchService();
   }

   @Override
   public final void run() {
      try {
         while (!Thread.interrupted()) {
            WatchKey key = this.service.take();
            Path directory = (Path)key.watchable();

            for (WatchEvent<?> event : key.pollEvents()) {
               Kind<?> kind = event.kind();
               if (kind == StandardWatchEventKinds.OVERFLOW) {
                  LOGGER.at(Level.WARNING)
                     .log(
                        "Event Overflow, Unable to detect all file changed! This may cause server instability!! More than AbstractWatchKey.MAX_EVENT_LIST_SIZE queued events (512)"
                     );
               } else {
                  Path path = directory.resolve((Path)event.context());
                  if (!HAS_FILE_TREE_SUPPORT && event.kind() == StandardWatchEventKinds.ENTRY_CREATE && Files.isDirectory(path)) {
                     this.addPath(path);
                  }

                  this.consumer.accept(path, EventKind.parse((Kind<Path>)event.kind()));
               }
            }

            if (!key.reset()) {
               break;
            }
         }
      } catch (InterruptedException var9) {
         Thread.currentThread().interrupt();
      } catch (Throwable var10) {
         ((HytaleLogger.Api)LOGGER.at(Level.SEVERE).withCause(var10)).log("Exception occurred when polling:");
      }

      LOGGER.at(Level.INFO).log("Stopped polling for changes in assets. Server will need to be rebooted to load changes!");

      try {
         this.service.close();
      } catch (IOException var8) {
      }

      this.registered.clear();
   }

   public void start() {
      this.thread.start();
   }

   public void shutdown() {
      this.thread.interrupt();

      try {
         this.thread.join(1000L);

         try {
            this.service.close();
         } catch (IOException var2) {
         }

         this.registered.clear();
      } catch (InterruptedException var3) {
         Thread.currentThread().interrupt();
      }
   }

   public void addPath(Path path) throws IOException {
      path = path.toAbsolutePath();
      if (Files.isRegularFile(path)) {
         path = path.getParent();
      }

      Path parent = path;

      do {
         WatchKey keys = this.registered.get(parent);
         if (keys != null) {
            if (!HAS_FILE_TREE_SUPPORT) {
               this.watchPath(path);
            }

            return;
         }
      } while ((parent = parent.getParent()) != null);

      this.watchPath(path);
   }

   private void watchPath(@Nonnull Path path) throws IOException {
      if (HAS_FILE_TREE_SUPPORT) {
         this.registered.put(path, path.register(this.service, WATCH_EVENT_KINDS, SensitivityWatchEventModifier.HIGH, ExtendedWatchEventModifier.FILE_TREE));
         LOGGER.at(Level.FINEST).log("Register path: %s", path);
      } else {
         Modifier[] modifiers = new Modifier[]{SensitivityWatchEventModifier.HIGH};

         try (Stream<Path> stream = Files.walk(path, FileUtil.DEFAULT_WALK_TREE_OPTIONS_ARRAY)) {
            stream.forEach(SneakyThrow.sneakyConsumer(childPath -> {
               if (Files.isDirectory(childPath)) {
                  this.registered.put(childPath, childPath.register(this.service, WATCH_EVENT_KINDS, modifiers));
                  LOGGER.at(Level.FINEST).log("Register path: %s", childPath);
               }
            }));
         }
      }

      LOGGER.at(Level.FINER).log("Watching path: %s", path);
   }
}