
user table 구조를 만들었다.
users 는 id, email, name을 갖고, auth_users는 id, user_id, provider, provider_id, password를 갖는다.
provider_id가 있으면 ouath2이고, password가 있으면 email(local) 가입자이다.
그렇게 만들어진 adapter layer의 구조

entity 와 repository는 결국 DB에 접근하기 위한 보조이고,adpater는 실제 구현체가 존재한다.
@Transactional
override fun saveProvider(user: User, provider: String, providerId: String): User {
val userEntity = UserEntity.fromDomain(user)
val savedUserEntity = userRepository.save(userEntity)
val authUserEntity = AuthUserEntity.fromProvider(user, provider, providerId)
authUserRepository.save(authUserEntity)
return savedUserEntity.toDomain()
}
@Transactional
override fun saveEmail(user: User, password: String): User {
val userEntity = UserEntity.fromDomain(user)
val savedUserEntity = userRepository.save(userEntity)
val authUserEntity = AuthUserEntity.formEmail(user, password)
authUserRepository.save(authUserEntity)
return savedUserEntity.toDomain()
}
그리고 구현체 내부는 이처럼 필요한 entity를 각자 형태에 따라서 넣을 수 있도록 했다.
그러면 실제 service에서는 entity의 형태를 크게 신경쓰지 않고, saveProivder를 통해 필요한 인자만 넘겨주면 된다.
테이블이 바뀌더라도 인자가 바뀌지 않는 이상 서비스레이어에서는 DB를 신경 쓸 일이 없다!
그렇게 나온 email 기반 User 저장 서비스이다
@Transactional
override fun createUserByEmail(input: CreateUserByEmailInput): User {
userDomainService.validateUserByEmail(input.toDomain())
if (userPort.findByEmail(input.email) != null) {
throw IllegalStateException("이미 존재하는 이메일입니다")
}
val savedUser = userPort.saveEmail(
user = input.toDomain(),
password = input.password
)
return savedUser
}
@Transactional
override fun createUserByProvider(input: CreateUserByProviderInput): User {
userDomainService.validateUserByProvider(input.toDomain())
if (userPort.findByEmail(input.email) != null) {
throw IllegalStateException("이미 존재하는 이메일입니다")
}
return userPort.saveProvider(
user = input.toDomain(),
provider = input.provider,
providerId = input.providerId
)
}
user는 email로 저장하기 위해서 port의 save email만 의존하면 된다. (userPort는 application layer에 있다)
port의 구현체는 adapter layer에 존재하기 때문에 외부와 연결되는 역할을 한다.
서비스는 정말 행해야 하는 행동만 하고 있다. 필수값을 확인하고, 중복되는 지를 보고, 저장한다.
나중에 가입 확인 메일 등의 내용이 필요하다면 이 서비스에서 추가가 될 수 있다.
user는 이제 필요한 서비스들만 붙이면 되므로 일단 끝,
Oauth2는 조금 나중에 생각하기로 하고, 다음에는 악기 판매의 핵심 부분을 만들어볼까한다.
'PROJECT' 카테고리의 다른 글
week3-1: Product Entity와 Domain 설계 (0) | 2025.03.10 |
---|---|
week2-1: Spring Security 비활성화 및 Swagger 적용, User 테이블 만들기 (0) | 2025.02.24 |
week1-1: 요구사항 나열해보기 feat. 도메인 주도 개발 (0) | 2025.02.22 |
Week 1: 사이드프로젝트를 시작해보자 with AI (0) | 2025.02.21 |
[SpringBoot] WebSecurityConfigurerAdapter 지원 불가 (6) | 2022.04.05 |