blob: d4eb22d97889d07533fcd0fe2489aa29364ab951 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2007, 2018 compeople AG 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:
* compeople AG (Stefan Liebig) - initial API and implementation
* IBM Corporation - adaptation to JAR deltas and on-going development
*******************************************************************************/
package org.eclipse.equinox.internal.p2.artifact.optimizers.jardelta;
import java.io.IOException;
import java.io.OutputStream;
import java.util.*;
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.NullProgressMonitor;
import org.eclipse.equinox.internal.p2.metadata.ArtifactKey;
import org.eclipse.equinox.internal.provisional.p2.artifact.repository.processing.ProcessingStep;
import org.eclipse.equinox.internal.provisional.p2.artifact.repository.processing.ProcessingStepHandler;
import org.eclipse.equinox.p2.core.ProvisionException;
import org.eclipse.equinox.p2.metadata.IArtifactKey;
import org.eclipse.equinox.p2.metadata.Version;
import org.eclipse.equinox.p2.query.IQueryResult;
import org.eclipse.equinox.p2.repository.artifact.*;
import org.eclipse.equinox.p2.repository.artifact.spi.ArtifactDescriptor;
import org.eclipse.equinox.p2.repository.artifact.spi.ProcessingStepDescriptor;
public class Optimizer {
private final IArtifactRepository repository;
private final int width;
private final int depth;
private static final String JAR_DELTA_FORMAT = "jarDelta"; //$NON-NLS-1$
private static final String JAR_DELTA_PATCH_STEP = "org.eclipse.equinox.p2.processing.JarDeltaPatchStep"; //$NON-NLS-1$
private static final Comparator<IArtifactDescriptor> ARTIFACT_DESCRIPTOR_VERSION_COMPARATOR = new ArtifactDescriptorVersionComparator();
private static final Comparator<IArtifactKey> ARTIFACT_KEY_VERSION_COMPARATOR = new ArtifactKeyVersionComparator();
/**
* This optimizer performs delta generation based on (currently) jbdiff. The
* optimization can be controlled with the ´width´ and the ´depth´ parameter.
* ´width´ defines for how many ´related´ artifact keys a delta should be
* generated, starting from the most up-to-date. ´depth´ defines to how many
* predecessor a delta should be generated.
*
* With AK(c-v) : AK - artifact key, c - artifact id, v - artifact version the
* ´repository content´ can be viewed a two dimensional array, where the
* artifact keys for the same component are in order of their version:
*
* <pre>
* <code>
* w=1 w=2
* | |
* | +------.------------+ d=2
* | | +----.---+ d=1 |
* | | | | | v
* [ v | | v v v
* [ AK(x,2.0) AK(x,1.5) AK(x,1.1) ]
* [ AK(y,2.0) AK(y,1.9) ]
* [ AK(z,2.0) AK(z,1.5) AK(z,1.3) AK(z,1.0) ]
* ]
* </code>
* </pre>
*
* E.g: with a ´width´ of one and a ´depth´ of two the optimizer would create
* two deltas for component ´x´ from 1.5 to 2.0 and from 1.1 to 2.0.
*
* @param repository
* @param width
* @param depth
*/
public Optimizer(IArtifactRepository repository, int width, int depth) {
this.repository = repository;
this.width = width;
this.depth = depth;
}
public void run() {
System.out.println("Starting delta (jardelta) optimizations (width=" + width + ", depth=" + depth + ")"); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
IQueryResult<IArtifactKey> queryResult = repository.query(ArtifactKeyQuery.ALL_KEYS, null);
IArtifactKey[][] keys = getSortedRelatedArtifactKeys(queryResult);
for (IArtifactKey[] key2 : keys) {
if (key2.length < 2)
// Nothing to diff here!
continue;
int minWidth = Math.min(width, key2.length);
for (int j = 0; j < minWidth; j++) {
IArtifactKey key = key2[j];
optimize(key2, key);
}
}
System.out.println("Done."); //$NON-NLS-1$
}
private void optimize(IArtifactKey[] keys, IArtifactKey key) {
IArtifactDescriptor[] descriptors = repository.getArtifactDescriptors(key);
IArtifactDescriptor canonical = null;
for (IArtifactDescriptor descriptor : descriptors) {
boolean optimized = false;
if (isCanonical(descriptor))
canonical = descriptor;
else
optimized |= isOptimized(descriptor);
if (!optimized)
optimize(canonical, keys);
}
}
private IArtifactKey getVersionlessKey(IArtifactKey key) {
return new ArtifactKey(key.getClassifier(), key.getId(), Version.emptyVersion);
}
/**
* This method retrieves a list of list of IArtifactKeys. The artifact keys in
* the list of artifact keys are all ´strongly related´ to each other such that
* are equal but not considering the versions. This list is sorted such that the
* newer versions are first in the list.
* <p>
* With AK(c-v) : AK - artifact key, c - artifact id, v - artifact version the
* result is than, e.g.
*
* <pre>
* <code>
* [
* [ AK(x,2.0) AK(x,1.5) AK(x,1.1) ]
* [ AK(y,2.0) AK(y,1.9) ]
* [ AK(z,2.0) AK(z,1.5) AK(z,1.3) AK(z,1.0) ]
* ]
* </code>
* </pre>
*
* @param artifactKeys
* @return the sorted artifact keys
*/
private IArtifactKey[][] getSortedRelatedArtifactKeys(IQueryResult<IArtifactKey> artifactKeys) {
Map<IArtifactKey, List<IArtifactKey>> map = new HashMap<>();
for (IArtifactKey nxt : artifactKeys) {
IArtifactKey freeKey = getVersionlessKey(nxt);
List<IArtifactKey> values = map.get(freeKey);
if (values == null) {
values = new ArrayList<>();
map.put(freeKey, values);
}
values.add(nxt);
}
IArtifactKey[][] lists = new IArtifactKey[map.size()][];
int i = 0;
for (List<IArtifactKey> artifactKeyList : map.values()) {
IArtifactKey[] relatedArtifactKeys = artifactKeyList.toArray(new IArtifactKey[artifactKeyList.size()]);
Arrays.sort(relatedArtifactKeys, ARTIFACT_KEY_VERSION_COMPARATOR);
lists[i++] = relatedArtifactKeys;
}
int candidates = 0;
for (IArtifactKey[] list : lists) {
for (IArtifactKey element : list) {
System.out.println(element + ", "); //$NON-NLS-1$
}
System.out.println(""); //$NON-NLS-1$
if (list.length > 1)
candidates++;
}
System.out.println("Candidates found: " + candidates); //$NON-NLS-1$
return lists;
}
private void optimize(IArtifactDescriptor canonical, IArtifactKey[] relatedArtifactKeys) {
System.out.println("Optimizing " + canonical); //$NON-NLS-1$
IArtifactDescriptor[] descriptors = getSortedCompletePredecessors(canonical.getArtifactKey(),
relatedArtifactKeys);
int minDepth = Math.min(depth, descriptors.length);
for (int i = 0; i < minDepth; i++) {
System.out.println("\t with jar delta against " + descriptors[i].getArtifactKey()); //$NON-NLS-1$
String predecessorData = descriptors[i].getArtifactKey().toExternalForm();
ArtifactDescriptor newDescriptor = new ArtifactDescriptor(canonical);
IProcessingStepDescriptor patchStep = new ProcessingStepDescriptor(JAR_DELTA_PATCH_STEP, predecessorData,
true);
IProcessingStepDescriptor[] steps = new IProcessingStepDescriptor[] { patchStep };
newDescriptor.setProcessingSteps(steps);
newDescriptor.setProperty(IArtifactDescriptor.FORMAT, JAR_DELTA_FORMAT);
try (OutputStream repositoryStream = repository.getOutputStream(newDescriptor)) {
// Add in all the processing steps needed to optimize (e.g., pack200, ...)
ProcessingStep optimizerStep = new JarDeltaOptimizerStep(repository);
optimizerStep.initialize(repository.getProvisioningAgent(), patchStep, newDescriptor);
ProcessingStepHandler handler = new ProcessingStepHandler();
OutputStream destination = handler.link(new ProcessingStep[] { optimizerStep }, repositoryStream, null);
// Do the actual work by asking the repo to get the artifact and put it in the
// destination.
IStatus status = repository.getArtifact(canonical, destination, new NullProgressMonitor());
if (!status.isOK()) {
System.out.println("Getting the artifact is not ok."); //$NON-NLS-1$
System.out.println(status);
}
repositoryStream.close();
status = ProcessingStepHandler.checkStatus(repositoryStream);
if (!status.isOK()) {
System.out.println("Skipping optimization of: " + descriptors[i].getArtifactKey()); //$NON-NLS-1$
System.out.println(status.toString());
}
} catch (ProvisionException e) {
System.out.println("Skipping optimization of: " + descriptors[i].getArtifactKey()); //$NON-NLS-1$
System.out.println(e.getMessage());
e.printStackTrace();
} catch (IOException e) {
System.out.println("Skipping optimization of: " + descriptors[i].getArtifactKey()); //$NON-NLS-1$
System.out.println(e.getMessage());
e.printStackTrace();
}
}
}
private IArtifactDescriptor[] getSortedCompletePredecessors(IArtifactKey artifactKey,
IArtifactKey[] relatedArtifactKeys) {
// get all artifact keys
List<IArtifactDescriptor> completeDescriptors = new ArrayList<>(relatedArtifactKeys.length);
for (IArtifactKey relatedArtifactKey : relatedArtifactKeys) {
// if we find ´our self´ skip
if (relatedArtifactKey.equals(artifactKey))
continue;
// look for a complete artifact descriptor of the current key
IArtifactDescriptor[] descriptors = repository.getArtifactDescriptors(relatedArtifactKey);
for (IArtifactDescriptor descriptor : descriptors) {
if (isCanonical(descriptor)) {
completeDescriptors.add(descriptor);
break;
}
}
}
IArtifactDescriptor[] completeSortedDescriptors = completeDescriptors
.toArray(new IArtifactDescriptor[completeDescriptors.size()]);
// Sort, so to allow a depth lookup!
Arrays.sort(completeSortedDescriptors, ARTIFACT_DESCRIPTOR_VERSION_COMPARATOR);
return completeSortedDescriptors;
}
private boolean isOptimized(IArtifactDescriptor descriptor) {
if (descriptor.getProcessingSteps().length != 1)
return false;
return JAR_DELTA_FORMAT.equals(descriptor.getProperty(IArtifactDescriptor.FORMAT));
}
private boolean isCanonical(IArtifactDescriptor descriptor) {
// TODO length != 0 is not necessarily an indicator for not being canonical!
return descriptor.getProcessingSteps().length == 0;
}
static final class ArtifactDescriptorVersionComparator implements Comparator<IArtifactDescriptor> {
@Override
public int compare(IArtifactDescriptor artifactDescriptor0, IArtifactDescriptor artifactDescriptor1) {
return -1 * artifactDescriptor0.getArtifactKey().getVersion()
.compareTo(artifactDescriptor1.getArtifactKey().getVersion());
}
}
static final class ArtifactKeyVersionComparator implements Comparator<IArtifactKey> {
@Override
public int compare(IArtifactKey artifactKey0, IArtifactKey artifactKey1) {
return -1 * artifactKey0.getVersion().compareTo(artifactKey1.getVersion());
}
}
}