view client/swing/src/test/java/com/redhat/thermostat/client/swing/internal/MainTest.java @ 1019:f84a49f6c797 0.6-branch

Remove UiFacadeFactory This commit removes UiFacadeFactory. Previous uses of this class have been converted to use trackers instead. This includes accessing the Host and VmInfoDAOs, and Host/VMContextActions. Instead of using UiFacadeFactory, these trackers now have getter methods for retrieving the list of tracked services. Trackers have been moved from ThermostatActivator to MainControllerImpl as neither the activator nor Main have any use for the tracked services. Shutdown has also been modified due to MainControllerImpl having trackers. When the shutdown latch in Main is triggered, Main then calls shutdownApplication in MainControllerImpl to close the trackers and perform other previously existing cleanup. I have also added a test for VMContextActionServiceTracker, since only the Host version was tested before. I have also encapsulated the runnable that instantiates MainControllerImpl in Main into a nested class, allowing us to use a mock in MainTest instead of actually instantiating it. Reviewed-by: neugens Review-thread: http://icedtea.classpath.org/pipermail/thermostat/2013-February/005944.html
author Elliott Baron <ebaron@redhat.com>
date Mon, 04 Mar 2013 17:04:10 -0500
parents b0879f47909f
children
line wrap: on
line source

/*
 * Copyright 2012, 2013 Red Hat, Inc.
 *
 * This file is part of Thermostat.
 *
 * Thermostat is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published
 * by the Free Software Foundation; either version 2, or (at your
 * option) any later version.
 *
 * Thermostat is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with Thermostat; see the file COPYING.  If not see
 * <http://www.gnu.org/licenses/>.
 *
 * Linking this code with other modules is making a combined work
 * based on this code.  Thus, the terms and conditions of the GNU
 * General Public License cover the whole combination.
 *
 * As a special exception, the copyright holders of this code give
 * you permission to link this code with independent modules to
 * produce an executable, regardless of the license terms of these
 * independent modules, and to copy and distribute the resulting
 * executable under terms of your choice, provided that you also
 * meet, for each linked independent module, the terms and conditions
 * of the license of that module.  An independent module is a module
 * which is not derived from or based on this code.  If you modify
 * this code, you may extend this exception to your version of the
 * library, but you are not obligated to do so.  If you do not wish
 * to do so, delete this exception statement from your version.
 */

package com.redhat.thermostat.client.swing.internal;

import static org.mockito.Matchers.anyString;
import static org.mockito.Matchers.isA;
import static org.mockito.Mockito.doAnswer;
import static org.mockito.Mockito.doNothing;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;

import javax.swing.SwingUtilities;

import org.junit.Before;
import org.junit.Ignore;
import org.junit.Test;
import org.mockito.ArgumentCaptor;
import org.mockito.invocation.InvocationOnMock;
import org.mockito.stubbing.Answer;

import com.redhat.thermostat.client.swing.internal.Main.MainWindowRunnable;
import com.redhat.thermostat.common.ApplicationService;
import com.redhat.thermostat.common.TimerFactory;
import com.redhat.thermostat.storage.core.Connection.ConnectionListener;
import com.redhat.thermostat.storage.core.Connection.ConnectionStatus;
import com.redhat.thermostat.storage.core.DbService;
import com.redhat.thermostat.storage.core.DbServiceFactory;
import com.redhat.thermostat.storage.dao.HostInfoDAO;
import com.redhat.thermostat.storage.dao.VmInfoDAO;
import com.redhat.thermostat.testutils.StubBundleContext;
import com.redhat.thermostat.utils.keyring.Keyring;

public class MainTest {

    private ExecutorService executorService;

    private MainWindowRunnable mainWindowRunnable;

    private ArgumentCaptor<ConnectionListener> connectionListenerCaptor;

    private DbServiceFactory dbServiceFactory;
    private DbService dbService;

    private TimerFactory timerFactory;
    private StubBundleContext context;
    private ApplicationService appService;
    private Keyring keyring;
    private CountDownLatch shutdown;

    @Before
    public void setUp() {
        context = new StubBundleContext();
        appService = mock(ApplicationService.class);
        context.registerService(ApplicationService.class, appService, null);

        executorService = mock(ExecutorService.class);
        doAnswer(new Answer<Void>() {
            @Override
            public Void answer(InvocationOnMock invocation) throws Throwable {
                Runnable runnable = (Runnable) invocation.getArguments()[0];
                runnable.run();
                return null;
            }
        }).when(executorService).execute(isA(Runnable.class));

        when(appService.getApplicationExecutor()).thenReturn(executorService);
        
        mainWindowRunnable = mock(MainWindowRunnable.class);
        
        dbServiceFactory = mock(DbServiceFactory.class);
        dbService = mock(DbService.class);
        when(dbServiceFactory.createDbService(anyString(), anyString(), anyString())).thenReturn(dbService);
        
        connectionListenerCaptor = ArgumentCaptor.forClass(ConnectionListener.class);
        doNothing().when(dbService).addConnectionListener(connectionListenerCaptor.capture());

        timerFactory = mock(TimerFactory.class);
        when(appService.getTimerFactory()).thenReturn(timerFactory);

        keyring = mock(Keyring.class);
        shutdown = mock(CountDownLatch.class);
    }

    /**
     * Handle all outstanding EDT events by posting a no-op event and waiting
     * until it completes.
     */
    private void handleAllEdtEvents() throws Exception {
        SwingUtilities.invokeAndWait(new Runnable() {
            @Override
            public void run() {
                /* NO-OP */
            }
        });
    }

    @Test
    public void verifyRunWaitsForShutdown() throws Exception {
        Main main = new Main(context, appService, dbServiceFactory, keyring, shutdown, mainWindowRunnable);

        main.run();

        handleAllEdtEvents();

        verify(shutdown).await();
    }

    @Test
    public void verifyConnectionIsMade() throws Exception {
        Main main = new Main(context, appService, dbServiceFactory, keyring, shutdown, mainWindowRunnable);

        main.run();

        handleAllEdtEvents();

        verify(dbService).connect();

    }

    @Test
    public void verifySuccessfulConnectionTriggersMainWindowToBeShown() throws Exception {
        HostInfoDAO hostInfoDAO = mock(HostInfoDAO.class);
        context.registerService(HostInfoDAO.class, hostInfoDAO, null);
        VmInfoDAO vmInfoDAO = mock(VmInfoDAO.class);
        context.registerService(VmInfoDAO.class, vmInfoDAO, null);
        Main main = new Main(context, appService, dbServiceFactory, keyring, shutdown, mainWindowRunnable);

        main.run();

        handleAllEdtEvents();

        ConnectionListener connectionListener = connectionListenerCaptor.getValue();
        connectionListener.changed(ConnectionStatus.CONNECTED);

        handleAllEdtEvents();

        verify(mainWindowRunnable).run();
    }

    @Ignore("this prompts the user with some gui")
    @Test
    public void verifyFailedConnectionTriggersShutdown() throws Exception {

        Main main = new Main(context, appService, dbServiceFactory, keyring, shutdown, mainWindowRunnable);

        main.run();

        handleAllEdtEvents();

        ConnectionListener connectionListener = connectionListenerCaptor.getValue();
        connectionListener.changed(ConnectionStatus.FAILED_TO_CONNECT);

        handleAllEdtEvents();

        verify(shutdown).countDown();
    }
}