| /******************************************************************************* |
| * Copyright (c) 2020, 2021 Paul Pazderski and others. |
| * |
| * This program and the accompanying materials |
| * are made available under the terms of the Eclipse Public License 2.0 |
| * which accompanies this distribution, and is available at |
| * https://www.eclipse.org/legal/epl-2.0/ |
| * |
| * SPDX-License-Identifier: EPL-2.0 |
| * |
| * Contributors: |
| * Paul Pazderski - initial API and implementation |
| * Hannes Wellmann - add tests regarding termination of descendants and timeout |
| *******************************************************************************/ |
| package org.eclipse.debug.tests.console; |
| |
| import static org.hamcrest.CoreMatchers.is; |
| import static org.hamcrest.MatcherAssert.assertThat; |
| import static org.junit.Assert.assertEquals; |
| import static org.junit.Assert.assertFalse; |
| import static org.junit.Assert.assertThrows; |
| import static org.junit.Assert.assertTrue; |
| |
| import java.util.List; |
| import java.util.Map; |
| import java.util.concurrent.atomic.AtomicInteger; |
| |
| import org.eclipse.debug.core.DebugEvent; |
| import org.eclipse.debug.core.DebugException; |
| import org.eclipse.debug.core.DebugPlugin; |
| import org.eclipse.debug.core.model.RuntimeProcess; |
| import org.eclipse.debug.internal.core.DebugCoreMessages; |
| import org.eclipse.debug.tests.AbstractDebugTest; |
| import org.eclipse.debug.tests.TestUtil; |
| import org.junit.Test; |
| |
| public class RuntimeProcessTests extends AbstractDebugTest { |
| |
| /** |
| * Test behavior of {@link RuntimeProcess} if the wrapped process |
| * terminates. |
| */ |
| @Test |
| public void testProcessTerminated() throws Exception { |
| AtomicInteger processTerminateEvents = new AtomicInteger(); |
| DebugPlugin.getDefault().addDebugEventListener(events -> { |
| for (DebugEvent event : events) { |
| if (event.getKind() == DebugEvent.TERMINATE) { |
| processTerminateEvents.incrementAndGet(); |
| } |
| } |
| }); |
| |
| MockProcess mockProcess = new MockProcess(MockProcess.RUN_FOREVER); |
| RuntimeProcess runtimeProcess = mockProcess.toRuntimeProcess(); |
| |
| assertFalse("RuntimeProcess already terminated.", runtimeProcess.isTerminated()); |
| assertTrue(runtimeProcess.canTerminate()); |
| |
| mockProcess.setExitValue(1); |
| mockProcess.destroy(); |
| |
| TestUtil.waitWhile(p -> !p.isTerminated(), runtimeProcess, 1000, p -> "RuntimePocess not terminated."); |
| TestUtil.waitForJobs(name.getMethodName(), 25, 500); |
| assertEquals("Wrong number of terminate events.", 1, processTerminateEvents.get()); |
| assertEquals("RuntimeProcess reported wrong exit code.", 1, runtimeProcess.getExitValue()); |
| } |
| |
| /** Test {@link RuntimeProcess} terminating the wrapped process. */ |
| @Test |
| public void testTerminateProcess() throws Exception { |
| AtomicInteger processTerminateEvents = new AtomicInteger(); |
| DebugPlugin.getDefault().addDebugEventListener(events -> { |
| for (DebugEvent event : events) { |
| if (event.getKind() == DebugEvent.TERMINATE) { |
| processTerminateEvents.incrementAndGet(); |
| } |
| } |
| }); |
| |
| MockProcess mockProcess = new MockProcess(MockProcess.RUN_FOREVER); |
| RuntimeProcess runtimeProcess = mockProcess.toRuntimeProcess(); |
| |
| assertFalse("RuntimeProcess already terminated.", runtimeProcess.isTerminated()); |
| assertTrue(runtimeProcess.canTerminate()); |
| |
| mockProcess.setExitValue(1); |
| runtimeProcess.terminate(); |
| assertFalse("RuntimeProcess failed to terminate wrapped process.", mockProcess.isAlive()); |
| |
| TestUtil.waitWhile(p -> !p.isTerminated(), runtimeProcess, 1000, p -> "RuntimePocess not terminated."); |
| TestUtil.waitForJobs(name.getMethodName(), 25, 500); |
| assertEquals("Wrong number of terminate events.", 1, processTerminateEvents.get()); |
| assertEquals("RuntimeProcess reported wrong exit code.", 1, runtimeProcess.getExitValue()); |
| } |
| |
| /** |
| * Test {@link RuntimeProcess} terminating the wrapped process and its |
| * descendants. |
| */ |
| @Test |
| public void testTerminateProcessWithSubProcesses() throws Exception { |
| |
| MockProcess grandChildProcess = new MockProcess(MockProcess.RUN_FOREVER); |
| |
| MockProcess childProcess1 = new MockProcess(MockProcess.RUN_FOREVER); |
| childProcess1.setHandle(new MockProcessHandle(childProcess1, List.of(grandChildProcess))); |
| |
| MockProcess childProcess2 = new MockProcess(MockProcess.RUN_FOREVER); |
| |
| MockProcess mockProcess = new MockProcess(MockProcess.RUN_FOREVER); |
| mockProcess.setHandle(new MockProcessHandle(childProcess1, List.of(childProcess1, childProcess2))); |
| |
| RuntimeProcess runtimeProcess = mockProcess.toRuntimeProcess(); |
| |
| assertTrue("RuntimeProcess already terminated.", grandChildProcess.isAlive()); |
| assertTrue("RuntimeProcess already terminated.", childProcess1.isAlive()); |
| assertTrue("RuntimeProcess already terminated.", childProcess2.isAlive()); |
| assertFalse("RuntimeProcess already terminated.", runtimeProcess.isTerminated()); |
| |
| runtimeProcess.terminate(); |
| |
| assertFalse("RuntimeProcess failed to terminate wrapped process.", mockProcess.isAlive()); |
| assertFalse("RuntimeProcess failed to terminate child of wrapped process.", childProcess1.isAlive()); |
| assertFalse("RuntimeProcess failed to terminate child of wrapped process.", childProcess2.isAlive()); |
| assertFalse("RuntimeProcess failed to terminate descendant of wrapped process.", grandChildProcess.isAlive()); |
| |
| TestUtil.waitWhile(p -> !p.isTerminated(), runtimeProcess, 1000, p -> "RuntimePocess not terminated."); |
| } |
| |
| /** |
| * Test {@link RuntimeProcess} terminating the wrapped process while not |
| * terminating its descendants. |
| */ |
| @Test |
| public void testTerminateProcessWithoutTerminatingDescendents() throws Exception { |
| |
| MockProcess childProcess = new MockProcess(MockProcess.RUN_FOREVER); |
| |
| MockProcess mockProcess = new MockProcess(MockProcess.RUN_FOREVER); |
| mockProcess.setHandle(new MockProcessHandle(mockProcess, List.of(childProcess))); |
| |
| RuntimeProcess runtimeProcess = mockProcess.toRuntimeProcess("MockProcess", Map.of(DebugPlugin.ATTR_TERMINATE_DESCENDANTS, false)); |
| |
| assertTrue("RuntimeProcess already terminated.", childProcess.isAlive()); |
| assertFalse("RuntimeProcess already terminated.", runtimeProcess.isTerminated()); |
| |
| runtimeProcess.terminate(); |
| |
| assertFalse("RuntimeProcess failed to terminate wrapped process.", mockProcess.isAlive()); |
| assertTrue("RuntimeProcess terminated child of wrapped process, unlike configured.", childProcess.isAlive()); |
| |
| TestUtil.waitWhile(p -> !p.isTerminated(), runtimeProcess, 1000, p -> "RuntimePocess not terminated."); |
| } |
| |
| /** |
| * Test {@link RuntimeProcess} terminating the wrapped process which does |
| * not support {@link Process#toHandle()}. |
| */ |
| @Test |
| public void testTerminateProcessNotSupportingProcessToHandle() throws Exception { |
| |
| MockProcess mockProcess = new MockProcess(MockProcess.RUN_FOREVER); |
| // set handle to null, so the standard java.lang.Process.toHandle() |
| // implementation is called which throws an |
| // UnsupportedOperationException |
| mockProcess.setHandle(null); |
| assertThrows(UnsupportedOperationException.class, mockProcess::toHandle); |
| RuntimeProcess runtimeProcess = mockProcess.toRuntimeProcess(); |
| runtimeProcess.terminate(); // must not throw, even toHandle() does |
| |
| TestUtil.waitWhile(p -> !p.isTerminated(), runtimeProcess, 1000, p -> "RuntimePocess not terminated."); |
| } |
| |
| /** |
| * Test {@link RuntimeProcess} terminating the wrapped process which does |
| * only terminate with a delay. |
| */ |
| @Test |
| public void testTerminateProcessWithTimeoutExeedingTermination() { |
| |
| MockProcess mockProcess = new MockProcess(MockProcess.RUN_FOREVER); |
| mockProcess.setTerminationDelay(6000); |
| |
| RuntimeProcess runtimeProcess = mockProcess.toRuntimeProcess(); |
| |
| DebugException timeoutException = assertThrows(DebugException.class, runtimeProcess::terminate); |
| assertThat(timeoutException.getMessage(), is(DebugCoreMessages.RuntimeProcess_terminate_failed)); |
| } |
| |
| /** |
| * Test {@link RuntimeProcess} terminating the wrapped process which does |
| * only terminate with a delay. |
| */ |
| @Test |
| public void testTerminateProcessWithDescendentExceedingTimeoutForTermination() { |
| |
| MockProcess childProcess = new MockProcess(MockProcess.RUN_FOREVER); |
| childProcess.setTerminationDelay(6000); |
| |
| MockProcess mockProcess = new MockProcess(MockProcess.RUN_FOREVER); |
| mockProcess.setHandle(new MockProcessHandle(mockProcess, List.of(childProcess))); |
| |
| RuntimeProcess runtimeProcess = mockProcess.toRuntimeProcess(); |
| |
| DebugException timeoutException = assertThrows(DebugException.class, runtimeProcess::terminate); |
| assertThat(timeoutException.getMessage(), is(DebugCoreMessages.RuntimeProcess_terminate_failed)); |
| } |
| } |