io.leonard.maven.plugins.jspc.JspcMojo Maven / Gradle / Ivy
//========================================================================
//$Id: JspcMojo.java 6430 2011-03-15 17:52:17Z joakime $
//Copyright 2006 Mort Bay Consulting Pty. Ltd.
//------------------------------------------------------------------------
//Licensed under the Apache License, Version 2.0 (the "License");
//you may not use this file except in compliance with the License.
//You may obtain a copy of the License at
//http://www.apache.org/licenses/LICENSE-2.0
//Unless required by applicable law or agreed to in writing, software
//distributed under the License is distributed on an "AS IS" BASIS,
//WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//See the License for the specific language governing permissions and
//limitations under the License.
//========================================================================
package io.leonard.maven.plugins.jspc;
import java.io.*;
import java.net.*;
import java.nio.charset.StandardCharsets;
import java.nio.file.*;
import java.util.*;
import java.util.concurrent.*;
import javax.xml.XMLConstants;
import javax.xml.parsers.*;
import javax.xml.transform.Source;
import javax.xml.transform.stream.StreamSource;
import javax.xml.validation.*;
import org.apache.jasper.*;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.plugin.*;
import org.apache.maven.plugins.annotations.*;
import org.apache.maven.plugins.annotations.Mojo;
import org.apache.maven.project.MavenProject;
import org.codehaus.plexus.util.*;
import org.eclipse.jdt.internal.compiler.env.NameEnvironmentAnswer;
import org.xml.sax.SAXException;
/**
*
* This goal will compile jsps for a webapp so that they can be included in a
* war.
*
*
* It is a fork of jetty-jspc-maven-plugin
* but has the following improvements:
*
*
* - Faster: ability of multi-threading see threads parameter
* - Not stop at the first error : see stopAtFirstError parameter
*
*
* The compiler used in this plugin the Apache Jasper 9.0.12 but it can be
* overloaded.
*
*
* @author Leonard Ehrenfried
* @description Runs jspc compiler to produce .java and .class files
*/
@Mojo(name = "compile", defaultPhase = LifecyclePhase.PROCESS_CLASSES, requiresDependencyResolution = ResolutionScope.COMPILE, threadSafe = true)
public class JspcMojo extends AbstractMojo {
private static final String WEB_XML = "web.xml";
public static final String END_OF_WEBAPP = "";
/**
* The maven project.
*/
@Parameter(defaultValue = "${project}", readonly = true, required = true)
private MavenProject project;
/**
* File into which to generate the <servlet> and <servlet-mapping>
* tags for the compiled jsps.
*
* If multithreading mode is active (threads > 1), then this filename will be
* suffixed by ".threadIndex" (example : webfrag.xml.3).
*/
@Parameter(defaultValue = "${basedir}/target/webfrag.xml")
private String webXmlFragment;
/**
* Optional. A marker string in the src web.xml file which indicates where to
* merge in the generated web.xml fragment. Note that the marker string will NOT
* be preserved during the insertion. Can be left blank, in which case the
* generated fragment is inserted just before the </web-app> line
*/
@Parameter
private String insertionMarker;
/**
* Merge the generated fragment file with the web.xml from
* webAppSourceDirectory. The merged file will go into the same directory as the
* webXmlFragment.
*/
@Parameter(defaultValue = "true")
private boolean mergeFragment;
/**
* The destination directory into which to put the compiled jsps.
*/
@Parameter(defaultValue = "${project.build.outputDirectory}")
private String generatedClasses;
/**
* Controls whether or not .java files generated during compilation will be
* preserved.
*/
@Parameter(defaultValue = "false")
private boolean keepSources;
/**
* Default root package for all generated classes
*/
@Parameter(defaultValue = "jsp")
private String packageRoot;
/**
* Root directory for all html/jsp etc files
*/
@Parameter(defaultValue = "${basedir}/src/main/webapp")
private String webAppSourceDirectory;
/**
* Location of web.xml. Defaults to src/main/webapp/web.xml.
*/
@Parameter(defaultValue = "${basedir}/src/main/webapp/WEB-INF/web.xml")
private String webXml;
/**
* The comma separated list of patterns for file extensions to be processed. By
* default will include all .jsp and .jspx files.
*/
@Parameter(defaultValue = "**\\/*.jsp, **\\/*.jspx, **\\/*.jspf")
private String[] includes;
/**
* The comma separated list of file name patters to exclude from compilation.
*/
@Parameter(defaultValue = "**\\/.svn\\/**")
private String[] excludes;
/**
* The location of the compiled classes for the webapp
*/
@Parameter(defaultValue = "${project.build.outputDirectory}")
private File classesDirectory;
/**
* Whether or not to output more verbose messages during compilation.
*/
@Parameter(defaultValue = "false")
private boolean verbose;
/**
* If true, validates tlds when parsing.
*/
@Parameter(defaultValue = "false")
private boolean validateXml;
/**
* If true, validates web.xml file (xml format)
* after beeing merge if mergeFragment parameter is true
*/
@Parameter(defaultValue = "true")
private boolean validateWebXmlAfterMerge;
/**
* If true, validates web.xml file (xsd see webXmlXsdSchema parameter)
* after beeing merge if mergeFragment parameter is true
*/
@Parameter(defaultValue = "false")
private boolean validateWebXmlWithXsdAfterMerge;
/**
* The link to xsd schema to validate web xml file after merging, if
* mergeFragment parameter is true.
*/
@Parameter(defaultValue = "http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd")
private String webXmlXsdSchema;
/**
* Optionnal hostname of http proxy (use when validating dtd/xsd with external URL)
*/
@Parameter
private String httpProxyHost;
/**
* Optionnal port of http proxy (use when validating dtd/xsd with external URL)
*/
@Parameter
private String httpProxyPort;
/**
* Optionnal no proxy hosts (use when validating dtd/xsd with external URL)
* A list of hosts that should be reached directly, bypassing the proxy.
* This is a list of patterns separated by '|'.
* The patterns may start or end with a '*' for wildcards.
* Any host matching one of these patterns will be reached through a direct connection instead of through a proxy.
* See https://docs.oracle.com/javase/8/docs/technotes/guides/net/proxies.html
*/
@Parameter
private String httpNoProxyHosts;
private boolean proxyEnvSet;
private String httpProxyHostBackup;
private String httpProxyPortBackup;
private String httpNoProxyHostsBackup;
/**
* The encoding scheme to use.
*/
@Parameter(defaultValue = "UTF-8")
private String javaEncoding;
/**
* Whether or not to generate JSR45 compliant debug info
*/
@Parameter(defaultValue = "true")
private boolean suppressSmap;
/**
* Whether or not to ignore precompilation errors caused by jsp fragments.
*/
@Parameter(defaultValue = "false")
private boolean ignoreJspFragmentErrors;
/**
* Fail the build and stop at the first jspc error. If set to "false", all jsp
* will be compiled even if they raise errors, and all errors will be listed
* when they raise. In this case the build will fail too. In case of threads > 1
* and stopAtFirstError=true, each thread can have is own first error.
*/
@Parameter(defaultValue = "true")
private boolean stopAtFirstError;
/**
* The number of threads will be used for compile all of the jsps. Number total
* of jsps will be divided by thread number. Each part will be given to
* differents thread.
*/
@Parameter(defaultValue = "1")
private int threads;
/**
* Whether Jsp Tag Pooling should be enabled.
*/
@Parameter(defaultValue = "true")
private boolean enableJspTagPooling;
/**
* Should white spaces in template text between actions or directives be
* trimmed?
*/
@Parameter(defaultValue = "false")
private boolean trimSpaces;
/**
* Should text strings be generated as char arrays, to improve performance in
* some cases?
*/
@Parameter(defaultValue = "false")
private boolean genStringAsCharArray;
/**
* Version of Java used to compile the jsp files.
*/
@Parameter(defaultValue = "1.8")
private String compilerVersion;
/**
* Name of the compiler class used to compile the jsp files. If threads
* parameter is greater than 2, then maybe the compilerClass
* "org.apache.jasper.compiler.ParallelJDTCompiler" will be more efficient
*/
@Parameter(defaultValue = "org.apache.jasper.compiler.JDTCompiler")
private String compilerClass;
private Map resourcesCache = new ConcurrentHashMap<>();
@Override
public void execute() throws MojoExecutionException, MojoFailureException {
if (getLog().isDebugEnabled()) {
getLog().info("verbose=" + verbose);
getLog().info("webAppSourceDirectory=" + webAppSourceDirectory);
getLog().info("generatedClasses=" + generatedClasses);
getLog().info("webXmlFragment=" + webXmlFragment);
getLog().info("webXml=" + webXml);
getLog().info("validateXml=" + validateXml);
getLog().info("packageRoot=" + packageRoot);
getLog().info("javaEncoding=" + javaEncoding);
getLog().info("insertionMarker="
+ (insertionMarker == null || insertionMarker.equals("") ? END_OF_WEBAPP : insertionMarker));
getLog().info("keepSources=" + keepSources);
getLog().info("mergeFragment=" + mergeFragment);
getLog().info("suppressSmap=" + suppressSmap);
getLog().info("ignoreJspFragmentErrors=" + ignoreJspFragmentErrors);
getLog().info("webXmlXsdSchema=" + webXmlXsdSchema);
getLog().info("stopAtFirstError=" + stopAtFirstError);
getLog().info("threads=" + threads);
getLog().info("enableJspTagPooling=" + enableJspTagPooling);
getLog().info("trimSpaces=" + trimSpaces);
getLog().info("genStringAsCharArray=" + genStringAsCharArray);
getLog().info("compilerVersion=" + compilerVersion);
getLog().info("compilerClass=" + compilerClass);
}
try {
long start = System.currentTimeMillis();
prepare();
compile();
cleanupSrcs();
mergeWebXml();
long finish = System.currentTimeMillis();
long millis = finish - start;
String time = String.format("%d min, %d sec", TimeUnit.MILLISECONDS.toMinutes(millis),
TimeUnit.MILLISECONDS.toSeconds(millis)
- TimeUnit.MINUTES.toSeconds(TimeUnit.MILLISECONDS.toMinutes(millis)));
getLog().info("Compilation completed in " + time);
} catch (Exception e) {
throw new MojoExecutionException("Failure processing jsps", e);
}
}
public void compile()
throws IOException, InterruptedException, MojoExecutionException, ExecutionException, JasperException {
ClassLoader currentClassLoader = Thread.currentThread().getContextClassLoader();
ArrayList urls = new ArrayList();
setUpClassPath(urls);
URLClassLoader ucl = new URLClassLoader(urls.toArray(new URL[0]), currentClassLoader);
StringBuilder classpathStr = new StringBuilder();
for (int i = 0; i < urls.size(); i++) {
if (getLog().isDebugEnabled()) {
getLog().debug("webappclassloader contains: " + urls.get(i));
}
classpathStr.append(urls.get(i).getFile());
if (getLog().isDebugEnabled()) {
getLog().debug("added to classpath: " + urls.get(i).getFile());
}
classpathStr.append(System.getProperty("path.separator"));
}
Thread.currentThread().setContextClassLoader(ucl);
String[] jspFiles = getJspFiles(webAppSourceDirectory);
if (verbose) {
getLog().info("Files selected to precompile: " + StringUtils.join(jspFiles, ", "));
}
ExecutorService executor = Executors.newFixedThreadPool(threads);
List> results = executor.invokeAll(initJspcWorkers(classpathStr, jspFiles, initJspList(jspFiles)));
executor.shutdown();
getLog().info("Number total of jsps : " + jspFiles.length);
manageResults(results);
Thread.currentThread().setContextClassLoader(currentClassLoader);
}
private List initJspList(String[] jspFiles) {
List jspFilesList = new ArrayList<>();
Collections.addAll(jspFilesList, jspFiles);
return jspFilesList;
}
private List initJspcWorkers(StringBuilder classpathStr, String[] jspFiles, List jspFilesList)
throws JasperException, IOException {
List workers = new ArrayList<>();
int minItem = jspFiles.length / threads;
int maxItem = minItem + 1;
int threadsWithMaxItems = jspFiles.length - threads * minItem;
int start = 0;
JspCContextAccessor topJspC = initJspc(classpathStr, -1, null);
for (int index = 0; index < threads; index++) {
int threadNumber = index + 1;
int itemsCount = (index < threadsWithMaxItems ? maxItem : minItem);
int end = start + itemsCount;
List jspFilesSubList = jspFilesList.subList(start, end);
if (jspFilesSubList.isEmpty()) {
getLog().info("Thread " + threadNumber + " have nothing to do, skip it");
} else {
JspC firstJspC = initJspc(classpathStr, index, topJspC);
JspcWorker worker = new JspcWorker(firstJspC, jspFilesSubList);
workers.add(worker);
start = end;
getLog().info("Number of jsps for thread " + threadNumber + " : " + jspFilesSubList.size());
}
}
return workers;
}
private JspCContextAccessor initJspc(StringBuilder classpathStr, int threadIndex, JspCContextAccessor topJspC)
throws IOException, JasperException {
JspCContextAccessor jspc = new JspCContextAccessor();
jspc.setWebXmlInclude(getwebXmlFragmentFilename(threadIndex));
jspc.setUriroot(webAppSourceDirectory);
jspc.setPackage(packageRoot);
jspc.setOutputDir(generatedClasses);
jspc.setValidateXml(validateXml);
jspc.setClassPath(classpathStr.toString());
jspc.setCompile(true);
jspc.setSmapSuppressed(suppressSmap);
jspc.setSmapDumped(!suppressSmap);
jspc.setJavaEncoding(javaEncoding);
jspc.setFailOnError(stopAtFirstError);
jspc.setPoolingEnabled(enableJspTagPooling);
jspc.setTrimSpaces(trimSpaces ? TrimSpacesOption.TRUE : TrimSpacesOption.FALSE);
jspc.setGenStringAsCharArray(genStringAsCharArray);
jspc.setCompilerSourceVM(compilerVersion);
jspc.setCompilerTargetVM(compilerVersion);
jspc.setcompilerClass(compilerClass);
jspc.setResourcesCache(resourcesCache);
if (topJspC == null) {
jspc.initClassLoader();
jspc.initServletContext();
} else {
jspc.initContext(topJspC);
}
// JspC#setExtensions() does not exist, so
// always set concrete list of files that will be processed.
if (topJspC != null) {
getLog().info("Includes=" + StringUtils.join(includes, ","));
if (excludes != null) {
getLog().info("Excludes=" + StringUtils.join(excludes, ","));
}
}
if (verbose) {
jspc.setVerbose(99);
} else {
jspc.setVerbose(0);
}
// force jspc Thread Count to 1 to avoid parallelism because it has already done
// in this plugin
jspc.setThreadCount("1");
return jspc;
}
private void manageResults(List> results)
throws InterruptedException, ExecutionException, MojoExecutionException {
boolean failTheBuild = false;
for (Future result : results) {
if (result.get() != null) {
getLog().error(result.get());
failTheBuild = true;
}
}
if (failTheBuild) {
throw new MojoExecutionException("see previous errors");
}
}
private String[] getJspFiles(String webAppSrcDir) {
DirectoryScanner scanner = new DirectoryScanner();
scanner.setBasedir(new File(webAppSrcDir));
if ((excludes != null) && (excludes.length != 0)) {
scanner.setExcludes(excludes);
}
scanner.addDefaultExcludes();
scanner.setIncludes(includes);
scanner.setCaseSensitive(false);
scanner.scan();
String[] includedFiles = scanner.getIncludedFiles();
getLog()
.debug(String.format("Included files returned from directory scan: %s", StringUtils.join(includedFiles, ",")));
getLog().debug(String.format("Excluded files returned from directory scan: %s",
StringUtils.join(scanner.getExcludedFiles(), ",")));
getLog().debug(String.format("Excluded directories returned from directory scan: %s",
StringUtils.join(scanner.getExcludedDirectories(), ",")));
return includedFiles;
}
/**
* Until Jasper supports the option to generate the srcs in a different dir than
* the classes, this is the best we can do.
*/
public void cleanupSrcs() {
// delete the .java files - depending on keepGenerated setting
if (!keepSources) {
File generatedClassesDir = new File(generatedClasses);
if (generatedClassesDir.exists() && generatedClassesDir.isDirectory()) {
delete(generatedClassesDir, f -> f.isDirectory() || f.getName().endsWith(".java"));
}
}
}
static void delete(File dir, FileFilter filter) {
File[] files = dir.listFiles(filter);
for (int i = 0; i < files.length; i++) {
File f = files[i];
if (f.isDirectory()) {
delete(f, filter);
} else {
f.delete();
}
}
}
/**
* Take the web fragment (for each thread if we active multithreading mode) and
* put it inside a copy of the web.xml.
*
* You can specify the insertion point by specifying the string in the
* insertionMarker configuration entry.
*
* If you dont specify the insertionMarker, then the fragment will be inserted
* at the end of the file just before the </webapp>
*
* @throws IOException maybe thrown during writing mergedXml file
* @throws MojoExecutionException maybe thrown when validating mergedXml file
*/
public void mergeWebXml() throws IOException, MojoExecutionException {
if (mergeFragment) {
// open the src web.xml
File webXmlFile = getWebXmlFile();
if (!webXmlFile.exists()) {
getLog().info(webXmlFile.toString() + " does not exist, cannot merge with generated fragment");
return;
}
File mergedWebXml = new File(new File(getwebXmlFragmentFilename(0)).getParentFile(), WEB_XML);
Path mergedWebXmlPath = createAndGetMergeWebXml(mergedWebXml);
try (BufferedReader webXmlReader = new BufferedReader(
new InputStreamReader(new FileInputStream(webXmlFile), StandardCharsets.UTF_8))) {
writeWebXmlMergedFile(webXmlReader, mergedWebXmlPath);
}
if (validateWebXmlAfterMerge) {
validateXmlContent(mergedWebXml);
}
if (validateWebXmlWithXsdAfterMerge) {
validateWithXsd(mergedWebXml);
}
}
}
private Path createAndGetMergeWebXml(File mergedWebXml) throws IOException {
Path mergedWebXmlPath = Paths.get(mergedWebXml.toURI());
Files.deleteIfExists(mergedWebXmlPath);
Files.createFile(mergedWebXmlPath);
return mergedWebXmlPath;
}
private void validateXmlContent(File mergedWebXml) throws IOException, MojoExecutionException {
try {
setHttpProxyIfNecessary();
DocumentBuilder parser = DocumentBuilderFactory.newInstance().newDocumentBuilder();
parser.parse(mergedWebXml);
} catch (ParserConfigurationException e) {
getLog().debug("Unable to instanciate Document Builder, so web.xml merged validation is not possible", e);
} catch (SAXException e) {
throw new MojoExecutionException("Error when validating XML content of merged web.xml !", e);
} finally {
restoreHttpProxy();
}
}
private void validateWithXsd(File mergedWebXml) throws IOException, MojoExecutionException {
try {
setHttpProxyIfNecessary();
Source webXmlSource = new StreamSource(mergedWebXml);
SchemaFactory schemaFactory = SchemaFactory.newInstance(XMLConstants.W3C_XML_SCHEMA_NS_URI);
Schema webXmlSchema = schemaFactory.newSchema(getWebXmlSchema());
Validator validator = webXmlSchema.newValidator();
validator.validate(webXmlSource);
} catch (SAXException e) {
throw new MojoExecutionException("Error when validating with XSD merged web.xml !", e);
} finally {
restoreHttpProxy();
}
}
private void setHttpProxyIfNecessary() {
if (!proxyEnvSet && httpProxyHost != null && !httpProxyHost.isEmpty()) {
proxyEnvSet = true;
httpProxyHostBackup = System.getProperty("http.proxyHost");
System.setProperty("http.proxyHost", httpProxyHost);
httpProxyPortBackup = System.getProperty("http.proxyPort");
System.setProperty("http.proxyPort", httpProxyPort);
if (httpNoProxyHosts != null && !httpNoProxyHosts.isEmpty()) {
httpNoProxyHostsBackup = System.getProperty("http.nonProxyHosts");
System.setProperty("http.nonProxyHosts", httpNoProxyHosts);
}
}
}
private void restoreHttpProxy() {
if (proxyEnvSet) {
System.setProperty("http.proxyHost", httpProxyHostBackup != null ? httpProxyHostBackup : "");
System.setProperty("http.proxyPort", httpProxyPortBackup != null ? httpProxyPortBackup : "");
if (httpNoProxyHosts != null && !httpNoProxyHosts.isEmpty()) {
System.setProperty("http.nonProxyHosts", httpNoProxyHostsBackup != null ? httpNoProxyHostsBackup : "");
}
proxyEnvSet = false;
}
}
private StreamSource[] getWebXmlSchema() throws MalformedURLException {
URL webXmlXsdUrl = new URL(webXmlXsdSchema);
return new StreamSource[] {new StreamSource(webXmlXsdUrl.toExternalForm())};
}
private String writeWebXmlMergedFile(BufferedReader webXmlReader, Path mergedWebXmlPath) throws IOException {
// read up to the insertion marker or the if there is no marker
String marker = (insertionMarker == null || insertionMarker.equals("") ? END_OF_WEBAPP : insertionMarker);
String line = "";
while ((line = webXmlReader.readLine()) != null) {
if (line.indexOf(marker) >= 0) {
writeXmlFragments(mergedWebXmlPath);
writeEndOfWebappIfNecessary(mergedWebXmlPath, marker);
} else {
Files.write(mergedWebXmlPath, (line + System.lineSeparator()).getBytes(StandardCharsets.UTF_8),
StandardOpenOption.APPEND);
}
}
return marker;
}
private void writeEndOfWebappIfNecessary(Path mergedWebXmlPath, String marker) throws IOException {
if (marker.equals(END_OF_WEBAPP)) {
Files.write(mergedWebXmlPath, END_OF_WEBAPP.getBytes(StandardCharsets.UTF_8), StandardOpenOption.APPEND);
}
}
private void writeXmlFragments(Path mergedWebXmlPath) throws IOException {
for (int index = 0; index < threads; index++) {
File fragmentWebXml = new File(getwebXmlFragmentFilename(index));
if (!fragmentWebXml.exists()) {
getLog().info("No fragment web.xml file generated for thread " + index);
} else {
// put in the generated fragment for the current thread
Files.write(mergedWebXmlPath, Files.readAllBytes(Paths.get(fragmentWebXml.toURI())), StandardOpenOption.APPEND);
}
}
}
private void prepare() {
// For some reason JspC doesn't like it if the dir doesn't
// already exist and refuses to create the web.xml fragment
File generatedSourceDirectoryFile = new File(generatedClasses);
if (!generatedSourceDirectoryFile.exists()) {
generatedSourceDirectoryFile.mkdirs();
}
}
/**
* Set up the execution classpath for Jasper.
*
* Put everything in the classesDirectory and all of the dependencies on the
* classpath.
*
* @param urls a list to which to add the urls of the dependencies
* @throws IOException
*/
private void setUpClassPath(List urls) throws IOException {
String classesDir = classesDirectory.getCanonicalPath();
classesDir = classesDir + (classesDir.endsWith(File.pathSeparator) ? "" : File.separator);
// we must keep deprecated usage of File.toURL because URLClassloader seem not
// working with path with %20 for example.
urls.add(new File(classesDir).toURL());
if (getLog().isDebugEnabled()) {
getLog().debug("Adding to classpath classes dir: " + classesDir);
}
for (Iterator> iter = project.getArtifacts().iterator(); iter.hasNext();) {
Artifact artifact = (Artifact) iter.next();
// Include runtime and compile time libraries
if (!Artifact.SCOPE_TEST.equals(artifact.getScope())) {
String filePath = artifact.getFile().getCanonicalPath();
if (getLog().isDebugEnabled()) {
getLog().debug("Adding to classpath dependency file: " + filePath);
}
urls.add(artifact.getFile().toURL());
}
}
}
private File getWebXmlFile() throws IOException {
File file = null;
File baseDir = project.getBasedir().getCanonicalFile();
File defaultWebAppSrcDir = new File(baseDir, "src/main/webapp").getCanonicalFile();
File webAppSrcDir = new File(webAppSourceDirectory).getCanonicalFile();
File defaultWebXml = new File(defaultWebAppSrcDir, WEB_XML).getCanonicalFile();
// If the web.xml has been changed from the default, try that
File webXmlFile = new File(webXml).getCanonicalFile();
if (webXmlFile.compareTo(defaultWebXml) != 0) {
file = new File(webXml);
return file;
}
// If the web app src directory has not been changed from the default, use
// whatever
// is set for the web.xml location
file = new File(webAppSrcDir, WEB_XML);
return file;
}
/**
* Add thread index at the end of webXmlFragment filename to deal with
* multithreading. If the number of threads is equal to 1 (no multithreading) we
* don't add suffix to maintain the same behavior as in the mode without
* multithreading.
*
* @param threadNumber the index of current thread
* @return web xml fragment filename with thread index
*/
private String getwebXmlFragmentFilename(int threadIndex) {
return threads == 1 ? webXmlFragment : webXmlFragment + "." + threadIndex;
}
}