web-dev-qa-db-ja.com

Spring Bootがバッチジョブを実行する方法

このサンプル Boot with Spring Batchに従いました。

Mainメソッドを実行すると、ジョブが実行されます。この方法では、ジョブの実行をどのように制御できるかわかりません。たとえば、ジョブのスケジュール方法、ジョブ実行へのアクセス方法、ジョブパラメーターの設定方法などです。

自分のJobLauncherを登録しようとしました

@Bean
public JobLauncher jobLauncher(JobRepository jobRepo){
    SimpleJobLauncher simpleJobLauncher = new SimpleJobLauncher();
    simpleJobLauncher.setJobRepository(jobRepo);
    return simpleJobLauncher;
}

しかし、メインメソッドでそれを使用しようとすると:

public static void main(String[] args) {
    ConfigurableApplicationContext ctx = SpringApplication.run(Application.class, args);    
    JobLauncher jobLauncher = ctx.getBean(JobLauncher.class);
    //try catch removed for readability
    jobLauncher.run(ctx.getBean(Job.class), new JobParameters());   
}

コンテキストがロードされるとジョブが再び実行され、手動で実行しようとするとJobInstanceAlreadyCompleteExceptionが返されます。自動ジョブ実行を防ぐ方法はありますか?

41
Evgeni Dimitrov

設定することでジョブの実行を防ぐことができます

spring.batch.job.enabled=false

application.propertiesで。または、spring.batch.job.namesを使用できます。これは、実行されるジョブ名のコンマ区切りリストを取ります。

ここから: コードを実行するときに、スプリングバッチスケジュールされたジョブが最初に実行されるのを停止する方法?

53
Evgeni Dimitrov

レストコントローラーのPOSTを使用して、ジョブの実行を有効にできます。

@RestController
@RequestMapping(value="/job/")
public class JobLauncherController {

    private static final Log LOG = LogFactory.getLog(JobLauncherController.class);

    @Autowired
    private JobLauncher jobLauncher;

    @Autowired
    private Job job;

    @Autowired
    private JobRepository jobRepository;

    @Autowired
    private JobRegistry jobRegistry;

    @RequestMapping("/launchjob/{jobName}")
    public String handle(@PathVariable("jobName") String jobName, @RequestBody Map<String,Object> request) throws Exception {
        try {           
            request.put("timeJobStarted", DateUtil.getDateFormatted(new Date(), DateUtil.DATE_UUUUMMDDHHMMSS));
            Map<String,Object> mapMessage = this.enrichJobMessage(request);
            Map<String, JobParameter> jobParameters = new HashMap<>();
            mapMessage.forEach((k,v)->{
                MapperUtil.castParameter(jobParameters, k, v);
            });
            jobParameters.put(Field.Batch.JOB_INSTANCE_NAME, new JobParameter(jobName));
            jobLauncher.run(job, new JobParameters(jobParameters));
            assertNotNull(jobRegistry.getJob(job.getName()));
        }catch( NoSuchJobException ex){
            jobRegistry.register(new ReferenceJobFactory(job));
        } catch (Exception e) {
            LOG.error(e.getMessage(),e);
        }

        return "Done";
    }

public static void castParameter(Map<String, JobParameter> jobParameters, String k, Object v){
    if(v instanceof String){
        jobParameters.put(k, new JobParameter((String)v));
    }else if(v instanceof Date){
        jobParameters.put(k, new JobParameter((Date)v));
    }else if(v instanceof Double){
        jobParameters.put(k, new JobParameter((Double)v));
    }else if(v instanceof Long){
        jobParameters.put(k, new JobParameter((Long)v));
    }else{
        DslJson dslJson = new DslJson<>();          
        JsonWriter writer = dslJson.newWriter();
        try {
            dslJson.serialize(writer,v);
            jobParameters.put(k, new JobParameter(writer.toString()));
        } catch (IOException e) {
            LOG.warn(e.getMessage(), e);
        }                       
    }
}

}
1
dmotta