# Spring Cloud Task Reference Guide
Michael Minella, Glenn Renfro, Jay Bryant
Table of Contents
* [Preface](#preface)
* [1. About the documentation](#about-the-documentation)
* [2. Getting help](#task-documentation-getting-help)
* [3. First Steps](#task-documentation-first-steps)
* [Getting started](#getting-started)
* [4. Introducing Spring Cloud Task](#getting-started-introducing-spring-cloud-task)
* [5. System Requirements](#getting-started-system-requirements)
* [5.1. Database Requirements](#database-requirements)
* [6. Developing Your First Spring Cloud Task Application](#getting-started-developing-first-task)
* [6.1. Creating the Spring Task Project using Spring Initializr](#getting-started-creating-project)
* [6.2. Writing the Code](#getting-started-writing-the-code)
* [6.3. Running the Example](#getting-started-running-the-example)
* [Features](#features)
* [7. The lifecycle of a Spring Cloud Task](#features-lifecycle)
* [7.1. The TaskExecution](#features-task-execution-details)
* [7.2. Mapping Exit Codes](#features-lifecycle-exit-codes)
* [8. Configuration](#features-configuration)
* [8.1. DataSource](#features-data-source)
* [8.2. Table Prefix](#features-table-prefix)
* [8.3. Enable/Disable table initialization](#features-table-initialization)
* [8.4. Externally Generated Task ID](#features-generated_task_id)
* [8.5. External Task Id](#features-external_task_id)
* [8.6. Parent Task Id](#features-parent_task_id)
* [8.7. TaskConfigurer](#features-task-configurer)
* [8.8. Task Name](#features-task-name)
* [8.9. Task Execution Listener](#features-task-execution-listener)
* [8.10. Restricting Spring Cloud Task Instances](#features-single-instance-enabled)
* [8.11. Disabling Spring Cloud Task Auto Configuration](#disabling-spring-cloud-task-auto-configuration)
* [8.12. Closing the Context](#closing-the-context)
* [Batch](#batch)
* [9. Associating a Job Execution to the Task in which It Was Executed](#batch-association)
* [9.1. Overriding the TaskBatchExecutionListener](#batch-association-override)
* [10. Remote Partitioning](#batch-partitioning)
* [10.1. Notes on Developing a Batch-partitioned application for the Kubernetes Platform](#notes-on-developing-a-batch-partitioned-application-for-the-kubernetes-platform)
* [10.2. Notes on Developing a Batch-partitioned Application for the Cloud Foundry Platform](#notes-on-developing-a-batch-partitioned-application-for-the-cloud-foundry-platform)
* [11. Batch Informational Messages](#batch-informational-messages)
* [12. Batch Job Exit Codes](#batch-failures-and-tasks)
* [Single Step Batch Job Starter](#batch-job-starter)
* [13. Defining a Job](#job-definition)
* [13.1. Properties](#job-definition-properties)
* [14. Autoconfiguration for ItemReader Implementations](#item-readers)
* [14.1. AmqpItemReader](#amqpitemreader)
* [14.2. FlatFileItemReader](#flatfileitemreader)
* [14.3. JdbcCursorItemReader](#jdbcCursorItemReader)
* [14.4. KafkaItemReader](#kafkaItemReader)
* [15. ItemProcessor Configuration](#item-processors)
* [16. Autoconfiguration for ItemWriter implementations](#item-writers)
* [16.1. AmqpItemWriter](#amqpitemwriter)
* [16.2. FlatFileItemWriter](#flatfileitemwriter)
* [16.3. JdbcBatchItemWriter](#jdbcitemwriter)
* [16.4. KafkaItemWriter](#kafkaitemwriter)
* [Spring Cloud Stream Integration](#stream-integration)
* [17. Launching a Task from a Spring Cloud Stream](#stream-integration-launching-sink)
* [17.1. Spring Cloud Data Flow](#stream-integration-launching-sink-dataflow)
* [18. Spring Cloud Task Events](#stream-integration-events)
* [18.1. Disabling Specific Task Events](#stream-integration-disable-task-events)
* [19. Spring Batch Events](#stream-integration-batch-events)
* [19.1. Sending Batch Events to Different Channels](#sending-batch-events-to-different-channels)
* [19.2. Disabling Batch Events](#disabling-batch-events)
* [19.3. Emit Order for Batch Events](#emit-order-for-batch-events)
* [Appendices](#appendix)
* [20. Task Repository Schema](#appendix-task-repository-schema)
* [20.1. Table Information](#table-information)
* [20.2. SQL Server](#sql-server)
* [21. Building This Documentation](#appendix-building-the-documentation)
* [22. Running a Task App on Cloud Foundry](#appendix-cloud-foundry)
Version 2.4.1
© 2009-2021 VMware, Inc. All rights reserved.
Copies of this document may be made for your own use and for distribution to
others, provided that you do not charge any fee for such copies and further
provided that each copy contains this Copyright Notice, whether distributed in
print or electronically.
# [](#preface)[Preface](#preface)
This section provides a brief overview of the Spring Cloud Task reference documentation.
Think of it as a map for the rest of the document. You can read this reference guide in a
linear fashion or you can skip sections if something does not interest you.
## [](#about-the-documentation)[1. About the documentation](#about-the-documentation)
The Spring Cloud Task reference guide is available in [html](https://docs.spring.io/spring-cloud-task/docs/current/reference)and [pdf](https://docs.spring.io/spring-cloud-task/docs/current/reference/index.pdf),[epub](https://docs.spring.io/spring-cloud-task/docs/current/reference/index.epub) . The
latest copy is available at [docs.spring.io/spring-cloud-task/docs/current-SNAPSHOT/reference/html/](https://docs.spring.io/spring-cloud-task/docs/current-SNAPSHOT/reference/html/).
Copies of this document may be made for your own use and for distribution to others,
provided that you do not charge any fee for such copies and further provided that each
copy contains this Copyright Notice, whether distributed in print or electronically.
## [](#task-documentation-getting-help)[2. Getting help](#task-documentation-getting-help)
Having trouble with Spring Cloud Task? We would like to help!
* Ask a question. We monitor [stackoverflow.com](https://stackoverflow.com) for questions
tagged with [`spring-cloud-task`](https://stackoverflow.com/tags/spring-cloud-task).
* Report bugs with Spring Cloud Task at[github.com/spring-cloud/spring-cloud-task/issues](https://github.com/spring-cloud/spring-cloud-task/issues).
| |All of Spring Cloud Task is open source, including the documentation. If you find
a problem with the docs or if you just want to improve them, please [get
involved](https://github.com/spring-cloud/spring-cloud-task/tree/master).|
|---|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
## [](#task-documentation-first-steps)[3. First Steps](#task-documentation-first-steps)
If you are just getting started with Spring Cloud Task or with 'Spring' in general, we
suggesting reading the [Getting started](#getting-started) chapter.
To get started from scratch, read the following sections:
* [Introducing Spring Cloud Task](#getting-started-introducing-spring-cloud-task)
* [System Requirements](#getting-started-system-requirements)
To follow the tutorial, read[Developing Your First Spring Cloud Task Application](#getting-started-developing-first-task)
To run your example, read[Running the Example](#getting-started-running-the-example)
# [](#getting-started)[Getting started](#getting-started)
If you are just getting started with Spring Cloud Task, you should read this section.
Here, we answer the basic “what?”, “how?”, and “why?” questions. We start with a
gentle introduction to Spring Cloud Task. We then build a Spring Cloud Task application,
discussing some core principles as we go.
## [](#getting-started-introducing-spring-cloud-task)[4. Introducing Spring Cloud Task](#getting-started-introducing-spring-cloud-task)
Spring Cloud Task makes it easy to create short-lived microservices. It provides
capabilities that let short lived JVM processes be executed on demand in a production
environment.
## [](#getting-started-system-requirements)[5. System Requirements](#getting-started-system-requirements)
You need to have Java installed (Java 8 or better). To build, you need to have Maven
installed as well.
### [](#database-requirements)[5.1. Database Requirements](#database-requirements)
Spring Cloud Task uses a relational database to store the results of an executed task.
While you can begin developing a task without a database (the status of the task is logged
as part of the task repository’s updates), for production environments, you want to
use a supported database. Spring Cloud Task currently supports the following databases:
* DB2
* H2
* HSQLDB
* MySql
* Oracle
* Postgres
* SqlServer
## [](#getting-started-developing-first-task)[6. Developing Your First Spring Cloud Task Application](#getting-started-developing-first-task)
A good place to start is with a simple “Hello, World!” application, so we create the
Spring Cloud Task equivalent to highlight the features of the framework. Most IDEs have
good support for Apache Maven, so we use it as the build tool for this project.
| |The spring.io web site contains many [“`Getting Started`”
guides](https://spring.io/guides) that use Spring Boot. If you need to solve a specific problem, check there first.
You can shortcut the following steps by going to the[Spring Initializr](https://start.spring.io/) and creating a new project. Doing so
automatically generates a new project structure so that you can start coding right away.
We recommend experimenting with the Spring Initializr to become familiar with it.|
|---|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
### [](#getting-started-creating-project)[6.1. Creating the Spring Task Project using Spring Initializr](#getting-started-creating-project)
Now we can create and test an application that prints `Hello, World!` to the console.
To do so:
1. Visit the [Spring Initialzr](https://start.spring.io/) site.
1. Create a new Maven project with a **Group** name of `io.spring.demo` and an **Artifact** name of `helloworld`.
2. In the Dependencies text box, type `task` and then select the `Cloud Task` dependency.
3. In the Dependencies text box, type `jdbc` and then select the `JDBC` dependency.
4. In the Dependencies text box, type `h2` and then select the `H2`. (or your favorite database)
5. Click the **Generate Project** button
2. Unzip the helloworld.zip file and import the project into your favorite IDE.
### [](#getting-started-writing-the-code)[6.2. Writing the Code](#getting-started-writing-the-code)
To finish our application, we need to update the generated `HelloworldApplication` with the following contents so that it launches a Task.
```
package io.spring.demo.helloworld;
import org.springframework.boot.CommandLineRunner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.Bean;
@SpringBootApplication
@EnableTask
public class HelloworldApplication {
@Bean
public CommandLineRunner commandLineRunner() {
return new HelloWorldCommandLineRunner();
}
public static void main(String[] args) {
SpringApplication.run(HelloworldApplication.class, args);
}
public static class HelloWorldCommandLineRunner implements CommandLineRunner {
@Override
public void run(String... strings) throws Exception {
System.out.println("Hello, World!");
}
}
}
```
While it may seem small, quite a bit is going on. For more about Spring
Boot specifics, see the[Spring Boot reference documentation](https://docs.spring.io/spring-boot/docs/current/reference/html/).
Now we can open the `application.properties` file in `src/main/resources`.
We need to configure two properties in `application.properties`:
* `application.name`: To set the application name (which is translated to the task name)
* `logging.level`: To set the logging for Spring Cloud Task to `DEBUG` in order to
get a view of what is going on.
The following example shows how to do both:
```
logging.level.org.springframework.cloud.task=DEBUG
spring.application.name=helloWorld
```
#### [](#getting-started-at-task)[6.2.1. Task Auto Configuration](#getting-started-at-task)
When including Spring Cloud Task Starter dependency, Task auto configures all beans to bootstrap it’s functionality.
Part of this configuration registers the `TaskRepository` and the infrastructure for its use.
In our demo, the `TaskRepository` uses an embedded H2 database to record the results
of a task. This H2 embedded database is not a practical solution for a production environment, since
the H2 DB goes away once the task ends. However, for a quick getting-started
experience, we can use this in our example as well as echoing to the logs what is being updated
in that repository. In the [Configuration](#features-configuration) section (later in this
documentation), we cover how to customize the configuration of the pieces provided by
Spring Cloud Task.
When our sample application runs, Spring Boot launches our `HelloWorldCommandLineRunner`and outputs our “Hello, World!” message to standard out. The `TaskLifecycleListener`records the start of the task and the end of the task in the repository.
#### [](#getting-started-main-method)[6.2.2. The main method](#getting-started-main-method)
The main method serves as the entry point to any java application. Our main method
delegates to Spring Boot’s [SpringApplication](https://docs.spring.io/spring-boot/docs/current/reference/html/boot-features-spring-application.html) class.
#### [](#getting-started-clr)[6.2.3. The CommandLineRunner](#getting-started-clr)
Spring includes many ways to bootstrap an application’s logic. Spring Boot provides
a convenient method of doing so in an organized manner through its `*Runner` interfaces
(`CommandLineRunner` or `ApplicationRunner`). A well behaved task can bootstrap any
logic by using one of these two runners.
The lifecycle of a task is considered from before the `*Runner#run` methods are executed
to once they are all complete. Spring Boot lets an application use multiple`*Runner` implementations, as does Spring Cloud Task.
| |Any processing bootstrapped from mechanisms other than a `CommandLineRunner` or`ApplicationRunner` (by using `InitializingBean#afterPropertiesSet` for example) is not
recorded by Spring Cloud Task.|
|---|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
### [](#getting-started-running-the-example)[6.3. Running the Example](#getting-started-running-the-example)
At this point, our application should work. Since this application is Spring Boot-based,
we can run it from the command line by using `$ mvn spring-boot:run` from the root
of our application, as shown (with its output) in the following example:
```
$ mvn clean spring-boot:run
....... . . .
....... . . . (Maven log output here)
....... . . .
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v2.0.3.RELEASE)
2018-07-23 17:44:34.426 INFO 1978 --- [ main] i.s.d.helloworld.HelloworldApplication : Starting HelloworldApplication on Glenns-MBP-2.attlocal.net with PID 1978 (/Users/glennrenfro/project/helloworld/target/classes started by glennrenfro in /Users/glennrenfro/project/helloworld)
2018-07-23 17:44:34.430 INFO 1978 --- [ main] i.s.d.helloworld.HelloworldApplication : No active profile set, falling back to default profiles: default
2018-07-23 17:44:34.472 INFO 1978 --- [ main] s.c.a.AnnotationConfigApplicationContext : Refreshing org.spring[email protected]1d24f32d: startup date [Mon Jul 23 17:44:34 EDT 2018]; root of context hierarchy
2018-07-23 17:44:35.280 INFO 1978 --- [ main] com.zaxxer.hikari.HikariDataSource : HikariPool-1 - Starting...
2018-07-23 17:44:35.410 INFO 1978 --- [ main] com.zaxxer.hikari.HikariDataSource : HikariPool-1 - Start completed.
2018-07-23 17:44:35.419 DEBUG 1978 --- [ main] o.s.c.t.c.SimpleTaskConfiguration : Using org.springframework.cloud.task.configuration.DefaultTaskConfigurer TaskConfigurer
2018-07-23 17:44:35.420 DEBUG 1978 --- [ main] o.s.c.t.c.DefaultTaskConfigurer : No EntityManager was found, using DataSourceTransactionManager
2018-07-23 17:44:35.522 DEBUG 1978 --- [ main] o.s.c.t.r.s.TaskRepositoryInitializer : Initializing task schema for h2 database
2018-07-23 17:44:35.525 INFO 1978 --- [ main] o.s.jdbc.datasource.init.ScriptUtils : Executing SQL script from class path resource [org/springframework/cloud/task/schema-h2.sql]
2018-07-23 17:44:35.558 INFO 1978 --- [ main] o.s.jdbc.datasource.init.ScriptUtils : Executed SQL script from class path resource [org/springframework/cloud/task/schema-h2.sql] in 33 ms.
2018-07-23 17:44:35.728 INFO 1978 --- [ main] o.s.j.e.a.AnnotationMBeanExporter : Registering beans for JMX exposure on startup
2018-07-23 17:44:35.730 INFO 1978 --- [ main] o.s.j.e.a.AnnotationMBeanExporter : Bean with name 'dataSource' has been autodetected for JMX exposure
2018-07-23 17:44:35.733 INFO 1978 --- [ main] o.s.j.e.a.AnnotationMBeanExporter : Located MBean 'dataSource': registering with JMX server as MBean [com.zaxxer.hikari:name=dataSource,type=HikariDataSource]
2018-07-23 17:44:35.738 INFO 1978 --- [ main] o.s.c.support.DefaultLifecycleProcessor : Starting beans in phase 0
2018-07-23 17:44:35.762 DEBUG 1978 --- [ main] o.s.c.t.r.support.SimpleTaskRepository : Creating: TaskExecution{executionId=0, parentExecutionId=null, exitCode=null, taskName='application', startTime=Mon Jul 23 17:44:35 EDT 2018, endTime=null, exitMessage='null', externalExecutionId='null', errorMessage='null', arguments=[]}
2018-07-23 17:44:35.772 INFO 1978 --- [ main] i.s.d.helloworld.HelloworldApplication : Started HelloworldApplication in 1.625 seconds (JVM running for 4.764)
Hello, World!
2018-07-23 17:44:35.782 DEBUG 1978 --- [ main] o.s.c.t.r.support.SimpleTaskRepository : Updating: TaskExecution with executionId=1 with the following {exitCode=0, endTime=Mon Jul 23 17:44:35 EDT 2018, exitMessage='null', errorMessage='null'}
```
The preceding output has three lines that of interest to us here:
* `SimpleTaskRepository` logged the creation of the entry in the `TaskRepository`.
* The execution of our `CommandLineRunner`, demonstrated by the “Hello, World!” output.
* `SimpleTaskRepository` logs the completion of the task in the `TaskRepository`.
| |A simple task application can be found in the samples module of the Spring Cloud
Task Project[here](https://github.com/spring-cloud/spring-cloud-task/tree/master/spring-cloud-task-samples/timestamp).|
|---|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
# [](#features)[Features](#features)
This section goes into more detail about Spring Cloud Task, including how to use it, how
to configure it, and the appropriate extension points.
## [](#features-lifecycle)[7. The lifecycle of a Spring Cloud Task](#features-lifecycle)
In most cases, the modern cloud environment is designed around the execution of processes
that are not expected to end. If they do end, they are typically restarted. While most
platforms do have some way to run a process that is not restarted when it ends, the
results of that run are typically not maintained in a consumable way. Spring Cloud
Task offers the ability to execute short-lived processes in an environment and record the
results. Doing so allows for a microservices architecture around short-lived processes as
well as longer running services through the integration of tasks by messages.
While this functionality is useful in a cloud environment, the same issues can arise in a
traditional deployment model as well. When running Spring Boot applications with a
scheduler such as cron, it can be useful to be able to monitor the results of the
application after its completion.
Spring Cloud Task takes the approach that a Spring Boot application can have a start and
an end and still be successful. Batch applications are one example of how processes that
are expected to end (and that are often short-lived) can be helpful.
Spring Cloud Task records the lifecycle events of a given task. Most long-running
processes, typified by most web applications, do not save their lifecycle events. The
tasks at the heart of Spring Cloud Task do.
The lifecycle consists of a single task execution. This is a physical execution of a
Spring Boot application configured to be a task (that is, it has the Sprint Cloud Task dependencies).
At the beginning of a task, before any `CommandLineRunner` or `ApplicationRunner`implementations have been run, an entry in the `TaskRepository` that records the start
event is created. This event is triggered through `SmartLifecycle#start` being triggered
by the Spring Framework. This indicates to the system that all beans are ready for use and
comes before running any of the `CommandLineRunner` or `ApplicationRunner` implementations
provided by Spring Boot.
| |The recording of a task only occurs upon the successful bootstrapping of an`ApplicationContext`. If the context fails to bootstrap at all, the task’s run is not
recorded.|
|---|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
Upon completion of all of the `*Runner#run` calls from Spring Boot or the failure of an`ApplicationContext` (indicated by an `ApplicationFailedEvent`), the task execution is
updated in the repository with the results.
| |If the application requires the `ApplicationContext` to be closed at the
completion of a task (all `*Runner#run` methods have been called and the task
repository has been updated), set the property `spring.cloud.task.closecontextEnabled`to true.|
|---|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
### [](#features-task-execution-details)[7.1. The TaskExecution](#features-task-execution-details)
The information stored in the `TaskRepository` is modeled in the `TaskExecution` class and
consists of the following information:
| Field | Description |
|--------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
|`executionid` | The unique ID for the task’s run. |
| `exitCode` |The exit code generated from an `ExitCodeExceptionMapper` implementation. If there is no
exit code generated but an `ApplicationFailedEvent` is thrown, 1 is set. Otherwise, it is
assumed to be 0.|
| `taskName` | The name for the task, as determined by the configured `TaskNameResolver`. |
| `startTime` | The time the task was started, as indicated by the `SmartLifecycle#start` call. |
| `endTime` | The time the task was completed, as indicated by the `ApplicationReadyEvent`. |
|`exitMessage` | Any information available at the time of exit. This can programmatically be set by a`TaskExecutionListener`. |
|`errorMessage`| If an exception is the cause of the end of the task (as indicated by an`ApplicationFailedEvent`), the stack trace for that exception is stored here. |
| `arguments` | A `List` of the string command line arguments as they were passed into the executable
boot application. |
### [](#features-lifecycle-exit-codes)[7.2. Mapping Exit Codes](#features-lifecycle-exit-codes)
When a task completes, it tries to return an exit code to the OS. If we take a look
at our [original example](#getting-started-developing-first-task), we can see that we are
not controlling that aspect of our application. So, if an exception is thrown, the JVM
returns a code that may or may not be of any use to you in debugging.
Consequently, Spring Boot provides an interface, `ExitCodeExceptionMapper`, that lets you
map uncaught exceptions to exit codes. Doing so lets you indicate, at the level of exit
codes, what went wrong. Also, by mapping exit codes in this manner, Spring Cloud Task
records the returned exit code.
If the task terminates with a SIG-INT or a SIG-TERM, the exit code is zero unless
otherwise specified within the code.
| |While the task is running, the exit code is stored as a null in the repository.
Once the task completes, the appropriate exit code is stored based on the guidelines described
earlier in this section.|
|---|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
## [](#features-configuration)[8. Configuration](#features-configuration)
Spring Cloud Task provides a ready-to-use configuration, as defined in the`DefaultTaskConfigurer` and `SimpleTaskConfiguration` classes. This section walks through
the defaults and how to customize Spring Cloud Task for your needs.
### [](#features-data-source)[8.1. DataSource](#features-data-source)
Spring Cloud Task uses a datasource for storing the results of task executions. By
default, we provide an in-memory instance of H2 to provide a simple method of
bootstrapping development. However, in a production environment, you probably want to
configure your own `DataSource`.
If your application uses only a single `DataSource` and that serves as both your business
schema and the task repository, all you need to do is provide any `DataSource` (the
easiest way to do so is through Spring Boot’s configuration conventions). This`DataSource` is automatically used by Spring Cloud Task for the repository.
If your application uses more than one `DataSource`, you need to configure the task
repository with the appropriate `DataSource`. This customization can be done through an
implementation of `TaskConfigurer`.
### [](#features-table-prefix)[8.2. Table Prefix](#features-table-prefix)
One modifiable property of `TaskRepository` is the table prefix for the task tables. By
default, they are all prefaced with `TASK_`. `TASK_EXECUTION` and `TASK_EXECUTION_PARAMS`are two examples. However, there are potential reasons to modify this prefix. If the
schema name needs to be prepended to the table names or if more than one set of task
tables is needed within the same schema, you must change the table prefix. You can do so
by setting the `spring.cloud.task.tablePrefix` to the prefix you need, as follows:
`spring.cloud.task.tablePrefix=yourPrefix`
By using the `spring.cloud.task.tablePrefix`, a user assumes the responsibility to
create the task tables that meet both the criteria for the task table schema but
with modifications that are required for a user’s business needs.
You can utilize the Spring Cloud Task Schema DDL as a guide when creating your own Task DDL as seen[here](https://github.com/spring-cloud/spring-cloud-task/tree/master/spring-cloud-task-core/src/main/resources/org/springframework/cloud/task).
### [](#features-table-initialization)[8.3. Enable/Disable table initialization](#features-table-initialization)
In cases where you are creating the task tables and do not wish for Spring Cloud Task to
create them at task startup, set the `spring.cloud.task.initialize-enabled` property to`false`, as follows:
`spring.cloud.task.initialize-enabled=false`
It defaults to `true`.
| |The property `spring.cloud.task.initialize.enable` has been deprecated.|
|---|-----------------------------------------------------------------------|
### [](#features-generated_task_id)[8.4. Externally Generated Task ID](#features-generated_task_id)
In some cases, you may want to allow for the time difference between when a task is
requested and when the infrastructure actually launches it. Spring Cloud Task lets you
create a `TaskExecution` when the task is requested. Then pass the execution ID of the
generated `TaskExecution` to the task so that it can update the `TaskExecution` through
the task’s lifecycle.
A `TaskExecution` can be created by calling the `createTaskExecution` method on an
implementation of the `TaskRepository` that references the datastore that holds
the `TaskExecution` objects.
In order to configure your Task to use a generated `TaskExecutionId`, add the
following property:
`spring.cloud.task.executionid=yourtaskId`
### [](#features-external_task_id)[8.5. External Task Id](#features-external_task_id)
Spring Cloud Task lets you store an external task ID for each`TaskExecution`. An example of this would be a task ID provided by
Cloud Foundry when a task is launched on the platform.
In order to configure your Task to use a generated `TaskExecutionId`, add the
following property:
`spring.cloud.task.external-execution-id=`
### [](#features-parent_task_id)[8.6. Parent Task Id](#features-parent_task_id)
Spring Cloud Task lets you store a parent task ID for each `TaskExecution`. An example of
this would be a task that executes another task or tasks and you want to record which task
launched each of the child tasks. In order to configure your Task to set a parent`TaskExecutionId` add the following property on the child task:
`spring.cloud.task.parent-execution-id=`
### [](#features-task-configurer)[8.7. TaskConfigurer](#features-task-configurer)
The `TaskConfigurer` is a strategy interface that lets you customize the way components of
Spring Cloud Task are configured. By default, we provide the `DefaultTaskConfigurer` that
provides logical defaults: `Map`-based in-memory components (useful for development if no`DataSource` is provided) and JDBC based components (useful if there is a `DataSource`available).
The `TaskConfigurer` lets you configure three main components:
| Component | Description | Default (provided by `DefaultTaskConfigurer`) |
|----------------------------|------------------------------------------------------------------------------------------------------------------|-------------------------------------------------------------------------------------------------------|
| `TaskRepository` | The implementation of the `TaskRepository` to be used. | `SimpleTaskRepository` |
| `TaskExplorer` |The implementation of the `TaskExplorer` (a component for read-only access to the task
repository) to be used.| `SimpleTaskExplorer` |
|`PlatformTransactionManager`| A transaction manager to be used when running updates for tasks. |`DataSourceTransactionManager` if a `DataSource` is used.`ResourcelessTransactionManager` if it is not.|
You can customize any of the components described in the preceding table by creating a
custom implementation of the `TaskConfigurer` interface. Typically, extending the`DefaultTaskConfigurer` (which is provided if a `TaskConfigurer` is not found) and
overriding the required getter is sufficient. However, implementing your own from scratch
may be required.
| |Users should not directly use getter methods from a `TaskConfigurer` directly
unless they are using it to supply implementations to be exposed as Spring Beans.|
|---|-------------------------------------------------------------------------------------------------------------------------------------------------------------------|
### [](#features-task-name)[8.8. Task Name](#features-task-name)
In most cases, the name of the task is the application name as configured in Spring
Boot. However, there are some cases where you may want to map the run of a task to a
different name. Spring Cloud Data Flow is an example of this (because you probably want
the task to be run with the name of the task definition). Because of this, we offer the
ability to customize how the task is named, through the `TaskNameResolver` interface.
By default, Spring Cloud Task provides the `SimpleTaskNameResolver`, which uses the
following options (in order of precedence):
1. A Spring Boot property (configured in any of the ways Spring Boot allows) called`spring.cloud.task.name`.
2. The application name as resolved using Spring Boot’s rules (obtained through`ApplicationContext#getId`).
### [](#features-task-execution-listener)[8.9. Task Execution Listener](#features-task-execution-listener)
`TaskExecutionListener` lets you register listeners for specific events that occur during
the task lifecycle. To do so, create a class that implements the`TaskExecutionListener` interface. The class that implements the `TaskExecutionListener`interface is notified of the following events:
* `onTaskStartup`: Prior to storing the `TaskExecution` into the `TaskRepository`.
* `onTaskEnd`: Prior to updating the `TaskExecution` entry in the `TaskRepository` and
marking the final state of the task.
* `onTaskFailed`: Prior to the `onTaskEnd` method being invoked when an unhandled
exception is thrown by the task.
Spring Cloud Task also lets you add `TaskExecution` Listeners to methods within a bean
by using the following method annotations:
* `@BeforeTask`: Prior to the storing the `TaskExecution` into the `TaskRepository`
* `@AfterTask`: Prior to the updating of the `TaskExecution` entry in the `TaskRepository`marking the final state of the task.
* `@FailedTask`: Prior to the `@AfterTask` method being invoked when an unhandled
exception is thrown by the task.
The following example shows the three annotations in use:
```
public class MyBean {
@BeforeTask
public void methodA(TaskExecution taskExecution) {
}
@AfterTask
public void methodB(TaskExecution taskExecution) {
}
@FailedTask
public void methodC(TaskExecution taskExecution, Throwable throwable) {
}
}
```
| |Inserting an `ApplicationListener` earlier in the chain than `TaskLifecycleListener` exists may cause unexpected effects.|
|---|-------------------------------------------------------------------------------------------------------------------------|
#### [](#features-task-execution-listener-Exceptions)[8.9.1. Exceptions Thrown by Task Execution Listener](#features-task-execution-listener-Exceptions)
If an exception is thrown by a `TaskExecutionListener` event handler, all listener
processing for that event handler stops. For example, if three `onTaskStartup` listeners
have started and the first `onTaskStartup` event handler throws an exception, the other
two `onTaskStartup` methods are not called. However, the other event handlers (`onTaskEnd`and `onTaskFailed`) for the `TaskExecutionListeners` are called.
The exit code returned when a exception is thrown by a `TaskExecutionListener`event handler is the exit code that was reported by the[ExitCodeEvent](https://docs.spring.io/spring-boot/docs/current/api/org/springframework/boot/ExitCodeEvent.html).
If no `ExitCodeEvent` is emitted, the Exception thrown is evaluated to see
if it is of type[ExitCodeGenerator](https://docs.spring.io/spring-boot/docs/current/reference/htmlsingle/#boot-features-application-exit).
If so, it returns the exit code from the `ExitCodeGenerator`. Otherwise, `1`is returned.
In the case that an exception is thrown in an `onTaskStartup` method, the exit code for the application will be `1`.
If an exception is thrown in either a `onTaskEnd` or `onTaskFailed`method, the exit code for the application will be the one established using the rules enumerated above.
| |In the case of an exception being thrown in a `onTaskStartup`, `onTaskEnd`, or `onTaskFailed`you can not override the exit code for the application using `ExitCodeExceptionMapper`.|
|---|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
#### [](#features-task-execution-listener-exit-messages)[8.9.2. Exit Messages](#features-task-execution-listener-exit-messages)
You can set the exit message for a task programmatically by using a`TaskExecutionListener`. This is done by setting the `TaskExecution’s` `exitMessage`,
which then gets passed into the `TaskExecutionListener`. The following example shows
a method that is annotated with the `@AfterTask` `ExecutionListener` :
```
@AfterTask
public void afterMe(TaskExecution taskExecution) {
taskExecution.setExitMessage("AFTER EXIT MESSAGE");
}
```
An `ExitMessage` can be set at any of the listener events (`onTaskStartup`,`onTaskFailed`, and `onTaskEnd`). The order of precedence for the three listeners follows:
1. `onTaskEnd`
2. `onTaskFailed`
3. `onTaskStartup`
For example, if you set an `exitMessage` for the `onTaskStartup` and `onTaskFailed`listeners and the task ends without failing, the `exitMessage` from the `onTaskStartup`is stored in the repository. Otherwise, if a failure occurs, the `exitMessage` from
the `onTaskFailed` is stored. Also if you set the `exitMessage` with an`onTaskEnd` listener, the `exitMessage` from the `onTaskEnd` supersedes
the exit messages from both the `onTaskStartup` and `onTaskFailed`.
### [](#features-single-instance-enabled)[8.10. Restricting Spring Cloud Task Instances](#features-single-instance-enabled)
Spring Cloud Task lets you establish that only one task with a given task name can be run
at a time. To do so, you need to establish the [task name](#features-task-name) and set`spring.cloud.task.single-instance-enabled=true` for each task execution. While the first
task execution is running, any other time you try to run a task with the same[task name](#features-task-name) and`spring.cloud.task.single-instance-enabled=true`, the
task fails with the following error message: `Task with name "application" is already
running.` The default value for `spring.cloud.task.single-instance-enabled` is `false`. The
following example shows how to set `spring.cloud.task.single-instance-enabled` to `true`:
`spring.cloud.task.single-instance-enabled=true or false`
To use this feature, you must add the following Spring Integration dependencies to your
application:
```
org.springframework.integration
spring-integration-core
org.springframework.integration
spring-integration-jdbc
```
| |The exit code for the application will be 1 if the task fails because this feature
is enabled and another task is running with the same task name.|
|---|------------------------------------------------------------------------------------------------------------------------------------------------------|
### [](#disabling-spring-cloud-task-auto-configuration)[8.11. Disabling Spring Cloud Task Auto Configuration](#disabling-spring-cloud-task-auto-configuration)
In cases where Spring Cloud Task should not be auto configured for an implementation, you can disable Task’s auto configuration.
This can be done either by adding the following annotation to your Task application:
```
@EnableAutoConfiguration(exclude={SimpleTaskAutoConfiguration.class})
```
You may also disable Task auto configuration by setting the `spring.cloud.task.autoconfiguration.enabled` property to `false`.
### [](#closing-the-context)[8.12. Closing the Context](#closing-the-context)
If the application requires the `ApplicationContext` to be closed at the
completion of a task (all `*Runner#run` methods have been called and the task
repository has been updated), set the property `spring.cloud.task.closecontextEnabled`to `true`.
Another case to close the context is when the Task Execution completes however the application does not terminate.
In these cases the context is held open because a thread has been allocated
(for example: if you are using a TaskExecutor). In these cases
set the `spring.cloud.task.closecontextEnabled` property to `true` when launching your task.
This will close the application’s context once the task is complete.
Thus allowing the application to terminate.
# [](#batch)[Batch](#batch)
This section goes into more detail about Spring Cloud Task’s integration with Spring
Batch. Tracking the association between a job execution and the task in which it was
executed as well as remote partitioning through Spring Cloud Deployer are covered in
this section.
## [](#batch-association)[9. Associating a Job Execution to the Task in which It Was Executed](#batch-association)
Spring Boot provides facilities for the execution of batch jobs within an über-jar.
Spring Boot’s support of this functionality lets a developer execute multiple batch jobs
within that execution. Spring Cloud Task provides the ability to associate the execution
of a job (a job execution) with a task’s execution so that one can be traced back to the
other.
Spring Cloud Task achieves this functionality by using the `TaskBatchExecutionListener`.
By default,
this listener is auto configured in any context that has both a Spring Batch Job
configured (by having a bean of type `Job` defined in the context) and the`spring-cloud-task-batch` jar on the classpath. The listener is injected into all jobs
that meet those conditions.
### [](#batch-association-override)[9.1. Overriding the TaskBatchExecutionListener](#batch-association-override)
To prevent the listener from being injected into any batch jobs within the current
context, you can disable the autoconfiguration by using standard Spring Boot mechanisms.
To only have the listener injected into particular jobs within the context, override the`batchTaskExecutionListenerBeanPostProcessor` and provide a list of job bean IDs, as shown
in the following example:
```
public TaskBatchExecutionListenerBeanPostProcessor batchTaskExecutionListenerBeanPostProcessor() {
TaskBatchExecutionListenerBeanPostProcessor postProcessor =
new TaskBatchExecutionListenerBeanPostProcessor();
postProcessor.setJobNames(Arrays.asList(new String[] {"job1", "job2"}));
return postProcessor;
}
```
| |You can find a sample batch application in the samples module of the Spring Cloud
Task Project,[here](https://github.com/spring-cloud/spring-cloud-task/tree/master/spring-cloud-task-samples/batch-job).|
|---|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
## [](#batch-partitioning)[10. Remote Partitioning](#batch-partitioning)
Spring Cloud Deployer provides facilities for launching Spring Boot-based applications on
most cloud infrastructures. The `DeployerPartitionHandler` and`DeployerStepExecutionHandler` delegate the launching of worker step executions to Spring
Cloud Deployer.
To configure the `DeployerStepExecutionHandler`, you must provide a `Resource`representing the Spring Boot über-jar to be executed, a `TaskLauncher`, and a`JobExplorer`. You can configure any environment properties as well as the max number of
workers to be executing at once, the interval to poll for the results (defaults to 10
seconds), and a timeout (defaults to -1 or no timeout). The following example shows how
configuring this `PartitionHandler` might look:
```
@Bean
public PartitionHandler partitionHandler(TaskLauncher taskLauncher,
JobExplorer jobExplorer) throws Exception {
MavenProperties mavenProperties = new MavenProperties();
mavenProperties.setRemoteRepositories(new HashMap<>(Collections.singletonMap("springRepo",
new MavenProperties.RemoteRepository(repository))));
Resource resource =
MavenResource.parse(String.format("%s:%s:%s",
"io.spring.cloud",
"partitioned-batch-job",
"1.1.0.RELEASE"), mavenProperties);
DeployerPartitionHandler partitionHandler =
new DeployerPartitionHandler(taskLauncher, jobExplorer, resource, "workerStep");
List commandLineArgs = new ArrayList<>(3);
commandLineArgs.add("--spring.profiles.active=worker");
commandLineArgs.add("--spring.cloud.task.initialize.enable=false");
commandLineArgs.add("--spring.batch.initializer.enabled=false");
partitionHandler.setCommandLineArgsProvider(
new PassThroughCommandLineArgsProvider(commandLineArgs));
partitionHandler.setEnvironmentVariablesProvider(new NoOpEnvironmentVariablesProvider());
partitionHandler.setMaxWorkers(2);
partitionHandler.setApplicationName("PartitionedBatchJobTask");
return partitionHandler;
}
```
| |When passing environment variables to partitions, each partition may
be on a different machine with different environment settings.
Consequently, you should pass only those environment variables that are required.|
|---|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
Notice in the example above that we have set the maximum number of workers to 2.
Setting the maximum of workers establishes the maximum number of
partitions that should be running at one time.
The `Resource` to be executed is expected to be a Spring Boot über-jar with a`DeployerStepExecutionHandler` configured as a `CommandLineRunner` in the current context.
The repository enumerated in the preceding example should be the remote repository in
which the über-jar is located. Both the manager and worker are expected to have visibility
into the same data store being used as the job repository and task repository. Once the
underlying infrastructure has bootstrapped the Spring Boot jar and Spring Boot has
launched the `DeployerStepExecutionHandler`, the step handler executes the requested`Step`. The following example shows how to configure the `DeployerStepExecutionHandler`:
```
@Bean
public DeployerStepExecutionHandler stepExecutionHandler(JobExplorer jobExplorer) {
DeployerStepExecutionHandler handler =
new DeployerStepExecutionHandler(this.context, jobExplorer, this.jobRepository);
return handler;
}
```
| |You can find a sample remote partition application in the samples module of the
Spring Cloud Task project,[here](https://github.com/spring-cloud/spring-cloud-task/tree/master/spring-cloud-task-samples/partitioned-batch-job).|
|---|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
### [](#notes-on-developing-a-batch-partitioned-application-for-the-kubernetes-platform)[10.1. Notes on Developing a Batch-partitioned application for the Kubernetes Platform](#notes-on-developing-a-batch-partitioned-application-for-the-kubernetes-platform)
* When deploying partitioned apps on the Kubernetes platform, you must use the following
dependency for the Spring Cloud Kubernetes Deployer:
```
org.springframework.cloud
spring-cloud-starter-deployer-kubernetes
```
* The application name for the task application and its partitions need to follow
the following regex pattern: `[a-z0-9]([-a-z0-9]*[a-z0-9])`.
Otherwise, an exception is thrown.
### [](#notes-on-developing-a-batch-partitioned-application-for-the-cloud-foundry-platform)[10.2. Notes on Developing a Batch-partitioned Application for the Cloud Foundry Platform](#notes-on-developing-a-batch-partitioned-application-for-the-cloud-foundry-platform)
* When deploying partitioned apps on the Cloud Foundry platform, you must use the
following dependencies for the Spring Cloud Foundry Deployer:
```
org.springframework.cloud
spring-cloud-deployer-cloudfoundry
io.projectreactor
reactor-core
3.1.5.RELEASE
io.projectreactor.ipc
reactor-netty
0.7.5.RELEASE
```
* When configuring the partition handler, Cloud Foundry Deployment
environment variables need to be established so that the partition handler
can start the partitions. The following list shows the required environment
variables:
* `spring_cloud_deployer_cloudfoundry_url`
* `spring_cloud_deployer_cloudfoundry_org`
* `spring_cloud_deployer_cloudfoundry_space`
* `spring_cloud_deployer_cloudfoundry_domain`
* `spring_cloud_deployer_cloudfoundry_username`
* `spring_cloud_deployer_cloudfoundry_password`
* `spring_cloud_deployer_cloudfoundry_services`
* `spring_cloud_deployer_cloudfoundry_taskTimeout`
An example set of deployment environment variables for a partitioned task that
uses a `mysql` database service might resemble the following:
```
spring_cloud_deployer_cloudfoundry_url=https://api.local.pcfdev.io
spring_cloud_deployer_cloudfoundry_org=pcfdev-org
spring_cloud_deployer_cloudfoundry_space=pcfdev-space
spring_cloud_deployer_cloudfoundry_domain=local.pcfdev.io
spring_cloud_deployer_cloudfoundry_username=admin
spring_cloud_deployer_cloudfoundry_password=admin
spring_cloud_deployer_cloudfoundry_services=mysql
spring_cloud_deployer_cloudfoundry_taskTimeout=300
```
| |When using PCF-Dev, the following environment variable is also required:`spring_cloud_deployer_cloudfoundry_skipSslValidation=true`|
|---|-----------------------------------------------------------------------------------------------------------------------------------|
## [](#batch-informational-messages)[11. Batch Informational Messages](#batch-informational-messages)
Spring Cloud Task provides the ability for batch jobs to emit informational messages. The
“[Spring Batch Events](#stream-integration-batch-events)” section covers this feature in detail.
## [](#batch-failures-and-tasks)[12. Batch Job Exit Codes](#batch-failures-and-tasks)
As discussed [earlier](#features-lifecycle-exit-codes), Spring Cloud Task
applications support the ability to record the exit code of a task execution. However, in
cases where you run a Spring Batch Job within a task, regardless of how the Batch Job
Execution completes, the result of the task is always zero when using the default
Batch/Boot behavior. Keep in mind that a task is a boot application and that the exit code
returned from the task is the same as a boot application.
To override this behavior and allow the task to return an exit code other than zero when a
batch job returns an[BatchStatus](https://docs.spring.io/spring-batch/4.0.x/reference/html/step.html#batchStatusVsExitStatus)of `FAILED`, set `spring.cloud.task.batch.fail-on-job-failure` to `true`. Then the exit code
can be 1 (the default) or be based on the[specified`ExitCodeGenerator`](https://docs.spring.io/spring-boot/docs/current/reference/html/boot-features-spring-application.html#boot-features-application-exit))
This functionality uses a new `CommandLineRunner` that replaces the one provided by Spring
Boot. By default, it is configured with the same order. However, if you want to customize
the order in which the `CommandLineRunner` is run, you can set its order by setting the`spring.cloud.task.batch.commandLineRunnerOrder` property. To have your task return the
exit code based on the result of the batch job execution, you need to write your own`CommandLineRunner`.
# [](#batch-job-starter)[Single Step Batch Job Starter](#batch-job-starter)
This section goes into how to develop a Spring Batch `Job` with a single `Step` by using the
starter included in Spring Cloud Task. This starter lets you use configuration
to define an `ItemReader`, an `ItemWriter`, or a full single-step Spring Batch `Job`.
For more about Spring Batch and its capabilities, see the[Spring Batch documentation](https://spring.io/projects/spring-batch).
To obtain the starter for Maven, add the following to your build:
```
org.springframework.cloud
spring-cloud-starter-single-step-batch-job
2.3.0
```
To obtain the starter for Gradle, add the following to your build:
```
compile "org.springframework.cloud:spring-cloud-starter-single-step-batch-job:2.3.0"
```
## [](#job-definition)[13. Defining a Job](#job-definition)
You can use the starter to define as little as an `ItemReader` or an `ItemWriter` or as much as a full `Job`.
In this section, we define which properties are required to be defined to configure a`Job`.
### [](#job-definition-properties)[13.1. Properties](#job-definition-properties)
To begin, the starter provides a set of properties that let you configure the basics of a Job with one Step:
| Property | Type |Default Value| Description |
|----------------------------|---------|-------------|----------------------------------------------------|
| `spring.batch.job.jobName` |`String` | `null` | The name of the job. |
|`spring.batch.job.stepName` |`String` | `null` | The name of the step. |
|`spring.batch.job.chunkSize`|`Integer`| `null` |The number of items to be processed per transaction.|
With the above properties configured, you have a job with a single, chunk-based step.
This chunk-based step reads, processes, and writes `Map` instances as the
items. However, the step does not yet do anything. You need to configure an `ItemReader`, an
optional `ItemProcessor`, and an `ItemWriter` to give it something to do. To configure one
of these, you can either use properties and configure one of the options that has provided
autoconfiguration or you can configure your own with the standard Spring configuration
mechanisms.
| |If you configure your own, the input and output types must match the others in the step.
The `ItemReader` implementations and `ItemWriter` implementations in this starter all use
a `Map` as the input and the output item.|
|---|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
## [](#item-readers)[14. Autoconfiguration for ItemReader Implementations](#item-readers)
This starter provides autoconfiguration for four different `ItemReader` implementations:`AmqpItemReader`, `FlatFileItemReader`, `JdbcCursorItemReader`, and `KafkaItemReader`.
In this section, we outline how to configure each of these by using the provided
autoconfiguration.
### [](#amqpitemreader)[14.1. AmqpItemReader](#amqpitemreader)
You can read from a queue or topic with AMQP by using the `AmqpItemReader`. The
autoconfiguration for this `ItemReader` implementation is dependent upon two sets of
configuration. The first is the configuration of an `AmqpTemplate`. You can either
configure this yourself or use the autoconfiguration provided by Spring Boot. See the[Spring Boot AMQP documentation](https://docs.spring.io/spring-boot/docs/2.4.x/reference/htmlsingle/#boot-features-amqp).
Once you have configured the `AmqpTemplate`, you can enable the batch capabilities to support it
by setting the following properties:
| Property | Type |Default Value| Description |
|------------------------------------------------------|---------|-------------|---------------------------------------------------------------------------------------|
| `spring.batch.job.amqpitemreader.enabled` |`boolean`| `false` | If `true`, the autoconfiguration will execute. |
|`spring.batch.job.amqpitemreader.jsonConverterEnabled`|`boolean`| `true` |Indicates if the `Jackson2JsonMessageConverter` should be registered to parse messages.|
For more information, see the [`AmqpItemReader` documentation](https://docs.spring.io/spring-batch/docs/4.3.x/api/org/springframework/batch/item/amqp/AmqpItemReader.html).
### [](#flatfileitemreader)[14.2. FlatFileItemReader](#flatfileitemreader)
`FlatFileItemReader` lets you read from flat files (such as CSVs
and other file formats). To read from a file, you can provide some components
yourself through normal Spring configuration (`LineTokenizer`, `RecordSeparatorPolicy`,`FieldSetMapper`, `LineMapper`, or `SkippedLinesCallback`). You can also use the
following properties to configure the reader:
| Property | Type | Default Value | Description |
|------------------------------------------------------|---------------|------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| `spring.batch.job.flatfileitemreader.saveState` | `boolean` | `true` | Determines if the state should be saved for restarts. |
| `spring.batch.job.flatfileitemreader.name` | `String` | `null` | Name used to provide unique keys in the `ExecutionContext`. |
| `spring.batch.job.flatfileitemreader.maxItemcount` | `int` | `Integer.MAX_VALUE` | Maximum number of items to be read from the file. |
|`spring.batch.job.flatfileitemreader.currentItemCount`| `int` | 0 | Number of items that have already been read. Used on restarts. |
| `spring.batch.job.flatfileitemreader.comments` |`List` | empty List | A list of Strings that indicate commented lines (lines to be ignored) in the file. |
| `spring.batch.job.flatfileitemreader.resource` | `Resource` | `null` | The resource to be read. |
| `spring.batch.job.flatfileitemreader.strict` | `boolean` | `true` | If set to `true`, the reader throws an exception if the resource is not found. |
| `spring.batch.job.flatfileitemreader.encoding` | `String` | `FlatFileItemReader.DEFAULT_CHARSET` | Encoding to be used when reading the file. |
| `spring.batch.job.flatfileitemreader.linesToSkip` | `int` | 0 | Indicates the number of lines to skip at the start of a file. |
| `spring.batch.job.flatfileitemreader.delimited` | `boolean` | `false` | Indicates whether the file is a delimited file (CSV and other formats). Only one of this property or `spring.batch.job.flatfileitemreader.fixedLength` can be `true` at the same time. |
| `spring.batch.job.flatfileitemreader.delimiter` | `String` | `DelimitedLineTokenizer.DELIMITER_COMMA` | If reading a delimited file, indicates the delimiter to parse on. |
| `spring.batch.job.flatfileitemreader.quoteCharacter` | `char` |`DelimitedLineTokenizer.DEFAULT_QUOTE_CHARACTER`| Used to determine the character used to quote values. |
| `spring.batch.job.flatfileitemreader.includedFields` |`List`| empty list | A list of indices to determine which fields in a record to include in the item. |
| `spring.batch.job.flatfileitemreader.fixedLength` | `boolean` | `false` | Indicates if a file’s records are parsed by column numbers. Only one of this property or `spring.batch.job.flatfileitemreader.delimited` can be `true` at the same time. |
| `spring.batch.job.flatfileitemreader.ranges` | `List` | empty list |List of column ranges by which to parse a fixed width record. See the [Range documentation](https://docs.spring.io/spring-batch/docs/4.3.x/api/org/springframework/batch/item/file/transform/Range.html).|
| `spring.batch.job.flatfileitemreader.names` | `String []` | `null` | List of names for each field parsed from a record. These names are the keys in the `Map` in the items returned from this `ItemReader`. |
| `spring.batch.job.flatfileitemreader.parsingStrict` | `boolean` | `true` | If set to `true`, the mapping fails if the fields cannot be mapped. |
See the [`FlatFileItemReader` documentation](https://docs.spring.io/spring-batch/docs/4.3.x/api/org/springframework/batch/item/file/FlatFileItemReader.html).
### [](#jdbcCursorItemReader)[14.3. JdbcCursorItemReader](#jdbcCursorItemReader)
The `JdbcCursorItemReader` runs a query against a relational database and iterates over
the resulting cursor (`ResultSet`) to provide the resulting items. This autoconfiguration
lets you provide a `PreparedStatementSetter`, a `RowMapper`, or both. You
can also use the following properties to configure a `JdbcCursorItemReader`:
| Property | Type | Default Value | Description |
|-------------------------------------------------------------------|---------|-------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| `spring.batch.job.jdbccursoritemreader.saveState` |`boolean`| `true` | Determines whether the state should be saved for restarts. |
| `spring.batch.job.jdbccursoritemreader.name` |`String` | `null` | Name used to provide unique keys in the `ExecutionContext`. |
| `spring.batch.job.jdbccursoritemreader.maxItemcount` | `int` |`Integer.MAX_VALUE`| Maximum number of items to be read from the file. |
| `spring.batch.job.jdbccursoritemreader.currentItemCount` | `int` | 0 | Number of items that have already been read. Used on restarts. |
| `spring.batch.job.jdbccursoritemreader.fetchSize` | `int` | |A hint to the driver to indicate how many records to retrieve per call to the database system. For best performance, you usually want to set it to match the chunk size.|
| `spring.batch.job.jdbccursoritemreader.maxRows` | `int` | | Maximum number of items to read from the database. |
| `spring.batch.job.jdbccursoritemreader.queryTimeout` | `int` | | Number of milliseconds for the query to timeout. |
| `spring.batch.job.jdbccursoritemreader.ignoreWarnings` |`boolean`| `true` | Determines whether the reader should ignore SQL warnings when processing. |
| `spring.batch.job.jdbccursoritemreader.verifyCursorPosition` |`boolean`| `true` | Indicates whether the cursor’s position should be verified after each read to verify that the `RowMapper` did not advance the cursor. |
| `spring.batch.job.jdbccursoritemreader.driverSupportsAbsolute` |`boolean`| `false` | Indicates whether the driver supports absolute positioning of a cursor. |
|`spring.batch.job.jdbccursoritemreader.useSharedExtendedConnection`|`boolean`| `false` | Indicates whether the connection is shared with other processing (and is therefore part of a transaction). |
| `spring.batch.job.jdbccursoritemreader.sql` |`String` | `null` | SQL query from which to read. |
See the [`JdbcCursorItemReader` documentation](https://docs.spring.io/spring-batch/docs/4.3.x/api/org/springframework/batch/item/database/JdbcCursorItemReader.html).
### [](#kafkaItemReader)[14.4. KafkaItemReader](#kafkaItemReader)
Ingesting a partition of data from a Kafka topic is useful and exactly what the`KafkaItemReader` can do. To configure a `KafkaItemReader`, two pieces
of configuration are required. First, configuring Kafka with Spring Boot’s Kafka
autoconfiguration is required (see the[Spring Boot Kafka documentation](https://docs.spring.io/spring-boot/docs/2.4.x/reference/htmlsingle/#boot-features-kafka)).
Once you have configured the Kafka properties from Spring Boot, you can configure the `KafkaItemReader`itself by setting the following properties:
| Property | Type |Default Value| Description |
|-------------------------------------------------------|---------------|-------------|-----------------------------------------------------------|
| `spring.batch.job.kafkaitemreader.name` | `String` | `null` |Name used to provide unique keys in the `ExecutionContext`.|
| `spring.batch.job.kafkaitemreader.topic` | `String` | `null` | Name of the topic from which to read. |
| `spring.batch.job.kafkaitemreader.partitions` |`List`| empty list | List of partition indices from which to read. |
|`spring.batch.job.kafkaitemreader.pollTimeOutInSeconds`| `long` | 30 | Timeout for the `poll()` operations. |
| `spring.batch.job.kafkaitemreader.saveState` | `boolean` | `true` |Determines whether the state should be saved for restarts. |
See the [`KafkaItemReader` documentation](https://docs.spring.io/spring-batch/docs/4.3.x/api/org/springframework/batch/item/kafka/KafkaItemReader.html).
## [](#item-processors)[15. ItemProcessor Configuration](#item-processors)
The single-step batch job autoconfiguration accepts an `ItemProcessor` if one
is available within the `ApplicationContext`. If one is found of the correct type
(`ItemProcessor