EMMA Coverage Report (generated Fri Aug 21 15:59:46 BST 2009)
[all classes][org.springframework.batch.core.job.flow.support.state]

COVERAGE SUMMARY FOR SOURCE FILE [SplitState.java]

nameclass, %method, %block, %line, %
SplitState.java100% (2/2)100% (6/6)88%  (113/128)88%  (23/26)

COVERAGE BREAKDOWN BY CLASS AND METHOD

nameclass, %method, %block, %line, %
     
class SplitState100% (1/1)100% (4/4)86%  (95/110)88%  (21/24)
handle (FlowExecutor): FlowExecutionStatus 100% (1/1)83%  (72/87)81%  (13/16)
SplitState (Collection, String): void 100% (1/1)100% (17/17)100% (5/5)
isEndState (): boolean 100% (1/1)100% (2/2)100% (1/1)
setTaskExecutor (TaskExecutor): void 100% (1/1)100% (4/4)100% (2/2)
     
class SplitState$1100% (1/1)100% (2/2)100% (18/18)100% (3/3)
SplitState$1 (SplitState, Flow, FlowExecutor): void 100% (1/1)100% (12/12)100% (2/2)
call (): FlowExecution 100% (1/1)100% (6/6)100% (1/1)

1/*
2 * Copyright 2006-2007 the original author or authors.
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16package org.springframework.batch.core.job.flow.support.state;
17 
18import java.util.ArrayList;
19import java.util.Collection;
20import java.util.concurrent.Callable;
21import java.util.concurrent.ExecutionException;
22import java.util.concurrent.Future;
23import java.util.concurrent.FutureTask;
24 
25import org.springframework.batch.core.job.flow.Flow;
26import org.springframework.batch.core.job.flow.FlowExecution;
27import org.springframework.batch.core.job.flow.FlowExecutionException;
28import org.springframework.batch.core.job.flow.FlowExecutionStatus;
29import org.springframework.batch.core.job.flow.FlowExecutor;
30import org.springframework.batch.core.job.flow.State;
31import org.springframework.core.task.SyncTaskExecutor;
32import org.springframework.core.task.TaskExecutor;
33import org.springframework.core.task.TaskRejectedException;
34 
35/**
36 * A {@link State} implementation that splits a {@link Flow} into multiple
37 * parallel subflows.
38 * 
39 * @author Dave Syer
40 * @since 2.0
41 */
42public class SplitState extends AbstractState {
43 
44        private final Collection<Flow> flows;
45 
46        private TaskExecutor taskExecutor = new SyncTaskExecutor();
47 
48        private FlowExecutionAggregator aggregator = new MaxValueFlowExecutionAggregator();
49 
50        /**
51         * @param name
52         */
53        public SplitState(Collection<Flow> flows, String name) {
54                super(name);
55                this.flows = flows;
56        }
57 
58        /**
59         * Public setter for the taskExecutor.
60         * @param taskExecutor the taskExecutor to set
61         */
62        public void setTaskExecutor(TaskExecutor taskExecutor) {
63                this.taskExecutor = taskExecutor;
64        }
65 
66        /**
67         * Execute the flows in parallel by passing them to the {@link TaskExecutor}
68         * and wait for all of them to finish before proceeding.
69         * 
70         * @see State#handle(FlowExecutor)
71         */
72        @Override
73        public FlowExecutionStatus handle(final FlowExecutor executor) throws Exception {
74 
75                // TODO: collect the last StepExecution from the flows as well, so they
76                // can be abandoned if necessary
77                Collection<Future<FlowExecution>> tasks = new ArrayList<Future<FlowExecution>>();
78 
79                for (final Flow flow : flows) {
80 
81                        final FutureTask<FlowExecution> task = new FutureTask<FlowExecution>(new Callable<FlowExecution>() {
82                                public FlowExecution call() throws Exception {
83                                        return flow.start(executor);
84                                }
85                        });
86 
87                        tasks.add(task);
88 
89                        try {
90                                taskExecutor.execute(task);
91                        }
92                        catch (TaskRejectedException e) {
93                                throw new FlowExecutionException("TaskExecutor rejected task for flow=" + flow.getName());
94                        }
95 
96                }
97 
98                Collection<FlowExecution> results = new ArrayList<FlowExecution>();
99 
100                // Could use a CompletionService here?
101                for (Future<FlowExecution> task : tasks) {
102                        try {
103                                results.add(task.get());
104                        }
105                        catch (ExecutionException e) {
106                                // Unwrap the expected exceptions
107                                Throwable cause = e.getCause();
108                                if (cause instanceof Exception) {
109                                        throw (Exception) cause;
110                                } else {
111                                        throw e;
112                                }
113                        }
114                }
115 
116                return aggregator.aggregate(results);
117 
118        }
119 
120        /*
121         * (non-Javadoc)
122         * 
123         * @see org.springframework.batch.core.job.flow.State#isEndState()
124         */
125        public boolean isEndState() {
126                return false;
127        }
128}

[all classes][org.springframework.batch.core.job.flow.support.state]
EMMA 2.0.5312 (C) Vladimir Roubtsov