blob: 87dd5dcc2753814ac61315fd9baa9a16dcda5ed8 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2007 IBM Corporation 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:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.team.internal.ccvs.ui.subscriber;
import java.lang.reflect.InvocationTargetException;
import java.util.*;
import org.eclipse.compare.structuremergeviewer.IDiffElement;
import org.eclipse.core.resources.IProject;
import org.eclipse.core.resources.IResource;
import org.eclipse.core.resources.mapping.ResourceMappingContext;
import org.eclipse.core.runtime.IProgressMonitor;
import org.eclipse.core.runtime.OperationCanceledException;
import org.eclipse.jface.dialogs.IDialogConstants;
import org.eclipse.jface.dialogs.MessageDialog;
import org.eclipse.osgi.util.NLS;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.team.core.TeamException;
import org.eclipse.team.core.mapping.provider.SynchronizationScopeManager;
import org.eclipse.team.core.synchronize.SyncInfo;
import org.eclipse.team.core.synchronize.SyncInfoSet;
import org.eclipse.team.internal.ccvs.core.*;
import org.eclipse.team.internal.ccvs.core.client.Command;
import org.eclipse.team.internal.ccvs.core.resources.CVSWorkspaceRoot;
import org.eclipse.team.internal.ccvs.ui.CVSUIMessages;
import org.eclipse.team.internal.ccvs.ui.CVSUIPlugin;
import org.eclipse.team.internal.ccvs.ui.Policy;
import org.eclipse.team.internal.ccvs.ui.mappings.ChangeSetComparator;
import org.eclipse.team.internal.ccvs.ui.operations.*;
import org.eclipse.team.internal.ccvs.ui.repo.RepositoryManager;
import org.eclipse.team.internal.core.subscribers.ChangeSet;
import org.eclipse.team.internal.ui.Utils;
import org.eclipse.team.ui.synchronize.ISynchronizePageConfiguration;
public class WorkspaceCommitOperation extends CVSSubscriberOperation {
private String comment;
private SyncInfoSet syncSet;
private boolean override;
public WorkspaceCommitOperation(ISynchronizePageConfiguration configuration, IDiffElement[] elements, boolean override) {
super(configuration, elements);
this.override = override;
}
@Override
protected String getErrorTitle() {
return CVSUIMessages.CommitAction_commitFailed;
}
@Override
protected String getJobName() {
SyncInfoSet syncSet = getSyncInfoSet();
return NLS.bind(CVSUIMessages.CommitAction_jobName, new String[] { Integer.valueOf(syncSet.size()).toString() });
}
@Override
public boolean shouldRun() {
SyncInfoSet set = getSyncInfoSet();
return !set.isEmpty();
}
@Override
protected SyncInfoSet getSyncInfoSet() {
if (syncSet == null) {
syncSet = super.getSyncInfoSet();
if (!promptForConflictHandling(syncSet)) {
syncSet.clear();
return syncSet;
}
try {
if (!promptForUnaddedHandling(syncSet)) {
syncSet.clear();
return syncSet;
}
} catch (CVSException e) {
Utils.handle(e);
syncSet.clear();
}
}
return syncSet;
}
protected boolean promptForConflictHandling(SyncInfoSet syncSet) {
if (syncSet.hasConflicts() || syncSet.hasIncomingChanges()) {
if (override) {
// If overriding, prompt to ensure that is what the user wants
switch (promptForConflicts(syncSet)) {
case 0:
// Yes, synchronize conflicts as well
break;
case 1:
// No, stop here
return false;
case 2:
default:
// Cancel
return false;
}
} else {
// If there is a conflict in the syncSet, remove from sync set.
syncSet.removeConflictingNodes();
syncSet.removeIncomingNodes();
}
}
return true;
}
@Override
public void runWithProjectRule(IProject project, SyncInfoSet syncSet, IProgressMonitor monitor) throws TeamException {
final SyncInfo[] changed = syncSet.getSyncInfos();
if (changed.length == 0) return;
// A list of files to be committed
final List<IResource> commits = new ArrayList<>();
// New resources that are not yet under CVS control and need a "cvs add"
final List<IResource> additions = new ArrayList<>();
// A list of incoming or conflicting file changes to be made outgoing changes
final List<SyncInfo> makeOutgoing = new ArrayList<>();
// A list of out-of-sync folders that must be made in-sync
final List<SyncInfo> makeInSync = new ArrayList<>();
for (int i = 0; i < changed.length; i++) {
SyncInfo changedNode = changed[i];
int kind = changedNode.getKind();
IResource resource = changedNode.getLocal();
// Any parent folders should be made in-sync.
// Steps will be taken after the commit to prune any empty folders
SyncInfo parent = getParent(changedNode);
if (parent != null) {
if (isOutOfSync(parent)) {
makeInSync.add(parent);
}
}
if (resource.getType() == IResource.FILE) {
// By default, all files are committed
commits.add(resource);
// Determine what other work needs to be done for the file
switch (kind & SyncInfo.DIRECTION_MASK) {
case SyncInfo.INCOMING:
// Convert the incoming change to an outgoing change
makeOutgoing.add(changedNode);
break;
case SyncInfo.OUTGOING:
switch (kind & SyncInfo.CHANGE_MASK) {
case SyncInfo.ADDITION:
// Outgoing addition. 'add' it before committing.
if (!isAdded(resource))
additions.add(resource);
break;
case SyncInfo.DELETION:
// Outgoing deletion is handled by move/delete
// hook and EclipseSynchronizer
break;
case SyncInfo.CHANGE:
// Outgoing change. Just commit it.
break;
}
break;
case SyncInfo.CONFLICTING:
// Convert the conflicting change to an outgoing change
makeOutgoing.add(changedNode);
break;
}
} else {
if (((kind & SyncInfo.DIRECTION_MASK) == SyncInfo.OUTGOING)
&& ((kind & SyncInfo.CHANGE_MASK) == SyncInfo.ADDITION)) {
// Outgoing folder additions must be added
additions.add(changedNode.getLocal());
} else if (isOutOfSync(changedNode)) {
// otherwise, make any out-of-sync folders in-sync using the remote info
makeInSync.add(changedNode);
}
}
}
monitor.beginTask(null, 200);
if (makeInSync.size() > 0) {
makeInSync(makeInSync.toArray(new SyncInfo[makeInSync.size()]), Policy.subMonitorFor(monitor, 25));
}
if (makeOutgoing.size() > 0) {
makeOutgoing(makeOutgoing.toArray(new SyncInfo[makeInSync.size()]), Policy.subMonitorFor(monitor, 25));
}
if (additions.size() != 0) {
add(project, additions.toArray(new IResource[0]), Policy.subMonitorFor(monitor, 50));
}
commit(project, commits.toArray(new IResource[commits.size()]), Policy.subMonitorFor(monitor, 100));
}
private void commit(final IProject project, IResource[] commits, IProgressMonitor monitor) throws TeamException {
try {
CommitOperation commitOperation = new CommitOperation(getPart(), RepositoryProviderOperation.asResourceMappers(commits),
new Command.LocalOption[0], comment) {
@Override
protected ResourceMappingContext getResourceMappingContext() {
return new SingleProjectSubscriberContext(CVSProviderPlugin.getPlugin().getCVSWorkspaceSubscriber(), false, project);
}
@Override
protected SynchronizationScopeManager createScopeManager(boolean consultModels) {
return new SingleProjectScopeManager(getJobName(), getSelectedMappings(), getResourceMappingContext(), consultModels, project);
}
};
commitOperation
.run(monitor);
} catch (InvocationTargetException e) {
throw TeamException.asTeamException(e);
} catch (InterruptedException e) {
throw new OperationCanceledException();
}
}
private void add(final IProject project, IResource[] additions, IProgressMonitor monitor) throws TeamException {
try {
new AddOperation(getPart(), RepositoryProviderOperation.asResourceMappers(additions)) {
@Override
protected ResourceMappingContext getResourceMappingContext() {
return new SingleProjectSubscriberContext(CVSProviderPlugin.getPlugin().getCVSWorkspaceSubscriber(), false, project);
}
@Override
protected SynchronizationScopeManager createScopeManager(boolean consultModels) {
return new SingleProjectScopeManager(getJobName(), getSelectedMappings(), getResourceMappingContext(), consultModels, project);
}
}.run(monitor);
} catch (InvocationTargetException e1) {
throw TeamException.asTeamException(e1);
} catch (InterruptedException e1) {
throw new OperationCanceledException();
}
}
/**
* Prompts the user to determine how conflicting changes should be handled.
* Note: This method is designed to be overridden by test cases.
* @return 0 to sync conflicts, 1 to sync all non-conflicts, 2 to cancel
*/
protected int promptForConflicts(SyncInfoSet syncSet) {
String[] buttons = new String[] {IDialogConstants.YES_LABEL, IDialogConstants.NO_LABEL, IDialogConstants.CANCEL_LABEL};
String question = CVSUIMessages.CommitSyncAction_questionRelease;
String title = CVSUIMessages.CommitSyncAction_titleRelease;
String[] tips = new String[] {
CVSUIMessages.CommitSyncAction_releaseAll,
CVSUIMessages.CommitSyncAction_releasePart,
CVSUIMessages.CommitSyncAction_cancelRelease
};
Shell shell = getShell();
final ToolTipMessageDialog dialog = new ToolTipMessageDialog(shell, title, null, question, MessageDialog.QUESTION, buttons, tips, 0);
shell.getDisplay().syncExec(() -> dialog.open());
return dialog.getReturnCode();
}
/**
* Prompts the user for a release comment.
* Note: This method is designed to be overridden by test cases.
* @return the comment, or null to cancel
*/
protected String promptForComment(RepositoryManager manager, IResource[] resourcesToCommit) {
String proposedComment = getProposedComment(resourcesToCommit);
return manager.promptForComment(getShell(), resourcesToCommit, proposedComment);
}
private String getProposedComment(IResource[] resourcesToCommit) {
StringBuffer comment = new StringBuffer();
ChangeSet[] sets = CVSUIPlugin.getPlugin().getChangeSetManager().getSets();
Arrays.sort(sets, new ChangeSetComparator());
int numMatchedSets = 0;
for (int i = 0; i < sets.length; i++) {
ChangeSet set = sets[i];
if (containsOne(set, resourcesToCommit)) {
if(numMatchedSets > 0) comment.append(System.getProperty("line.separator")); //$NON-NLS-1$
comment.append(set.getComment());
numMatchedSets++;
}
}
return comment.toString();
}
private boolean containsOne(ChangeSet set, IResource[] resourcesToCommit) {
for (int j = 0; j < resourcesToCommit.length; j++) {
IResource resource = resourcesToCommit[j];
if (set.contains(resource)) {
return true;
}
}
return false;
}
protected IResource[] promptForResourcesToBeAdded(RepositoryManager manager, IResource[] unadded) {
return manager.promptForResourcesToBeAdded(getShell(), unadded);
}
private boolean promptForUnaddedHandling(SyncInfoSet syncSet) throws CVSException {
if (syncSet.isEmpty()) return false;
// accumulate any resources that are not under version control
IResource[] unadded = getUnaddedResources(syncSet);
// prompt to get comment and any resources to be added to version control
RepositoryManager manager = CVSUIPlugin.getPlugin().getRepositoryManager();
IResource[] toBeAdded = promptForResourcesToBeAdded(manager, unadded);
if (toBeAdded == null) return false; // User cancelled.
comment = promptForComment(manager, syncSet.getResources());
if (comment == null) return false; // User cancelled.
// remove unshared resources that were not selected by the user
if (unadded != null && unadded.length > 0) {
List<IResource> resourcesToRemove = new ArrayList<>(unadded.length);
for (int i = 0; i < unadded.length; i++) {
IResource unaddedResource = unadded[i];
boolean included = false;
for (int j = 0; j < toBeAdded.length; j++) {
IResource resourceToAdd = toBeAdded[j];
if (unaddedResource.equals(resourceToAdd)) {
included = true;
break;
}
}
if (!included)
resourcesToRemove.add(unaddedResource);
}
syncSet.removeAll(resourcesToRemove.toArray(new IResource[resourcesToRemove.size()]));
}
return true;
}
private IResource[] getUnaddedResources(SyncInfoSet syncSet) throws CVSException {
// TODO: Should only get outgoing additions (since conflicting additions
// could be considered to be under version control already)
IResource[] resources = syncSet.getResources();
List<IResource> result = new ArrayList<>();
for (int i = 0; i < resources.length; i++) {
IResource resource = resources[i];
if (!isAdded(resource)) {
result.add(resource);
}
}
return result.toArray(new IResource[result.size()]);
}
private boolean isAdded(IResource resource) throws CVSException {
ICVSResource cvsResource = CVSWorkspaceRoot.getCVSResourceFor(resource);
if (cvsResource.isFolder()) {
return ((ICVSFolder)cvsResource).isCVSFolder();
} else {
return cvsResource.isManaged();
}
}
}