com.canoo.dolphin.server.context.DolphinContextTaskQueue Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of dolphin-platform-server Show documentation
Show all versions of dolphin-platform-server Show documentation
The Dolphin Platform is a framework that implements the presentation model pattern and provides amodern way to create enterprise applications. The Platform provides several client implementations that all canbe used in combination with a general sever API.
/*
* Copyright 2015-2017 Canoo Engineering AG.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.canoo.dolphin.server.context;
import com.canoo.dolphin.server.DolphinSession;
import com.canoo.dolphin.util.Assert;
import com.google.common.util.concurrent.SettableFuture;
import java.util.concurrent.atomic.AtomicBoolean;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.Future;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
/**
* Internal class to call tasks (see {@link Runnable}) in a Dolphin Platform context
* (see {@link DolphinContext}). Tasks can come from an "invokeLater" call or the event bus.
*/
public class DolphinContextTaskQueue {
private static final Logger LOG = LoggerFactory.getLogger(DolphinContextTaskQueue.class);
private final BlockingQueue tasks;
private final String dolphinSessionId;
private final long maxExecutionTime;
private final TimeUnit maxExecutionTimeUnit;
private final DolphinSessionProvider sessionProvider;
private final CommunicationManager communicationManager;
private final Lock taskLock = new ReentrantLock();
private final Condition taskCondition = taskLock.newCondition();
private final AtomicBoolean interrupted = new AtomicBoolean(false);
public DolphinContextTaskQueue(final String dolphinSessionId, final DolphinSessionProvider sessionProvider, final CommunicationManager communicationManager, final long maxExecutionTime, final TimeUnit maxExecutionTimeUnit) {
this.dolphinSessionId = Assert.requireNonBlank(dolphinSessionId, "dolphinSessionId");
this.tasks = new LinkedBlockingQueue<>();
this.communicationManager = Assert.requireNonNull(communicationManager, "communicationManager");;
this.sessionProvider = Assert.requireNonNull(sessionProvider, "sessionProvider");
this.maxExecutionTime = maxExecutionTime;
this.maxExecutionTimeUnit = Assert.requireNonNull(maxExecutionTimeUnit, "maxExecutionTimeUnit");
}
public Future addTask(final Runnable task) {
Assert.requireNonNull(task, "task");
final SettableFuture future = SettableFuture.create();
tasks.offer(new Runnable() {
@Override
public void run() {
try {
task.run();
future.set(null);
} catch (Exception e) {
future.setException(e);
}
}
});
LOG.trace("Tasks added to Dolphin Platform context {}", dolphinSessionId);
taskLock.lock();
try {
taskCondition.signal();
} finally {
taskLock.unlock();
}
return future;
}
public void interrupt() {
taskLock.lock();
try {
interrupted.set(true);
LOG.trace("Tasks in Dolphin Platform context {} interrupted", dolphinSessionId);
taskCondition.signal();
} finally {
taskLock.unlock();
}
}
public void executeTasks() {
final DolphinSession currentSession = sessionProvider.getCurrentDolphinSession();
if (currentSession == null || !dolphinSessionId.equals(currentSession.getId())) {
throw new IllegalStateException("Not in Dolphin Platform session " + dolphinSessionId);
}
LOG.trace("Running {} tasks in Dolphin Platform session {}", tasks.size(), dolphinSessionId);
long endTime = System.currentTimeMillis() + maxExecutionTimeUnit.toMillis(maxExecutionTime);
while (!communicationManager.hasResponseCommands()) {
if (interrupted.get()) {
interrupted.set(false);
break;
}
final Runnable task = tasks.poll();
if (task == null) {
try {
taskLock.lock();
try {
if (tasks.isEmpty() && !taskCondition.await(endTime - System.currentTimeMillis(), TimeUnit.MILLISECONDS)) {
interrupted.set(false);
break;
}
} finally {
taskLock.unlock();
}
if (tasks.isEmpty()) {
break;
}
} catch (InterruptedException e) {
LOG.error("Concurrency error in task executor for Dolphin Platform session {}", dolphinSessionId);
throw new IllegalStateException("Concurrency error in task executor for Dolphin Platform session " + dolphinSessionId);
}
} else {
try {
task.run();
LOG.trace("Task executor executed task in Dolphin Platform session {}", dolphinSessionId);
} catch (Exception e) {
throw new DolphinTaskException("Error in running task in Dolphin Platform session " + dolphinSessionId, e);
}
}
}
LOG.trace("Task executor for Dolphin Platform session {} ended after {} seconds with {} task still open", dolphinSessionId, maxExecutionTimeUnit.toSeconds(maxExecutionTime), tasks.size());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy