/* * This file is part of LanternServer, licensed under the MIT License (MIT). * * Copyright (c) LanternPowered <https://www.lanternpowered.org> * Copyright (c) SpongePowered <https://www.spongepowered.org> * Copyright (c) contributors * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the Software), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED AS IS, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ package org.lanternpowered.server.scheduler; import static com.google.common.base.Preconditions.checkNotNull; import static org.lanternpowered.server.util.Conditions.checkPlugin; import com.google.common.collect.ImmutableSet; import com.google.common.util.concurrent.ListenableFutureTask; import org.lanternpowered.server.game.Lantern; import org.lanternpowered.server.game.LanternGame; import org.spongepowered.api.plugin.PluginContainer; import org.spongepowered.api.scheduler.Scheduler; import org.spongepowered.api.scheduler.SpongeExecutorService; import org.spongepowered.api.scheduler.Task; import org.spongepowered.api.util.Functional; import org.spongepowered.api.util.GuavaCollectors; import java.util.Optional; import java.util.Set; import java.util.UUID; import java.util.concurrent.Callable; import java.util.concurrent.CompletableFuture; import java.util.concurrent.Future; import java.util.regex.Pattern; public class LanternScheduler implements Scheduler { private final AsyncScheduler asyncScheduler = new AsyncScheduler(); private final SyncScheduler syncScheduler = new SyncScheduler(); @Override public LanternTaskBuilder createTaskBuilder() { return new LanternTaskBuilder(this); } @Override public Optional<Task> getTaskById(UUID id) { final Optional<Task> task = this.syncScheduler.getTask(id); if (task.isPresent()) { return task; } return this.asyncScheduler.getTask(id); } @Override public Set<Task> getTasksByName(String pattern) { final Pattern searchPattern = Pattern.compile(checkNotNull(pattern, "pattern")); return this.getScheduledTasks().stream() .filter(task -> searchPattern.matcher(task.getName()).matches()) .collect(GuavaCollectors.toImmutableSet()); } @Override public Set<Task> getScheduledTasks() { final ImmutableSet.Builder<Task> builder = ImmutableSet.builder(); builder.addAll(this.asyncScheduler.getScheduledTasks()); builder.addAll(this.syncScheduler.getScheduledTasks()); return builder.build(); } @Override public Set<Task> getScheduledTasks(boolean async) { if (async) { return this.asyncScheduler.getScheduledTasks(); } else { return this.syncScheduler.getScheduledTasks(); } } @Override public Set<Task> getScheduledTasks(Object plugin) { final PluginContainer pluginContainer = checkPlugin(plugin, "plugin"); return this.getScheduledTasks().stream() .filter(task -> task.getOwner().equals(pluginContainer)) .collect(GuavaCollectors.toImmutableSet()); } @Override public int getPreferredTickInterval() { return LanternGame.TICK_DURATION; } /** * Calls the callable from the main thread. * * @param callable the callable * @return the future result */ public <V> Future<V> callSync(Callable<V> callable) { final ListenableFutureTask<V> future = ListenableFutureTask.create(callable); this.createTaskBuilder().execute(future).submit(Lantern.getMinecraftPlugin()); return future; } private SchedulerBase getDelegate(Task task) { if (task.isAsynchronous()) { return this.asyncScheduler; } else { return this.syncScheduler; } } private SchedulerBase getDelegate(ScheduledTask.TaskSynchronicity syncType) { if (syncType == ScheduledTask.TaskSynchronicity.ASYNCHRONOUS) { return this.asyncScheduler; } else { return this.syncScheduler; } } String getNameFor(PluginContainer plugin, ScheduledTask.TaskSynchronicity syncType) { return this.getDelegate(syncType).nextName(plugin); } void submit(ScheduledTask task) { this.getDelegate(task).addTask(task); } /** * Pulses the synchronous scheduler. */ public void pulseSyncScheduler() { this.syncScheduler.tick(); } public void shutdownAsyncScheduler() { this.asyncScheduler.shutdown(); } @Override public SpongeExecutorService createSyncExecutor(Object plugin) { return new TaskExecutorService(this::createTaskBuilder, this.syncScheduler, checkPlugin(plugin, "plugin")); } @Override public SpongeExecutorService createAsyncExecutor(Object plugin) { return new TaskExecutorService(() -> this.createTaskBuilder().async(), this.asyncScheduler, checkPlugin(plugin, "plugin")); } public <T> CompletableFuture<T> submitAsyncTask(Callable<T> callable) { return Functional.asyncFailableFuture(callable, this.asyncScheduler.getExecutor()); } }