blob: b4b1a441535b78069cda5ea6efeb177889ea8c8a [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.easyant.core;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail;
import java.io.File;
import java.io.IOException;
import java.io.PrintStream;
import java.net.URISyntaxException;
import java.net.URL;
import org.apache.easyant.core.configuration.EasyAntConfiguration;
import org.apache.easyant.core.configuration.EasyantConfigurationFactory;
import org.apache.tools.ant.BuildEvent;
import org.apache.tools.ant.BuildException;
import org.apache.tools.ant.BuildListener;
import org.apache.tools.ant.Project;
import org.apache.tools.ant.types.LogLevel;
import org.junit.After;
import org.junit.Rule;
import org.junit.rules.TemporaryFolder;
/**
* A EasyAntBaseTest is a TestCase which executes targets from an easyant module for testing.
*
* This class provides a number of utility methods for particular build file tests which extend this class.
*
*/
public abstract class EasyAntBaseTest {
protected static final String EASYANT_CACHE_DIR = "ivy.cache.dir";
protected Project project;
protected EasyAntConfiguration conf;
private StringBuilder logBuffer;
private StringBuilder fullLogBuffer;
private StringBuilder outBuffer;
private StringBuilder errBuffer;
private BuildException buildException;
@Rule
public TemporaryFolder folder = new TemporaryFolder();
/**
* Automatically calls the target called "tearDown" from the build file tested if it exits.
*
* This allows to use Ant tasks directly in the build file to clean up after each test. Note that no "setUp" target
* is automatically called, since it's trivial to have a test target depend on it.
*/
@After
public void tearDown() throws Exception {
if (project == null) {
/*
* Maybe the EasyAntBaseTest was subclassed and there is no initialized project. So we could avoid getting a
* NPE. If there is an initialized project getTargets() does not return null as it is initialized by an
* empty HashSet.
*/
return;
}
final String tearDown = "tearDown";
if (project.getTargets().containsKey(tearDown)) {
project.executeTarget(tearDown);
}
cleanTargetDirectory();
}
public void cleanTargetDirectory() {
if (project == null) {
throw new IllegalStateException("Project is not configured !");
}
File targetDirectory = project.resolveFile(project.replaceProperties("${basedir}/target"));
targetDirectory.delete();
}
public void cleanCache() {
if (project == null) {
throw new IllegalStateException("Project is not configured !");
}
File cacheDirectory = project.resolveFile(project.replaceProperties(project.getProperty(EASYANT_CACHE_DIR)));
cacheDirectory.delete();
}
/**
* run a target, expect for any build exception
*
* @param target
* target to run
* @param cause
* information string to reader of report
*/
public void expectBuildException(String target, String cause) {
expectSpecificBuildException(target, cause, null);
}
/**
* Assert that only the given message has been logged with a priority <= INFO when running the given target.
*/
public void expectLog(String target, String log) {
executeTarget(target);
String realLog = getLog();
assertEquals(log, realLog);
}
/**
* Assert that the given substring is in the log messages.
*/
public void assertLogContaining(String substring) {
String realLog = getLog();
assertTrue("expecting log to contain \"" + substring + "\" log was \"" + realLog + "\"",
realLog.contains(substring));
}
/**
* Assert that the given substring is not in the log messages.
*/
public void assertLogNotContaining(String substring) {
String realLog = getLog();
assertFalse("didn't expect log to contain \"" + substring + "\" log was \"" + realLog + "\"",
realLog.contains(substring));
}
/**
* Assert that the given substring is in the output messages.
*
* @since Ant1.7
*/
public void assertOutputContaining(String substring) {
assertOutputContaining(null, substring);
}
/**
* Assert that the given substring is in the output messages.
*
* @param message
* Print this message if the test fails. Defaults to a meaningful text if <tt>null</tt> is passed.
* @since Ant1.7
*/
public void assertOutputContaining(String message, String substring) {
String realOutput = getOutput();
String realMessage = (message != null) ? message : "expecting output to contain \"" + substring
+ "\" output was \"" + realOutput + "\"";
assertTrue(realMessage, realOutput.contains(substring));
}
/**
* Assert that the given substring is not in the output messages.
*
* @param message
* Print this message if the test fails. Defaults to a meaningful text if <tt>null</tt> is passed.
* @since Ant1.7
*/
public void assertOutputNotContaining(String message, String substring) {
String realOutput = getOutput();
String realMessage = (message != null) ? message : "expecting output to not contain \"" + substring
+ "\" output was \"" + realOutput + "\"";
assertFalse(realMessage, realOutput.contains(substring));
}
/**
* Assert that the given message has been logged with a priority &lt;= INFO when running the given target.
*/
public void expectLogContaining(String target, String log) {
executeTarget(target);
assertLogContaining(log);
}
/**
* Assert that the given message has not been logged with a priority &lt;= INFO when running the given target.
*/
public void expectLogNotContaining(String target, String log) {
executeTarget(target);
assertLogNotContaining(log);
}
/**
* Gets the log the BuildFileTest object. Only valid if configureProject() has been called.
*
* @pre logBuffer!=null
* @return The log value
*/
public String getLog() {
return logBuffer.toString();
}
/**
* Assert that the given message has been logged with a priority &gt;= VERBOSE when running the given target.
*/
public void expectDebuglog(String target, String log) {
executeTarget(target);
String realLog = getFullLog();
assertEquals(log, realLog);
}
/**
* Assert that the given substring is in the log messages.
*/
public void assertDebuglogContaining(String substring) {
String realLog = getFullLog();
assertTrue("expecting debug log to contain \"" + substring + "\" log was \"" + realLog + "\"",
realLog.contains(substring));
}
/**
* Gets the log the BuildFileTest object.
*
* Only valid if configureProject() has been called.
*
* @pre fullLogBuffer!=null
* @return The log value
*/
public String getFullLog() {
return fullLogBuffer.toString();
}
/**
* execute the target, verify output matches expectations
*
* @param target
* target to execute
* @param output
* output to look for
*/
public void expectOutput(String target, String output) {
executeTarget(target);
String realOutput = getOutput();
assertEquals(output, realOutput.trim());
}
/**
* Executes the target, verify output matches expectations and that we got the named error at the end
*
* @param target
* target to execute
* @param output
* output to look for
* @param error
* Description of Parameter
*/
public void expectOutputAndError(String target, String output, String error) {
executeTarget(target);
String realOutput = getOutput();
assertEquals(output, realOutput);
String realError = getError();
assertEquals(error, realError);
}
public String getOutput() {
return cleanBuffer(outBuffer);
}
public String getError() {
return cleanBuffer(errBuffer);
}
public BuildException getBuildException() {
return buildException;
}
private String cleanBuffer(StringBuilder buffer) {
StringBuilder cleanedBuffer = new StringBuilder();
for (int i = 0; i < buffer.length(); i++) {
char ch = buffer.charAt(i);
if (ch != '\r') {
cleanedBuffer.append(ch);
}
}
return cleanedBuffer.toString();
}
/**
* configure and init to run the named project
*
* @param url
* path to project file to run
* @param logLevel
* a given {@link LogLevel}
*
* @throws BuildException
*/
public void configureAndInitProject(URL url, int logLevel) {
configureProject(url, logLevel);
initProject();
}
/**
* configure and init to run the named project
*
* @param url
* path to project file to run
* @throws BuildException
*/
public void configureAndInitProject(URL url) {
configureProject(url);
initProject();
}
/**
* Set up to run the named project
*
* @param url
* path to project file to run
* @throws BuildException
*/
public void configureProject(URL url) throws BuildException {
File f;
try {
f = new File(url.toURI());
} catch (URISyntaxException e) {
throw new BuildException("Can't load project from url " + url.toString(), e);
}
configureProject(f.getAbsolutePath(), Project.MSG_DEBUG);
}
/**
* Set up to run the named project
*
* @param url
* path to project file to run
* @param logLevel
* a given {@link LogLevel}
*
* @throws BuildException
*/
public void configureProject(URL url, int logLevel) throws BuildException {
File f;
try {
f = new File(url.toURI());
} catch (URISyntaxException e) {
throw new BuildException("Can't load project from url " + url.toString(), e);
}
configureProject(f.getAbsolutePath(), logLevel);
}
/**
* Sets up to run the named project
*
* @param filename
* name of project file to run
*/
public void configureProject(String filename) throws BuildException {
configureProject(filename, Project.MSG_DEBUG);
}
/**
* Sets up to run the named project If you want to modify a few thing on the default configuration you should
* override this method
*
* @param filename
* name of project file to run
* @param logLevel
* a given {@link LogLevel}
*/
public void configureProject(String filename, int logLevel) throws BuildException {
conf = EasyantConfigurationFactory.getInstance().createDefaultConfiguration();
conf.setMsgOutputLevel(logLevel);
conf.setBuildModule(new File(filename));
conf.getDefinedProps().put(EasyAntMagicNames.SKIP_CORE_REVISION_CHECKER, "true");
// to avoid side effects due to user settings we ignore this setting by
// default for test
conf.getDefinedProps().put(EasyAntMagicNames.IGNORE_USER_IVYSETTINGS, "true");
conf.getDefinedProps().put(EasyAntMagicNames.PROJECT_IVY_INSTANCE, EasyAntMagicNames.EASYANT_IVY_INSTANCE);
// Configure easyant ivy instance
conf.setEasyantIvySettingsUrl(this.getClass().getResource("/repositories/easyant-ivysettings-test.xml"));
try {
conf.getDefinedProps().put("ivy.cache.dir", folder.newFolder().getAbsolutePath());
} catch (IOException e) {
throw new BuildException(e);
}
File projectModule = new File(filename);
if (!projectModule.exists()) {
throw new BuildException("Project " + projectModule.getAbsolutePath() + " does not exists");
}
}
/**
* Init the project instance with configuration
*
* @pre configureProject has been called
*/
public void initProject() {
if (conf == null) {
throw new RuntimeException("You must call the configureProject method before initProject()");
}
// Flush the buffer
logBuffer = new StringBuilder();
fullLogBuffer = new StringBuilder();
// init the new project instance
project = new Project();
project.addBuildListener(new AntTestListener(conf.getMsgOutputLevel()));
EasyAntEngine eaEngine = new EasyAntEngine(conf);
eaEngine.configureEasyAnt(project);
eaEngine.loadProject(project);//FIXME should be better handled
}
/**
* Executes a target we have set up
*
* @pre initProject has been called
* @param targetName
* target to run
*/
public void executeTarget(String targetName) {
PrintStream sysOut = System.out;
PrintStream sysErr = System.err;
try {
sysOut.flush();
sysErr.flush();
outBuffer = new StringBuilder();
PrintStream out = new PrintStream(new AntOutputStream(outBuffer));
System.setOut(out);
errBuffer = new StringBuilder();
PrintStream err = new PrintStream(new AntOutputStream(errBuffer));
System.setErr(err);
logBuffer = new StringBuilder();
fullLogBuffer = new StringBuilder();
buildException = null;
project.executeTarget(targetName);
} finally {
System.setOut(sysOut);
System.setErr(sysErr);
}
}
/**
* Get the project which has been configured for a test.
*
* @return the Project instance for this test.
*/
public Project getProject() {
return project;
}
/**
* Gets the directory of the project.
*
* @return the base dir of the project
*/
public File getProjectDir() {
return project.getBaseDir();
}
/**
* Runs a target, wait for a build exception.
*
* @param target
* target to run
* @param cause
* information string to reader of report
* @param msg
* the message value of the build exception we are waiting for set to null for any build exception to be
* valid
*/
public void expectSpecificBuildException(String target, String cause, String msg) {
try {
executeTarget(target);
} catch (org.apache.tools.ant.BuildException ex) {
buildException = ex;
if ((null != msg) && (!ex.getMessage().equals(msg))) {
fail("Should throw BuildException because '" + cause + "' with message '" + msg + "' (actual message '"
+ ex.getMessage() + "' instead)");
}
return;
}
fail("Should throw BuildException because: " + cause);
}
/**
* run a target, expect an exception string containing the substring we look for (case sensitive match)
*
* @param target
* target to run
* @param cause
* information string to reader of report
* @param contains
* substring of the build exception to look for
*/
public void expectBuildExceptionContaining(String target, String cause, String contains) {
try {
executeTarget(target);
} catch (org.apache.tools.ant.BuildException ex) {
buildException = ex;
if ((null != contains) && (!ex.getMessage().contains(contains))) {
fail("Should throw BuildException because '" + cause + "' with message containing '" + contains
+ "' (actual message '" + ex.getMessage() + "' instead)");
}
return;
}
fail("Should throw BuildException because: " + cause);
}
/**
* call a target, verify property is as expected
*
* @param target
* build file target
* @param property
* property name
* @param value
* expected value
*/
public void expectPropertySet(String target, String property, String value) {
executeTarget(target);
assertPropertyEquals(property, value);
}
/**
* assert that a property equals a value; comparison is case sensitive.
*
* @param property
* property name
* @param value
* expected value
*/
public void assertPropertyEquals(String property, String value) {
String result = project.getProperty(property);
assertEquals("property " + property, value, result);
}
/**
* assert that a property equals "true".
*
* @param property
* property name
*/
public void assertPropertySet(String property) {
assertPropertyEquals(property, "true");
}
/**
* assert that a property is null.
*
* @param property
* property name
*/
public void assertPropertyUnset(String property) {
assertPropertyEquals(property, null);
}
/**
* call a target, verify named property is "true".
*
* @param target
* build file target
* @param property
* property name
*/
public void expectPropertySet(String target, String property) {
expectPropertySet(target, property, "true");
}
/**
* Call a target, verify property is null.
*
* @param target
* build file target
* @param property
* property name
*/
public void expectPropertyUnset(String target, String property) {
expectPropertySet(target, property, null);
}
/**
* Retrieve a resource from the caller classloader to avoid assuming a vm working directory. The resource path must
* be relative to the package name or absolute from the root path.
*
* @param resource
* the resource to retrieve its url.
* @throws junit.framework.AssertionFailedError
* if the resource is not found.
*/
public URL getResource(String resource) {
URL url = getClass().getResource(resource);
assertNotNull("Could not find resource :" + resource, url);
return url;
}
/**
* an output stream which saves stuff to our buffer.
*/
private static class AntOutputStream extends java.io.OutputStream {
private StringBuilder buffer;
public AntOutputStream(StringBuilder buffer) {
this.buffer = buffer;
}
public void write(int b) {
buffer.append((char) b);
}
}
/**
* Our own personal build listener.
*/
private class AntTestListener implements BuildListener {
private int logLevel;
/**
* Constructs a test listener which will ignore log events above the given level.
*/
public AntTestListener(int logLevel) {
this.logLevel = logLevel;
}
/**
* Fired before any targets are started.
*/
public void buildStarted(BuildEvent event) {
}
/**
* Fired after the last target has finished. This event will still be thrown if an error occurred during the
* build.
*
* @see BuildEvent#getException()
*/
public void buildFinished(BuildEvent event) {
}
/**
* Fired when a target is started.
*
* @see BuildEvent#getTarget()
*/
public void targetStarted(BuildEvent event) {
// System.out.println("targetStarted " +
// event.getTarget().getName());
}
/**
* Fired when a target has finished. This event will still be thrown if an error occurred during the build.
*
* @see BuildEvent#getException()
*/
public void targetFinished(BuildEvent event) {
// System.out.println("targetFinished " +
// event.getTarget().getName());
}
/**
* Fired when a task is started.
*
* @see BuildEvent#getTask()
*/
public void taskStarted(BuildEvent event) {
// System.out.println("taskStarted " +
// event.getTask().getTaskName());
}
/**
* Fired when a task has finished. This event will still be throw if an error occurred during the build.
*
* @see BuildEvent#getException()
*/
public void taskFinished(BuildEvent event) {
// System.out.println("taskFinished " +
// event.getTask().getTaskName());
}
/**
* Fired whenever a message is logged.
*
* @see BuildEvent#getMessage()
* @see BuildEvent#getPriority()
*/
public void messageLogged(BuildEvent event) {
if (event.getPriority() > logLevel) {
// ignore event
return;
}
if (event.getPriority() == Project.MSG_INFO || event.getPriority() == Project.MSG_WARN
|| event.getPriority() == Project.MSG_ERR) {
logBuffer.append(event.getMessage());
}
fullLogBuffer.append(event.getMessage());
}
}
}