mirror of
https://github.com/eclipse-cdt/cdt
synced 2025-06-07 17:56:01 +02:00
Provide a base UI test case for JUnit5
This is the analogous class to BaseUITestCase, but for JUnit5.
This commit is contained in:
parent
8ff58a5262
commit
cbbbe44553
3 changed files with 379 additions and 1 deletions
|
@ -84,7 +84,7 @@ These tags can only be applied to JUnit5 (aka Jupiter) tests. If a test needs co
|
||||||
|
|
||||||
## Converting tests to JUnit5
|
## 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
|
- 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`.
|
- 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`)
|
- add `@Test` annotation (make sure to use `org.junit.jupiter.api.Test` and not JUnit4's `org.junit.Test`)
|
||||||
|
|
|
@ -54,6 +54,10 @@ import org.eclipse.ui.WorkbenchException;
|
||||||
import org.eclipse.ui.handlers.IHandlerService;
|
import org.eclipse.ui.handlers.IHandlerService;
|
||||||
import org.eclipse.ui.internal.WorkbenchPartReference;
|
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 abstract class BaseUITestCase extends BaseTestCase {
|
||||||
public BaseUITestCase() {
|
public BaseUITestCase() {
|
||||||
super();
|
super();
|
||||||
|
|
|
@ -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 <code>readTaggedComment(getClass(), tag)</code>
|
||||||
|
* @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<Control> result = new ArrayList<>();
|
||||||
|
findControls(w, clazz, result);
|
||||||
|
return result.toArray(new Control[result.size()]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void findControls(Control w, Class<?> clazz, List<Control> 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.
|
Loading…
Add table
Reference in a new issue