ResourceService.java

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

  2. import com.github.benmanes.caffeine.cache.Cache;
  3. import com.github.benmanes.caffeine.cache.Caffeine;
  4. import io.prometheus.client.Gauge;
  5. import lombok.extern.slf4j.Slf4j;
  6. import no.nav.data.common.security.azure.AzureAdService;
  7. import no.nav.data.common.storage.StorageService;
  8. import no.nav.data.common.storage.domain.GenericStorage;
  9. import no.nav.data.common.utils.MetricUtils;
  10. import no.nav.data.team.resource.domain.ResourcePhoto;
  11. import no.nav.data.team.resource.domain.ResourcePhotoRepository;
  12. import org.springframework.scheduling.annotation.Scheduled;
  13. import org.springframework.stereotype.Service;
  14. import org.springframework.transaction.annotation.Transactional;

  15. import java.time.Duration;
  16. import java.time.LocalDateTime;
  17. import java.util.List;

  18. @Slf4j
  19. @Service
  20. public class ResourceService {

  21.     private static final Gauge photos = MetricUtils.gauge()
  22.             .name("team_profile_picture_count_gauge").help("Number of profile pictures cached")
  23.             .register();

  24.     private static final Duration PHOTO_DB_DURATION = Duration.ofDays(1);
  25.     private static final Duration PHOTO_MEM_DURATION = Duration.ofHours(1);
  26.     private final Cache<String, ResourcePhoto> photoCache;

  27.     private final StorageService storage;
  28.     private final ResourcePhotoRepository resourcePhotoRepository;
  29.     private final AzureAdService azureAdService;

  30.     public ResourceService(StorageService storage, ResourcePhotoRepository resourcePhotoRepository, AzureAdService azureAdService) {
  31.         this.storage = storage;
  32.         this.resourcePhotoRepository = resourcePhotoRepository;
  33.         this.azureAdService = azureAdService;
  34.         this.photoCache = Caffeine.newBuilder().recordStats()
  35.                 .expireAfterWrite(PHOTO_MEM_DURATION)
  36.                 .maximumSize(200).build();
  37.         MetricUtils.register("photoCache", photoCache);
  38.     }

  39.     @Transactional
  40.     public ResourcePhoto getPhoto(String ident, boolean forceUpdate) {
  41.         var cached = photoCache.getIfPresent(ident);
  42.         if (!forceUpdate && cached != null) {
  43.             return cached;
  44.         }
  45.         List<GenericStorage> photoStorage = resourcePhotoRepository.findByIdent(ident);
  46.         if (forceUpdate) {
  47.             photoStorage.forEach(photo -> storage.delete(photo.getId(), ResourcePhoto.class));
  48.             photoStorage = List.of();
  49.         }

  50.         if (photoStorage.isEmpty()) {
  51.             log.info("Get photo id={} calling graph", ident);
  52.             var picture = azureAdService.lookupProfilePictureByNavIdent(ident);
  53.             return getPhoto(storage.save(ResourcePhoto.builder()
  54.                     .content(picture)
  55.                     .ident(ident)
  56.                     .missing(picture == null)
  57.                     .build()
  58.             ));
  59.         }
  60.         if (photoStorage.size() > 1) {
  61.             // Cleanup duplicates from race conditions
  62.             photoStorage.subList(1, photoStorage.size()).forEach(ps -> storage.delete(ps.getId(), ResourcePhoto.class));
  63.         }
  64.         return getPhoto(photoStorage.get(0).getDomainObjectData(ResourcePhoto.class));
  65.     }

  66.     private ResourcePhoto getPhoto(ResourcePhoto photo) {
  67.         photoCache.put(photo.getIdent(), photo);
  68.         return photo;
  69.     }

  70.     @Scheduled(initialDelayString = "PT1M", fixedRateString = "PT10M")
  71.     public void cleanOld() {
  72.         log.debug("Deleted {} old photos", storage.deleteCreatedOlderThan(ResourcePhoto.class, LocalDateTime.now().minus(PHOTO_DB_DURATION)));
  73.     }

  74.     @Scheduled(initialDelayString = "PT1M", fixedRateString = "PT1M")
  75.     public void gatherMetrics() {
  76.         photos.set(storage.count(ResourcePhoto.class));
  77.     }
  78. }