AuditDiffService.java

  1. package no.nav.data.team.notify;

  2. import lombok.Value;
  3. import lombok.extern.slf4j.Slf4j;
  4. import no.nav.data.common.auditing.domain.Action;
  5. import no.nav.data.common.auditing.domain.AuditVersionRepository;
  6. import no.nav.data.common.auditing.dto.AuditMetadata;
  7. import no.nav.data.team.notify.domain.Notification;
  8. import no.nav.data.team.notify.domain.Notification.NotificationChannel;
  9. import no.nav.data.team.notify.domain.Notification.NotificationType;
  10. import no.nav.data.team.notify.domain.NotificationTask;
  11. import no.nav.data.team.notify.domain.NotificationTask.AuditTarget;
  12. import no.nav.data.team.notify.domain.TeamAuditMetadata;
  13. import org.springframework.stereotype.Service;

  14. import java.util.ArrayList;
  15. import java.util.HashMap;
  16. import java.util.HashSet;
  17. import java.util.List;
  18. import java.util.Map;
  19. import java.util.Map.Entry;
  20. import java.util.Set;
  21. import java.util.UUID;
  22. import java.util.stream.Stream;

  23. import static java.util.stream.Collectors.groupingBy;
  24. import static java.util.stream.Collectors.toList;
  25. import static no.nav.data.common.utils.StreamUtils.convert;
  26. import static no.nav.data.common.utils.StreamUtils.filter;
  27. import static no.nav.data.common.utils.StreamUtils.find;
  28. import static no.nav.data.common.utils.StreamUtils.tryFind;
  29. import static no.nav.data.common.utils.StreamUtils.union;

  30. @Slf4j
  31. @Service
  32. public class AuditDiffService {

  33.     private final AuditVersionRepository auditVersionRepository;

  34.     public AuditDiffService(AuditVersionRepository auditVersionRepository) {
  35.         this.auditVersionRepository = auditVersionRepository;
  36.     }

  37.     public List<NotificationTask> createTask(List<AuditMetadata> audits, List<Notification> notifications) {
  38.         var allTasks = new ArrayList<NotificationTask>();
  39.         if (audits.isEmpty()) {
  40.             return allTasks;
  41.         }

  42.         var lastAudit = audits.get(audits.size() - 1);
  43.         var auditsStart = audits.get(0).getTime();
  44.         var auditsEnd = lastAudit.getTime();
  45.         log.info("Notification {} audits", audits.size());

  46.         var teamsPrev = auditVersionRepository.getTeamMetadataBefore(auditsStart);
  47.         var teamsCurr = auditVersionRepository.getTeamMetadataBetween(auditsStart, auditsEnd);
  48.         notifications = expandProductAreaNotifications(notifications, teamsPrev, teamsCurr);
  49.         var auditsByTargetId = audits.stream().collect(groupingBy(AuditMetadata::getTableId));
  50.         notifications.removeIf(n -> {
  51.                     boolean notAllEventNotification = n.getType() != NotificationType.ALL_EVENTS;
  52.                     boolean noAuditsForNotification = !auditsByTargetId.containsKey(n.getTarget());
  53.                     boolean noDependentAuditsForNotification = auditsByTargetId.keySet().stream().noneMatch(n::isDependentOn);
  54.                     return notAllEventNotification && noAuditsForNotification && noDependentAuditsForNotification;
  55.                 }
  56.         );
  57.         notifications.forEach(n -> {
  58.             if (n.getTarget() != null && !auditsByTargetId.containsKey(n.getTarget())) {
  59.                 log.info("Adding empty audits for target {}", n.getTarget());
  60.                 auditsByTargetId.put(n.getTarget(), List.of());
  61.             }
  62.         });

  63.         var notificationsByIdent = notifications.stream().collect(groupingBy(Notification::getIdent));
  64.         log.info("Notification for {}", notificationsByIdent.keySet());
  65.         for (Entry<String, List<Notification>> entry : notificationsByIdent.entrySet()) {
  66.             String ident = entry.getKey();
  67.             List<Notification> notificationsForIdent = entry.getValue();
  68.             var notificationTargetAudits = unpackAndGroupTargets(notificationsForIdent, auditsByTargetId);
  69.             var tasksForIdent = createTasks(ident, notificationTargetAudits);
  70.             allTasks.addAll(filter(tasksForIdent, t -> !t.getTargets().isEmpty()));
  71.         }
  72.         return allTasks;
  73.     }

  74.     private List<Notification> expandProductAreaNotifications(List<Notification> notifications, List<TeamAuditMetadata> teamsPrev, List<TeamAuditMetadata> teamsCurr) {
  75.         var allNotifications = new ArrayList<>(notifications);
  76.         for (Notification notification : notifications) {
  77.             if (notification.getType() == NotificationType.PA) {
  78.                 var paTeamsPrev = filter(teamsPrev, t -> notification.getTarget().equals(t.getProductAreaId()));
  79.                 var paTeamsCurr = filter(teamsCurr, t -> notification.getTarget().equals(t.getProductAreaId()));
  80.                 var allTeams = union(paTeamsPrev, paTeamsCurr).stream().map(TeamAuditMetadata::getTableId).distinct().collect(toList());
  81.                 // Adding teams from product area to notifications, setting their level as Product area, to enforce correct channel overrides later
  82.                 allNotifications.addAll(convert(allTeams, teamId -> Notification.builder()
  83.                         .type(NotificationType.PA)
  84.                         .time(notification.getTime())
  85.                         .ident(notification.getIdent())
  86.                         .channels(notification.getChannels())
  87.                         .target(teamId)
  88.                         .build()));
  89.                 notification.setDependentTargets(allTeams);
  90.                 log.info("Notification PA {} DependentTargets {}", notification.getTarget(), allTeams);
  91.             }
  92.         }
  93.         return allNotifications;
  94.     }

  95.     private List<NotificationTargetAudits> unpackAndGroupTargets(List<Notification> notifications, Map<UUID, List<AuditMetadata>> auditsByTargetId) {
  96.         var allEventAudits = new ArrayList<NotificationTargetAudits>();
  97.         // unpack ALL_EVENTS
  98.         tryFind(notifications, n -> n.getType() == NotificationType.ALL_EVENTS)
  99.                 .ifPresent(n -> allEventAudits.addAll(convert(auditsByTargetId.entrySet(), e -> new NotificationTargetAudits(n, e.getKey(), e.getValue()))));

  100.         var targetAudits = notifications.stream()
  101.                 .filter(n -> n.getType() != NotificationType.ALL_EVENTS)
  102.                 .map(n -> new NotificationTargetAudits(n, n.getTarget(), auditsByTargetId.get(n.getTarget())))
  103.                 .collect(toList());
  104.         return union(allEventAudits, targetAudits);
  105.     }

  106.     private List<NotificationTask> createTasks(String ident, List<NotificationTargetAudits> targetAudits) {
  107.         // All times are equal down here
  108.         var time = targetAudits.get(0).getNotification().getTime();

  109.         // Calculate which type is the most specific for each target.
  110.         // ie. if a a user has a team marked as a different channel than it's product area, we will use the teams channel settings for that target, same goes for ALL_EVENTS.
  111.         Map<UUID, NotificationType> targetTypes = new HashMap<>();
  112.         targetAudits.forEach(ta -> targetTypes.compute(ta.getTargetId(), (uuid, existingType) -> NotificationType.min(ta.getNotification().getType(), existingType)));

  113.         var classifications = Stream.of(NotificationChannel.values()).map(TargetClassification::new).collect(toList());

  114.         targetAudits.forEach(ta -> {
  115.             Notification notification = ta.getNotification();
  116.             UUID targetId = ta.getTargetId();

  117.             var notificationTypeForTarget = targetTypes.get(targetId);
  118.             var silent = notificationTypeForTarget != notification.getType();

  119.             filter(classifications, c -> c.matches(notification.getChannels()))
  120.                     .forEach(c -> c.add(new Target(targetId, ta.getAudits(), silent)));
  121.         });

  122.         return classifications.stream()
  123.                 .filter(c -> c.getTargets().stream().anyMatch(t -> !t.isSilent()))
  124.                 .map(classification ->
  125.                         NotificationTask.builder()
  126.                                 .ident(ident)
  127.                                 .time(time)
  128.                                 .channel(classification.getChannel())
  129.                                 .targets(convertAuditTargets(ident, classification.getTargets()))
  130.                                 .build()
  131.                 )
  132.                 .collect(toList());
  133.     }

  134.     private List<AuditTarget> convertAuditTargets(String ident, List<Target> targets) {
  135.         return convert(targets, target -> {
  136.             var targetId = target.getTarget();
  137.             var audits = target.getAudits();
  138.             AuditMetadata oldestAudit;
  139.             UUID prev;
  140.             UUID curr;
  141.             if (audits.isEmpty()) {
  142.                 // If the target in question has not actually changed, ie. a team added/removed in a product area
  143.                 oldestAudit = auditVersionRepository.lastAuditForObject(targetId);
  144.                 prev = oldestAudit.getId();
  145.                 curr = oldestAudit.getId();
  146.             } else {
  147.                 oldestAudit = audits.get(0);
  148.                 var newestAudit = audits.get(audits.size() - 1);
  149.                 prev = getPreviousFor(oldestAudit);
  150.                 curr = newestAudit.getAction() == Action.DELETE ? null : newestAudit.getId();
  151.             }
  152.             if (prev == null && curr == null) {
  153.                 log.info("Create and delete target {}, ignoring", targetId);
  154.                 return null;
  155.             }
  156.             var tableName = oldestAudit.getTableName();
  157.             log.info("Notification to {} target {}: {} from {} to {}", ident, tableName, targetId, prev, curr);
  158.             return AuditTarget.builder()
  159.                     .targetId(targetId)
  160.                     .type(tableName)
  161.                     .prevAuditId(prev)
  162.                     .currAuditId(curr)
  163.                     .silent(target.isSilent())
  164.                     .build();
  165.         });
  166.     }

  167.     private UUID getPreviousFor(AuditMetadata oldestAudit) {
  168.         if (oldestAudit.getAction() == Action.CREATE) {
  169.             return null;
  170.         }
  171.         return UUID.fromString(auditVersionRepository.getPreviousAuditIdFor(oldestAudit.getId()));
  172.     }

  173.     @Value
  174.     static class NotificationTargetAudits {

  175.         Notification notification;
  176.         UUID targetId;
  177.         List<AuditMetadata> audits;
  178.     }

  179.     @Value
  180.     static class TargetClassification {

  181.         NotificationChannel channel;
  182.         List<Target> targets = new ArrayList<>();
  183.         Set<UUID> targetsAdded = new HashSet<>();

  184.         TargetClassification(NotificationChannel channel) {
  185.             this.channel = channel;
  186.         }

  187.         boolean isAdded(UUID targetId) {
  188.             return targetsAdded.contains(targetId);
  189.         }

  190.         void add(Target target) {
  191.             if (isAdded(target.getTarget())) {
  192.                 if (!target.isSilent()) {
  193.                     var existingTarget = find(targets, t -> t.getTarget().equals(target.getTarget()));
  194.                     if (existingTarget.isSilent()) {
  195.                         targets.remove(existingTarget);
  196.                         targets.add(target);
  197.                     }
  198.                 }
  199.             } else {
  200.                 targetsAdded.add(target.getTarget());
  201.                 targets.add(target);
  202.             }
  203.         }

  204.         boolean matches(List<NotificationChannel> channels) {
  205.             return channels.contains(channel);
  206.         }
  207.     }

  208.     @Value
  209.     static class Target {

  210.         UUID target;
  211.         List<AuditMetadata> audits;
  212.         boolean silent;
  213.     }
  214. }