/******************************************************************************* * Copyright (C) 2010, Jens Baumgart <jens.baumgart@sap.com> * Copyright (C) 2010, Roland Grunberg <rgrunber@redhat.com> * Copyright (C) 2012, 2014 Robin Stocker <robin@nibor.org> * Copyright (C) 2015, Stephan Hackstedt <stephan.hackstedt@googlemail.com> * * 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 * * Code extracted from org.eclipse.egit.ui.internal.actions.DiscardChangesAction * and reworked. *******************************************************************************/ package org.eclipse.egit.core.op; import java.util.Collection; import java.util.Map; import java.util.Map.Entry; import org.eclipse.core.resources.IResource; import org.eclipse.core.resources.IWorkspace; import org.eclipse.core.resources.IWorkspaceRunnable; import org.eclipse.core.resources.ResourcesPlugin; import org.eclipse.core.runtime.CoreException; import org.eclipse.core.runtime.IPath; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.core.runtime.IStatus; import org.eclipse.core.runtime.SubMonitor; import org.eclipse.core.runtime.jobs.ISchedulingRule; import org.eclipse.egit.core.Activator; import org.eclipse.egit.core.internal.CoreText; import org.eclipse.egit.core.internal.job.RuleUtil; import org.eclipse.egit.core.internal.util.ProjectUtil; import org.eclipse.egit.core.internal.util.ResourceUtil; import org.eclipse.jgit.api.CheckoutCommand; import org.eclipse.jgit.api.Git; import org.eclipse.jgit.api.errors.GitAPIException; import org.eclipse.jgit.lib.Repository; /** * The operation discards changes on a set of resources (checkout with paths). * In case of a folder resource all file resources in the sub tree are * processed. Untracked files are ignored. */ public class DiscardChangesOperation implements IEGitOperation { private final Map<Repository, Collection<String>> pathsByRepository; private final ISchedulingRule schedulingRule; private String revision; private Stage stage; /** * The index stage to check out for conflicting files. */ public enum Stage { /** * "base" stage */ BASE(CheckoutCommand.Stage.BASE), /** * "ours" stage */ OURS(CheckoutCommand.Stage.OURS), /** * "theirs" stage */ THEIRS(CheckoutCommand.Stage.THEIRS); private CheckoutCommand.Stage checkoutStage; private Stage(CheckoutCommand.Stage checkoutStage) { this.checkoutStage = checkoutStage; } } /** * Construct a {@link DiscardChangesOperation} object. * * @param files */ public DiscardChangesOperation(IResource[] files) { this(files, null); } /** * Construct a {@link DiscardChangesOperation} object. * * @param files * @param revision */ public DiscardChangesOperation(IResource[] files, String revision) { this(ResourceUtil.splitResourcesByRepository(files)); this.revision = revision; } /** * {@link DiscardChangesOperation} for absolute paths. * * @param paths */ public DiscardChangesOperation(Collection<IPath> paths) { this(ResourceUtil.splitPathsByRepository(paths)); } private DiscardChangesOperation( Map<Repository, Collection<String>> pathsByRepository) { this.pathsByRepository = pathsByRepository; this.schedulingRule = RuleUtil.getRuleForRepositories(pathsByRepository .keySet()); } /** * Set the index stage to check out for conflicting files. Not compatible * with revision. * * @param stage */ public void setStage(Stage stage) { if (revision != null) throw new IllegalStateException( "Either stage or revision can be set, but not both"); //$NON-NLS-1$ this.stage = stage; } /* * (non-Javadoc) * * @see org.eclipse.egit.core.op.IEGitOperation#getSchedulingRule() */ @Override public ISchedulingRule getSchedulingRule() { return schedulingRule; } @Override public void execute(IProgressMonitor m) throws CoreException { IWorkspaceRunnable action = new IWorkspaceRunnable() { @Override public void run(IProgressMonitor actMonitor) throws CoreException { discardChanges(actMonitor); } }; ResourcesPlugin.getWorkspace().run(action, getSchedulingRule(), IWorkspace.AVOID_UPDATE, m); } private void discardChanges(IProgressMonitor monitor) throws CoreException { SubMonitor progress = SubMonitor.convert(monitor, CoreText.DiscardChangesOperation_discardingChanges, pathsByRepository.size() * 2); boolean errorOccurred = false; for (Entry<Repository, Collection<String>> entry : pathsByRepository .entrySet()) { Repository repository = entry.getKey(); Collection<String> paths = entry.getValue(); try { discardChanges(repository, paths); } catch (GitAPIException e) { errorOccurred = true; Activator.logError( CoreText.DiscardChangesOperation_discardFailed, e); } progress.worked(1); try { ProjectUtil.refreshRepositoryResources(repository, paths, progress.newChild(1)); } catch (CoreException e) { errorOccurred = true; Activator.logError( CoreText.DiscardChangesOperation_refreshFailed, e); } } if (errorOccurred) { IStatus status = Activator.error( CoreText.DiscardChangesOperation_discardFailedSeeLog, null); throw new CoreException(status); } } private void discardChanges(Repository repository, Collection<String> paths) throws GitAPIException { ResourceUtil.saveLocalHistory(repository); try (Git git = new Git(repository)) { CheckoutCommand checkoutCommand = git.checkout(); if (revision != null) { checkoutCommand.setStartPoint(revision); } if (stage != null) { checkoutCommand.setStage(stage.checkoutStage); } if (paths.isEmpty() || paths.contains("")) { //$NON-NLS-1$ checkoutCommand.setAllPaths(true); } else { for (String path : paths) { checkoutCommand.addPath(path); } } checkoutCommand.call(); } } }