/******************************************************************************* * Copyright (c) 2004, 2010 BREDEX GmbH. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * BREDEX GmbH - initial API and implementation and/or initial documentation *******************************************************************************/ package org.eclipse.jubula.rc.swt.driver; import java.lang.reflect.InvocationTargetException; import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import java.util.concurrent.FutureTask; import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeoutException; import org.apache.commons.lang.Validate; import org.eclipse.jubula.rc.common.AUTServer; import org.eclipse.jubula.rc.common.driver.IEventThreadQueuer; import org.eclipse.jubula.rc.common.driver.IRunnable; import org.eclipse.jubula.rc.common.driver.RunnableWrapper; import org.eclipse.jubula.rc.common.exception.StepExecutionException; import org.eclipse.jubula.rc.common.logger.AutServerLogger; import org.eclipse.jubula.rc.swt.SwtAUTServer; import org.eclipse.swt.SWTException; import org.eclipse.swt.widgets.Display; /** * This class executes an <code>IRunnable</code> instance in the SWT event * thread. * * @author BREDEX GmbH * @created 26.07.2006 */ public class EventThreadQueuerSwtImpl implements IEventThreadQueuer { /** the logger */ private static AutServerLogger log = new AutServerLogger(EventThreadQueuerSwtImpl.class); /** {@inheritDoc} */ public <V> V invokeAndWait(String name, IRunnable<V> runnable) throws IllegalArgumentException, StepExecutionException { Validate.notNull(runnable, "runnable must not be null"); //$NON-NLS-1$ RunnableWrapper<V> wrapper = new RunnableWrapper<V>(name, runnable); try { Display display = getDisplay(); if (display.isDisposed()) { // this may happen e.g. during the shutdown process of the AUT // see http://bugzilla.bredex.de/907 for additional information log.warn("Display has already been disposed - skipping IRunnable invocation!"); //$NON-NLS-1$ return null; } try { display.syncExec(wrapper); } catch (SWTException e) { if (display.isDisposed()) { // this may happen e.g. during the shutdown process of the AUT // see http://bugzilla.bredex.de/1591 log.warn("Display has disposed while waiting for IRunnable execution!"); //$NON-NLS-1$ return null; } log.warn("SWTException while waiting for IRunnable execution!", e); //$NON-NLS-1$ } StepExecutionException exception = wrapper.getException(); if (exception != null) { throw new InvocationTargetException(exception); } } catch (InvocationTargetException ite) { // the run() method from IRunnable has thrown an exception // -> log on info // -> throw a StepExecutionException Throwable thrown = ite.getTargetException(); if (thrown instanceof StepExecutionException) { if (log.isInfoEnabled()) { log.info(ite); } throw (StepExecutionException)thrown; } // any other (unchecked) Exception from IRunnable.run() log.error("exception thrown by '" + wrapper.getName() //$NON-NLS-1$ + "':", thrown); //$NON-NLS-1$ throw new StepExecutionException(thrown); } return wrapper.getResult(); } /** {@inheritDoc} */ public void invokeLater(String name, Runnable runnable) throws StepExecutionException { Validate.notNull(runnable, "runnable must not be null"); //$NON-NLS-1$ getDisplay().asyncExec(runnable); } /** * * @return the {@link Display} associated with the * receiver. */ private Display getDisplay() { return ((SwtAUTServer)AUTServer.getInstance()).getAutDisplay(); } /** {@inheritDoc} */ public <V> V invokeAndWait(String name, Callable<V> callable, long timeout) throws StepExecutionException, TimeoutException { Display display = getDisplay(); if (display.isDisposed()) { // this may happen e.g. during the shutdown process of the AUT // see http://bugzilla.bredex.de/907 for additional information log.warn("Display has already been disposed - skipping IRunnable invocation!"); //$NON-NLS-1$ return null; } try { FutureTask<V> task = new FutureTask<V>(callable); getDisplay().asyncExec(task); return task.get(timeout, TimeUnit.MILLISECONDS); } catch (SWTException e) { if (display.isDisposed()) { // this may happen e.g. during the shutdown process of the AUT // see http://bugzilla.bredex.de/1591 log.warn("Display has disposed while waiting for IRunnable execution!"); //$NON-NLS-1$ return null; } log.warn("SWTException while waiting for IRunnable execution!", e); //$NON-NLS-1$ } catch (InterruptedException ie) { // this (the waiting) thread was interrupted -> error log.error(ie); throw new StepExecutionException(ie); } catch (ExecutionException ee) { // the run() method from IRunnable has thrown an exception // -> log on info // -> throw a StepExecutionException Throwable thrown = ee.getCause(); if (thrown instanceof StepExecutionException) { if (log.isInfoEnabled()) { log.info(ee); } throw (StepExecutionException) thrown; } // any other (unchecked) Exception from IRunnable.run() log.error("exception thrown by '" + name //$NON-NLS-1$ + "':", thrown); //$NON-NLS-1$ throw new StepExecutionException(thrown); } return null; } }