/******************************************************************************* * Copyright (c) 2012-2016 Codenvy, S.A. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * Codenvy, S.A. - initial API and implementation *******************************************************************************/ package org.eclipse.che.ide.ext.git.client.branch; import org.eclipse.che.api.git.shared.Branch; import org.eclipse.che.api.git.shared.CheckoutRequest; import org.eclipse.che.api.project.gwt.client.ProjectServiceClient; import org.eclipse.che.api.workspace.shared.dto.ProjectConfigDto; import org.eclipse.che.ide.api.editor.EditorAgent; import org.eclipse.che.ide.api.editor.EditorInput; import org.eclipse.che.ide.api.editor.EditorPartPresenter; import org.eclipse.che.ide.api.event.FileContentUpdateEvent; import org.eclipse.che.ide.api.event.project.ProjectUpdatedEvent; import org.eclipse.che.ide.api.notification.StatusNotification; import org.eclipse.che.ide.api.parts.WorkspaceAgent; import org.eclipse.che.ide.api.project.tree.VirtualFile; import org.eclipse.che.ide.dto.DtoFactory; import org.eclipse.che.ide.ext.git.client.BaseTest; import org.eclipse.che.ide.part.explorer.project.ProjectExplorerPresenter; import org.eclipse.che.ide.rest.AsyncRequestCallback; import org.eclipse.che.ide.ui.dialogs.ConfirmCallback; import org.eclipse.che.ide.ui.dialogs.DialogFactory; import org.eclipse.che.ide.ui.dialogs.InputCallback; import org.eclipse.che.ide.ui.dialogs.confirm.ConfirmDialog; import org.eclipse.che.ide.ui.dialogs.input.InputDialog; import org.eclipse.che.test.GwtReflectionUtils; import org.junit.Test; import org.mockito.ArgumentCaptor; import org.mockito.Captor; import org.mockito.Matchers; import org.mockito.Mock; import java.util.ArrayList; import java.util.List; import java.util.NavigableMap; import java.util.TreeMap; import static org.eclipse.che.api.git.shared.BranchListRequest.LIST_ALL; import static org.eclipse.che.ide.ext.git.client.patcher.WindowPatcher.RETURNED_MESSAGE; import static org.mockito.Matchers.any; import static org.mockito.Matchers.anyInt; import static org.mockito.Matchers.anyObject; import static org.mockito.Matchers.anyString; import static org.mockito.Matchers.eq; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.never; import static org.mockito.Mockito.reset; import static org.mockito.Mockito.times; import static org.mockito.Mockito.verify; import static org.mockito.Mockito.verifyNoMoreInteractions; import static org.mockito.Mockito.when; import static org.eclipse.che.ide.ext.git.client.branch.BranchPresenter.BRANCH_LIST_COMMAND_NAME; import static org.eclipse.che.ide.ext.git.client.branch.BranchPresenter.BRANCH_RENAME_COMMAND_NAME; import static org.eclipse.che.ide.ext.git.client.branch.BranchPresenter.BRANCH_DELETE_COMMAND_NAME; import static org.eclipse.che.ide.ext.git.client.branch.BranchPresenter.BRANCH_CREATE_COMMAND_NAME; /** * Testing {@link BranchPresenter} functionality. * * @author Andrey Plotnikov */ public class BranchPresenterTest extends BaseTest { @Captor private ArgumentCaptor<InputCallback> inputCallbackCaptor; @Captor private ArgumentCaptor<ConfirmCallback> confirmCallbackCaptor; @Captor private ArgumentCaptor<AsyncRequestCallback<Branch>> createBranchCallbackCaptor; @Captor private ArgumentCaptor<AsyncRequestCallback<List<Branch>>> branchListCallbackCaptor; @Captor private ArgumentCaptor<AsyncRequestCallback<String>> asyncRequestCallbackCaptor; @Captor private ArgumentCaptor<AsyncRequestCallback<ProjectConfigDto>> getProjectCallbackCaptor; public static final String BRANCH_NAME = "branchName"; public static final String REMOTE_BRANCH_NAME = "origin/branchName"; public static final boolean NEED_DELETING = true; public static final boolean IS_REMOTE = true; public static final boolean IS_ACTIVE = true; @Mock private BranchView view; @Mock private EditorInput editorInput; @Mock private EditorAgent editorAgent; @Mock private Branch selectedBranch; @Mock private EditorPartPresenter partPresenter; @Mock private WorkspaceAgent workspaceAgent; @Mock private DialogFactory dialogFactory; @Mock private DtoFactory dtoFactory; @Mock private ProjectExplorerPresenter projectExplorer; @Mock private CheckoutRequest checkoutRequest; @Mock private ProjectServiceClient projectService; private BranchPresenter presenter; @Override public void disarm() { super.disarm(); when(appContext.getWorkspaceId()).thenReturn("id"); presenter = new BranchPresenter(view, dtoFactory, editorAgent, service, projectService, constant, appContext, notificationManager, dtoUnmarshallerFactory, gitOutputConsoleFactory, consolesPanelPresenter, dialogFactory, projectExplorer, eventBus); NavigableMap<String, EditorPartPresenter> partPresenterMap = new TreeMap<>(); partPresenterMap.put("partPresenter", partPresenter); when(selectedBranch.getDisplayName()).thenReturn(BRANCH_NAME); when(selectedBranch.getName()).thenReturn(BRANCH_NAME); when(selectedBranch.isRemote()).thenReturn(IS_REMOTE); when(selectedBranch.isActive()).thenReturn(IS_ACTIVE); when(editorAgent.getOpenedEditors()).thenReturn(partPresenterMap); when(partPresenter.getEditorInput()).thenReturn(editorInput); } @Test public void testShowBranchesWhenGetBranchesRequestIsSuccessful() throws Exception { final List<Branch> branches = new ArrayList<>(); presenter.showBranches(); verify(service).branchList(anyString(), eq(rootProjectConfig), eq(LIST_ALL), branchListCallbackCaptor.capture()); AsyncRequestCallback<List<Branch>> branchListCallback = branchListCallbackCaptor.getValue(); GwtReflectionUtils.callOnSuccess(branchListCallback, branches); verify(appContext).getCurrentProject(); verify(view).showDialogIfClosed(); verify(view).setBranches(eq(branches)); verify(console, never()).printError(anyString()); verify(notificationManager, never()).notify(anyString(), any(ProjectConfigDto.class)); verify(constant, never()).branchesListFailed(); } @Test public void testShowBranchesWhenGetBranchesRequestIsFailed() throws Exception { presenter.showBranches(); verify(service).branchList(anyString(), eq(rootProjectConfig), eq(LIST_ALL), branchListCallbackCaptor.capture()); AsyncRequestCallback<List<Branch>> branchListCallback = branchListCallbackCaptor.getValue(); GwtReflectionUtils.callOnFailure(branchListCallback, mock(Throwable.class)); verify(appContext).getCurrentProject(); verify(view, never()).showDialogIfClosed(); verify(gitOutputConsoleFactory).create(BRANCH_LIST_COMMAND_NAME); verify(console).printError(anyString()); verify(consolesPanelPresenter).addCommandOutput(anyString(), eq(console)); verify(notificationManager).notify(anyString(), anyObject(), eq(true), eq(rootProjectConfig)); verify(constant, times(2)).branchesListFailed(); } @Test public void testOnCloseClicked() throws Exception { presenter.onCloseClicked(); verify(view).close(); } @Test public void testOnRenameClickedWhenLocalBranchSelected() throws Exception { reset(selectedBranch); when(selectedBranch.getDisplayName()).thenReturn(BRANCH_NAME); when(selectedBranch.isRemote()).thenReturn(false); InputDialog inputDialog = mock(InputDialog.class); when(dialogFactory.createInputDialog(anyString(), anyString(), anyString(), anyInt(), anyInt(), anyObject(), anyObject())) .thenReturn(inputDialog); selectBranch(); presenter.onRenameClicked(); verify(dialogFactory).createInputDialog(anyString(), anyString(), anyString(), anyInt(), anyInt(), inputCallbackCaptor.capture(), anyObject()); InputCallback inputCallback = inputCallbackCaptor.getValue(); inputCallback.accepted(RETURNED_MESSAGE); verify(service) .branchRename(anyString(), eq(rootProjectConfig), eq(BRANCH_NAME), eq(RETURNED_MESSAGE), asyncRequestCallbackCaptor.capture()); AsyncRequestCallback<String> renameBranchCallback = asyncRequestCallbackCaptor.getValue(); GwtReflectionUtils.callOnSuccess(renameBranchCallback, PROJECT_PATH); verify(selectedBranch, times(2)).getDisplayName(); verify(service, times(2)).branchList(anyString(), eq(rootProjectConfig), eq(LIST_ALL), anyObject()); verify(dialogFactory, never()).createConfirmDialog(anyString(), anyString(), anyObject(), anyObject()); verify(console, never()).printError(anyString()); verify(notificationManager, never()).notify(anyString(), eq(rootProjectConfig)); verify(constant, never()).branchRenameFailed(); } @Test public void testOnRenameClickedWhenRemoteBranchSelectedAndUserConfirmRename() throws Exception { reset(selectedBranch); when(selectedBranch.getDisplayName()).thenReturn(REMOTE_BRANCH_NAME); when(selectedBranch.isRemote()).thenReturn(true); InputDialog inputDialog = mock(InputDialog.class); when(dialogFactory.createInputDialog(anyString(), anyString(), anyString(), anyInt(), anyInt(), anyObject(), anyObject())) .thenReturn(inputDialog); ConfirmDialog confirmDialog = mock(ConfirmDialog.class); when(dialogFactory.createConfirmDialog(anyString(), anyString(), anyObject(), anyObject())).thenReturn(confirmDialog); selectBranch(); presenter.onRenameClicked(); verify(dialogFactory).createConfirmDialog(anyString(), anyString(), confirmCallbackCaptor.capture(), anyObject()); ConfirmCallback confirmCallback = confirmCallbackCaptor.getValue(); confirmCallback.accepted(); verify(dialogFactory).createInputDialog(anyString(), anyString(), anyString(), anyInt(), anyInt(), inputCallbackCaptor.capture(), anyObject()); InputCallback inputCallback = inputCallbackCaptor.getValue(); inputCallback.accepted(RETURNED_MESSAGE); verify(service).branchRename(anyString(), eq(rootProjectConfig), eq(REMOTE_BRANCH_NAME), eq(RETURNED_MESSAGE), asyncRequestCallbackCaptor.capture()); AsyncRequestCallback<String> renameBranchCallback = asyncRequestCallbackCaptor.getValue(); GwtReflectionUtils.callOnSuccess(renameBranchCallback, PROJECT_PATH); verify(selectedBranch, times(2)).getDisplayName(); verify(service, times(2)) .branchList(anyString(), eq(rootProjectConfig), eq(LIST_ALL), anyObject()); verify(console, never()).printError(anyString()); verify(notificationManager, never()).notify(anyString(), eq(rootProjectConfig)); verify(constant, never()).branchRenameFailed(); } /** Select mock branch for testing. */ private void selectBranch() { presenter.showBranches(); presenter.onBranchSelected(selectedBranch); } @Test public void testOnRenameClickedWhenBranchRenameRequestIsFailed() throws Exception { when(selectedBranch.getDisplayName()).thenReturn(BRANCH_NAME); when(selectedBranch.isRemote()).thenReturn(false); InputDialog inputDialog = mock(InputDialog.class); when(dialogFactory.createInputDialog(anyString(), anyString(), anyString(), anyInt(), anyInt(), anyObject(), anyObject())) .thenReturn(inputDialog); selectBranch(); presenter.onRenameClicked(); verify(dialogFactory).createInputDialog(anyString(), anyString(), anyString(), anyInt(), anyInt(), inputCallbackCaptor.capture(), anyObject()); InputCallback inputCallback = inputCallbackCaptor.getValue(); inputCallback.accepted(RETURNED_MESSAGE); verify(service) .branchRename(anyString(), eq(rootProjectConfig), eq(BRANCH_NAME), eq(RETURNED_MESSAGE), asyncRequestCallbackCaptor.capture()); AsyncRequestCallback<String> renameBranchCallback = asyncRequestCallbackCaptor.getValue(); GwtReflectionUtils.callOnFailure(renameBranchCallback, mock(Throwable.class)); verify(selectedBranch, times(2)).getDisplayName(); verify(gitOutputConsoleFactory).create(BRANCH_RENAME_COMMAND_NAME); verify(console).printError(anyString()); verify(consolesPanelPresenter).addCommandOutput(anyString(), eq(console)); verify(notificationManager).notify(anyString(), anyObject(), eq(true), eq(rootProjectConfig)); verify(constant, times(2)).branchRenameFailed(); } @Test public void testOnDeleteClickedWhenBranchDeleteRequestIsSuccessful() throws Exception { selectBranch(); presenter.onDeleteClicked(); verify(service).branchDelete(anyString(), eq(rootProjectConfig), eq(BRANCH_NAME), eq(NEED_DELETING), asyncRequestCallbackCaptor.capture()); AsyncRequestCallback<String> deleteBranchCallback = asyncRequestCallbackCaptor.getValue(); GwtReflectionUtils.callOnSuccess(deleteBranchCallback, PROJECT_PATH); verify(selectedBranch).getName(); verify(service, times(2)).branchList(anyString(), eq(rootProjectConfig), eq(LIST_ALL), anyObject()); verify(constant, never()).branchDeleteFailed(); verify(console, never()).printError(anyString()); verify(notificationManager, never()).notify(anyString(), eq(rootProjectConfig)); } @Test public void testOnDeleteClickedWhenBranchDeleteRequestIsFailed() throws Exception { selectBranch(); presenter.onDeleteClicked(); verify(service).branchDelete(anyString(), eq(rootProjectConfig), eq(BRANCH_NAME), eq(NEED_DELETING), asyncRequestCallbackCaptor.capture()); AsyncRequestCallback<String> deleteBranchCallback = asyncRequestCallbackCaptor.getValue(); GwtReflectionUtils.callOnFailure(deleteBranchCallback, mock(Throwable.class)); verify(selectedBranch).getName(); verify(constant, times(2)).branchDeleteFailed(); verify(gitOutputConsoleFactory).create(BRANCH_DELETE_COMMAND_NAME); verify(console).printError(anyString()); verify(consolesPanelPresenter).addCommandOutput(anyString(), eq(console)); verify(notificationManager).notify(anyString(), anyObject(), eq(true), eq(rootProjectConfig)); } @Test public void testOnCheckoutClickedWhenSelectedNotRemoteBranch() throws Exception { when(selectedBranch.isRemote()).thenReturn(false); when(dtoFactory.createDto(CheckoutRequest.class)).thenReturn(checkoutRequest); selectBranch(); presenter.onCheckoutClicked(); verify(checkoutRequest).setName(eq(BRANCH_NAME)); verifyNoMoreInteractions(checkoutRequest); verify(service).checkout(anyString(), eq(rootProjectConfig), eq(checkoutRequest), asyncRequestCallbackCaptor.capture()); } @Test public void testOnCheckoutClickedWhenSelectedRemoteBranch() throws Exception { when(dtoFactory.createDto(CheckoutRequest.class)).thenReturn(checkoutRequest); selectBranch(); presenter.onCheckoutClicked(); verify(checkoutRequest).setTrackBranch(eq(BRANCH_NAME)); verifyNoMoreInteractions(checkoutRequest); verify(service).checkout(anyString(), eq(rootProjectConfig), eq(checkoutRequest), asyncRequestCallbackCaptor.capture()); } @Test public void testOnCheckoutClickedWhenCheckoutRequestAndRefreshProjectIsSuccessful() throws Exception { when(dtoFactory.createDto(CheckoutRequest.class)).thenReturn(checkoutRequest); VirtualFile virtualFile = mock(VirtualFile.class); when(editorInput.getFile()).thenReturn(virtualFile); when(virtualFile.getPath()).thenReturn("/foo"); selectBranch(); presenter.onCheckoutClicked(); verify(checkoutRequest).setTrackBranch(eq(BRANCH_NAME)); verifyNoMoreInteractions(checkoutRequest); verify(service).checkout(anyString(), eq(rootProjectConfig), eq(checkoutRequest), asyncRequestCallbackCaptor.capture()); AsyncRequestCallback<String> checkoutBranchCallback = asyncRequestCallbackCaptor.getValue(); GwtReflectionUtils.callOnSuccess(checkoutBranchCallback, PROJECT_PATH); AsyncRequestCallback<ProjectConfigDto> getProjectCallback = getProjectCallbackCaptor.getValue(); GwtReflectionUtils.callOnSuccess(getProjectCallback, PROJECT_PATH); verify(editorAgent).getOpenedEditors(); verify(selectedBranch, times(2)).getDisplayName(); verify(selectedBranch).isRemote(); verify(service).checkout(anyString(), eq(rootProjectConfig), eq(checkoutRequest), anyObject()); verify(service, times(2)).branchList(anyString(), eq(rootProjectConfig), eq(LIST_ALL), anyObject()); verify(appContext).getCurrentProject(); verify(console, never()).printError(anyString()); verify(notificationManager, never()).notify(anyString(), eq(rootProjectConfig)); verify(eventBus).fireEvent(Matchers.<FileContentUpdateEvent>anyObject()); verify(constant, never()).branchCheckoutFailed(); verify(projectService).getProject(anyString(), anyString(), anyObject()); verify(eventBus).fireEvent(Matchers.<ProjectUpdatedEvent>anyObject()); } @Test public void testOnCheckoutClickedWhenCheckoutRequestAndRefreshProjectIsSuccessfulButOpenFileIsNotExistInBranch() throws Exception { when(dtoFactory.createDto(CheckoutRequest.class)).thenReturn(checkoutRequest); VirtualFile virtualFile = mock(VirtualFile.class); when(editorInput.getFile()).thenReturn(virtualFile); when(virtualFile.getPath()).thenReturn("/foo"); selectBranch(); presenter.onCheckoutClicked(); verify(checkoutRequest).setTrackBranch(eq(BRANCH_NAME)); verifyNoMoreInteractions(checkoutRequest); verify(service).checkout(anyString(), eq(rootProjectConfig), eq(checkoutRequest), asyncRequestCallbackCaptor.capture()); AsyncRequestCallback<String> checkoutBranchCallback = asyncRequestCallbackCaptor.getValue(); GwtReflectionUtils.callOnSuccess(checkoutBranchCallback, PROJECT_PATH); verify(editorAgent).getOpenedEditors(); verify(selectedBranch, times(2)).getDisplayName(); verify(selectedBranch).isRemote(); verify(service, times(2)).branchList(anyString(), eq(rootProjectConfig), eq(LIST_ALL), anyObject()); verify(eventBus).fireEvent(Matchers.<FileContentUpdateEvent>anyObject()); verify(appContext).getCurrentProject(); } @Test public void testOnCheckoutClickedWhenCheckoutRequestIsFailed() throws Exception { when(dtoFactory.createDto(CheckoutRequest.class)).thenReturn(checkoutRequest); selectBranch(); presenter.onCheckoutClicked(); verify(checkoutRequest).setTrackBranch(eq(BRANCH_NAME)); verifyNoMoreInteractions(checkoutRequest); verify(service).checkout(anyString(), eq(rootProjectConfig), eq(checkoutRequest), asyncRequestCallbackCaptor.capture()); AsyncRequestCallback<String> checkoutBranchCallback = asyncRequestCallbackCaptor.getValue(); GwtReflectionUtils.callOnFailure(checkoutBranchCallback, mock(Throwable.class)); verify(selectedBranch, times(2)).getDisplayName(); verify(selectedBranch).isRemote(); verify(notificationManager).notify(anyString(), eq(StatusNotification.Status.FAIL), eq(true), eq(rootProjectConfig)); } @Test public void testOnCreateClickedWhenBranchCreateRequestIsSuccessful() throws Exception { InputDialog inputDialog = mock(InputDialog.class); when(dialogFactory.createInputDialog(anyString(), anyString(), anyObject(), anyObject())).thenReturn(inputDialog); presenter.showBranches(); presenter.onCreateClicked(); verify(dialogFactory).createInputDialog(anyString(), anyString(), inputCallbackCaptor.capture(), anyObject()); InputCallback inputCallback = inputCallbackCaptor.getValue(); inputCallback.accepted(BRANCH_NAME); verify(service).branchCreate(anyString(), anyObject(), anyString(), anyString(), createBranchCallbackCaptor.capture()); AsyncRequestCallback<Branch> createBranchCallback = createBranchCallbackCaptor.getValue(); GwtReflectionUtils.callOnSuccess(createBranchCallback, selectedBranch); verify(constant).branchTypeNew(); verify(service).branchCreate(anyString(), eq(rootProjectConfig), anyString(), anyString(), anyObject()); verify(service, times(2)).branchList(anyString(), eq(rootProjectConfig), eq(LIST_ALL), anyObject()); } @Test public void testOnCreateClickedWhenBranchCreateRequestIsFailed() throws Exception { Throwable exception = mock(Exception.class); InputDialog inputDialog = mock(InputDialog.class); when(dialogFactory.createInputDialog(anyString(), anyString(), anyObject(), anyObject())).thenReturn(inputDialog); presenter.showBranches(); presenter.onCreateClicked(); verify(dialogFactory).createInputDialog(anyString(), anyString(), inputCallbackCaptor.capture(), anyObject()); InputCallback inputCallback = inputCallbackCaptor.getValue(); inputCallback.accepted(BRANCH_NAME); verify(service).branchCreate(anyString(), anyObject(), anyString(), anyString(), createBranchCallbackCaptor.capture()); AsyncRequestCallback<Branch> createBranchCallback = createBranchCallbackCaptor.getValue(); GwtReflectionUtils.callOnFailure(createBranchCallback, exception); verify(constant, times(2)).branchCreateFailed(); verify(gitOutputConsoleFactory).create(BRANCH_CREATE_COMMAND_NAME); verify(console).printError(anyString()); verify(consolesPanelPresenter).addCommandOutput(anyString(), eq(console)); verify(notificationManager).notify(anyString(), anyObject(), eq(true), eq(rootProjectConfig)); } @Test public void checkoutButtonShouldBeEnabled() throws Exception { when(selectedBranch.isActive()).thenReturn(false); presenter.onBranchSelected(selectedBranch); verify(view).setEnableCheckoutButton(eq(ENABLE_BUTTON)); } @Test public void checkoutButtonShouldBeDisabled() throws Exception { when(selectedBranch.isActive()).thenReturn(true); presenter.onBranchSelected(selectedBranch); verify(view).setEnableCheckoutButton(eq(DISABLE_BUTTON)); } @Test public void renameButtonShouldBeEnabledWhenLocalBranchSelected() throws Exception { when(selectedBranch.isRemote()).thenReturn(false); presenter.onBranchSelected(selectedBranch); verify(view).setEnableRenameButton(eq(ENABLE_BUTTON)); } @Test public void renameButtonShouldBeEnabledWhenRemoteBranchSelected() throws Exception { when(selectedBranch.isRemote()).thenReturn(true); presenter.onBranchSelected(selectedBranch); verify(view).setEnableRenameButton(eq(ENABLE_BUTTON)); } @Test public void deleteButtonShouldBeEnabled() throws Exception { when(selectedBranch.isActive()).thenReturn(false); presenter.onBranchSelected(selectedBranch); verify(view).setEnableDeleteButton(eq(ENABLE_BUTTON)); } @Test public void deleteButtonShouldBeDisabled() throws Exception { when(selectedBranch.isActive()).thenReturn(true); presenter.onBranchSelected(selectedBranch); verify(view).setEnableDeleteButton(eq(DISABLE_BUTTON)); } @Test public void checkoutDeleteRenameButtonsShouldBeDisabled() throws Exception { presenter.onBranchUnselected(); verify(view).setEnableCheckoutButton(eq(DISABLE_BUTTON)); verify(view).setEnableDeleteButton(eq(DISABLE_BUTTON)); verify(view).setEnableRenameButton(eq(DISABLE_BUTTON)); } }