2016-08-05 117 views
0

我正在通過命令行調用我的批處理服務併爲作業提供幾個參數。創建作業時需要訪問這些參數,因爲我需要從數據庫中查找數據作爲參數提供的「站點名稱」,並動態創建多個步驟。問題出現在'createJob'方法中。我硬編碼的網站ID,但目前沒有爲itemizedReader方法異常:創建作業時需要訪問作業啓動器參數

Error creating bean with name 'scopedTarget.itemizedReader' defined in billing.BillingConfig: Unsatisfied dependency expressed through method 'itemizedReader' parameter 1: No qualifying bean of type [java.lang.String] 

Spring配置

package billing; 

import billing.components.AspiviaFieldSetter; 
import billing.components.AspiviaPrepStatementSetter; 
import billing.components.SummaryProcessor; 
import billing.mapper.ItemizedCostingMapper; 
import billing.model.BillingItem; 
import billing.model.ItemizedCosting; 
import billing.tasklet.SummaryOutputTasklet; 
import billing.batch.common.AppProps; 
import billing.batch.common.SqlConst; 
import billing.batch.common.model.ItemizedPartner; 
import billing.batch.common.repo.PartnerBillingRepo; 
import com.zaxxer.hikari.HikariConfig; 
import com.zaxxer.hikari.HikariDataSource; 
import java.sql.Timestamp; 
import java.text.SimpleDateFormat; 
import java.util.HashMap; 
import java.util.List; 
import java.util.Map; 
import javax.sql.DataSource; 
import org.apache.logging.log4j.LogManager; 
import org.apache.logging.log4j.Logger; 
import org.springframework.batch.core.Job; 
import org.springframework.batch.core.Step; 
import org.springframework.batch.core.configuration.annotation.EnableBatchProcessing; 
import org.springframework.batch.core.configuration.annotation.JobBuilderFactory; 
import org.springframework.batch.core.configuration.annotation.StepBuilderFactory; 
import org.springframework.batch.core.configuration.annotation.StepScope; 
import org.springframework.batch.core.job.builder.SimpleJobBuilder; 
import org.springframework.batch.core.launch.support.RunIdIncrementer; 
import org.springframework.batch.item.ItemReader; 
import org.springframework.batch.item.ItemWriter; 
import org.springframework.batch.item.database.JdbcBatchItemWriter; 
import org.springframework.batch.item.database.JdbcCursorItemReader; 
import org.springframework.batch.item.file.FlatFileItemReader; 
import org.springframework.batch.item.file.FlatFileItemWriter; 
import org.springframework.batch.item.file.mapping.DefaultLineMapper; 
import org.springframework.batch.item.file.transform.DelimitedLineAggregator; 
import org.springframework.batch.item.file.transform.DelimitedLineTokenizer; 
import org.springframework.batch.item.file.transform.FieldExtractor; 
import org.springframework.batch.repeat.RepeatStatus; 
import org.springframework.beans.factory.annotation.Autowired; 
import org.springframework.beans.factory.annotation.Value; 
import org.springframework.context.annotation.Bean; 
import org.springframework.context.annotation.ComponentScan; 
import org.springframework.context.annotation.Configuration; 
import org.springframework.context.annotation.EnableAspectJAutoProxy; 
import org.springframework.context.annotation.Profile; 
import org.springframework.context.annotation.PropertySource; 
import org.springframework.context.support.PropertySourcesPlaceholderConfigurer; 
import org.springframework.core.io.FileSystemResource; 
import org.springframework.jdbc.core.JdbcTemplate; 
import org.springframework.jdbc.datasource.DataSourceTransactionManager; 

@ComponentScan(basePackages = {"billing", "billing.batch.common"}) 
@Configuration 
@EnableBatchProcessing 
@EnableAspectJAutoProxy 
@PropertySource("classpath:/app.properties") 
public class BillingConfig { 

    private static final Logger LOG = LogManager.getLogger(); 

    @Autowired 
    private AppProps appProps; 

    @Autowired 
    private PartnerBillingRepo billingRepo; 

    @Bean 
    @Profile("prod") 
    public DataSource datasource() { 
     final HikariConfig cfg = new HikariConfig(); 
     cfg.setJdbcUrl(appProps.getPartnerBillingUrl()); 
     cfg.setUsername(appProps.getPartnerBillingUsername()); 
     cfg.setPassword(appProps.getPartnerBillingPassword()); 
     cfg.addDataSourceProperty("cachePrepStmts", appProps.getCachePrepStatements()); 
     cfg.addDataSourceProperty("prepStmtCacheSize", appProps.getPrepStatementCacheSize()); 
     cfg.addDataSourceProperty("prepStmtCacheSqlLimit", appProps.getPrepStatementCacheSqlLimit()); 

     HikariDataSource ds = new HikariDataSource(cfg); 

     return ds; 
    } 

    @Bean 
    public JdbcTemplate template(DataSource ds) { 
     return new JdbcTemplate(ds); 
    } 

    @Bean 
    @StepScope 
    public FlatFileItemReader billingFileReader(@Value("#{jobParameters['input.file']}") String inputFile) { 
     DefaultLineMapper lineMapper = new DefaultLineMapper(); 
     lineMapper.setFieldSetMapper(new BillingFieldSetter()); 
     lineMapper.setLineTokenizer(new DelimitedLineTokenizer()); 

     FlatFileItemReader reader = new FlatFileItemReader(); 
     reader.setLineMapper(lineMapper); 
     reader.setResource(new FileSystemResource(inputFile)); 

     return reader; 
    } 

    @Bean 
    @StepScope 
    public JdbcBatchItemWriter BillingWriter(DataSource ds, BillingPrepStatementSetter setter) { 
     JdbcBatchItemWriter writer = new JdbcBatchItemWriter(); 
     writer.setDataSource(ds); 
     writer.setItemPreparedStatementSetter(setter); 
     writer.setSql(SqlConst.INSERT_INTO_BILLING); 

     return writer; 
    } 

    @Bean 
    @StepScope 
    public BillingPrepStatementSetter prepStatementSetter() { 
     return new BillingPrepStatementSetter(); 
    } 

    @Bean 
    @StepScope 
    public SummaryProcessor summaryProc() { 
     return new SummaryProcessor(); 
    } 

    @Bean 
    @StepScope 
    public SummaryOutputTasklet summaryTask() { 
     return new SummaryOutputTasklet(); 
    } 

    @Bean 
    @StepScope 
    public ItemReader<ItemizedCosting> itemizedReader(@Value("#{jobParameters['site.id']}") Integer siteId, String accountCodes, 
      @Value("#{jobParameter['start.date']") String startDate, @Value("#{jobParameters['end.date']") String endDate) { 

     JdbcCursorItemReader reader = new JdbcCursorItemReader(); 
     reader.setDataSource(datasource()); 
     reader.setSql(SqlConst.SELECT_ITEMIZED_BILLING_FOR_ACCOUNT_CODES); 
     reader.setRowMapper(new ItemizedCostingMapper()); 
     reader.setPreparedStatementSetter((ps) -> { 
      try { 
       SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd"); 

       ps.setTimestamp(0, new Timestamp(formatter.parse(startDate).getTime())); 
       ps.setTimestamp(1, new Timestamp(formatter.parse(endDate).getTime())); 
      } catch (Exception err) { 
       LOG.error("Unable to parse dates, start: {} end: {}", startDate, endDate); 
      } 
      ps.setString(2, accountCodes); 
      ps.setInt(3, siteId); 
     }); 

     return reader; 
    } 

    @Bean 
    @StepScope 
    public ItemWriter<ItemizedCosting> itemizedWriter(@Value("start.date") String startDate, 
      String partnerName) { 

     DelimitedLineAggregator lineAgg = new DelimitedLineAggregator(); 
     FieldExtractor<ItemizedCosting> extractor = (f) -> { 
      Object[] output = new Object[9]; 
      output[0] = f.getExtension(); 
      output[1] = f.getPbxCallTime(); 
      output[2] = f.getDuration(); 
      output[3] = f.getAccountCode(); 
      output[4] = f.getDigits(); 
      output[5] = f.getCost(); 
      output[6] = f.getDestination(); 
      output[7] = f.getCarrier(); 
      output[8] = f.getAttribute(); 

      return output; 
     }; 
     lineAgg.setFieldExtractor(extractor); 

     Timestamp start = null; 

     try { 
      SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd"); 

      start = new Timestamp(formatter.parse(startDate).getTime()); 
     } catch (Exception e) { 
      LOG.error("Unable to parse date: {}", startDate); 
     } 

     FlatFileItemWriter<ItemizedCosting> writer = new FlatFileItemWriter<>(); 
     writer.setEncoding("UTF-8"); 
     writer.setLineAggregator(lineAgg); 
     writer.setResource(new FileSystemResource(String.format("%s/%2$tY-%2$tm_%s_", 
       appProps.getItemizedBillingOutputPath(), start, partnerName))); 

     return writer; 
    } 

    @Bean 
    public Job createJob(JobBuilderFactory jobBuilder, StepBuilderFactory stepBuilders, DataSource ds, FlatFileItemReader reader) 
      throws Exception { 
     Step findSiteIdStep = stepBuilders.get("find.site.id").tasklet((contribution, chunkContext) -> { 
      String siteName 
        = (String) chunkContext.getStepContext().getJobParameters().get(BillingConst.PARAM_SITE); 

      Integer siteId = billingRepo.findSiteIdByName(siteName); 

      chunkContext.getStepContext().getStepExecution().getJobExecution().getExecutionContext().put(
        BillingConst.SITE_ID, siteId); 

      return RepeatStatus.FINISHED; 
     }).build(); 

     Step processFileStep = stepBuilders.get("process.file").<BillingItem, BillingItem>chunk(appProps.getChunkSize()) 
       .reader(reader) 
       .processor(summaryProc()) 
       .writer(aspiviaWriter(ds, prepStatementSetter())).build(); 

     Step outputSummary = stepBuilders.get("output.summary").tasklet(summaryTask()).build(); 

     SimpleJobBuilder builder = jobBuilder.get("process.aspivia").incrementer(new RunIdIncrementer()) 
       .start(findSiteIdStep) 
       .next(processFileStep) 
       .next(outputSummary); 

     List<ItemizedPartner> partners = billingRepo.findPartnersForSite("CPT"); 
     Integer siteId = billingRepo.findSiteIdByName("CPT"); 

     Map<String, String> partnerAccCodes = new HashMap<>(); 

     partners.stream().forEach(i -> { 
      if (!partnerAccCodes.containsKey(i.getPartnerName())) { 
       partnerAccCodes.put(i.getPartnerName(), ""); 
      } 

      String accCodes = partnerAccCodes.get(i.getPartnerName()); 
      accCodes += i.getAccountCode().toString() + ", "; 

      partnerAccCodes.put(i.getPartnerName(), accCodes); 
     }); 

     partnerAccCodes.forEach((k, v) -> { 
      Step itemizedReport = stepBuilders.get("itemized." + k).<ItemizedCosting, ItemizedCosting>chunk(appProps.getChunkSize()) 
        .reader(itemizedReader(siteId, v, null, null)) 
        .writer(itemizedWriter(null, k)).build(); 

      builder.next(itemizedReport); 
     }); 

     return builder.build(); 
    } 

    @Bean 
    public static PropertySourcesPlaceholderConfigurer propCfg() { 
     return new PropertySourcesPlaceholderConfigurer(); 
    } 

    @Bean 
    public DataSourceTransactionManager transactionManager(DataSource datasource) { 
     return new DataSourceTransactionManager(datasource); 
    } 
} 

回答

0

此問題是由於對Spring Batch的是如何工作的生命週期。如果bean是用@StepScope裝飾的,則作業參數只有在啓動後纔可用。

final Job loadAspiviaDataJob = context.getBean(Job.class); 
    final JobLauncher launcher = context.getBean(JobLauncher.class); 

    JobParametersBuilder paramBuilder = new JobParametersBuilder(); 
    paramBuilder.addString(AspiviaConst.PARAM_INPUT_FILE, inputFile); 
    paramBuilder.addString(AspiviaConst.PARAM_SITE, site); 
    paramBuilder.addString(AspiviaConst.PARAM_OUTPUT_FILE_PATH, summaryFile); 

JobExecution runStatus = launcher.run(loadAspiviaDataJob,paramBuilder.toJobParameters());

在上面的代碼相同,我們獲取工作這是通過我的配置createJob Bean方法設置。作業參數不可用。

我做了什麼讓獲得我所需要的值如下:

  • 增加了額外的@PropertySource(「類路徑:cli-runtime.properties」)
  • 的Application.java說啓動spring批處理作業會將我們需要的屬性保存到cli-runtime.properties。在@Configuration類中創建作業時,將從屬性文件加載值,並且可以在需要的作業中創建附加步驟