no image
[Spring] Gradle 캐시 적용으로 CI 파이프라인과 빌드를 빠르게 하기
$GRADLE_USER_HOME아래 내용을 이해하려면 이 환경 변수를 우선 알고 있어야 한다.Gradle 은 GRADLE_USER_HOME 이라는 환경 변수를 지원하여 사용자가 직접 지정하여 이용할 수 있도록 지원해주고 있다. 이 환경 변수에는 단순히 폴더 경로만 지정해주면 되는데 이 환경 변수는 전역 설정 property, 초기화 스크립트, 캐시, 로그 파일 등을 저장하는 용도로 쓰인다.GRADLE_USER_HOME 이라는 환경 변수를 읽지 못했을 때는 사용자 홈 디렉토리의 .gradle 폴더를 바라본다. 프로젝트 저장소의 .gradle 폴더와는 다르기 때문에 혼동하면 안 된다. Wrapper 및 의존성Wrapperhttps://docs.gradle.org/current/userguide/gradle_..
2024.11.15
no image
[Spring] OpenFeign @SpringQueryMap 사용 시 별도 파라미터로 사용하기
어떤 API 에 쿼리 파라미터로 전달해야 하는 항목 중 파라미터를 다르게 전달해야 하는 경우가 있다고 생각해보자.예를 들어, keywordType 을 keyword_type 이라는 이름으로 전달해야 하는 상황이라고 생각해보자. @SpringQueryMap 을 사용하면 아래와 같이 작성할 수 있을 것이다.@FeignClientpublic interface MemberApiClient { @GetMapping(value = "/users") UserResponseDto userSearch(@SpringQueryMap UserRequestDto request);}@Builder@Getterpublic class UserRequestDto { @Builder.Default private U..
2024.11.15
no image
[JPA] Querydsl LocalDateTime 에서 Date 로 변환할 때 오류나는 경우 (feat. Oracle)
@Entity@Table(name = "SOME_ENTITY")public class SomeEntity { @Id @Column(name = "id") private Long id; @Column(name = "created_at") private LocalDateTime createdAt;}@Getter@Builder@NoArgsConstructor@AllArgsConstructorpublic class SomeDto { private LocalDateTime createdAt;}위와 같은 Entity와 결과를 받아올 클래스가 있다고 가정하자.여기서 Querydsl의 Projections.field를 이용하여 createdAt 이라는 값을 select 한다고 해보자. 그러..
2024.03.06
no image
[JPA] QueryDSL 에서 enumPath 사용하기
회사에서 SQL 쿼리를 작성하다 보니 CASE WHEN을 사용해야 하는 경우가 있었는데(안티패턴으로 취급되기도 하지만) QueryDSL에서 CASE WHEN 과 함께 enum 으로 받고 싶어서 사용해보다가 뭔가 이상한 구석이 있어서 한번 기록으로 적어보았다. QueryDSL에서 CASE WHEN 사용법의 경우 이곳에 잘 정리되어 있다. 사용법 우선 EnumPath 를 사용할 경우 예시로 아래와 같이 사용할 수 있다. @RequiredArgsConstructor public class PartnerSubAccountRepositorySupportImpl implements PartnerSubAccountRepositorySupport { private final JPAQueryFactory queryFac..
2023.11.30
no image
[Spring] RestTemplate 과 MessageConverter 순서의 관계
회사에서 내가 만들어둔 라이브러리를 다른 팀에서 사용하고 있는 경우가 있었는데, 어느 날 갑자기 다른 팀에서 아래와 같은 메세지가 나타난다고 확인 요청이 왔다. java.lang.IllegalStateException: org.springframework.web.client.HttpClientErrorException$BadRequest: 400 Bad Request: "{"errors":["failed to parse JSON input: invalid character '\u003c' looking for beginning of value"]}" 이게 갑자기 나타날리가 없어서 어떤 작업을 한 후에 나타났냐고 물으니, build.gradle 에 아래와 같은 의존성을 추가하면 위와 같은 에러가 나타난다고..
2023.02.05
no image
[Spring] ConversionService 활용해보기
배경 회사에서 100개가 넘는 컬럼들을 다루어야 하는 경우가 있었다. 100개가 넘는 컬럼들 중 섹션별로 일부의 데이터를 추가하거나 수정할 수 있어야 했고, 기존의 데이터와 서로 다른지 판단할 수 있어야 하며 이를 로그로도 남겨야 하는 요구사항이 있었다. 예를 들면 Entity 가 아래와 같은데, @ExampleSection 이라는 Annotation 으로 섹션을 분리한 뒤 사용자가 원하는 Section 별로 데이터를 저장하거나 수정할 수 있어야 하는 것이다. @Entity(name = "exampleEntity") @Table(name = "EXAMPLE_ENTITY", schema = "GREATSCHEMA") @DynamicInsert @DynamicUpdate @Builder @Getter @No..
2022.11.21
no image
[Spring] AutoConfiguration 직접 만들어서 라이브러리로 만들어보기
Spring 에서는 상당히 많은 AutoConfiguration 을 제공해주고 있는데, 따로 라이브러리를 만들면서 직접 만들 수도 있다. 테스트로 하나 만들어보자 구성 우선 구조는 아래처럼 잡고 멀티모듈로 구성했다. autoconfigure - Spring Boot 에서 자동으로 AutoConfiguration 이 적용되는 로직 관련 somelogic - 비즈니스 로직 // root - build.gradle buildscript { ext { projectVersion = '1.0.0' projectGroup = 'kr.pe.karsei' springBootVersion = '2.7.5' springDependencyVersion = '1.0.15.RELEASE' } repositories { mave..
2022.11.11
[JPA] QueryDSL 에서 오늘 날짜 활용해서 사용하기 (feat. Oracle)
아래와 같은 클래스가 있다. @Entity @Builder @Getter @NoArgsConstructor @AllArgsConstructor public class PremiumPrice { @Id private Long id; @Column private BigDecimal usdPrice; @Column @CreationTimestamp private LocalDateTime registDate; } 만약, 데이터를 찾을 때 registDate 를 기준으로 오늘 이후의 데이터를 찾고 싶으면 어떻게 할까? 여러 가지 방법이 있겠지만 이 글에서는 두 가지 방법을 소개한다. Expressions.dateTemplate Expressions.dateTemplate(LocalDateTime.class, "s..
2022.10.24
no image
[JPA] QueryDSL 에서 문자열을 숫자로 뽑아서 변환하기
Entity 안에서 String 으로 관리되고 있는 값을 다른 DTO 클래스에서 받을 때 BigDecimal 과 같은 데이터 타입으로 받고 싶다면 어떻게 할까? 물론, Entity 안에 있는 필드 데이터 타입과 똑같이 관리되면 좋겠지만 뭔가 크롤러 같은 프로그램에서 어딘가의 가격 데이터를 가져올 때 아래와 같은 데이터들이 포함된다고 해보자. '판매중이지 않음' '1200000' '없음' 만약, 위 데이터 중에서 숫자만 이루어진 것을 뽑아내려면 어떻게 해야 할까? 충분히 있을 수 있는 상황이다. 우선 아래와 같은 클래스가 있다고 하고, renewalPrice 라는 필드 값에 위와 같은 데이터들이 포함된다고 생각해보자 @Builder @Getter @NoArgsConstructor @AllArgsConstr..
2022.10.24