/* * Copyright (C) 2013 SeqWare * * This program 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 3 of the License, or * (at your option) any later version. * * This program 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 this program. If not, see <http://www.gnu.org/licenses/>. */ package net.sourceforge.seqware.pipeline.plugins; import io.seqware.common.model.WorkflowRunStatus; import io.seqware.pipeline.SqwKeys; import java.util.ArrayList; import java.util.List; import java.util.Map; import joptsimple.OptionSet; import net.sourceforge.seqware.common.model.WorkflowRun; import net.sourceforge.seqware.common.module.ReturnValue; import net.sourceforge.seqware.common.util.filetools.FileTools; import net.sourceforge.seqware.common.util.filetools.FileTools.LocalhostPair; import net.sourceforge.seqware.pipeline.tools.RunLock; import org.apache.oozie.client.OozieClient; import org.apache.oozie.client.WorkflowJob; import org.junit.After; import org.junit.Assert; import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.mockito.InjectMocks; import static org.mockito.Matchers.any; import static org.mockito.Matchers.anyString; import org.mockito.Mock; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.reset; import static org.mockito.Mockito.times; import static org.mockito.Mockito.verify; import static org.mockito.Mockito.verifyNoMoreInteractions; import static org.mockito.Mockito.when; import org.mockito.MockitoAnnotations; import org.powermock.api.mockito.PowerMockito; import org.powermock.core.classloader.annotations.PrepareForTest; import org.powermock.modules.junit4.PowerMockRunner; /** * * @author dyuen */ @PrepareForTest({ FileTools.class, WorkflowStatusChecker.class, RunLock.class }) @RunWith(PowerMockRunner.class) public class WorkflowStatusCheckerTest { @Mock private Map<String, String> config; @Mock private OptionSet options; @Mock private net.sourceforge.seqware.common.metadata.Metadata metadata; @InjectMocks private WorkflowStatusChecker workflowStatusChecker; @Before public void initMocks() throws Exception { reset(config, options, metadata); workflowStatusChecker = new WorkflowStatusChecker(); // this is kind of hacky // apparantly testNG retains the state of mocks and statuschecker from test to test, so we need to rebuild everything MockitoAnnotations.initMocks(this); when(options.has("force-host")).thenReturn(true); when(options.valueOf("force-host")).thenReturn("localhost"); when(config.get(SqwKeys.SW_REST_USER.getSettingKey())).thenReturn("user"); PowerMockito.mockStatic(RunLock.class); } @After public void cleanMocks() { } @Test public void testShouldInjectMocks() { Assert.assertNotNull(metadata); Assert.assertNotNull(workflowStatusChecker); Assert.assertNotNull(workflowStatusChecker.getMetadata()); } @Test public void testEmptyRun() { final ReturnValue ret1 = workflowStatusChecker.init(); Assert.assertTrue("workflowStatusChecker could not init", ret1.getExitStatus() == ReturnValue.SUCCESS); final ReturnValue ret2 = workflowStatusChecker.do_run(); Assert.assertTrue("workflowStatusChecker ran properly", ret2.getExitStatus() == ReturnValue.SUCCESS); verify(metadata).getWorkflowRunsByStatus(WorkflowRunStatus.running); verify(metadata).getWorkflowRunsByStatus(WorkflowRunStatus.pending); verify(metadata).getWorkflowRunsByStatus(WorkflowRunStatus.submitted_cancel); verify(metadata).getWorkflowRunsByStatus(WorkflowRunStatus.submitted_retry); verifyNoMoreInteractions(metadata); } @Test public void testNormalRun() throws Exception { final ReturnValue ret1 = workflowStatusChecker.init(); Assert.assertTrue("workflowStatusChecker could not init", ret1.getExitStatus() == ReturnValue.SUCCESS); mockupFakeRuns(); final ReturnValue ret2 = workflowStatusChecker.do_run(); verifyNormalRun(ret2); } @Test public void testDoubleThreadedRun() throws Exception { final ReturnValue ret1 = workflowStatusChecker.init(); Assert.assertTrue("workflowStatusChecker could not init", ret1.getExitStatus() == ReturnValue.SUCCESS); mockupFakeRuns(); when(options.has("threads-in-thread-pool")).thenReturn(true); when(options.valueOf("threads-in-thread-pool")).thenReturn(2); final ReturnValue ret2 = workflowStatusChecker.do_run(); verifyNormalRun(ret2); } @Test public void testManyThreadedRun() throws Exception { final ReturnValue ret1 = workflowStatusChecker.init(); Assert.assertTrue("workflowStatusChecker could not init", ret1.getExitStatus() == ReturnValue.SUCCESS); mockupFakeRuns(); when(options.has("threads-in-thread-pool")).thenReturn(true); when(options.valueOf("threads-in-thread-pool")).thenReturn(100); final ReturnValue ret2 = workflowStatusChecker.do_run(); verifyNormalRun(ret2); } /** * For testing purposes, create some workflow runs and make our mocks aware of them * * @throws Exception */ private void mockupFakeRuns() throws Exception { // mock up some fake workflow_runs so that their status can be checked List<WorkflowRun> wrList = new ArrayList<>(); for (int i = 0; i < 100; i++) { WorkflowRun wr = new WorkflowRun(); wr.setOwnerUserName("user"); wr.setWorkflowAccession(42); wr.setWorkflowRunId(42 + i); wr.setSwAccession(42 + i); wr.setCommand("dummyValue"); wr.setTemplate("dummyValue"); wr.setCurrentWorkingDir("dummyValue"); wr.setDax("dummyValue"); wr.setIniFile("dummyValue"); wr.setWorkflowEngine("oozie"); wr.setHost("localhost"); wr.setStatusCmd("pegasus-status -l /home/seqware/pegasus-dax/seqware/pegasus/FastqQualityReportAndFilter_0.10.0/run00" + 42 + i); wrList.add(wr); } PowerMockito.mockStatic(FileTools.class); when(FileTools.getLocalhost(options)).thenReturn(new LocalhostPair("localhost", new ReturnValue(ReturnValue.SUCCESS))); when(FileTools.isFileOwner(anyString())).thenReturn(true); final OozieClient oozieClient = mock(OozieClient.class); PowerMockito.whenNew(OozieClient.class).withAnyArguments().thenReturn(oozieClient); final WorkflowJob workflowJob = mock(WorkflowJob.class); when(oozieClient.getJobInfo(anyString())).thenReturn(workflowJob); when(workflowJob.getStatus()).thenReturn(WorkflowJob.Status.RUNNING); ReturnValue fakeReturn = new ReturnValue(ReturnValue.SUCCESS); fakeReturn.setAttribute("currStep", "1"); fakeReturn.setAttribute("totalSteps", "1"); when(metadata.getWorkflowRunsByStatus(WorkflowRunStatus.running)).thenReturn(wrList); } /** * Verify that the run returned normally and that the appropriate number of updates were make to the database * * @param ret2 */ private void verifyNormalRun(final ReturnValue ret2) { Assert.assertTrue("workflowStatusChecker ran properly", ret2.getExitStatus() == ReturnValue.SUCCESS); verify(metadata).getWorkflowRunsByStatus(WorkflowRunStatus.running); verify(metadata).getWorkflowRunsByStatus(WorkflowRunStatus.pending); verify(metadata).getWorkflowRunsByStatus(WorkflowRunStatus.submitted_cancel); verify(metadata).getWorkflowRunsByStatus(WorkflowRunStatus.submitted_retry); verify(metadata, times(100)).updateWorkflowRun(any(WorkflowRun.class)); } }