NotificationController.java

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

  2. import io.swagger.v3.oas.annotations.Operation;
  3. import io.swagger.v3.oas.annotations.responses.ApiResponse;
  4. import io.swagger.v3.oas.annotations.responses.ApiResponses;
  5. import io.swagger.v3.oas.annotations.tags.Tag;
  6. import lombok.extern.slf4j.Slf4j;
  7. import no.nav.data.common.exceptions.ValidationException;
  8. import no.nav.data.common.rest.RestResponsePage;
  9. import no.nav.data.common.security.SecurityUtils;
  10. import no.nav.data.common.storage.StorageService;
  11. import no.nav.data.common.storage.domain.GenericStorage;
  12. import no.nav.data.team.notify.domain.Notification;
  13. import no.nav.data.team.notify.domain.Notification.NotificationType;
  14. import no.nav.data.team.notify.domain.NotificationRepository;
  15. import no.nav.data.team.notify.dto.Changelog;
  16. import no.nav.data.team.notify.dto.NotificationDto;
  17. import org.springframework.format.annotation.DateTimeFormat;
  18. import org.springframework.format.annotation.DateTimeFormat.ISO;
  19. import org.springframework.http.HttpStatus;
  20. import org.springframework.http.ResponseEntity;
  21. import org.springframework.web.bind.annotation.DeleteMapping;
  22. import org.springframework.web.bind.annotation.GetMapping;
  23. import org.springframework.web.bind.annotation.PathVariable;
  24. import org.springframework.web.bind.annotation.PostMapping;
  25. import org.springframework.web.bind.annotation.RequestBody;
  26. import org.springframework.web.bind.annotation.RequestMapping;
  27. import org.springframework.web.bind.annotation.RequestParam;
  28. import org.springframework.web.bind.annotation.RestController;

  29. import java.time.Duration;
  30. import java.time.LocalDate;
  31. import java.time.LocalDateTime;
  32. import java.time.LocalTime;
  33. import java.time.temporal.ChronoUnit;
  34. import java.util.List;
  35. import java.util.UUID;
  36. import java.util.stream.Collectors;

  37. import static no.nav.data.common.utils.StreamUtils.convert;

  38. @Slf4j
  39. @RestController
  40. @Tag(name = "Notification")
  41. @RequestMapping("/notification")
  42. public class NotificationController {

  43.     private final StorageService storage;
  44.     private final NotificationService service;
  45.     private final NotificationRepository repository;

  46.     public NotificationController(StorageService storage, NotificationService service, NotificationRepository repository) {
  47.         this.storage = storage;
  48.         this.service = service;
  49.         this.repository = repository;
  50.     }

  51.     @Operation(summary = "Get Notifications for current user")
  52.     @ApiResponses(value = {@ApiResponse(description = "Notifications fetched")})
  53.     @GetMapping
  54.     public ResponseEntity<RestResponsePage<NotificationDto>> getForCurrentUser() {
  55.         var ident = SecurityUtils.lookupCurrentIdent();
  56.         if (ident.isEmpty()) {
  57.             return ResponseEntity.ok(new RestResponsePage<>());
  58.         }
  59.         return ResponseEntity.ok(new RestResponsePage<>(getAll(ident.get())));
  60.     }

  61.     @Operation(summary = "Get Notification")
  62.     @ApiResponses(value = {@ApiResponse(description = "Notification fetched")})
  63.     @GetMapping("/{id}")
  64.     public ResponseEntity<NotificationDto> get(@PathVariable("id") UUID id) {
  65.         return ResponseEntity.ok(storage.get(id, Notification.class).convertToDto());
  66.     }

  67.     @Operation(summary = "Save new Notification - immutable")
  68.     @ApiResponses(value = {@ApiResponse(description = "Notification saved")})
  69.     @PostMapping
  70.     public ResponseEntity<NotificationDto> save(@RequestBody NotificationDto dto) {
  71.         return ResponseEntity.ok(service.save(dto).convertToDto());
  72.     }

  73.     @Operation(summary = "Delete Notification")
  74.     @ApiResponses(value = {@ApiResponse(description = "Notification deleted")})
  75.     @DeleteMapping("/{id}")
  76.     public ResponseEntity<NotificationDto> delete(@PathVariable("id") UUID id) {
  77.         return ResponseEntity.ok(service.delete(id).convertToDto());
  78.     }

  79.     @Operation(summary = "diff test")
  80.     @ApiResponses(value = {@ApiResponse(description = "diff")})
  81.     @GetMapping(value = "/diff", produces = "text/html")
  82.     public ResponseEntity<String> diff(
  83.             @RequestParam(value = "type") NotificationType type,
  84.             @RequestParam(value = "targetId", required = false) UUID targetId,
  85.             @RequestParam(value = "start") @DateTimeFormat(iso = DateTimeFormat.ISO.DATE_TIME) LocalDateTime start,
  86.             @RequestParam(value = "end", required = false) @DateTimeFormat(iso = DateTimeFormat.ISO.DATE_TIME) LocalDateTime end
  87.     ) {
  88.         if (targetId == null && type != NotificationType.ALL_EVENTS) {
  89.             throw new ValidationException("need targetId for " + type);
  90.         }
  91.         if (end == null) {
  92.             end = LocalDateTime.now();
  93.         }
  94.         try {
  95.             String changelog = service.changelogMail(type, targetId, start, end);
  96.             return ResponseEntity.ok(changelog);
  97.         } catch (Exception e) {
  98.             log.error("notification diff failed", e);
  99.             return new ResponseEntity<>("error", HttpStatus.INTERNAL_SERVER_ERROR);
  100.         }
  101.     }

  102.     @Operation(summary = "Changelog")
  103.     @ApiResponses(value = {@ApiResponse(description = "Changelog for team updates")})
  104.     @GetMapping(value = "/changelog")
  105.     public ResponseEntity<Changelog> changelog(
  106.             @RequestParam(value = "type") NotificationType type,
  107.             @RequestParam(value = "targetId", required = false) UUID targetId,
  108.             @RequestParam(value = "start") @DateTimeFormat(iso = ISO.DATE_TIME) LocalDateTime start,
  109.             @RequestParam(value = "end", required = false) @DateTimeFormat(iso = ISO.DATE_TIME) LocalDateTime end
  110.     ) {
  111.         if (targetId == null && type != NotificationType.ALL_EVENTS) {
  112.             throw new ValidationException("need targetId for " + type);
  113.         }
  114.         if (end == null) {
  115.             end = LocalDateTime.now();
  116.         }
  117.         if (!Duration.between(start, end).minusDays(32).isNegative()) {
  118.             throw new ValidationException("Duration is more than 31 days");
  119.         }
  120.         try {
  121.             var changelog = service.changelogJson(type, targetId, start, end);
  122.             return ResponseEntity.ok(changelog);
  123.         } catch (Exception e) {
  124.             log.error("notification diff failed", e);
  125.             return new ResponseEntity<>(HttpStatus.INTERNAL_SERVER_ERROR);
  126.         }
  127.     }

  128.     @Operation(summary = "Changelog")
  129.     @ApiResponses(value = {@ApiResponse(description = "Changelog for team updates")})
  130.     @GetMapping(value = "/changelog/day")
  131.     public ResponseEntity<RestResponsePage<Changelog>> changelogDay(
  132.             @RequestParam(value = "type") NotificationType type,
  133.             @RequestParam(value = "targetId", required = false) UUID targetId,
  134.             @RequestParam(value = "start") @DateTimeFormat(iso = ISO.DATE) LocalDate start,
  135.             @RequestParam(value = "end", required = false) @DateTimeFormat(iso = ISO.DATE) LocalDate end
  136.     ) {
  137.         if (targetId == null && type != NotificationType.ALL_EVENTS) {
  138.             throw new ValidationException("need targetId for " + type);
  139.         }
  140.         if (end == null) {
  141.             end = LocalDate.now();
  142.         }
  143.         if (ChronoUnit.DAYS.between(start, end) > 31) {
  144.             throw new ValidationException("Duration is more than 31 days");
  145.         }
  146.         try {
  147.             var changelog = start.datesUntil(end)
  148.                     .map(d -> service.changelogJson(type, targetId, LocalDateTime.of(d, LocalTime.MIN), LocalDateTime.of(d, LocalTime.MAX)))
  149.                     .collect(Collectors.toList());
  150.             return ResponseEntity.ok(new RestResponsePage<>(changelog));
  151.         } catch (Exception e) {
  152.             log.error("notification diff failed", e);
  153.             return new ResponseEntity<>(HttpStatus.INTERNAL_SERVER_ERROR);
  154.         }
  155.     }

  156.     private List<NotificationDto> getAll(String ident) {
  157.         return convert(GenericStorage.to(repository.findByIdent(ident), Notification.class), Notification::convertToDto);
  158.     }

  159.     static class NotificationPage extends RestResponsePage<NotificationDto> {

  160.     }
  161. }