ResourceController.java

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

  2. import io.swagger.v3.oas.annotations.Operation;
  3. import io.swagger.v3.oas.annotations.responses.ApiResponse;
  4. import io.swagger.v3.oas.annotations.tags.Tag;
  5. import lombok.RequiredArgsConstructor;
  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.security.dto.UserInfo;
  11. import no.nav.data.common.validator.Validator;
  12. import no.nav.data.team.naisteam.NaisConsoleClient;
  13. import no.nav.data.team.resource.domain.Resource;
  14. import no.nav.data.team.resource.dto.ResourceResponse;
  15. import no.nav.data.team.resource.dto.ResourceUnitsResponse;
  16. import org.apache.commons.lang3.StringUtils;
  17. import org.springframework.http.HttpStatus;
  18. import org.springframework.http.MediaType;
  19. import org.springframework.http.ResponseEntity;
  20. import org.springframework.security.core.context.SecurityContextHolder;
  21. import org.springframework.web.bind.annotation.GetMapping;
  22. import org.springframework.web.bind.annotation.PathVariable;
  23. import org.springframework.web.bind.annotation.PostMapping;
  24. import org.springframework.web.bind.annotation.RequestBody;
  25. import org.springframework.web.bind.annotation.RequestMapping;
  26. import org.springframework.web.bind.annotation.RequestParam;
  27. import org.springframework.web.bind.annotation.RestController;

  28. import java.util.List;
  29. import java.util.Optional;
  30. import java.util.stream.Collectors;
  31. import java.util.stream.Stream;

  32. import static java.util.stream.Collectors.toList;

  33. @Slf4j
  34. @RestController
  35. @RequestMapping("/resource")
  36. @Tag(name = "Resource")
  37. @RequiredArgsConstructor
  38. public class ResourceController {

  39.     private final NomClient nomClient;
  40.     private final NomGraphClient nomGraphClient;
  41.     private final ResourceService resourceService;
  42.     private final NaisConsoleClient naisTeamService;

  43.     @Operation(summary = "Search resources")
  44.     @ApiResponse(description = "Resources fetched")
  45.     @GetMapping("/search/{name}")
  46.     public ResponseEntity<RestResponsePage<ResourceResponse>> searchResourceName(@PathVariable String name) {
  47.         log.info("Resource search '{}'", name);
  48.         if (Stream.of(name.split(" ")).sorted().distinct().collect(Collectors.joining("")).length() < 3) {
  49.             throw new ValidationException("Search resource must be at least 3 characters");
  50.         }
  51.         var resources = nomClient.search(name);
  52.         log.info("Returned {} resources", resources.getPageSize());
  53.         return new ResponseEntity<>(resources.convert(Resource::convertToResponse), HttpStatus.OK);
  54.     }


  55.     @Operation(summary = "Get Resource")
  56.     @ApiResponse(description = "ok")
  57.     @GetMapping("/{id}")
  58.     public ResponseEntity<ResourceResponse> getById(@PathVariable String id) {
  59.         log.info("Resource get id={}", id);
  60.         var resource = nomClient.getByNavIdent(id);
  61.         if (resource.isEmpty()) {
  62.             return ResponseEntity.notFound().build();
  63.         }
  64.         return ResponseEntity.ok(resource.get().convertToResponse());
  65.     }

  66.     @Operation(summary = "Get Resource Units")
  67.     @ApiResponse(description = "ok")
  68.     @GetMapping("/{id}/units")
  69.     public ResponseEntity<ResourceUnitsResponse> getUnitsById(@PathVariable String id) {
  70.         log.info("Resource get units id={}", id);

  71.         temporaryLogConsumer();

  72.         try {
  73.             var units = nomGraphClient.getUnits(id);
  74.             if (units.isEmpty()) {
  75.                 return ResponseEntity.notFound().build();
  76.             }
  77.             return ResponseEntity.ok(units.get());
  78.         } catch (Exception e) {
  79.             log.error("Failed to get units for " + id, e);
  80.             return ResponseEntity.ok(null);
  81.         }
  82.     }

  83.     @Operation(summary = "Get Resources")
  84.     @ApiResponse(description = "ok")
  85.     @PostMapping("/multi")
  86.     public ResponseEntity<RestResponsePage<ResourceResponse>> getById(@RequestBody List<String> ids) {
  87.         log.info("Resource get ids={}", ids);

  88.         var resources = ids.stream()
  89.                 .map(nomClient::getByNavIdent)
  90.                 .filter(Optional::isPresent)
  91.                 .map(Optional::get)
  92.                 .map(Resource::convertToResponse)
  93.                 .collect(toList());
  94.         return ResponseEntity.ok(new RestResponsePage<>(resources));
  95.     }

  96.     @Operation(summary = "Get Resource Photo")
  97.     @ApiResponse(description = "ok")
  98.     @GetMapping(value = "/{id}/photo", produces = MediaType.IMAGE_JPEG_VALUE)
  99.     public ResponseEntity<byte[]> getPhoto(
  100.             @PathVariable String id,
  101.             @RequestParam(name = "forceUpdate", required = false, defaultValue = "false") boolean forceUpdate
  102.     ) {
  103.         id = StringUtils.upperCase(id);
  104.         if (!Validator.NAV_IDENT_PATTERN.matcher(id).matches()) {
  105.             log.info("Resource get photo id={} invalid id", id);
  106.             return ResponseEntity.notFound().build();
  107.         }
  108.         var photo = resourceService.getPhoto(id, forceUpdate);

  109.         if (photo.isMissing()) {
  110.             log.info("Resource get photo id={} not found", id);
  111.             return ResponseEntity.notFound().build();
  112.         }
  113.         log.info("Resource get photo id={}", id);
  114.         return ResponseEntity.ok(photo.getContent());
  115.     }

  116.     static class ResourcePageResponse extends RestResponsePage<ResourceResponse> {

  117.     }

  118.     private void temporaryLogConsumer(){
  119.         var auth = SecurityContextHolder.getContext().getAuthentication();
  120.         var c = auth.getPrincipal().getClass();
  121.         var x = SecurityUtils.getCurrentUser().map(UserInfo::getAppName);
  122.         var y = SecurityUtils.getCurrentUser().map(UserInfo::getAppId);
  123.         log.info("/resource/{id}/units called by: name = " + x.orElse("<>") + " , id = "  + y.orElse("<>") + " . Principal class = " + c.getName() + " , Authentication class = " + auth.getClass().getName());
  124.     }

  125. }