blob: dda495e8a8e75d133f79266271dc527559e1cfc5 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2005 IBM Corporation and others.
* 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:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.update.internal.operations;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.Hashtable;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.osgi.util.NLS;
import org.eclipse.update.configuration.IConfiguredSite;
import org.eclipse.update.configuration.IInstallConfiguration;
import org.eclipse.update.core.IFeature;
import org.eclipse.update.core.IFeatureReference;
import org.eclipse.update.core.IIncludedFeatureReference;
import org.eclipse.update.core.VersionedIdentifier;
import org.eclipse.update.internal.core.Messages;
import org.eclipse.update.operations.IInstallFeatureOperation;
/**
*
*/
public class DuplicateConflictsValidator {
public static class IdEntry {
IConfiguredSite csite;
IFeature feature;
public IdEntry(IFeature feature, IConfiguredSite csite) {
this.feature = feature;
this.csite = csite;
if (csite == null) {
System.out.println("csite null"); //$NON-NLS-1$
}
}
public boolean isInstallCandidate() {
return csite != null;
}
public IFeature getFeature() {
return feature;
}
public String getIdentifier() {
return feature.getVersionedIdentifier().getIdentifier();
}
public IConfiguredSite getConfiguredSite() {
if (csite != null)
return csite;
return feature.getSite().getCurrentConfiguredSite();
}
public boolean sameLevel(IdEntry entry) {
VersionedIdentifier vid = feature.getVersionedIdentifier();
VersionedIdentifier evid =
entry.getFeature().getVersionedIdentifier();
return vid.equals(evid);
}
public String toString() {
IConfiguredSite configSite = getConfiguredSite();
String version =
feature.getVersionedIdentifier().getVersion().toString();
String location = configSite.getSite().getURL().getFile();
return NLS.bind(Messages.DuplicateConflictsDialog_conflict, (new String[] { version, location }));
}
}
public static ArrayList computeDuplicateConflicts(
IInstallFeatureOperation job,
IInstallConfiguration config,
IConfiguredSite targetSite,
IFeatureReference[] optionalFeatures) {
Hashtable featureTable = new Hashtable();
try {
computePresentState(featureTable, config);
computeNewFeature(
job.getFeature(),
targetSite,
featureTable,
optionalFeatures);
return computeConflicts(featureTable);
} catch (CoreException e) {
return null;
}
}
public static ArrayList computeDuplicateConflicts(
IInstallFeatureOperation[] jobs,
IInstallConfiguration config) {
Hashtable featureTable = new Hashtable();
computePresentState(featureTable, config);
computeNewFeatures(jobs, featureTable);
return computeConflicts(featureTable);
}
private static ArrayList computeConflicts(Hashtable featureTable) {
ArrayList result = null;
for (Enumeration iterator = featureTable.elements();
iterator.hasMoreElements();
) {
ArrayList candidate = (ArrayList) iterator.nextElement();
if (candidate.size() == 1)
continue;
ArrayList conflict = checkForConflict(candidate);
if (conflict != null) {
if (result == null)
result = new ArrayList();
result.add(conflict);
}
}
return result;
}
private static ArrayList checkForConflict(ArrayList candidate) {
IdEntry firstEntry = null;
for (int i = 0; i < candidate.size(); i++) {
IdEntry entry = (IdEntry) candidate.get(i);
if (firstEntry == null)
firstEntry = entry;
else if (!entry.sameLevel(firstEntry))
return candidate;
}
return null;
}
private static void computePresentState(
Hashtable table,
IInstallConfiguration config) {
IConfiguredSite[] csites = config.getConfiguredSites();
for (int i = 0; i < csites.length; i++) {
IConfiguredSite csite = csites[i];
IFeatureReference[] refs = csite.getConfiguredFeatures();
for (int j = 0; j < refs.length; j++) {
try {
addEntry(refs[j].getFeature(null), csite, table);
} catch (CoreException e) {
// don't let one bad feature stop the loop
}
}
}
}
private static void computeNewFeatures(
IInstallFeatureOperation[] jobs,
Hashtable featureTable) {
for (int i = 0; i < jobs.length; i++) {
IInstallFeatureOperation job = jobs[i];
IConfiguredSite targetSite = job.getTargetSite();
IFeature newFeature = job.getFeature();
try {
computeNewFeature(newFeature, targetSite, featureTable, null);
} catch (CoreException e) {
}
}
}
private static void computeNewFeature(
IFeature feature,
IConfiguredSite csite,
Hashtable table,
IFeatureReference[] optionalFeatures)
throws CoreException {
addEntry(feature, csite, table);
IIncludedFeatureReference[] irefs =
feature.getIncludedFeatureReferences();
for (int i = 0; i < irefs.length; i++) {
IIncludedFeatureReference iref = irefs[i];
boolean add = true;
if (iref.isOptional() && optionalFeatures != null) {
boolean found = false;
for (int j = 0; j < optionalFeatures.length; j++) {
IFeatureReference checked = optionalFeatures[j];
if (checked.equals(iref)) {
found = true;
break;
}
}
add = found;
}
if (add)
computeNewFeature(
iref.getFeature(null),
csite,
table,
optionalFeatures);
}
}
private static void addEntry(
IFeature feature,
IConfiguredSite csite,
Hashtable featureTable) {
String id = feature.getVersionedIdentifier().getIdentifier();
ArrayList entries = (ArrayList) featureTable.get(id);
if (entries == null) {
entries = new ArrayList();
featureTable.put(id, entries);
}
IdEntry entry = new IdEntry(feature, csite);
boolean replaced = false;
for (int i = 0; i < entries.size(); i++) {
IdEntry existingEntry = (IdEntry) entries.get(i);
IConfiguredSite existingSite = existingEntry.getConfiguredSite();
if (existingSite.equals(entry.getConfiguredSite())) {
// same site - replace it if not new
if (entry.isInstallCandidate()) {
entries.set(i, entry);
entries.remove(existingEntry);
}
replaced = true;
break;
}
}
if (!replaced)
entries.add(entry);
}
}