AuthService.java

  1. package no.nav.data.common.security;

  2. import io.prometheus.client.Gauge;
  3. import lombok.extern.slf4j.Slf4j;
  4. import no.nav.data.common.exceptions.NotFoundException;
  5. import no.nav.data.common.exceptions.UnauthorizedException;
  6. import no.nav.data.common.security.domain.Auth;
  7. import no.nav.data.common.security.domain.AuthRepository;
  8. import no.nav.data.common.utils.Constants;
  9. import no.nav.data.common.utils.MetricUtils;
  10. import no.nav.data.common.utils.StringUtils;
  11. import org.springframework.scheduling.annotation.Scheduled;
  12. import org.springframework.security.crypto.keygen.KeyGenerators;
  13. import org.springframework.security.crypto.keygen.StringKeyGenerator;
  14. import org.springframework.stereotype.Service;
  15. import org.springframework.transaction.annotation.Transactional;

  16. import java.time.Duration;
  17. import java.time.LocalDateTime;
  18. import java.util.List;
  19. import java.util.UUID;

  20. import static org.apache.commons.lang3.StringUtils.isBlank;

  21. @Slf4j
  22. @Service
  23. @Transactional
  24. public class AuthService {

  25.     private static final StringKeyGenerator keyGenerator = KeyGenerators.string();
  26.     private static final Gauge uniqueUsers = MetricUtils.gauge()
  27.             .labels("hour").labels("day").labels("week").labels("twoweek")
  28.             .labelNames("period")
  29.             .name("team_auth_users_active").help("Users active")
  30.             .register();

  31.     private final AuthRepository authRepository;
  32.     private final Encryptor encryptor;

  33.     public AuthService(AuthRepository authRepository, Encryptor refreshTokenEncryptor) {
  34.         this.authRepository = authRepository;
  35.         this.encryptor = refreshTokenEncryptor;
  36.     }

  37.     public String getCodeVerifier(String sessionId) {
  38.         return authRepository.findById(StringUtils.toUUID(sessionId))
  39.                 .orElseThrow(() -> new NotFoundException("couldn't find session"))
  40.                 .getCodeVerifier();
  41.     }

  42.     public Auth getAuth(String sessionId, String sessionKey) {
  43.         var sessUuid = StringUtils.toUUID(sessionId);
  44.         Auth auth = authRepository.findById(sessUuid)
  45.                 .orElseThrow(() -> new NotFoundException("couldn't find session"))
  46.                 .addSecret(encryptor, sessionKey);
  47.         if (isBlank(auth.getEncryptedRefreshToken())) {
  48.             throw new UnauthorizedException("session is terminated");
  49.         }
  50.         if (auth.getLastActive().isBefore(LocalDateTime.now().minusMinutes(1))) {
  51.             auth.setLastActive(LocalDateTime.now());
  52.         }
  53.         return auth;
  54.     }

  55.     public Auth createAuth() {
  56.         String codeVerifier = genChallengeKey();
  57.         return authRepository.save(Auth.builder()
  58.                 .generateId()
  59.                 .userId("")
  60.                 .encryptedRefreshToken(codeVerifier)
  61.                 .initiated(LocalDateTime.now())
  62.                 .lastActive(LocalDateTime.now())
  63.                 .build());
  64.     }

  65.     public String initAuth(String userId, String refreshToken, String id) {
  66.         var enc = encryptor.encrypt(refreshToken);
  67.         var auth = authRepository.findById(UUID.fromString(id)).orElseThrow();
  68.         auth.setUserId(userId);
  69.         auth.setEncryptedRefreshToken(enc.cipher());
  70.         auth.setLastActive(LocalDateTime.now());
  71.         auth.addSecret(encryptor, enc.salt());
  72.         return auth.session();
  73.     }

  74.     public void endSession(UUID id) {
  75.         Auth auth = authRepository.findById(id).orElseThrow();
  76.         auth.setEncryptedRefreshToken("");
  77.     }

  78.     @Scheduled(initialDelayString = "PT1M", fixedRateString = "PT10M")
  79.     public void cleanOldAuth() {
  80.         List<Auth> auths = authRepository.findByLastActiveBefore(LocalDateTime.now().minus(Constants.SESSION_LENGTH.plusHours(1)));
  81.         for (Auth auth : auths) {
  82.             authRepository.delete(auth);
  83.             log.debug("Deleting old auth for user {}", auth.getUserId());
  84.         }
  85.     }

  86.     @Scheduled(initialDelayString = "PT1M", fixedRateString = "PT1M")
  87.     public void gatherMetrics() {
  88.         uniqueUsers.labels("hour").set(countActiveLast(Duration.ofHours(1)));
  89.         uniqueUsers.labels("day").set(countActiveLast(Duration.ofDays(1)));
  90.         uniqueUsers.labels("week").set(countActiveLast(Duration.ofDays(7)));
  91.         uniqueUsers.labels("twoweek").set(countActiveLast(Duration.ofDays(14)));
  92.     }

  93.     private long countActiveLast(Duration duration) {
  94.         return authRepository.countDistinctUserIdByLastActiveAfter(LocalDateTime.now().minus(duration));
  95.     }

  96.     private String genChallengeKey() {
  97.         var len = 64;
  98.         StringBuilder sb = new StringBuilder();
  99.         do {
  100.             sb.append(keyGenerator.generateKey());
  101.         } while (sb.length() < len);
  102.         return sb.toString();
  103.     }
  104. }