Spring Data JPA는 엔티티가 언제, 누구에 의해 생성되거나 수정되었는지를 투명하게 추적할 수 있도록 고급 지원을 제공합니다. 이를 위해 엔티티 클래스에 감사 메타데이터를 추가해야 하며, 이 메타데이터는 어노테이션을 사용하거나 인터페이스를 구현하여 정의할 수 있습니다. 또한 감사 기능을 활성화하려면 어노테이션 구성 또는 XML 구성을 통해 필요한 인프라 구성 요소를 등록해야 합니다.
Annotation-based Auditing Metadata
Spring Data JPA에서는 @CreatedBy, @LastModifiedBy, @CreatedDate, @LastModifiedDate 어노테이션을 사용하여 엔티티의 생성자와 수정자, 그리고 변경 시점을 캡처할 수 있습니다.
예시: 감사 메타데이터가 포함된 엔티티
import javax.persistence.*;
import java.time.Instant;
@Entity
public class Customer {
@CreatedBy
private User user; // 생성한 사용자
@CreatedDate
private Instant createdDate; // 생성 일시
// 기타 속성 생략
}
위의 코드에서 볼 수 있듯이, 감사 정보를 캡처할 필드에 해당 어노테이션을 적용하여 필요한 정보만 캡처할 수 있습니다. 이 어노테이션은 JDK8의 날짜 및 시간 타입, long, Long, 구식 Java Date 및 Calendar 타입의 속성에 사용할 수 있습니다.
임베디드 엔티티의 감사 메타데이터
감사 메타데이터는 루트 엔티티에만 존재할 필요는 없으며, 임베디드 엔티티에도 추가할 수 있습니다.
예시: 임베디드 엔티티의 감사 메타데이터
@Entity
public class Customer {
private AuditMetadata auditingMetadata; // 감사 메타데이터
// 기타 속성 생략
}
@Embeddable
class AuditMetadata {
@CreatedBy
private User user; // 생성한 사용자
@CreatedDate
private Instant createdDate; // 생성 일시
}
Interface-based Auditing Metadata
어노테이션을 사용하지 않고 감사 메타데이터를 정의하려면, 도메인 클래스가 Auditable 인터페이스를 구현하도록 할 수 있습니다. 이 인터페이스는 모든 감사 속성에 대한 setter 메서드를 노출합니다.
AuditorAware
@CreatedBy 또는 @LastModifiedBy를 사용할 경우, 현재 사용자를 인식할 수 있도록 AuditorAware<T> SPI 인터페이스를 구현해야 합니다. 이 인터페이스는 감사 메타데이터에서 사용할 속성의 타입을 정의합니다.
예시: Spring Security 기반 AuditorAware 구현
import org.springframework.security.core.Authentication;
import org.springframework.security.core.context.SecurityContext;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.data.domain.AuditorAware;
import java.util.Optional;
public class SpringSecurityAuditorAware implements AuditorAware<User> {
@Override
public Optional<User> getCurrentAuditor() {
return Optional.ofNullable(SecurityContextHolder.getContext())
.map(SecurityContext::getAuthentication)
.filter(Authentication::isAuthenticated)
.map(Authentication::getPrincipal)
.map(User.class::cast);
}
}
위 구현에서는 Spring Security에서 제공하는 Authentication 객체를 사용하여 현재 사용자를 조회합니다. 여기서 UserDetailsService에서 생성한 사용자 세부 정보를 가져올 수 있습니다.
ReactiveAuditorAware
반응형 인프라를 사용하는 경우 ReactiveAuditorAware<T> SPI 인터페이스를 구현하여 현재 사용자를 인식할 수 있습니다.
예시: Spring Security 기반 ReactiveAuditorAware 구현
import org.springframework.security.core.Authentication;
import org.springframework.security.core.context.ReactiveSecurityContextHolder;
import org.springframework.data.domain.ReactiveAuditorAware;
import reactor.core.publisher.Mono;
public class SpringSecurityAuditorAware implements ReactiveAuditorAware<User> {
@Override
public Mono<User> getCurrentAuditor() {
return ReactiveSecurityContextHolder.getContext()
.map(SecurityContext::getAuthentication)
.filter(Authentication::isAuthenticated)
.map(Authentication::getPrincipal)
.map(User.class::cast);
}
}
이 구현은 반응형 Spring Security에서 제공하는 Authentication 객체를 사용하여 현재 사용자를 조회합니다.
감사 기능 일반 구성(General Auditing Configuration)
예시 1: orm.xml에서 감사 구성
<persistence-unit-metadata>
<persistence-unit-defaults>
<entity-listeners>
<entity-listener class="org.springframework.data.jpa.domain.support.AuditingEntityListener" />
</entity-listeners>
</persistence-unit-defaults>
</persistence-unit-metadata>
또는 특정 엔티티에 대해 @EntityListeners 어노테이션을 사용하여 AuditingEntityListener를 활성화할 수도 있습니다.
예시 2: 특정 엔티티에서 감사 리스너 사용
import javax.persistence.Entity;
import javax.persistence.EntityListeners;
@Entity
@EntityListeners(AuditingEntityListener.class)
public class MyEntity {
// 엔티티 정의
}
감사 기능은 spring-aspects.jar가 클래스패스에 있어야 합니다. orm.xml 파일을 수정한 후, 다음과 같이 Spring Data JPA 감사 네임스페이스 요소를 추가하여 감사 기능을 활성화할 수 있습니다.
예시 3: XML 구성을 통한 감사 기능 활성화
<jpa:auditing auditor-aware-ref="yourAuditorAwareBean" />
Spring Data JPA 1.5부터는 다음과 같이 구성 클래스를 어노테이션으로 사용하여 감사 기능을 활성화할 수 있습니다.
예시 4: Java 구성을 통한 감사 기능 활성화
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.jpa.config.EnableJpaAuditing;
@Configuration
@EnableJpaAuditing
class Config {
@Bean
public AuditorAware<User> auditorProvider() {
return new SpringSecurityAuditorAware(); // AuditorAware 구현체
}
}
ApplicationContext에 AuditorAware 타입의 빈을 노출하면, 감사 인프라가 이를 자동으로 감지하고 도메인 타입에 설정할 현재 사용자를 결정합니다. 만약 여러 구현체가 등록된 경우, @EnableJpaAuditing의 auditorAwareRef 속성을 통해 사용할 구현체를 명시할 수 있습니다.
Spring Data JPA의 감사 기능은 데이터 변경의 원인과 시점을 효과적으로 추적할 수 있는 기능을 제공합니다. 이를 통해 애플리케이션의 데이터 무결성을 보장하고, 감사 로그를 작성하는 데 유용하게 활용할 수 있습니다.
'JPA > Spring JPA Official' 카테고리의 다른 글
Locking (0) | 2024.12.02 |
---|---|
Transactionality (0) | 2024.12.02 |
Projections (0) | 2024.12.02 |
Query Creation (0) | 2024.12.02 |
Defining Query Methods (1) | 2024.12.02 |