From cbbbe44553c7e26df4dedf109a79d863e67ac221 Mon Sep 17 00:00:00 2001 From: Jonah Graham Date: Mon, 30 Jan 2023 14:08:34 -0500 Subject: [PATCH] Provide a base UI test case for JUnit5 This is the analogous class to BaseUITestCase, but for JUnit5. --- TESTING.md | 2 +- .../eclipse/cdt/ui/tests/BaseUITestCase.java | 4 + .../eclipse/cdt/ui/tests/BaseUITestCase5.java | 374 ++++++++++++++++++ 3 files changed, 379 insertions(+), 1 deletion(-) create mode 100644 core/org.eclipse.cdt.ui.tests/ui/org/eclipse/cdt/ui/tests/BaseUITestCase5.java diff --git a/TESTING.md b/TESTING.md index 054c7eea94b..e9981712a8d 100644 --- a/TESTING.md +++ b/TESTING.md @@ -84,7 +84,7 @@ These tags can only be applied to JUnit5 (aka Jupiter) tests. If a test needs co ## Converting tests to JUnit5 -To take advantage of new features, such as excluding flaky and slow tests, individual tests need to JUnit5 (aka Jupiter). If a test is currently written in JUnit4 or JUnit3 style it needs to be converted to JUnit5 first. Those tests that currently derive from `org.eclipse.cdt.core.testplugin.util.BaseTestCase` can change to `org.eclipse.cdt.core.testplugin.util.BaseTestCase5` and make further adjustments. Common adjustments are: +To take advantage of new features, such as excluding flaky and slow tests, individual tests need to JUnit5 (aka Jupiter). If a test is currently written in JUnit4 or JUnit3 style it needs to be converted to JUnit5 first. Those tests that currently derive from `org.eclipse.cdt.core.testplugin.util.BaseTestCase` (or `org.eclipse.cdt.ui.tests.BaseUITestCase` for UI tests) can change to `org.eclipse.cdt.core.testplugin.util.BaseTestCase5` (`org.eclipse.cdt.ui.tests.BaseUITestCase5` for UI tests) and make further adjustments. Common adjustments are: - refactoring `setUp`/`tearDown` methods to use `@BeforeEach` and `@AfterEach` annotations - refactor complicated uses of TestSuites in JUnit3 that were workarounds for the lack of JUnit features like `@BeforeAll` and `@AfterAll`. - add `@Test` annotation (make sure to use `org.junit.jupiter.api.Test` and not JUnit4's `org.junit.Test`) diff --git a/core/org.eclipse.cdt.ui.tests/ui/org/eclipse/cdt/ui/tests/BaseUITestCase.java b/core/org.eclipse.cdt.ui.tests/ui/org/eclipse/cdt/ui/tests/BaseUITestCase.java index 5fec6d0cd01..5e92646fd9a 100644 --- a/core/org.eclipse.cdt.ui.tests/ui/org/eclipse/cdt/ui/tests/BaseUITestCase.java +++ b/core/org.eclipse.cdt.ui.tests/ui/org/eclipse/cdt/ui/tests/BaseUITestCase.java @@ -54,6 +54,10 @@ import org.eclipse.ui.handlers.IHandlerService; import org.eclipse.ui.internal.WorkbenchPartReference; +/** + * @deprecated Use {@link BaseUITestCase5} for new code. See TESTING.md for details on converting to JUnit5 + */ +@Deprecated public abstract class BaseUITestCase extends BaseTestCase { public BaseUITestCase() { super(); diff --git a/core/org.eclipse.cdt.ui.tests/ui/org/eclipse/cdt/ui/tests/BaseUITestCase5.java b/core/org.eclipse.cdt.ui.tests/ui/org/eclipse/cdt/ui/tests/BaseUITestCase5.java new file mode 100644 index 00000000000..427b65e693f --- /dev/null +++ b/core/org.eclipse.cdt.ui.tests/ui/org/eclipse/cdt/ui/tests/BaseUITestCase5.java @@ -0,0 +1,374 @@ +/******************************************************************************* + * Copyright (c) 2006, 2023 Wind River Systems, Inc. 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: + * Markus Schorn - initial API and implementation + *******************************************************************************/ +package org.eclipse.cdt.ui.tests; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertFalse; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.fail; + +import java.io.IOException; +import java.util.ArrayList; +import java.util.List; + +import org.eclipse.cdt.core.dom.ast.IASTTranslationUnit; +import org.eclipse.cdt.core.index.IIndex; +import org.eclipse.cdt.core.model.CModelException; +import org.eclipse.cdt.core.model.ICProject; +import org.eclipse.cdt.core.testplugin.util.BaseTestCase5; +import org.eclipse.cdt.core.testplugin.util.TestSourceReader; +import org.eclipse.cdt.ui.testplugin.CTestPlugin; +import org.eclipse.cdt.ui.testplugin.util.StringAsserts; +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.core.commands.NotEnabledException; +import org.eclipse.core.commands.NotHandledException; +import org.eclipse.core.commands.common.NotDefinedException; +import org.eclipse.core.resources.IContainer; +import org.eclipse.core.resources.IFile; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.Path; +import org.eclipse.swt.SWT; +import org.eclipse.swt.SWTException; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Display; +import org.eclipse.swt.widgets.Event; +import org.eclipse.swt.widgets.Tree; +import org.eclipse.swt.widgets.TreeItem; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.IEditorReference; +import org.eclipse.ui.IViewPart; +import org.eclipse.ui.IViewReference; +import org.eclipse.ui.IWorkbenchPage; +import org.eclipse.ui.IWorkbenchPartSite; +import org.eclipse.ui.IWorkbenchWindow; +import org.eclipse.ui.PartInitException; +import org.eclipse.ui.PlatformUI; +import org.eclipse.ui.WorkbenchException; +import org.eclipse.ui.handlers.IHandlerService; +import org.eclipse.ui.internal.WorkbenchPartReference; +import org.junit.jupiter.api.AfterEach; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.TestInfo; + +public abstract class BaseUITestCase5 extends BaseTestCase5 { + + @BeforeEach + protected void setupBaseUI(TestInfo testInfo) throws Exception { + + final IWorkbenchPage activePage = PlatformUI.getWorkbench().getActiveWorkbenchWindow().getActivePage(); + IViewPart view = activePage.findView("org.eclipse.cdt.ui.tests.DOMAST.DOMAST"); + if (view != null) { + activePage.hideView(view); + } + } + + @AfterEach + protected void tearDownBaseUI() throws Exception { + runEventQueue(0); + } + + /** + * Reads a section in comments form the source of the given class. + */ + protected String readTaggedComment(Class clazz, final String tag) throws IOException { + return TestSourceReader.readTaggedComment(CTestPlugin.getDefault().getBundle(), "ui", clazz, tag); + } + + /** + * Reads a section in comments form the source of the given class. Fully + * equivalent to readTaggedComment(getClass(), tag) + * @since 4.0 + */ + protected String readTaggedComment(final String tag) throws IOException { + return readTaggedComment(getClass(), tag); + } + + /** + * Reads multiple sections in comments from the source of the given class. + * + * Trailing whitespace can be removed by editor/clean-up actions. To enforce whitespace + * at end of line, use ${whitspace_eol}, which will be removed, but cause the + * whitespace to the left of it to be preserved. + * + * @since 4.0 + */ + public StringBuilder[] getContentsForTest(int sections) throws IOException { + return TestSourceReader.getContentsForTest(CTestPlugin.getDefault().getBundle(), "ui", getClass(), getName(), + sections); + } + + public String getAboveComment() throws IOException { + return getContentsForTest(1)[0].toString(); + } + + protected IFile createFile(IContainer container, String fileName, String contents) throws Exception { + return TestSourceReader.createFile(container, new Path(fileName), contents); + } + + protected IASTTranslationUnit createIndexBasedAST(IIndex index, ICProject project, IFile file) + throws CModelException, CoreException { + return TestSourceReader.createIndexBasedAST(index, project, file); + } + + protected void runEventQueue(int time) { + final long endTime = System.currentTimeMillis() + time; + while (true) { + while (Display.getCurrent().readAndDispatch()) { + } + + long diff = endTime - System.currentTimeMillis(); + if (diff <= 0) { + break; + } + try { + Thread.sleep(Math.min(20, diff)); + } catch (InterruptedException e) { + return; + } + } + } + + protected void expandTreeItem(Tree tree, int idx) { + expandTreeItem(tree, new int[] { idx }); + } + + protected void expandTreeItem(Tree tree, int idx1, int idx2) { + expandTreeItem(tree, new int[] { idx1, idx2 }); + } + + protected void expandTreeItem(Tree tree, int[] idxs) { + TreeItem item = tree.getItem(idxs[0]); + assertNotNull(item); + expandTreeItem(item); + for (int i = 1; i < idxs.length; i++) { + item = item.getItem(idxs[i]); + assertNotNull(item); + expandTreeItem(item); + } + } + + protected void expandTreeItem(TreeItem item) { + Event event = new Event(); + event.item = item; + item.getParent().notifyListeners(SWT.Expand, event); + item.setExpanded(true); + runEventQueue(0); + } + + protected void selectTreeItem(Tree tree, int idx) { + selectTreeItem(tree, new int[] { idx }); + } + + protected void selectTreeItem(Tree tree, int idx1, int idx2) { + selectTreeItem(tree, new int[] { idx1, idx2 }); + } + + protected void selectTreeItem(Tree tree, int[] idxs) { + TreeItem item = tree.getItem(idxs[0]); + assertNotNull(item); + for (int i = 1; i < idxs.length; i++) { + item = item.getItem(idxs[i]); + assertNotNull(item); + } + tree.setSelection(item); + Event event = new Event(); + event.item = item; + item.getParent().notifyListeners(SWT.Selection, event); + runEventQueue(0); + } + + protected void closeEditor(IEditorPart editor) { + IWorkbenchPartSite site; + IWorkbenchPage page; + if (editor != null && (site = editor.getSite()) != null && (page = site.getPage()) != null) { + page.closeEditor(editor, false); + } + } + + protected void closeAllEditors() { + IWorkbenchWindow[] windows = PlatformUI.getWorkbench().getWorkbenchWindows(); + for (IWorkbenchWindow window : windows) { + IWorkbenchPage[] pages = window.getPages(); + for (IWorkbenchPage page : pages) { + page.closeAllEditors(false); + } + } + } + + protected void restoreAllParts() throws WorkbenchException { + IWorkbenchPage page = PlatformUI.getWorkbench().getActiveWorkbenchWindow().getActivePage(); + page.zoomOut(); + runEventQueue(0); + + IViewReference[] viewRefs = page.getViewReferences(); + for (IViewReference ref : viewRefs) { + page.setPartState(ref, IWorkbenchPage.STATE_RESTORED); + } + IEditorReference[] editorRefs = page.getEditorReferences(); + for (IEditorReference ref : editorRefs) { + page.setPartState(ref, IWorkbenchPage.STATE_RESTORED); + } + runEventQueue(0); + } + + protected IViewPart activateView(String id) throws PartInitException { + IViewPart view = PlatformUI.getWorkbench().getActiveWorkbenchWindow().getActivePage().showView(id); + assertNotNull(view); + runEventQueue(0); + return view; + } + + protected void executeCommand(IViewPart viewPart, String commandID) + throws ExecutionException, NotDefinedException, NotEnabledException, NotHandledException { + IHandlerService hs = viewPart.getSite().getService(IHandlerService.class); + assertNotNull(hs); + hs.executeCommand(commandID, null); + } + + private Control[] findControls(Control w, Class clazz) { + ArrayList result = new ArrayList<>(); + findControls(w, clazz, result); + return result.toArray(new Control[result.size()]); + } + + private void findControls(Control w, Class clazz, List result) { + if (clazz.isInstance(w)) { + result.add(w); + } + if (w instanceof Composite) { + Composite comp = (Composite) w; + Control[] children = comp.getChildren(); + for (Control element : children) { + findControls(element, clazz, result); + } + } + } + + final protected TreeItem checkTreeNode(IViewPart part, int i0, String label) { + assertNotNull(label); // we don't handle testing for a base node to not appear; can be added if/when needed + IViewReference viewRef = part.getViewSite().getPage().findViewReference(part.getViewSite().getId()); + Control viewControl = ((WorkbenchPartReference) viewRef).getPane().getControl(); + + Tree tree = null; + TreeItem root = null; + StringBuilder cands = new StringBuilder(); + for (int i = 0; i < 400; i++) { + cands.setLength(0); + Control[] trees = findControls(viewControl, Tree.class); + for (int j = 0; j < trees.length; j++) { + try { + tree = (Tree) trees[j]; + root = tree.getItem(i0); + if (label.equals(root.getText())) { + return root; + } + if (j > 0) { + cands.append('|'); + } + cands.append(root.getText()); + } catch (SWTException e) { + // in case widget was disposed, item may be replaced + } catch (IllegalArgumentException e) { + // item does not yet exist. + } + } + runEventQueue(10); + } + assertNotNull(tree, "No tree in viewpart"); + assertNotNull(root, "Tree node " + label + "{" + i0 + "} does not exist!"); + assertEquals(label, cands.toString()); + return root; + } + + final protected TreeItem checkTreeNode(Tree tree, int i0, String label) { + assertNotNull(label); // we don't handle testing for a base node to not appear; can be added if/when needed + TreeItem root = null; + for (int millis = 0; millis < 5000; millis = millis == 0 ? 1 : millis * 2) { + runEventQueue(millis); + try { + root = tree.getItem(i0); + if (label.equals(root.getText())) { + return root; + } + } catch (SWTException e) { + // in case widget was disposed, item may be replaced + } catch (IllegalArgumentException e) { + // item does not yet exist. + } + } + fail("Tree node " + label + "{" + i0 + "} does not exist!"); + return null; + } + + /** + * Pass label=null to test that the {i0,i1} node doesn't exist + */ + final protected TreeItem checkTreeNode(Tree tree, int i0, int i1, String label) { + String firstItemText = null; + int timeout = (label == null) ? 1000 : 5000; // see footnote[0] + + // If {i0,i1} exists, whether or not it matches label (when label != null) + boolean nodePresent = false; + + for (int millis = 0; millis < timeout; millis = millis == 0 ? 1 : millis * 2) { + nodePresent = false; + runEventQueue(millis); + TreeItem i0Node = tree.getItem(i0); + if (!i0Node.getExpanded()) { + expandTreeItem(i0Node); + } + try { + TreeItem firstItem = i0Node.getItem(0); + firstItemText = firstItem.getText(); + if (!firstItemText.isEmpty() && !firstItemText.equals("...")) { + TreeItem item = i0Node.getItem(i1); + nodePresent = true; + if (label != null && label.equals(item.getText())) { + return item; + } + } + } catch (SWTException e) { + // in case widget was disposed, item may be replaced + } catch (IllegalArgumentException e) { + // item does not yet exist. + } + } + + if (label == null) { + assertFalse(nodePresent, "Tree node {" + i0 + "," + i1 + "} exists but shouldn't!"); + } else { + fail("Tree node " + label + "{" + i0 + "," + i1 + "} does not exist!"); + } + return null; + } + + public static void assertEqualString(String actual, String expected) { + StringAsserts.assertEqualString(actual, expected); + } +} + +// Footnotes +// [0] Waiting for something to appear is very efficient; waiting for it to not +// appear is very inefficient. In the former case, regardless of how much time +// is alloted, we stop waiting as soon as the item appears, whereas in the +// latter we have to wait the entire timeout. In test suites with thousands of +// tests, efficiency is critical. Thus, in testing that a tree node doesn't have +// an Nth child, we shoot for efficiency and accept the risk of a false +// negative. More specifically, we wait only one second for the item TO NOT +// appear, whereas we give an item up to five seconds TO appear. This compromise +// is better than not having that sort of test at all, which some would argue is +// the better approach. In practice, it takes about 60-150 ms for the item to +// appear (on my machine), but we give it up to five seconds. Waiting one second +// for it to not appear should be more than adequate.