blob: 08af79d94ca604efd261730810f7110c72ab967b [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2018, 2020 Red Hat Inc. and others.
* This program and the accompanying materials are made
* available under the terms of the Eclipse Public License 2.0
* which is available at https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* Mickael Istria (Red Hat Inc.) - Added some suites
* Pierre-Yves B. <pyvesdev@gmail.com> - Bug 525411 - [rename] input field should be filled with symbol to rename
* Martin Lippert (Pivotal) - Bug 561373 - added async enablement for late language servers
*******************************************************************************/
package org.eclipse.lsp4e.test.rename;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertTrue;
import java.io.File;
import java.lang.reflect.Method;
import java.net.URI;
import java.nio.file.Files;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicReference;
import org.eclipse.core.commands.Command;
import org.eclipse.core.commands.ExecutionEvent;
import org.eclipse.core.filebuffers.ITextFileBuffer;
import org.eclipse.core.filebuffers.ITextFileBufferManager;
import org.eclipse.core.filesystem.EFS;
import org.eclipse.core.filesystem.IFileStore;
import org.eclipse.core.resources.IFile;
import org.eclipse.core.resources.IProject;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.NullProgressMonitor;
import org.eclipse.jface.dialogs.Dialog;
import org.eclipse.jface.text.IDocument;
import org.eclipse.jface.text.tests.util.DisplayHelper;
import org.eclipse.lsp4e.LSPEclipseUtils;
import org.eclipse.lsp4e.LanguageServiceAccessor;
import org.eclipse.lsp4e.operations.rename.LSPRenameHandler;
import org.eclipse.lsp4e.operations.rename.LSPRenameProcessor;
import org.eclipse.lsp4e.test.AllCleanRule;
import org.eclipse.lsp4e.test.TestUtils;
import org.eclipse.lsp4e.tests.mock.MockLanguageServer;
import org.eclipse.lsp4j.Position;
import org.eclipse.lsp4j.Range;
import org.eclipse.lsp4j.TextEdit;
import org.eclipse.lsp4j.WorkspaceEdit;
import org.eclipse.lsp4j.jsonrpc.messages.Either;
import org.eclipse.ltk.core.refactoring.RefactoringStatus;
import org.eclipse.ltk.core.refactoring.participants.ProcessorBasedRefactoring;
import org.eclipse.swt.SWT;
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.Listener;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.ui.IWorkbenchCommandConstants;
import org.eclipse.ui.PlatformUI;
import org.eclipse.ui.commands.ICommandService;
import org.eclipse.ui.handlers.IHandlerService;
import org.eclipse.ui.texteditor.ITextEditor;
import org.junit.Rule;
import org.junit.Test;
public class RenameTest {
@Rule public AllCleanRule clear = new AllCleanRule();
@Test
public void testRenameHandlerEnablement() throws Exception {
IProject project = TestUtils.createProject("blah");
IFile file = TestUtils.createUniqueTestFile(project, "old");
ITextEditor editor = (ITextEditor) TestUtils.openEditor(file);
editor.selectAndReveal(1, 0);
ICommandService commandService = PlatformUI.getWorkbench().getService(ICommandService.class);
Command command = commandService.getCommand(IWorkbenchCommandConstants.FILE_RENAME);
assertTrue(command.isEnabled() && command.isHandled());
}
@Test
public void testAsyncRenameHandlerEnablement() throws Exception {
long delay = 2000;
// this fixed value is not really an optimal solution, since it depends on the following things
// to happen within that time frame. Should maybe re-work this in the future towards a more
// precise way of steering the execution from the test here
MockLanguageServer.INSTANCE.setTimeToProceedQueries(delay);
IProject project = TestUtils.createProject("blah");
IFile file = TestUtils.createUniqueTestFile(project, "old");
ITextEditor editor = (ITextEditor) TestUtils.openEditor(file);
editor.selectAndReveal(1, 0);
ICommandService commandService = PlatformUI.getWorkbench().getService(ICommandService.class);
Command command = commandService.getCommand(IWorkbenchCommandConstants.FILE_RENAME);
assertFalse(command.isEnabled() && command.isHandled());
Thread.sleep(delay);
assertTrue(command.isEnabled() && command.isHandled());
}
@Test
public void testRenameRefactoring() throws Exception {
IProject project = TestUtils.createProject("blah");
IFile file = TestUtils.createUniqueTestFile(project, "old");
MockLanguageServer.INSTANCE.getTextDocumentService().setRenameEdit(createSimpleMockRenameEdit(LSPEclipseUtils.toUri(file)));
IDocument document = LSPEclipseUtils.getDocument(file);
LanguageServiceAccessor.getLanguageServers(document, LSPRenameHandler::isRenameProvider).thenAccept(languageServers -> {
LSPRenameProcessor processor = new LSPRenameProcessor(LSPEclipseUtils.getDocument(file), languageServers.get(0), 0);
processor.setNewName("new");
try {
ProcessorBasedRefactoring processorBasedRefactoring = new ProcessorBasedRefactoring(processor);
processorBasedRefactoring.checkAllConditions(new NullProgressMonitor());
processorBasedRefactoring.createChange(new NullProgressMonitor()).perform(new NullProgressMonitor());
} catch (CoreException e) {
e.printStackTrace();
}
}).join();
assertEquals("new", document.get());
}
@Test
public void testPrepareRenameRefactoring() throws Exception {
IProject project = TestUtils.createProject("testPrepareRenameRefactoring");
IFile file = TestUtils.createUniqueTestFile(project, "old");
MockLanguageServer.INSTANCE.getTextDocumentService().setRenameEdit(createSimpleMockRenameEdit(LSPEclipseUtils.toUri(file)));
IDocument document = LSPEclipseUtils.getDocument(file);
LanguageServiceAccessor.getLanguageServers(document, LSPRenameHandler::isRenameProvider).thenAccept(languageServers -> {
LSPRenameProcessor processor = new LSPRenameProcessor(LSPEclipseUtils.getDocument(file), languageServers.get(0), 0);
processor.setNewName("new");
try {
ProcessorBasedRefactoring processorBasedRefactoring = new ProcessorBasedRefactoring(processor);
processorBasedRefactoring.checkAllConditions(new NullProgressMonitor());
processorBasedRefactoring.createChange(new NullProgressMonitor()).perform(new NullProgressMonitor());
} catch (CoreException e) {
e.printStackTrace();
}
}).join();
assertEquals("new", document.get());
}
@Test
public void testPrepareRenameRefactoringError() throws Exception {
IProject project = TestUtils.createProject("testPrepareRenameRefactoring");
IFile file = TestUtils.createUniqueTestFile(project, "old");
MockLanguageServer.INSTANCE.getTextDocumentService().setRenameEdit(createSimpleMockRenameEdit(LSPEclipseUtils.toUri(file)));
MockLanguageServer.INSTANCE.getTextDocumentService().setPrepareRenameResult(null);
IDocument document = LSPEclipseUtils.getDocument(file);
LanguageServiceAccessor.getLanguageServers(document, LSPRenameHandler::isRenameProvider).thenAccept(languageServers -> {
LSPRenameProcessor processor = new LSPRenameProcessor(LSPEclipseUtils.getDocument(file), languageServers.get(0), 0);
processor.setNewName("new");
try {
ProcessorBasedRefactoring processorBasedRefactoring = new ProcessorBasedRefactoring(processor);
RefactoringStatus status = processorBasedRefactoring.checkAllConditions(new NullProgressMonitor());
assertEquals(RefactoringStatus.FATAL, status.getSeverity());
} catch (CoreException e) {
e.printStackTrace();
}
}).join();
}
@Test
public void testRenameRefactoringExternalFile() throws Exception {
File file = File.createTempFile("testPerformOperationExternalFile", ".lspt");
MockLanguageServer.INSTANCE.getTextDocumentService().setRenameEdit(createSimpleMockRenameEdit(file.toURI()));
IFileStore store = EFS.getStore(file.toURI());
ITextFileBufferManager manager = ITextFileBufferManager.DEFAULT;
try {
manager.connectFileStore(store, new NullProgressMonitor());
IDocument document = ((ITextFileBuffer)manager.getFileStoreFileBuffer(store)).getDocument();
document.set("old");
LanguageServiceAccessor.getLanguageServers(document, LSPRenameHandler::isRenameProvider).thenAccept(languageServers -> {
LSPRenameProcessor processor = new LSPRenameProcessor(document, languageServers.get(0), 0);
processor.setNewName("new");
try {
ProcessorBasedRefactoring processorBasedRefactoring = new ProcessorBasedRefactoring(processor);
processorBasedRefactoring.checkAllConditions(new NullProgressMonitor());
processorBasedRefactoring.createChange(new NullProgressMonitor()).perform(new NullProgressMonitor());
} catch (CoreException e) {
e.printStackTrace();
}
}).join();
assertEquals("new", document.get());
} finally {
manager.disconnectFileStore(store, new NullProgressMonitor());
Files.deleteIfExists(file.toPath());
}
}
@Test
public void testRenameChangeAlsoExternalFile() throws Exception {
IProject project = TestUtils.createProject("blah");
IFile workspaceFile = TestUtils.createUniqueTestFile(project, "old");
File externalFile = File.createTempFile("testRenameChangeAlsoExternalFile", ".lspt");
try {
Files.write(externalFile.toPath(), "old".getBytes());
Map<String, List<TextEdit>> edits = new HashMap<>(2, 1.f);
edits.put(LSPEclipseUtils.toUri(workspaceFile).toString(), Collections.singletonList(new TextEdit(new Range(new Position(0, 0), new Position(0, 3)), "new")));
edits.put(LSPEclipseUtils.toUri(externalFile).toString(), Collections.singletonList(new TextEdit(new Range(new Position(0, 0), new Position(0, 3)), "new")));
MockLanguageServer.INSTANCE.getTextDocumentService().setRenameEdit(new WorkspaceEdit(edits));
IDocument document = LSPEclipseUtils.getDocument(workspaceFile);
LanguageServiceAccessor.getLanguageServers(document, LSPRenameHandler::isRenameProvider).thenAccept(languageServers -> {
LSPRenameProcessor processor = new LSPRenameProcessor(LSPEclipseUtils.getDocument(workspaceFile), languageServers.get(0), 0);
processor.setNewName("new");
try {
ProcessorBasedRefactoring processorBasedRefactoring = new ProcessorBasedRefactoring(processor);
processorBasedRefactoring.checkAllConditions(new NullProgressMonitor());
processorBasedRefactoring.createChange(new NullProgressMonitor()).perform(new NullProgressMonitor());
} catch (CoreException e) {
e.printStackTrace();
}
}).join();
assertEquals("new", document.get());
assertEquals("new", new String(Files.readAllBytes(externalFile.toPath())));
} finally {
Files.deleteIfExists(externalFile.toPath());
}
}
@Test
public void testRenameHandlerExecution() throws Exception {
IProject project = TestUtils.createProject("blah");
IFile file = TestUtils.createUniqueTestFile(project, "old");
MockLanguageServer.INSTANCE.getTextDocumentService().setRenameEdit(createSimpleMockRenameEdit(LSPEclipseUtils.toUri(file)));
ITextEditor editor = (ITextEditor) TestUtils.openEditor(file);
editor.selectAndReveal(1, 0);
ICommandService commandService = PlatformUI.getWorkbench().getService(ICommandService.class);
IHandlerService handlerService = PlatformUI.getWorkbench().getService(IHandlerService.class);
Command command = commandService.getCommand(IWorkbenchCommandConstants.FILE_RENAME);
assertTrue(command.isEnabled() && command.isHandled());
Event e = new Event();
e.widget = editor.getAdapter(Control.class);
Shell ideShell = editor.getSite().getShell();
Display display = ideShell.getDisplay();
e.display = display;
AtomicBoolean renameDialogOkPressed = new AtomicBoolean();
Listener pressOKonRenameDialogPaint = event -> {
if (renameDialogOkPressed.get()) {
return;
}
if(event.widget instanceof Composite) {
Shell shell = ((Composite)event.widget).getShell();
if(shell != ideShell && "Rename".equals(shell.getText())) {
event.widget.getDisplay().asyncExec(() -> pressOk(shell));
renameDialogOkPressed.set(true);
}
}
};
try {
display.addFilter(SWT.Paint, pressOKonRenameDialogPaint);
ExecutionEvent executionEvent = handlerService.createExecutionEvent(command, e);
command.executeWithChecks(executionEvent);
assertTrue("Rename dialog not shown", new DisplayHelper() {
@Override
protected boolean condition() {
return renameDialogOkPressed.get();
}
}.waitForCondition(display, 3000));
IDocument document = LSPEclipseUtils.getDocument(editor);
assertTrue("document not modified, rename not applied", new DisplayHelper() {
@Override
protected boolean condition() {
return "new".equals(document.get());
}
}.waitForCondition(display, 3000));
} finally {
ideShell.getDisplay().removeFilter(SWT.Paint, pressOKonRenameDialogPaint);
}
}
@Test
public void testPlaceholderUsingPlaceholderFromPrepareRenameResult() throws Exception {
IProject project = TestUtils.createProject("blah");
IFile file = TestUtils.createUniqueTestFile(project, "old");
MockLanguageServer.INSTANCE.getTextDocumentService().setRenameEdit(createSimpleMockRenameEdit(LSPEclipseUtils.toUri(file)));
IDocument document = LSPEclipseUtils.getDocument(file);
AtomicReference<String> placeholder = new AtomicReference<>();
LanguageServiceAccessor.getLanguageServers(document, LSPRenameHandler::isRenameProvider).thenAccept(languageServers -> {
LSPRenameProcessor processor = new LSPRenameProcessor(LSPEclipseUtils.getDocument(file), languageServers.get(0), 0);
try {
processor.checkInitialConditions(new NullProgressMonitor());
} catch (CoreException e) {
e.printStackTrace();
}
placeholder.set(processor.getPlaceholder());
}).join();
assertEquals("placeholder", placeholder.get());
}
@Test
public void testPlaceholderUsingRangeFromPrepareRenameResult() throws Exception {
IProject project = TestUtils.createProject("blah");
IFile file = TestUtils.createUniqueTestFile(project, "old");
MockLanguageServer.INSTANCE.getTextDocumentService().setRenameEdit(createSimpleMockRenameEdit(LSPEclipseUtils.toUri(file)));
Range range = new Range(new Position(0, 1), new Position(0, 3)); // Two last letters of "old".
MockLanguageServer.INSTANCE.getTextDocumentService().setPrepareRenameResult(Either.forLeft(range));
IDocument document = LSPEclipseUtils.getDocument(file);
AtomicReference<String> placeholder = new AtomicReference<>();
LanguageServiceAccessor.getLanguageServers(document, LSPRenameHandler::isRenameProvider).thenAccept(languageServers -> {
LSPRenameProcessor processor = new LSPRenameProcessor(LSPEclipseUtils.getDocument(file), languageServers.get(0), 0);
try {
processor.checkInitialConditions(new NullProgressMonitor());
} catch (CoreException e) {
e.printStackTrace();
}
placeholder.set(processor.getPlaceholder());
}).join();
assertEquals("ld", placeholder.get());
}
private void pressOk(Shell dialogShell) {
try {
Dialog dialog = (Dialog)dialogShell.getData();
Method okPressedMethod = Dialog.class.getDeclaredMethod("okPressed");
okPressedMethod.setAccessible(true);
okPressedMethod.invoke(dialog);
} catch (Exception ex) {
throw new Error(ex);
}
}
private static WorkspaceEdit createSimpleMockRenameEdit(URI fileUri) {
WorkspaceEdit res = new WorkspaceEdit();
File f = new File(fileUri);
res.setChanges(Collections.singletonMap(LSPEclipseUtils.toUri(f).toString(),
Collections.singletonList(new TextEdit(new Range(new Position(0, 0), new Position(0, 3)), "new"))));
return res;
}
}