NotificationService.java

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

  2. import com.github.benmanes.caffeine.cache.Cache;
  3. import com.github.benmanes.caffeine.cache.Caffeine;
  4. import lombok.RequiredArgsConstructor;
  5. import lombok.extern.slf4j.Slf4j;
  6. import no.nav.data.common.auditing.domain.AuditVersionRepository;
  7. import no.nav.data.common.exceptions.NotFoundException;
  8. import no.nav.data.common.exceptions.ValidationException;
  9. import no.nav.data.common.mail.EmailService;
  10. import no.nav.data.common.mail.MailTask;
  11. import no.nav.data.common.security.SecurityUtils;
  12. import no.nav.data.common.storage.StorageService;
  13. import no.nav.data.common.utils.MetricUtils;
  14. import no.nav.data.team.contact.domain.ContactAddress;
  15. import no.nav.data.team.contact.domain.ContactMessage;
  16. import no.nav.data.team.integration.slack.SlackClient;
  17. import no.nav.data.team.notify.domain.GenericNotificationTask.InactiveMembers;
  18. import no.nav.data.team.notify.domain.Notification;
  19. import no.nav.data.team.notify.domain.Notification.NotificationChannel;
  20. import no.nav.data.team.notify.domain.Notification.NotificationTime;
  21. import no.nav.data.team.notify.domain.Notification.NotificationType;
  22. import no.nav.data.team.notify.domain.NotificationTask;
  23. import no.nav.data.team.notify.dto.Changelog;
  24. import no.nav.data.team.notify.dto.MailModels.UpdateModel;
  25. import no.nav.data.team.notify.dto.NotificationDto;
  26. import no.nav.data.team.resource.NomClient;
  27. import no.nav.data.team.resource.domain.Resource;
  28. import no.nav.data.team.shared.Lang;
  29. import no.nav.data.team.shared.domain.Membered;
  30. import no.nav.data.team.team.domain.Team;
  31. import no.nav.data.team.team.domain.TeamRole;
  32. import org.apache.commons.lang3.NotImplementedException;
  33. import org.apache.commons.lang3.StringUtils;
  34. import org.springframework.stereotype.Service;
  35. import org.springframework.util.CollectionUtils;

  36. import java.time.Duration;
  37. import java.time.LocalDateTime;
  38. import java.time.Month;
  39. import java.util.List;
  40. import java.util.Objects;
  41. import java.util.UUID;
  42. import java.util.function.Function;
  43. import java.util.stream.Collectors;

  44. import static no.nav.data.common.utils.StreamUtils.convert;
  45. import static no.nav.data.common.utils.StreamUtils.safeStream;
  46. import static no.nav.data.common.utils.StreamUtils.tryFind;
  47. import static no.nav.data.team.contact.domain.Channel.EPOST;

  48. @Slf4j
  49. @Service
  50. @RequiredArgsConstructor
  51. public class NotificationService {

  52.     private final StorageService storage;
  53.     private final NomClient nomClient;
  54.     private final EmailService emailService;
  55.     private final TemplateService templateService;
  56.     private final SlackClient slackClient;
  57.     private final NotificationSlackMessageConverter notificationSlackMessageConverter;

  58.     private final AuditVersionRepository auditVersionRepository;
  59.     private final NotificationMessageGenerator messageGenerator;
  60.     private final AuditDiffService auditDiffService;
  61.     private final Cache<String, Changelog> changelogCache = MetricUtils.register("changelogCache",
  62.             Caffeine.newBuilder()
  63.                     .expireAfterWrite(Duration.ofMinutes(15))
  64.                     .maximumSize(500)
  65.                     .recordStats().build());

  66.     // changes before this date have non-backwards compatible formats
  67.     private static final LocalDateTime earliestChangelog = LocalDateTime.of(2020, Month.APRIL, 24, 0, 0);

  68.     public Notification save(NotificationDto dto) {
  69.         dto.validate();
  70.         SecurityUtils.assertIsUserOrAdmin(dto.getIdent(), "Cannot edit other users notifications");

  71.         if (nomClient.getByNavIdent(dto.getIdent()).isEmpty()) {
  72.             throw new ValidationException("Couldn't find user " + dto.getIdent());
  73.         }

  74.         return storage.save(new Notification(dto));
  75.     }

  76.     public Notification delete(UUID id) {
  77.         var notification = storage.get(id, Notification.class);
  78.         SecurityUtils.assertIsUserOrAdmin(notification.getIdent(), "Cannot delete other users notifications");
  79.         return storage.delete(id, Notification.class);
  80.     }

  81.     public void notifyTask(NotificationTask task) {
  82.         log.info("Sending notification for task {}", task);
  83.         var email = getEmailForIdent(task.getIdent());

  84.         var message = messageGenerator.updateSummary(task);
  85.         if (message.isEmpty()) {
  86.             log.info("Skipping task, end message is empty taskId {}", task.getId());
  87.             return;
  88.         }
  89.         if (task.getChannel() == NotificationChannel.EMAIL) {
  90.             sendUpdateMail(email, message.getModel(), message.getSubject());
  91.         } else if (task.getChannel() == NotificationChannel.SLACK) {
  92.             var blocks = notificationSlackMessageConverter.convertTeamUpdateModel(message.getModel());
  93.             try {
  94.                 slackClient.sendMessageToUser(email, message.getSubject(), blocks);
  95.             } catch (NotFoundException e) {
  96.                 sendUpdateMail(email, message.getModel(), message.getSubject() + " - Erstatning for slack melding. Klarte ikke finne din slack bruker.");
  97.             }
  98.         }
  99.     }

  100.     private void sendUpdateMail(String email, UpdateModel model, String subject) {
  101.         String body = templateService.teamUpdate(model);

  102.         emailService.sendMail(MailTask.builder().to(email).subject(subject).body(body).build());
  103.     }

  104.     public void nudge(Membered object) {
  105.         sendMessage(object, recipients -> messageGenerator.nudgeTime(object, recipients.role()));
  106.     }

  107.     public void inactive(InactiveMembers task) {
  108.         Membered object = storage.get(task.getId(), task.getType());
  109.         sendMessage(object, recipients -> messageGenerator.inactive(object, recipients.role(), task.getIdentsInactive()));
  110.     }

  111.     private void sendMessage(Membered membered, Function<Recipients, ContactMessage> messageGenerator) {
  112.         try {
  113.             var recipients = getRecipients(membered);
  114.             if (recipients.isEmpty()) {
  115.                 log.info("No recipients found for contact to {}: {}", membered.type(), membered.getName());
  116.                 return;
  117.             }
  118.             var contactMessage = messageGenerator.apply(recipients);
  119.             // TODO consider schedule slack messages async (like email) to guard against slack downtime
  120.             for (var recipient : recipients.addresses) {
  121.                 switch (recipient.getType()) {
  122.                     case EPOST -> emailService.scheduleMail(MailTask.builder().to(recipient.getAddress()).subject(contactMessage.getTitle()).body(contactMessage.toHtml()).build());
  123.                     case SLACK -> slackClient.sendMessageToChannel(recipient.getAddress(), contactMessage.getTitle(), contactMessage.toSlack());
  124.                     case SLACK_USER -> slackClient.sendMessageToUserId(recipient.getAddress(), contactMessage.getTitle(), contactMessage.toSlack());
  125.                     default -> throw new NotImplementedException("%s is not an implemented varsel type".formatted(recipient.getType()));
  126.                 }
  127.             }
  128.         } catch (Exception e) {
  129.             log.error("Failed to send message to %s %s".formatted(membered.type(), membered.getName()), e);
  130.             throw e;
  131.         }
  132.     }

  133.     private Recipients getRecipients(Membered object) {
  134.         if (object instanceof Team team) {
  135.             if (!CollectionUtils.isEmpty(team.getContactAddresses())) {
  136.                 return new Recipients("Kontaktadresse", team.getContactAddresses());
  137.             } else if (!StringUtils.isBlank(team.getContactPersonIdent())) {
  138.                 return new Recipients("Kontaktperson", List.of(new ContactAddress(getEmailForIdent(team.getContactPersonIdent()), EPOST)));
  139.             }
  140.         }
  141.         var role = TeamRole.LEAD;
  142.         List<String> emails = getEmails(object, role);
  143.         if (emails.isEmpty()) {
  144.             role = TeamRole.PRODUCT_OWNER;
  145.             emails = getEmails(object, role);
  146.         }
  147.         return new Recipients(Lang.roleName(role), convert(emails, e -> new ContactAddress(e, EPOST)));
  148.     }

  149.     private List<String> getEmails(Membered object, TeamRole role) {
  150.         return safeStream(object.getMembers())
  151.                 .filter(m -> m.getRoles().contains(role))
  152.                 .map(l -> {
  153.                     try {
  154.                         return getEmailForIdent(l.getNavIdent());
  155.                     } catch (MailNotFoundException e) {
  156.                         log.warn("email not found", e);
  157.                         return null;
  158.                     }
  159.                 })
  160.                 .filter(Objects::nonNull)
  161.                 .collect(Collectors.toList());
  162.     }

  163.     private String getEmailForIdent(String ident) {
  164.         return nomClient.getByNavIdent(ident)
  165.                 .map(Resource::getEmail)
  166.                 .orElseThrow(() -> new MailNotFoundException("Can't find email for " + ident));
  167.     }

  168.     public String changelogMail(NotificationType type, UUID targetId, LocalDateTime start, LocalDateTime end) {
  169.         var model = changelog(type, targetId, start, end);
  170.         if (model == null) {
  171.             return "empty";
  172.         }
  173.         log.info("new {} removes {} updates {}", model.getCreated(), model.getDeleted(), model.getUpdated());
  174.         return templateService.teamUpdate(model);
  175.     }

  176.     public Changelog changelogJson(NotificationType type, UUID targetId, LocalDateTime start, LocalDateTime end) {
  177.         return changelogCache.get("" + type + targetId + start + end, k -> Changelog.from(changelog(type, targetId, start, end)));
  178.     }

  179.     private UpdateModel changelog(NotificationType type, UUID targetId, LocalDateTime start, LocalDateTime end) {
  180.         var notifications = List.of(Notification.builder()
  181.                 .channels(List.of(NotificationChannel.EMAIL))
  182.                 .target(targetId)
  183.                 .type(type)
  184.                 .ident("MANUAL")
  185.                 .time(NotificationTime.ALL)
  186.                 .build());
  187.         start = start.isBefore(earliestChangelog) ? earliestChangelog : start;
  188.         var audits = auditVersionRepository.findByTimeBetween(start, end);
  189.         var tasks = auditDiffService.createTask(audits, notifications);
  190.         var task = tryFind(tasks, t -> t.getChannel() == NotificationChannel.EMAIL);
  191.         if (task.isEmpty() || task.get().getTargets().isEmpty()) {
  192.             return null;
  193.         }
  194.         return messageGenerator.updateSummary(task.get()).getModel();
  195.     }

  196.     record Recipients(String role, List<ContactAddress> addresses) {

  197.         boolean isEmpty() {
  198.             return addresses.isEmpty();
  199.         }

  200.     }
  201. }