我spring-batch
在spring-boot
应用程序中使用。Spring Boot 版本是2.3.3.RELEASE
.
我打算达到的目标
我必须read a xml file
包含数千个Transactions
(header tag
fileInformation)。对事务执行一些业务逻辑,然后使用事务中的更新值将文件写回。我StaxEventItemReader
用于读取文件和StaxEventItemWriter
写入文件。然后我有几个ItemProcessors
用于处理业务逻辑。XML 文件看起来像:
<?xml version="1.0" encoding="UTF-8"?>
<reportFile>
<fileInformation>
<sender>200GH7XZ60</sender>
<timestamp>2020-12-23T09:05:34Z</timestamp>
<environment>PRO</environment>
<version>001.60</version>
</fileInformation>
<record>
<transaction>
<buyer><buyer/>
</transaction>
<transaction>
<buyer><buyer/>
</transaction>
<transaction>
<buyer><buyer/>
</transaction>
</record>
</reportFile>
我面临的问题是标题标签的值。
我已经配置了OmegaXmlHeaderCallBack
生成所需的标头标签,但这些标签中的值应该从输入文件中复制。据我所知,它StaxWriterCallback
是在阅读器、处理器和编写器之前初始化的。所以我无法使用late binding
. 这看起来像是一个基本要求,但在stackoverflow
.
这是配置spring批处理作业的代码片段。
@Slf4j
@Configuration
@EnableBatchProcessing
public class BatchConfiguration {
@Autowired
PIExtractorItemProcessor pIExtractorItemProcessor;
@Autowired
JobBuilderFactory jobBuilderFactory;
@Autowired
StepBuilderFactory stepBuilderFactory;
@Value( "${eugateway.batch.chunk.size}" )
private int chunkSize;
@Bean
public Step jobStep(ItemStreamReader<CustomHeaderTransactionXmlElement> reader,
CompositeItemProcessor<CustomHeaderTransactionXmlElement,
ProcessorWriterDto> processor,
CompositeItemWriter<ProcessorWriterDto> writer,
EdsClientItemWriteListener<ProcessorWriterDto> writeListener,
StepBuilderFactory stepBuilderFactory) {
return stepBuilderFactory.get("extractAndReplacePersonalDataStep")
.<CustomHeaderTransactionXmlElement, ProcessorWriterDto>chunk(chunkSize)
.reader(reader)
.processor(processor)
.listener(writeListener)
.writer(writer)
.build();
}
@Bean
public Job extractPersonalDataJob(Step jobStep, JobResultListener jobListener,
JobBuilderFactory jobBuilderFactory) {
return jobBuilderFactory.get("extractAndReplacePersonalDataJob")
.incrementer(new RunIdIncrementer())
.start(jobStep)
.listener(jobListener)
.build();
}
@Bean
@StepScope
public ItemStreamReader<CustomHeaderTransactionXmlElement> itemReader(@Value("#{jobParameters[file.path]}") String path) {
Jaxb2Marshaller transactionMarshaller = new Jaxb2Marshaller();
transactionMarshaller.setClassesToBeBound (FileInformation.class, TransactionPositionReport.class);
log.info("Generating StaxEventItemReader");
return new StaxEventItemReaderBuilder<CustomHeaderTransactionXmlElement>()
.name("headerTransaction")
.resource(new FileSystemResource(new FileSystemResource(path)))
.addFragmentRootElements("fileInformation", "transaction")
.unmarshaller(transactionMarshaller)
.build();
}
@Bean
@StepScope
OmegaXmlHeaderCallBack getOmegaXmlHeaderCallBack(@Value("#{jobExecutionContext['header.sender']}") String sender,
@Value("#{jobExecutionContext['header.timestamp']}") String timestamp,
@Value("#{jobExecutionContext['header.environment']}") String environment,
@Value("#{jobExecutionContext['header.version']}") String version){
return new OmegaXmlHeaderCallBack(sender, timestamp, environment, version);
}
@Bean
@StepScope
OmegaXmlFooterCallBack getOmegaXmlFooterCallBack(){
return new OmegaXmlFooterCallBack();
}
@StepScope
@Bean(name = "staxTransactionWriter")
public StaxEventItemWriter<TransactionPositionReport> staxTransactionItemWriter(@Value("#{jobExecutionContext['header.sender']}") String sender,
@Value("#{jobExecutionContext['header.timestamp']}") String timestamp,
@Value("#{jobExecutionContext['header.environment']}") String environment,
@Value("#{jobExecutionContext['header.version']}") String version) {
String exportFilePath = "C:\\Users\\sasharma\\Documents\\TO_BE_DELETED\\eugateway\\outputfile.xml";
Resource exportFileResource = new FileSystemResource(exportFilePath);
Jaxb2Marshaller marshaller = new Jaxb2Marshaller();
marshaller.setSupportDtd(true);
marshaller.setSupportJaxbElementClass(true);
marshaller.setClassesToBeBound(TransactionPositionReport.class);
return new StaxEventItemWriterBuilder<TransactionPositionReport>()
.name("transactionWriter")
.version("1.0")
.resource(exportFileResource)
.marshaller(marshaller)
.rootTagName("reportFile")
.headerCallback(getOmegaXmlHeaderCallBack(sender, timestamp, environment, version))
.footerCallback(getOmegaXmlFooterCallBack())
.shouldDeleteIfEmpty(true)
.build();
}
@Bean
@StepScope
public PIExtractorItemProcessor extractItemProcessor() {
log.info("Generating PIExtractorItemProcessor");
return new PIExtractorItemProcessor();
}
@Bean
public PIRemoverItemProcessor removeItemProcessor() {
log.info("Generating PIRemoverItemProcessor");
return new PIRemoverItemProcessor();
}
@Bean
@StepScope
CompositeItemProcessor<CustomHeaderTransactionXmlElement, ProcessorWriterDto> extractAndRemoveItemProcessor() {
log.info("Generating CompositeItemProcessor");
CompositeItemProcessor<CustomHeaderTransactionXmlElement, ProcessorWriterDto> itemProcessor = new CompositeItemProcessor<>();
itemProcessor.setDelegates((List<? extends ItemProcessor<?, ?>>) Arrays.asList(extractItemProcessor(), removeItemProcessor()));
return itemProcessor;
}
@Bean
@StepScope
public EdsClientItemWriter<ProcessorWriterDto> edsClientItemWriter() {
log.info("Generating EdsClientItemWriter");
return new EdsClientItemWriter<>();
}
@Bean
@StepScope
public OmegaXmlFileWriter<ProcessorWriterDto> omegaXmlFileWriter(@Value("#{jobExecutionContext['header.sender']}") String sender,
@Value("#{jobExecutionContext['header.timestamp']}") String timestamp,
@Value("#{jobExecutionContext['header.environment']}") String environment,
@Value("#{jobExecutionContext['header.version']}") String version) {
log.info("Generating OmegaXmlFileWriter");
return new OmegaXmlFileWriter(staxTransactionItemWriter(sender, timestamp, environment, version));
}
@Bean
@StepScope
public CompositeItemWriter<ProcessorWriterDto> compositeItemWriter(@Value("#{jobExecutionContext['header.sender']}") String sender,
@Value("#{jobExecutionContext['header.timestamp']}") String timestamp,
@Value("#{jobExecutionContext['header.environment']}") String environment,
@Value("#{jobExecutionContext['header.version']}") String version) {
log.info("Generating CompositeItemWriter");
CompositeItemWriter<ProcessorWriterDto> compositeItemWriter = new CompositeItemWriter<>();
compositeItemWriter.setDelegates(Arrays.asList(edsClientItemWriter(), omegaXmlFileWriter(sender, timestamp, environment, version)));
return compositeItemWriter;
}
}
下面是 OmegaXmlHeaderCallBack
课堂。由于没有后期绑定,我总是在标题标签中得到空值。
@Slf4j
public class OmegaXmlHeaderCallBack implements StaxWriterCallback {
private String sender;
private String timestamp;
private String environment;
private String version;
public OmegaXmlHeaderCallBack(String sender, String timestamp, String environment, String version) {
super();
this.sender = sender;
this.timestamp = timestamp;
this.environment = environment;
this.version = version;
}
@Override
public void write(XMLEventWriter writer) {
XMLEventFactory factory = XMLEventFactory.newInstance();
try {
writer.add(factory.createStartElement("", "", "fileInformation"));
writer.add(factory.createStartElement("", "", "sender"));
writer.add(factory.createCharacters(sender));
writer.add(factory.createEndElement("", "", "sender"));
writer.add(factory.createStartElement("", "", "timestamp"));
writer.add(factory.createCharacters(timestamp));
writer.add(factory.createEndElement("", "", "timestamp"));
writer.add(factory.createStartElement("", "", "environment"));
writer.add(factory.createCharacters(environment));
writer.add(factory.createEndElement("", "", "environment"));
writer.add(factory.createStartElement("", "", "version"));
writer.add(factory.createCharacters(version));
writer.add(factory.createEndElement("", "", "version"));
writer.add(factory.createEndElement("", "", "fileInformation"));
writer.add(factory.createStartElement("", "", "record"));
} catch (XMLStreamException e) {
log.error("Error writing OMEGA XML Header: {}", e.getMessage());
throw new OmegaXmlHeaderWriterException(e.getMessage());
}
}
}
代码ItemProcessor
如下。我正在设置ExecutionContext
headerCallback 打算读取的标题数据(遗憾的是不会发生)。
@Slf4j
public class PIExtractorItemProcessor implements ItemProcessor<CustomHeaderTransactionXmlElement, ProcessorWriterDto> {
@Autowired
PersonalDataExtractor personalDataExtractor;
@Value("#{jobParameters['submission.account']}")
private String subAccntId;
@Value("#{stepExecution}")
private StepExecution stepExecution;
@Override
public ProcessorWriterDto process(CustomHeaderTransactionXmlElement headerTransactionElement) throws Exception {
FileInformation header = null;
TransactionPositionReport transaction = null;
if(headerTransactionElement instanceof FileInformation) {
header = (FileInformation)headerTransactionElement;
stepExecution.getExecutionContext().put("header.sender", header.getSender());
stepExecution.getExecutionContext().put("header.timestamp", header.getTimestamp());
stepExecution.getExecutionContext().put("header.environment", header.getEnvironment());
stepExecution.getExecutionContext().put("header.version", header.getVersion());
log.debug("Header {} found.", header.toString());
return null;
} else {
transaction = (TransactionPositionReport)headerTransactionElement;
log.debug("NO header info found for transaction {}", transaction.getProcessingDetails().getCustomerTransactionId());
log.info("Extracting personal data for transaction customer id {} and create EDS requestDto.", transaction.getProcessingDetails().getCustomerTransactionId());
ProcessorWriterDto transferObject = new ProcessorWriterDto();
transferObject.setEdsRequestDtoList(personalDataExtractor.extract(transaction, subAccntId));
transferObject.setTransaction(transaction);
return transferObject;
}
}
}
我引用的链接: