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

net.nemerosa.ontrack.model.structure.SyncPolicy Maven / Gradle / Ivy

package net.nemerosa.ontrack.model.structure;

import lombok.Data;
import net.nemerosa.ontrack.model.exceptions.SyncTargetItemPresentException;
import net.nemerosa.ontrack.model.exceptions.SyncTargetItemUnknownException;

import java.util.Collection;
import java.util.HashSet;
import java.util.Map;
import java.util.function.Function;
import java.util.stream.Collectors;

/**
 * Behaviour to adopt when synchronising two lists.
 */
@Data
public class SyncPolicy {

    /**
     * Target already present.
     * 

*

    *
  • Can be ignored
  • *
  • Can be replaced
  • *
  • Can raise an error
  • *
*/ private final TargetPresentPolicy targetPresentPolicy; /** * Unknown target. *

*

    *
  • Can be ignored
  • *
  • Can be deleted
  • *
  • Can raise an error
  • *
*/ private final UnknownTargetPolicy unknownTargetPolicy; /** * This Sync policy is conservative, making sure existing items in the target are kept as they were. */ public static final SyncPolicy COPY = new SyncPolicy(TargetPresentPolicy.IGNORE, UnknownTargetPolicy.IGNORE); /** * This Sync policy makes sure the source and the target are exactly aligned. Any existing item in the target * which is not in the source will be deleted. */ public static final SyncPolicy SYNC = new SyncPolicy(TargetPresentPolicy.REPLACE, UnknownTargetPolicy.DELETE); /** * This Sync policy makes sure the source and the target common elements are synchronised. Any existing item in the * target is kept if not existing in the source. */ public static final SyncPolicy SYNC_KEEP = new SyncPolicy(TargetPresentPolicy.REPLACE, UnknownTargetPolicy.IGNORE); public static enum TargetPresentPolicy { IGNORE, REPLACE, ERROR } public enum UnknownTargetPolicy { IGNORE, DELETE, ERROR } public SyncResult sync(SyncConfig config) { SyncResult result = SyncResult.empty(); // Gets the source items Collection sourceItems = config.getSourceItems(); // Index of target items Map targetMap = config.getTargetItems().stream().collect( Collectors.toMap( config::getItemId, Function.identity() ) ); // For each source item for (T sourceItem : sourceItems) { // Gets its identifier for the target items D itemId = config.getItemId(sourceItem); // Gets a corresponding item T targetItem = targetMap.get(itemId); // If present if (targetItem != null && config.isTargetItemPresent(targetItem)) { // This depends on the policy switch (targetPresentPolicy) { case IGNORE: result.ignorePresentTarget(); break; case REPLACE: config.replaceTargetItem(sourceItem, targetItem); result.replacePresentTarget(); break; case ERROR: default: throw new SyncTargetItemPresentException(config.getItemType(), itemId); } } // If not present, creates it else { config.createTargetItem(sourceItem); result.create(); } } // Now that target items have been either created or updated, we need to know which ones were // not matched with the sources targetMap = config.getTargetItems().stream().collect( Collectors.toMap( config::getItemId, Function.identity() ) ); Collection targetIds = new HashSet<>(targetMap.keySet()); Collection sourceIds = sourceItems.stream().map(config::getItemId).collect(Collectors.toList()); targetIds.removeAll(sourceIds); int unknownTargetNumber = targetIds.size(); if (unknownTargetNumber > 0) { switch (unknownTargetPolicy) { case IGNORE: result.setUnknownTargetIgnored(unknownTargetNumber); break; case DELETE: for (D targetId : targetIds) { T targetItem = targetMap.get(targetId); config.deleteTargetItem(targetItem); } result.setUnknownTargetDeleted(unknownTargetNumber); break; case ERROR: default: throw new SyncTargetItemUnknownException(config.getItemType(), targetIds); } } // OK return result; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy