Examples of Jakarta Batch

This section provides an example of a Jakarta Batch application developed in JEUS.

1. Overview

The example provided in this section describes a simple application that executes a batch job requested from a JEUS servlet and returns the results to the client.

The following figure shows the structure of the WAR file of a Jakarta batch application.

Example.war
    |--META-INF
    |     |--MANIFEST.MF
    |--WEB-INF
          |--[X]jeus-web-dd.xml
          |--classes
               |--META-INF
                    |--batch-jobs
                    |     |--[X]simple_file_batch.xml
                    |--task
                          |--[T]personList.txt
                    |--tmaxsoft
                          |--jbatch
                               |--test
                                    |--[C]jBatchServlet.class
                                    |--handler
                                         |--[C]FileItemReader.class
                                         |--[C]LogWriter.class
                                         |--[C]Person.class
                                         |--[C]StringToPersonProcessor.class

* Legend
- [01]: binary or executable file
- [X] : XML document
- [J] : JAR file
- [T] : Text file
- [C] : Class file
- [V] : java source file
- [DD] : deployment descriptor

2. Configuring Data Source

Before running a Jakarta Batch application on JEUS, you should define a data source used to execute Jakarta Batch jobs.

In Linux, you can run derby by executing JEUS_HOME/bin/startderby. Specify the export name of the data source as "jdbc/batch".

Only Apache Derby is supported now. Other databases have not been verified yet, and thus are not recommended for use.

The following is an example of a data source defined in the domain.xml file.

Configuring a DataSource: <domain.xml>
<domain>
  <data-source>
  ...
    <database>
      <data-source-id>jdbc/batch</data-source-id>
      <data-source-class-name>
           org.apache.derby.jdbc.ClientConnectionPoolDataSource
      </data-source-class-name>
      <data-source-type>ConnectionPoolDataSource</data-source-type>
      <server-name>localhost</server-name>
      <port-number>1527</port-number>
      <database-name>derbyDB</database-name>
      <user>app</user>
      <password>app</password>
      <property>
        <name>ConnectionAttributes</name>
        <type>java.lang.String</type>
        <value>;create=true</value>
      </property>
    </database>
  </data-source>
</domain>

3. Configuring jeus-web-dd.xml

A thread pool can be configured with the batch-thread-pool element in jeus-web-dd to execute a job. If it is not specified, it is set to the default thread pool. For details, refer to Component DD Configuration.

4. Configuring a Job

Job specifications are defined in an XML file including the name, steps, ItemReader, ItemWriter, and ItemProcessor. The XML file should be located in the META-INF/batch-jobs/ directory.

For a WAR file, a META-INF directory should be placed in the /WEB-INF/classes directory for successful operation.

The following is an example of configuring the /WEB-INF/classes/META-INF/batch-jobs/simple_file_batch.xml file.

Configuring a Job: <simple_file_batch.xml>
<job id="demo" xmlns="http://xmlns.jcp.org/xml/ns/jakartaee" version="1.0">
    <step id="step1">
        <chunk>
            <reader ref="tmaxsoft.bhkim.test.filebatch.FileItemReader">
                <properties>
                    <property name="filePath" value="#{jobParameters['filePath']}" />
                </properties>
            </reader>
            <processor ref="tmaxsoft.bhkim.test.filebatch.StringToPersonProcessor" />
            <writer ref="tmaxsoft.bhkim.test.filebatch.LogWriter" />
        </chunk>
    </step>
</job>

5. JBatchServlet

The JBatchServlet hands the job configuration file (.xml) as well as the JobParameters to the JobOperator.

The following is an example of passing the path to the personList.txt file that contains data used for processing a filePath into a key.

JBatchServlet Example
@WebServlet("/JBatchServlet")
public class JBatchServlet extends HttpServlet {
    private static final long serialVersionUID = 1L;
    private final String FILE_PATH = "META-INF/task/personList.txt";

    public JBatchServlet() {
        super();
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        Properties jobParams = new Properties();
        URL personListURL = Thread.currentThread().getContextClassLoader().getResource(FILE_PATH);
        jobParams.setProperty("filePath", personListURL.getPath());

        JobOperator operator = BatchRuntime.getJobOperator();
        // xml file name without format
        long id = operator.start("simple_file_batch", jobParams);

        waitForEnd(operator, id);

        response.getWriter().println("File Batch is successfully precessed.");
    }

    protected void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        doGet(request, response);
    }

    public static void waitForEnd(final JobOperator jobOperator, final long id) {
        final Collection<BatchStatus> endStatuses = Arrays.asList(BatchStatus.COMPLETED, BatchStatus.FAILED);
        do {
            try {
                Thread.sleep(100);
            } catch (final InterruptedException e) {
                return;
            }
        } while (!endStatuses.contains(jobOperator.getJobExecution(id).getBatchStatus()));
    }
}

FILE_PATH contains the following definitions:

Name1,27,Computer Science
Name2,22,English Education
Name3,23,Electronic Engineering

6. ItemReader

An ItemReader reads an item in. Below is an example of FileItemReader, an ItemReader with four functions—​open, close, readItem, and checkpointInfo. In the following example, the ItemReader reads a line of data as an item.

Serializable, a parameter variable of the open function, verifies a checkpoint so that when a job is abnormally terminated, an ItemReader can begin at the last checkpoint.

Injections may not work for pure batch logic except when made into a servlet or an EJB.

The following is an example of a FileItemReader with the open function implemented to bookmark a recordNumber after every readltem so that the FileItemReader can begin from the last recordNumber in case the job has been abnormally terminated.

FileItemReader Example
public class FileItemReader implements ItemReader {

    @Inject
    @BatchProperty
    private String filePath; // JobProperties in xml setting is injected to the field

    private BufferedReader reader = null;

    private int recordNumber;

    @Override
    public void open(Serializable checkpoint) throws Exception {
        if (filePath == null)
            throw new RuntimeException("Can't find any input");

        final File file = new File(filePath);
        if (!file.exists())
            throw new RuntimeException("A file '" + filePath + "' doesn't exist");

        reader = new BufferedReader(new FileReader(file));

        if (checkpoint != null) {
            assert (checkpoint instanceof Integer);
            recordNumber = (Integer) checkpoint;

            // Pass over latest checkpoint record
            for (int i = 1; i < recordNumber; i++)
                reader.readLine();
        }
    }

    @Override
    public void close() throws Exception {
        if (reader != null)
            reader.close();
    }

    @Override
    public Object readItem() throws Exception {
        Object item = reader.readLine();

        // checkpoint line update
        recordNumber++;
        return item;
    }

    @Override
    public Serializable checkpointInfo() throws Exception {
        return recordNumber;
    }

}

7. ItemWriter

The following is an example of a LogWriter, which is an ItemWriter with four functions—​open, close, wirteItems, and checkpointInfo. The list of items to be processed by writeItem is handed.

Injections may not work for pure batch logic except when made into a servlet or an EJB.

The following is an example of using an internally defined logger to write out a list of items received.

ItemWriter Example
public class LogWriter implements ItemWriter {
    private static final Logger logger = Logger.getLogger(LogWriter.class.getSimpleName());
    int writeRecordNumber;

    @Override
    public void open(Serializable checkpoint) throws Exception {
    }

    @Override
    public void close() throws Exception {
    }


    @Override
    public void writeItems(List<Object> items) throws Exception {
        // simply print passed item to logger
        for (Object o: items) {
            logger.info("writeItems > " + o.toString());
        }
    }

    @Override
    public Serializable checkpointInfo() throws Exception {
        return null;
    }
}

8. ItemProcessor

An ItemProcessor processes items that have been read in by an ItemReader.

Injections may not work for pure batch logic except when made into a servlet or an EJB.

The following is an example of an ItemProcessor that receives as item parameter each line of a file and processes the character strings into the form of a Person object. The objects are passed to an ItemWriter after a phase.

ItemProcessor Example
public class StringToPersonProcessor implements ItemProcessor {

    @Override
    public Object processItem(Object item) throws Exception {
        // Items passed as parameters are arrayed like "name, age, department".
        final String[] line = String.class.cast(item).split(",");

        if (line == null || line.length != 3)
            return null;

        return new Person(line[0], Integer.parseInt(line[1]), line[2]);
    }
}

The following is the definition of the Person object to use in the ItemProcessor above.

Person Object
public class Person {
    private String name;
    private int age;
    private String department;

    public Person(String name, int age, String department) {
        this.name = name;
        this.age = age;
        this.department = department;
    }

    @Override
    public String toString() {
        return name + "," + age + "," + department;
    }
}