Spring Batch 5.0 마이그레이션
Spring Batch 5.0은 2022년 11월 16일에 출시되었습니다.
주요 변경 사항
JDK 17 +
Spring Batch 4는 JDK 8 이상, Spring Framework 5.x를 지원했는데,
Spring Batch 5는 최소 버전으로 JDK 17, Spring 6을 기반으로합니다.
- dependency 버전업그레이드
- Spring Framework 6
- Spring Integration 6
- Spring Data 3
- Spring AMQP 3
- Spring for Apache Kafka 3
- Micrometer 1.10
- Java EE 기반의 API에서 Jakarta EE 9 이상을 기반으로 하여 패키지명이 javax 에서 jakarta로 변경되었습니다.
- 기타
- `org.springframework:spring-jdbc`는 이제 `spring-batch-core`의 필수 dependency입니다.
- `junit:junit`은 더 이상 `spring-batch-test`에서 필수 dependency가 아닙니다.
- `com.fasterxml.jackson.core:jackson-core`는 이제 `spring-batch-core`에서 선택 사항입니다.
Database 스키마 업데이트
Oracle
- 새로운 애플리케이션을 위한 Oracle 시퀀스 순서가 정해졌습니다.
- 기존 애플리케이션은 migration-oracle.sql 스크립트를 사용하여 시퀀스를 변경할 수 있습니다.
- DDL 스크립트 이름 변경
- `schema-drop-oracle10g.sql` → `schema-drop-oracle.sql`
- `schema-oracle10g.sql` → `schema-oracle.sql`
MS SQLServer
v4까지는 테이블을 사용하여 시퀀스를 emulate했으나, 이제 실제 시퀀스를 사용합니다.
CREATE SEQUENCE BATCH_STEP_EXECUTION_SEQ START WITH 0 MINVALUE 0 MAXVALUE 9223372036854775807 NO CACHE NO CYCLE;
CREATE SEQUENCE BATCH_JOB_EXECUTION_SEQ START WITH 0 MINVALUE 0 MAXVALUE 9223372036854775807 NO CACHE NO CYCLE;
CREATE SEQUENCE BATCH_JOB_SEQ START WITH 0 MINVALUE 0 MAXVALUE 9223372036854775807 NO CACHE NO CYCLE;
기존 애플리케이션은 시퀀스 테이블의 마지막 값에서 시퀀스를 시작하도록 수정해야 합니다.
All platforms
`BATCH_JOB_EXECUTION` 테이블에서 `JOB_CONFIGURATION_LOCATION` 열이 제거되었습니다.
ALTER TABLE BATCH_JOB_EXECUTION DROP COLUMN JOB_CONFIGURATION_LOCATION;
다양한 Job 파라미터 타입 제공으로 `BATCH_JOB_EXECUTION_PARAMS` 테이블의 컬럼이 변경되었습니다.
CREATE TABLE BATCH_JOB_EXECUTION_PARAMS (
JOB_EXECUTION_ID BIGINT NOT NULL ,
--- TYPE_CD VARCHAR(6) NOT NULL ,
--- KEY_NAME VARCHAR(100) NOT NULL ,
--- STRING_VAL VARCHAR(250) ,
--- DATE_VAL DATETIME(6) DEFAULT NULL ,
--- LONG_VAL BIGINT ,
--- DOUBLE_VAL DOUBLE PRECISION ,
+++ PARAMETER_NAME VARCHAR(100) NOT NULL ,
+++ PARAMETER_TYPE VARCHAR(100) NOT NULL ,
+++ PARAMETER_VALUE VARCHAR(2500) ,
IDENTIFYING CHAR(1) NOT NULL ,
constraint JOB_EXEC_PARAMS_FK foreign key (JOB_EXECUTION_ID)
references BATCH_JOB_EXECUTION(JOB_EXECUTION_ID)
);
마이그레이션 스크립트는 `org/springframework/batch/core/migration/5.0` 경로에 있습니다.
-- v4.3 → v5
ALTER TABLE BATCH_JOB_EXECUTION_PARAMS DROP COLUMN DATE_VAL;
ALTER TABLE BATCH_JOB_EXECUTION_PARAMS DROP COLUMN LONG_VAL;
ALTER TABLE BATCH_JOB_EXECUTION_PARAMS DROP COLUMN DOUBLE_VAL;
ALTER TABLE BATCH_JOB_EXECUTION_PARAMS CHANGE COLUMN TYPE_CD PARAMETER_TYPE VARCHAR(100);
ALTER TABLE BATCH_JOB_EXECUTION_PARAMS CHANGE COLUMN KEY_NAME PARAMETER_NAME VARCHAR(100);
ALTER TABLE BATCH_JOB_EXECUTION_PARAMS CHANGE COLUMN STRING_VAL PARAMETER_VALUE VARCHAR(2500);
v4까지는 4개의 타입(Long, Double, String, Date)만 지원했는데, v5부터는 모든 타입을 JobParameter로 사용할 수 있습니다.
---public class JobParameter implements Serializable {
+++public class JobParameter<T> implements Serializable {
--- private Object parameter;
+++ private T value;
--- private ParameterType parameterType;
+++ private Class<T> type;
}
// sample with v4
parameterName(parameterType)=parameterValue
// sample with v5
// DefaultJobParametersConverter
parameterName=parameterValue,parameterType,identificationFlag
// JsonJobParametersConverter
parameterName='{"value": "parameterValue", "type":"parameterType", "identifying": "booleanValue"}'
`BATCH_STEP_EXECUTION` 테이블에 `CREATE_TIME` 열이 추가되고, `START_TIME` 열의 NOT NULL 제약 조건이 제거되었습니다.
ALTER TABLE BATCH_STEP_EXECUTION ADD CREATE_TIME DATETIME(6) NOT NULL DEFAULT '1970-01-01 00:00:00' AFTER JOB_EXECUTION_ID;
ALTER TABLE BATCH_STEP_EXECUTION MODIFY START_TIME DATETIME(6) NULL;
Infrastructure beans configuration with @EnableBatchProcessing
Map 기반 Job Repository/Explorer 구현은 v4에서 더이상 사용되지 않으며 v5에서 완전히 제거되었고, 대신 JDBC 기반 구현을 사용해야 합니다.
v4.3까지 @EnableBatchProcessing 어노테이션이 TransactionManager Bean을 노출했으나, v5에서는 더 이상 노출되지 않습니다.
// Sample with v4
@Configuration
@EnableBatchProcessing
public class MyStepConfig {
@Autowired
private StepBuilderFactory stepBuilderFactory;
@Bean
public Step myStep() {
return this.stepBuilderFactory.get("myStep")
.tasklet(..) // or .chunk()
.build();
}
}
// Sample with v5
@Configuration
@EnableBatchProcessing
public class MyStepConfig {
@Bean
public Tasklet myTasklet() {
return new MyTasklet();
}
@Bean
public Step myStep(JobRepository jobRepository, Tasklet myTasklet, PlatformTransactionManager transactionManager) {
return new StepBuilder("myStep", jobRepository)
.tasklet(myTasklet, transactionManager) // or .chunk(chunkSize, transactionManager)
.build();
}
}
JobBuilderFactory와 StepBuilderFactory가 Deprecated 되었고, v5.2에서 제거될 예정입니다.
// Sample with v4
@Configuration
@EnableBatchProcessing
public class MyJobConfig {
@Autowired
private JobBuilderFactory jobBuilderFactory;
@Bean
public Job myJob(Step step) {
return this.jobBuilderFactory.get("myJob")
.start(step)
.build();
}
}
// Sample with v5
@Configuration
@EnableBatchProcessing
public class MyJobConfig {
@Bean
public Job myJob(JobRepository jobRepository, Step step) {
return new JobBuilder("myJob", jobRepository)
.start(step)
.build();
}
}
DefaultBatchConfiguration
Spring Batch 5에서는 BatchConfigurer 및 DefaultBatchConfigurer가 제거되었고, 이를 대체하여 @EnableBatchProcessing 또는 DefaultBatchConfiguration을 사용해 설정을 관리합니다.
@EnableBatchProcessing이 JobRepository, StepScope 등의 Bean을 등록하고 처리하던 기능을 대신할 DefaultBatchConfiguration 클래스가 추가되었습니다.
// Sample with v4
@Configuration
public class BatchConfiguration extends DefaultBatchConfigurer {
@Resource(name = BATCH_DATA_SOURCE_NM)
private DataSource dataSource;
@Resource(name = BATCH_TRANSACTION_NM)
private PlatformTransactionManager platformTransactionManager;
protected JobRepository createJobRepository() throws Exception {
JobRepositoryFactoryBean factory = new JobRepositoryFactoryBean();
factory.setDataSource(dataSource);
factory.setTransactionManager(platformTransactionManager);
return factory.getObject();
}
}
// sample with v5
@Configuration
public class BatchConfiguration extends DefaultBatchConfiguration {
@Resource(name = BATCH_DATA_SOURCE_NM)
private DataSource dataSource;
@Resource(name = BATCH_TRANSACTION_NM)
private PlatformTransactionManager platformTransactionManager;
@Override
protected DataSource getDataSource() {
return dataSource;
}
@Override
protected PlatformTransactionManager getTransactionManager() {
return platformTransactionManager;
}
getDataSource, getTransactionManager 등의 다양한 메소드를 제공하고 있습니다.
참고로 DefaultBatchConfiguration을 사용하게되면 BatchAutoConfiguration이 활성화되지 않습니다.
@ConditionalOnMissingBean(value = DefaultBatchConfiguration.class, annotation = EnableBatchProcessing.class)
public class BatchAutoConfiguration {
/** 이하 생략 */
BatchAutoConfiguration 내부에서 정의해주는 JobLauncherApplicationRunner가 job.name으로 지정된 job 이름을 jobName으로 지정해주는 역할을 합니다. 그래서 DefaultBatchConfiguration을 사용할 경우 JobLauncherApplicationRunner을 Bean으로 등록해줘야 실행 argument로 --job.name을 설정할 수 있습니다.
@Configuration
@EnableConfigurationProperties(BatchProperties.class)
public class BatchConfiguration extends DefaultBatchConfiguration {
@Bean
@ConditionalOnMissingBean
@ConditionalOnProperty(prefix = "spring.batch.job", name = "enabled", havingValue = "true", matchIfMissing = true)
public JobLauncherApplicationRunner jobLauncherApplicationRunner(JobLauncher jobLauncher, JobExplorer jobExplorer,
JobRepository jobRepository, BatchProperties properties) {
JobLauncherApplicationRunner runner = new JobLauncherApplicationRunner(jobLauncher, jobExplorer, jobRepository);
String jobName = properties.getJob().getName();
if (StringUtils.hasText(jobName)) {
if (Arrays.stream(BatchJobCd.values()).map(BatchJobCd::getJobName).noneMatch(jobName::equals)) {
throw new IllegalArgumentException(jobName + "는 등록되지 않은 job name입니다. job name을 확인하세요.");
}
runner.setJobName(jobName);
}
return runner;
}
추가로 `spring.batch.job.names`으로 여러 job을 설정하는 것에서 `spring.batch.job.name`으로 하나의 job만 실행하도록 변경되었습니다. 여러 job이 동시에 실행되는 것을 피하고 한 번에 하나의 job만 실행 할 수 있도록 명확히 하기 위함입니다.
spring:
batch:
job:
name: ${job.name:NONE}
기타
Data type 변경
`org.springframework.batch.core.StepExecution` 및 `org.springframework.batch.core.StepContribution`의 메트릭 카운터가 int에서 long으로 변경되었습니다.
`JobExecution` 및 `StepExecution`의 시간 필드가 `java.util.Date`에서 `java.time.LocalDateTime`으로 변경되었습니다.
Observability 변경
Micrometer가 버전 1.10으로 업데이트되었습니다.
모든 태그가 미터 이름으로 접두어가 붙습니다.
Execution Context 직렬화 변경
기본 ExecutionContextSerializer가 Jackson에서 Base64로 변경되었습니다.
SystemCommandTasklet 변경
새로운 전략 인터페이스 `CommandRunner`가 도입되었고, 명령어 실행 방법이 개선되었습니다.
[Reference]
https://docs.spring.io/spring-batch/docs/4.3.x/reference/html/
https://docs.spring.io/spring-batch/docs/5.0.x/reference/html/
https://github.com/spring-projects/spring-batch/wiki/Spring-Batch-5.0-Migration-Guide