All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.gerrit.server.patch.GitPositionTransformer Maven / Gradle / Ivy

There is a newer version: 3.11.0
Show newest version
// Copyright (C) 2020 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package com.google.gerrit.server.patch;

import static com.google.common.collect.Comparators.emptiesFirst;
import static com.google.common.collect.ImmutableList.toImmutableList;
import static com.google.common.collect.ImmutableSet.toImmutableSet;
import static java.util.Comparator.comparing;
import static java.util.stream.Collectors.collectingAndThen;
import static java.util.stream.Collectors.groupingBy;

import com.google.auto.value.AutoValue;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Sets;
import com.google.common.collect.Streams;
import java.util.Collection;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.function.BiFunction;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.stream.Stream;

/**
 * Transformer of {@link Position}s in one Git tree to {@link Position}s in another Git tree given
 * the {@link Mapping}s between the trees.
 *
 * 

The base idea is that a {@link Position} in the source tree can be translated/mapped to a * corresponding {@link Position} in the target tree when we know how the target tree changed * compared to the source tree. As long as {@link Position}s are only defined via file path and line * range, we only need to know which file path in the source tree corresponds to which file path in * the target tree and how the lines within that file changed from the source to the target tree. * *

The algorithm is roughly: * *

    *
  1. Go over all positions and replace the file path for each of them with the corresponding one * in the target tree. If a file path maps to two file paths in the target tree (copied file), * duplicate the position entry and use each of the new file paths with it. If a file path * maps to no file in the target tree (deleted file), apply the specified conflict strategy * (e.g. drop position completely or map to next best guess). *
  2. Per file path, go through the file from top to bottom and keep track of how the range * mappings for that file shift the lines. Derive the shifted amount by comparing the number * of lines between source and target in the range mapping. While going through the file, * shift each encountered position by the currently tracked amount. If a position overlaps * with the lines of a range mapping, apply the specified conflict strategy (e.g. drop * position completely or map to next best guess). *
*/ public class GitPositionTransformer { private final PositionConflictStrategy positionConflictStrategy; /** * Creates a new {@code GitPositionTransformer} which uses the specified strategy for conflicts. */ public GitPositionTransformer(PositionConflictStrategy positionConflictStrategy) { this.positionConflictStrategy = positionConflictStrategy; } /** * Transforms the {@link Position}s of the specified entities as indicated via the {@link * Mapping}s. * *

This is typically used to transform the {@link Position}s in one Git tree (source) to the * corresponding {@link Position}s in another Git tree (target). The {@link Mapping}s need to * indicate all relevant changes between the source and target tree. {@link Mapping}s for files * not referenced by the given {@link Position}s need not be specified. They can be included, * though, as they aren't harmful. * * @param entities the entities whose {@link Position} should be mapped to the target tree * @param mappings the mappings describing all relevant changes between the source and the target * tree * @param an entity which has a {@link Position} * @return a list of entities with transformed positions. There are no guarantees about the order * of the returned elements. */ public ImmutableList> transform( Collection> entities, Set mappings) { // Update the file paths first as copied files might exist. For copied files, this operation // will duplicate the PositionedEntity instances of the original file. List> filePathUpdatedEntities = updateFilePaths(entities, mappings); return shiftRanges(filePathUpdatedEntities, mappings); } private ImmutableList> updateFilePaths( Collection> entities, Set mappings) { Map> newFilesPerOldFile = groupNewFilesByOldFiles(mappings); return entities.stream() .flatMap(entity -> mapToNewFileIfChanged(newFilesPerOldFile, entity)) .collect(toImmutableList()); } private static Map> groupNewFilesByOldFiles(Set mappings) { return mappings.stream() .map(Mapping::file) // Ignore file additions (irrelevant for mappings). .filter(mapping -> mapping.oldPath().isPresent()) .collect( groupingBy( mapping -> mapping.oldPath().orElse(""), collectingAndThen( Collectors.mapping(FileMapping::newPath, toImmutableSet()), // File deletion (empty Optional) -> empty set. GitPositionTransformer::unwrapOptionals))); } private static ImmutableSet unwrapOptionals(ImmutableSet> optionals) { return optionals.stream().flatMap(Streams::stream).collect(toImmutableSet()); } private Stream> mapToNewFileIfChanged( Map> newFilesPerOldFile, PositionedEntity entity) { if (!entity.position().filePath().isPresent()) { // No mapping of file paths necessary if no file path is set. -> Keep existing entry. return Stream.of(entity); } String oldFilePath = entity.position().filePath().get(); if (!newFilesPerOldFile.containsKey(oldFilePath)) { // Unchanged files don't have a mapping. -> Keep existing entries. return Stream.of(entity); } Set newFiles = newFilesPerOldFile.get(oldFilePath); if (newFiles.isEmpty()) { // File was deleted. return Streams.stream( positionConflictStrategy.getOnFileConflict(entity.position()).map(entity::withPosition)); } return newFiles.stream().map(entity::withFilePath); } private ImmutableList> shiftRanges( List> filePathUpdatedEntities, Set mappings) { Map> mappingsPerNewFilePath = groupRangeMappingsByNewFilePath(mappings); return Stream.concat( // Keep positions without a file. filePathUpdatedEntities.stream() .filter(entity -> !entity.position().filePath().isPresent()), // Shift ranges per file. groupByFilePath(filePathUpdatedEntities).entrySet().stream() .flatMap( newFilePathAndEntities -> shiftRangesInOneFileIfChanged( mappingsPerNewFilePath, newFilePathAndEntities.getKey(), newFilePathAndEntities.getValue()) .stream())) .collect(toImmutableList()); } private static Map> groupRangeMappingsByNewFilePath( Set mappings) { return mappings.stream() // Ignore range mappings of deleted files. .filter(mapping -> mapping.file().newPath().isPresent()) .collect( groupingBy( mapping -> mapping.file().newPath().orElse(""), collectingAndThen( Collectors.>reducing( new HashSet<>(), Mapping::ranges, Sets::union), ImmutableSet::copyOf))); } private static Map>> groupByFilePath( List> fileUpdatedEntities) { return fileUpdatedEntities.stream() .filter(entity -> entity.position().filePath().isPresent()) .collect(groupingBy(entity -> entity.position().filePath().orElse(""), toImmutableList())); } private ImmutableList> shiftRangesInOneFileIfChanged( Map> mappingsPerNewFilePath, String newFilePath, ImmutableList> sameFileEntities) { ImmutableSet sameFileRangeMappings = mappingsPerNewFilePath.getOrDefault(newFilePath, ImmutableSet.of()); if (sameFileRangeMappings.isEmpty()) { // Unchanged files and pure renames/copies don't have range mappings. -> Keep existing // entries. return sameFileEntities; } return shiftRangesInOneFile(sameFileEntities, sameFileRangeMappings); } private ImmutableList> shiftRangesInOneFile( List> sameFileEntities, Set sameFileRangeMappings) { ImmutableList> sortedEntities = sortByStartEnd(sameFileEntities); ImmutableList sortedMappings = sortByOldStartEnd(sameFileRangeMappings); int shiftedAmount = 0; int mappingIndex = 0; int entityIndex = 0; ImmutableList.Builder> resultingEntities = ImmutableList.builderWithExpectedSize(sortedEntities.size()); while (entityIndex < sortedEntities.size() && mappingIndex < sortedMappings.size()) { PositionedEntity entity = sortedEntities.get(entityIndex); if (entity.position().lineRange().isPresent()) { Range range = entity.position().lineRange().get(); RangeMapping mapping = sortedMappings.get(mappingIndex); if (mapping.oldLineRange().end() <= range.start()) { shiftedAmount = mapping.newLineRange().end() - mapping.oldLineRange().end(); mappingIndex++; } else if (range.end() <= mapping.oldLineRange().start()) { resultingEntities.add(entity.shiftPositionBy(shiftedAmount)); entityIndex++; } else { positionConflictStrategy .getOnRangeConflict(entity.position()) .map(entity::withPosition) .ifPresent(resultingEntities::add); entityIndex++; } } else { // No range -> no need to shift. resultingEntities.add(entity); entityIndex++; } } for (int i = entityIndex; i < sortedEntities.size(); i++) { resultingEntities.add(sortedEntities.get(i).shiftPositionBy(shiftedAmount)); } return resultingEntities.build(); } private static ImmutableList> sortByStartEnd( List> entities) { return entities.stream() .sorted( comparing( entity -> entity.position().lineRange(), emptiesFirst(comparing(Range::start).thenComparing(Range::end)))) .collect(toImmutableList()); } private static ImmutableList sortByOldStartEnd(Set mappings) { return mappings.stream() .sorted( comparing( RangeMapping::oldLineRange, comparing(Range::start).thenComparing(Range::end))) .collect(toImmutableList()); } /** * A mapping from a {@link Position} in one Git commit/tree (source) to a {@link Position} in * another Git commit/tree (target). */ @AutoValue public abstract static class Mapping { /** A mapping describing how the attributes of one file are mapped from source to target. */ public abstract FileMapping file(); /** * Mappings describing how line ranges within the file indicated by {@link #file()} are mapped * from source to target. */ public abstract ImmutableSet ranges(); public static Mapping create(FileMapping fileMapping, Iterable rangeMappings) { return new AutoValue_GitPositionTransformer_Mapping( fileMapping, ImmutableSet.copyOf(rangeMappings)); } } /** * A mapping of attributes from a file in one Git tree (source) to a file in another Git tree * (target). * *

At the moment, only the file path is considered. Other attributes like file mode would be * imaginable too but are currently not supported. */ @AutoValue public abstract static class FileMapping { /** File path in the source tree. For file additions, this is an empty {@link Optional}. */ public abstract Optional oldPath(); /** * File path in the target tree. Can be the same as {@link #oldPath()} if unchanged. For file * deletions, this is an empty {@link Optional}. */ public abstract Optional newPath(); /** * Creates a {@link FileMapping} for a file addition. * *

In the context of {@link GitPositionTransformer}, file additions are irrelevant as no * given position in the source tree can refer to such a new file in the target tree. We still * provide this factory method so that code outside of {@link GitPositionTransformer} doesn't * have to care about such details and can simply create {@link FileMapping}s for any * modifications between the trees. */ public static FileMapping forAddedFile(String filePath) { return new AutoValue_GitPositionTransformer_FileMapping( Optional.empty(), Optional.of(filePath)); } /** Creates a {@link FileMapping} for a file deletion. */ public static FileMapping forDeletedFile(String filePath) { return new AutoValue_GitPositionTransformer_FileMapping( Optional.of(filePath), Optional.empty()); } /** Creates a {@link FileMapping} for a file modification. */ public static FileMapping forModifiedFile(String filePath) { return new AutoValue_GitPositionTransformer_FileMapping( Optional.of(filePath), Optional.of(filePath)); } /** Creates a {@link FileMapping} for a file renaming. */ public static FileMapping forRenamedFile(String oldPath, String newPath) { return new AutoValue_GitPositionTransformer_FileMapping( Optional.of(oldPath), Optional.of(newPath)); } } /** * A mapping of a line range in one Git tree (source) to the corresponding line range in another * Git tree (target). */ @AutoValue public abstract static class RangeMapping { /** Range in the source tree. */ public abstract Range oldLineRange(); /** Range in the target tree. */ public abstract Range newLineRange(); /** * Creates a new {@code RangeMapping}. * * @param oldRange see {@link #oldLineRange()} * @param newRange see {@link #newLineRange()} */ public static RangeMapping create(Range oldRange, Range newRange) { return new AutoValue_GitPositionTransformer_RangeMapping(oldRange, newRange); } } /** * A position within the tree of a Git commit. * *

The term 'position' is our own invention. The underlying idea is that a Gerrit comment is at * a specific position within the commit of a patchset. That position is defined by the attributes * defined in this class. * *

The same thinking can be applied to diff hunks (= JGit edits). Each diff hunk maps a * position in one commit (e.g. in the parent of the patchset) to a position in another commit * (e.g. in the commit of the patchset). * *

We only refer to lines and not character offsets within the lines here as Git only works * with line precision. In theory, we could do better in Gerrit as we also have intraline diffs. * Incorporating those requires careful considerations, though. */ @AutoValue public abstract static class Position { /** Absolute file path. */ public abstract Optional filePath(); /** * Affected lines. An empty {@link Optional} indicates that this position does not refer to any * specific lines (e.g. used for a file comment). */ public abstract Optional lineRange(); /** * Creates a copy of this {@code Position} whose range is shifted by the indicated amount. * *

Note: There's no guarantee that this method returns a new instance. * * @param amount number of lines to shift. Negative values mean moving the range up, positive * values mean moving the range down. * @return a {@code Position} instance with the updated range */ public Position shiftBy(int amount) { return lineRange() .map(range -> toBuilder().lineRange(range.shiftBy(amount)).build()) .orElse(this); } /** * Creates a copy of this {@code Position} which doesn't refer to any specific lines. * *

Note: There's no guarantee that this method returns a new instance. * * @return a {@code Position} instance without a line range */ public Position withoutLineRange() { return toBuilder().lineRange(Optional.empty()).build(); } /** * Creates a copy of this {@code Position} whose file path is adjusted to the indicated value. * *

Note: There's no guarantee that this method returns a new instance. * * @param filePath the new file path to use * @return a {@code Position} instance with the indicated file path */ public Position withFilePath(String filePath) { return toBuilder().filePath(filePath).build(); } abstract Builder toBuilder(); public static Builder builder() { return new AutoValue_GitPositionTransformer_Position.Builder(); } /** Builder of a {@link Position}. */ @AutoValue.Builder public abstract static class Builder { /** See {@link #filePath()}. */ public abstract Builder filePath(String filePath); /** See {@link #lineRange()}. */ public abstract Builder lineRange(Range lineRange); /** See {@link #lineRange()}. */ public abstract Builder lineRange(Optional lineRange); public abstract Position build(); } } /** A range. In the context of {@link GitPositionTransformer}, this is a line range. */ @AutoValue public abstract static class Range { /** Start of the range. (inclusive) */ public abstract int start(); /** End of the range. (exclusive) */ public abstract int end(); /** * Creates a copy of this {@code Range} which is shifted by the indicated amount. A shift * equally applies to both {@link #start()} end {@link #end()}. * *

Note: There's no guarantee that this method returns a new instance. * * @param amount amount to shift. Negative values mean moving the range up, positive values mean * moving the range down. * @return a {@code Range} instance with updated start/end */ public Range shiftBy(int amount) { return create(start() + amount, end() + amount); } public static Range create(int start, int end) { return new AutoValue_GitPositionTransformer_Range(start, end); } } /** * Wrapper around an instance of {@code T} which annotates it with a {@link Position}. Methods * such as {@link #shiftPositionBy(int)} and {@link #withFilePath(String)} allow to update the * associated {@link Position}. Afterwards, use {@link #getEntityAtUpdatedPosition()} to get an * updated version of the {@code T} instance. * * @param an object/entity type which has a {@link Position} */ public static class PositionedEntity { private final T entity; private final Position position; private final BiFunction updatedEntityCreator; /** * Creates a new {@code PositionedEntity}. * * @param entity an instance which should be annotated with a {@link Position} * @param positionExtractor a function describing how a {@link Position} can be derived from the * given entity * @param updatedEntityCreator a function to create a new entity of type {@code T} from an * existing entity and a given {@link Position}. This must return a new instance of type * {@code T}! The existing instance must not be modified! * @param an object/entity type which has a {@link Position} */ public static PositionedEntity create( T entity, Function positionExtractor, BiFunction updatedEntityCreator) { Position position = positionExtractor.apply(entity); return new PositionedEntity<>(entity, position, updatedEntityCreator); } private PositionedEntity( T entity, Position position, BiFunction updatedEntityCreator) { this.entity = entity; this.position = position; this.updatedEntityCreator = updatedEntityCreator; } /** * Returns an updated version of the entity to which the internally stored {@link Position} was * written back to. * * @return an updated instance of {@code T} */ public T getEntityAtUpdatedPosition() { return updatedEntityCreator.apply(entity, position); } Position position() { return position; } /** * Shifts the tracked {@link Position} by the specified amount. * * @param amount number of lines to shift. Negative values mean moving the range up, positive * values mean moving the range down. * @return a {@code PositionedEntity} with updated {@link Position} */ public PositionedEntity shiftPositionBy(int amount) { return new PositionedEntity<>(entity, position.shiftBy(amount), updatedEntityCreator); } /** * Updates the file path of the tracked {@link Position}. * * @param filePath the new file path to use * @return a {@code PositionedEntity} with updated {@link Position} */ public PositionedEntity withFilePath(String filePath) { return new PositionedEntity<>(entity, position.withFilePath(filePath), updatedEntityCreator); } /** * Updates the tracked {@link Position}. * * @return a {@code PositionedEntity} with updated {@link Position} */ public PositionedEntity withPosition(Position newPosition) { return new PositionedEntity<>(entity, newPosition, updatedEntityCreator); } } /** * Strategy indicating how to handle {@link Position}s for which mapping conflicts exist. A * mapping conflict means that a {@link Position} can't be transformed such that it still refers * to exactly the same commit content afterwards. * *

Example: A {@link Position} refers to file foo.txt and lines 5-6 which contain the text * "Line 5\nLine 6". One of the {@link Mapping}s given to {@link #transform(Collection, Set)} * indicates that line 5 of foo.txt was modified to "Line five\nLine 5.1\n". We could derive a * transformed {@link Position} (foo.txt, lines 5-7) but that {@link Position} would then refer to * the content "Line five\nLine 5.1\nLine 6". If the modification started already in line 4, we * could even only guess what the transformed {@link Position} would be. */ public interface PositionConflictStrategy { /** * Determines an alternate {@link Position} when the range of the position can't be mapped * without a conflict. * * @param oldPosition position in the source tree * @return the new {@link Position} or an empty {@link Optional} if the position should be * dropped */ Optional getOnRangeConflict(Position oldPosition); /** * Determines an alternate {@link Position} when there is no file for the position (= file * deletion) in the target tree. * * @param oldPosition position in the source tree * @return the new {@link Position} or an empty {@link Optional} if the position should be * * dropped */ Optional getOnFileConflict(Position oldPosition); } /** * A strategy which drops any {@link Position}s on a conflicting mapping. Such a strategy is * useful if it's important that any mapped {@link Position} still refers to exactly the same * commit content as before. See more details at {@link PositionConflictStrategy}. * *

We need this strategy for computing edits due to rebase. */ public enum OmitPositionOnConflict implements PositionConflictStrategy { INSTANCE; @Override public Optional getOnRangeConflict(Position oldPosition) { return Optional.empty(); } @Override public Optional getOnFileConflict(Position oldPosition) { return Optional.empty(); } } /** * A strategy which tries to select the next suitable {@link Position} on a conflicting mapping. * At the moment, this strategy is very basic and only defers to the next higher level (e.g. range * unclear -> drop range but keep file reference). This could be improved in the future. * *

We need this strategy for ported comments. * *

Warning: With this strategy, mapped {@link Position}s are not guaranteed to * refer to exactly the same commit content as before. See more details at {@link * PositionConflictStrategy}. * *

Contract: This strategy will never drop any {@link Position}. */ public enum BestPositionOnConflict implements PositionConflictStrategy { INSTANCE; @Override public Optional getOnRangeConflict(Position oldPosition) { return Optional.of(oldPosition.withoutLineRange()); } @Override public Optional getOnFileConflict(Position oldPosition) { // If there isn't a target file, we can also drop any ranges. return Optional.of(Position.builder().build()); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy