org.killbill.billing.payment.dispatcher.PluginDispatcher Maven / Gradle / Ivy
/*
* Copyright 2010-2013 Ning, Inc.
* Copyright 2014 Groupon, Inc
* Copyright 2014 The Billing Project, LLC
*
* The Billing Project licenses this file to you 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 org.killbill.billing.payment.dispatcher;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import org.killbill.commons.profiling.Profiling;
import org.killbill.commons.profiling.ProfilingData;
public class PluginDispatcher {
private final TimeUnit DEEFAULT_PLUGIN_TIMEOUT_UNIT = TimeUnit.SECONDS;
private final long timeoutSeconds;
private final ExecutorService executor;
public PluginDispatcher(final long timeoutSeconds, final ExecutorService executor) {
this.timeoutSeconds = timeoutSeconds;
this.executor = executor;
}
// TODO Once we switch fully to automata, should this throw PaymentPluginApiException instead?
public ReturnType dispatchWithTimeout(final Callable> task) throws TimeoutException, ExecutionException, InterruptedException {
return dispatchWithTimeout(task, timeoutSeconds, DEEFAULT_PLUGIN_TIMEOUT_UNIT);
}
public ReturnType dispatchWithTimeout(final Callable> task, final long timeout, final TimeUnit unit)
throws TimeoutException, ExecutionException, InterruptedException {
final Future> future = executor.submit(task);
final PluginDispatcherReturnType pluginDispatcherResult = future.get(timeout, unit);
if (pluginDispatcherResult instanceof WithProfilingPluginDispatcherReturnType) {
// Transfer state from dispatch thread into current one.
final ProfilingData currentThreadProfilingData = Profiling.getPerThreadProfilingData();
if (currentThreadProfilingData != null) {
currentThreadProfilingData.merge(((WithProfilingPluginDispatcherReturnType)pluginDispatcherResult).getProfilingData());
}
}
return pluginDispatcherResult.getReturnType();
}
public interface PluginDispatcherReturnType {
public ReturnType getReturnType();
}
public interface WithProfilingPluginDispatcherReturnType extends PluginDispatcherReturnType {
public ProfilingData getProfilingData();
}
public static class DefaultWithProfilingPluginDispatcherReturnType implements WithProfilingPluginDispatcherReturnType {
private final ReturnType returnType;
private final ProfilingData profilingData;
public DefaultWithProfilingPluginDispatcherReturnType(final ReturnType returnType, final ProfilingData profilingData) {
this.returnType = returnType;
this.profilingData = profilingData;
}
@Override
public ReturnType getReturnType() {
return returnType;
}
@Override
public ProfilingData getProfilingData() {
return profilingData;
}
}
public static PluginDispatcherReturnType createPluginDispatcherReturnType(final ReturnType returnType) {
return new DefaultWithProfilingPluginDispatcherReturnType(returnType, Profiling.getPerThreadProfilingData());
}
}