mirror of
https://github.com/eclipse-cdt/cdt
synced 2025-06-07 09:46:02 +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
|
||||
|
||||
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`)
|
||||
|
|
|
@ -54,6 +54,10 @@ import org.eclipse.ui.WorkbenchException;
|
|||
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();
|
||||
|
|
|
@ -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