<html> <!-- <![CDATA[ putDataHere ]]> --> <title>Writing tests for the AspectJ compiler </title> <body> <h2>Writing tests for the AspectJ compiler </h2> The AspectJ project has a harness which reads test specification files and run tests. The tests are usually simple scenarios like "compile and run" or "compile expecting error", but may involve multiple files, incremental compilation, classpath or aspectpath entries, etc. This document shows how to write tests that can be run by the harness and suggests some patterns to use in test code, discussing <ul> <li><a href="#simple">Simple test definitions</a></li> <li><a href="#sourceFiles">Test source files</a></li> <li><a href="#incremental">Incremental tests</a></li> <li><a href="#verifying">Verifying test steps</a></li> <ul> <li><a href="#messages">Messages</a></li> <li><a href="#dirchanges">Changes in an output directory</a></li> <li><a href="#tester">Runtime behavior</a></li> </ul> <li><a href="#compilerOptions">Compiler Options</a></li> <li><a href="#background">Harness background</a></li> </ul> Most people just writing a test case need to know only the information in <a href="#simple">Simple test definitions</a> and <a href="#sourceFiles">Test source files</a>. <p>Related documents: <ul> <li>For information on running the harness, see <a href="readme-tests-module.html"> readme-tests-module.html</a> </li> <li>For example test definition files, see <a href="ajcTests.xml">ajcTests.xml</a> and <a href="ajcTestsFailing.xml">ajcTestsFailing.xml</a>. </li> </ul> <a name="simple"></a> <h4>Simple Test definitions</h4> Here is a simple example to compile <code>Main.java</code> and expect an error on line 10: <pre> <ajc-test dir="new" title="simple error test"> <compile files="Main.java"> <message kind="error" line="10"/> </compile> </ajc-test> </pre> <p>Here is an example to compile <code>pack/Aspect.java</code> and <code>pack2/Main.java</code> and run the main class: <pre> <ajc-test dir="new" title="simple run test"> <compile files="pack/Aspect.java,pack1/Main.java"/> <run class="pack1.Main"/> </ajc-test> </pre> The compile and run steps of a given ajc-test share a common sandbox, so (e.g.,) the run step knows to set its classpath using the classes directory generated by the compile step. <p>More complex compilations are discussed in <a href="#compilerOptions">Compiler Options</a> below. <a name="sourceFiles"></a> <h4>Test source files</h4> The <code>dir</code> attribute in the <code>ajc-test</code> element specifies a base test directory relative to the directory of the test specification file. All paths are specified relative to this base test directory. E.g., the last example used <code>dir="new"</code> and presumed the following directory structure: <pre> {some dir} # test specification directory {testDefinition}.xml new/ # test base directory pack/Aspect.java pack2/Main.java </pre> Test cases with only one file in the default package can often share directories (e.g., see the many files in <a href="new">new/</a>), but usually a test case has its own directory. <a name="incremental"></a> <h4>Incremental tests</h4> Incremental tests are more complex because they involve updating source files before recompiling. Here's an example that <ul> <li>compiles Main.java, </li><li>runs it, </li><li>updates the source to introduce an error, </li><li>incrementally compile it (and detect the error) </li><li>updates the source to fix the error, </li><li>incrementally compile it (successfully), and </li><li>runs it again. </li> </ul> <pre> <ajc-test dir="new/incremental1" title="incremental test"> <compile staging="true" sourceroots="." options="-incremental" /> <run class="Main"/> <inc-compile tag="20"> <message kind="error" line="15"> </inc-compile> <inc-compile tag="30"/> <run class="Main"/> </ajc-test> </pre> To understand what's happening in this test would require looking at the source directory to see which files are tagged "20" and "30". But before walking through that, there's a variation of incremental building for AJDE. (The AJDE wrapper around the <code>ajc</code> compiler can also be driven by the test harness.) <p> In AJDE, incremental tests also involve the notion of "fresh builds", i.e., when the test reuses the same compiler and build configuration but rebuilds from scratch. In that case, there is still the question of whether source files should be updated; if not, the tag can have the special value "same". For example, if the last example had two more lines: <pre> ... <inc-compile tag="30"/> <run class="Main"/> <inc-compile fresh="true" tag="same"/> <run class="Main"/> </ajc-test> </pre> The test would complete by completely rebuilding the same files and then running the main class. This option has no effect on the normal (ajc) compiler, and requires specifying <code>-ajdeCompiler</code> to the harness or compile step as an argument. <p> To recap the attributes of note for setting up incremental tests: <ul> <li><code>compile</code> <code>staging="true"</code>: Incremental tests require staging, which copies the test source files to a temporary directory so they can be updated during the test without changing the actual sources. </li> <li><code>compile</code> <code>sourceroots="{..}"</code>: incremental mode only takes source files in the form of <code>-sourceroots</code> entries. </li> <li><code>compile</code> <code>options="-incremental{,..}"</code>: Specify the <code>-incremental</code> option to signal incremental mode to the compiler. If you do not include this, the harness will still run an incremental test, but the compiler will not do its additional checking it on input options. </li> <li><code>inc-compile</code> <code>tag="{##}"</code>: The required tag attribute is a suffix identifying files in the test source directory specifying how the sources should be changed before that incremental compilation step. If there is a prefixing suffix "delete", then the file is deleted; otherwise, the file is copied (with the effect either of updating an existing file or adding a new file). If the tag is "same", then no files are changed. </li> <li><code>inc-compile</code> <code>fresh="true"</code>: With the AJDE compiler, you can rebuild the current build configuration in its entirety. (By contrast, doing another <compile> task would re-initialize the AJDE compiler.) This option is ignored unless <code>-ajdeCompiler</code> is passed to the harness on the command line or to the immediately preceding <compile> task in the options. </li> </ul> <p> Now, to get back to the question of what exactly is happening in an incremental test. To do so, compare the tags with the files specified in the test source directory; the tagged files are the updates for that particular step. (By convention the tags are numeric and in order, but they need not be.) For example, here are some sources for the test above: <pre> {some dir} {testDefinition}.xml new/ incremental1/ DeleteMe.delete.30.java DeleteMe.java Main.20.java Main.30.java Main.java NewFile.30.java </pre> Comparing this with the test specification, you can see the harness will run one compile and two re-compiles: <ol> <li>Initially compile <code>Main.java</code> and <code>DeleteMe.java</code> <pre> <compile staging="true" files="Main.java,DeleteMe.java"/> {some dir} {testDefinition}.xml new/ incremental1/ ... DeleteMe.java ... Main.java ... </pre> </li> <li>For incremental tag 20, update <code>Main.java</code> with the contents of <code>Main.20.java</code> and recompile, expecting an error on line 15: <pre> <inc-compile tag="20"> <message kind="error" line="15"> </inc-compile> {some dir} {testDefinition}.xml new/ incremental1/ ... Main.20.java ... </pre></li> <li>For incremental tag 30, delete <code>DeleteMe.java</code>, add <code>NewFile.java</code>, update <code>Main.java</code> with the contents of <code>Main.30.java</code> and recompile with no error or warning messages: <pre> <inc-compile tag="30"/> {some dir} {testDefinition}.xml new/ incremental1/ DeleteMe.delete.30.java ... Main.30.java ... NewFile.30.java </pre> </li> </ol> <a name="verifying"></a> <h4>Verifying test steps</h4> As seen above, two ways to verify that a compile was successful are to run the corresponding class or check the compiler messages. More generally, the harness can verify compile/run test steps by detecting the following things and comparing against expected behavior: <p> <table border="1" cellpadding="1"> <tr><th>Detect </th> <th>Evaluate</th> </tr> <tr><td>Exceptions </td> <td>signal failure </td> </tr> <tr><td>Result value </td> <td>heuristically compare with expected: compiles not expecting errors are expected to return a normal result status, and vice-versa.</td> </tr> <tr><td>Messages (e.g., compiler warnings and errors)</td> <td>Compare with expected messages</td> </tr> <tr><td>Directory changes (e.g., <code>.class</code> files created) </td> <td>Compare with expected changes</td> </tr> <tr><td>Runtime behavior</td> <td>Use <code>Tester</code> in test source code to signal events for comparison with expected events.</td> </tr> </table> <p> <a name="messages"></a> <h5>Messages</h5> In a test definition, a nested <code>message</code> element specifies a condition on the successful completion of the nesting ajc-test sub-element. In the earlier example, if the harness does not detect an error message on line 10 or if there are unexpected messages, then the compile step will be reported as failing: <pre> <ajc-test dir="new" title="simple error test"> <compile files="Main.java"> <message kind="error" line="10"/> </compile> </ajc-test> </pre> Expected messages can be specified as sub-elements for the three <code>ajc-test</code> elements <code>compile</code>, <code>inc-compile</code>, and <code>run</code>. Messages require a kind (error or warning) and a line. To make specification easier, if an error is specified for a line, the harness accepts as expected any number of errors on that line. <p> The current harness has only been tested to validate compilation based on line numbers. Filename-based comparison is disabled as untested/unused, and run messages are handled wrongly; line-number comparison will fail since run messages do not have line numbers. <a name="dirchanges"></a> <h5>Changes in an output directory</h5> As with messages, specifying directory changes as a nested element operates as a condition on the successful completion of the nesting element. The harness will check for files added, removed, updated, or unchanged from a given directory. The directory is specified explicitly or using a token for the shared classes or run directory. For even more brevity, the harness supports a default suffix for the files. <p> Directory changes have been used only to validate changes in the classes directory. The current harness defaults to using the classes directory, and when using the classes directory uses <code>.class</code> as a default suffix. <p> Here's an example specification: <pre> <ajc-test dir="new/dirchanges-test" title="dir-changes test"> <compile staging="true" files="Main.java,DeleteMe.java,Unchanged.java"/> <inc-compile tag="20"> <dir-changes updated="Main" removed="DeleteMe" unchanged="Unchanged"/> </inc-compile> </ajc-test> </pre> It checks after a recompile that <ul> <li><code>Main.class</code> was updated</li> <li><code>DeleteMe.class</code> was deleted</li> <li><code>Unchanged.class</code> was not touched</li> </ul> <a name="tester"></a> <h5>Runtime behavior</h5> Code that tests aspects often falls into the pattern of comparing expected and actual events/signals. For example, to prove that before advice in fact ran before a particular method execution, you might generated and expect signals corresponding to <ol> <li>method-call</li> <li>before advice execution</li> <li>method-execution</li> </ol> The <code>Tester</code> utility class provides API's for signalling actual and expecting events and comparing the two. Typically, events are symbolized and compared as String. Here's a small sample test case that for the scenario above: <pre> import org.aspectj.testing.Tester; public class Main implements Runnable { public static void main(String[] args) { Tester.expectEvent("before advice"); Tester.expectEvent("execute run"); new Main().run(); Tester.checkAllEvents(); } public void run() { Tester.event("execute run"); } } aspect A { before () : target(Runnable) && execution(void run()) { Tester.event("before advice"); } } </pre> If either the advice or the method does not run, the harness will report a failure. <p> <code>Tester</code> also has methods that operate like JUnit assertions as idioms to detect differences in expected and actual values, signalling appropriately. <p> <code>Tester</code> is at <a href="../testing-client/src/org/aspectj/testing/Tester.java"> ../testing-client/src/org/aspectj/testing/Tester.java</a> and is built into <a href="../lib/tests/testing-client.jar"> ../lib/tests/testing-client.jar</a> which is included on the classpath by the compile and run steps. <p>You can write runtime test cases without using Tester; simply throw some exception from the main thread to signal failure. <a name="compilerOptions"></a> <h4>Compiler options</h4> The harness does not support all of the AspectJ 1.1 compiler options. Flags are mainly supported through the a comma-delimited list in the <code>options</code> attribute: <pre> <ajc-test dir="new" title="lint test"> <compile files="LintTest.java" options="-Xlint,-emacssym"> <message kind="warning" line="22"> </compile> </pre> This should work even for complex single-arg options like <code>-g:none</code>, but will fail for comma-delimited single-arg options like <code>-g:lines,vars</code> because the comma delimiters are ambiguous (yes, a design bug!). For -source 1.4 and -source 1.3 options, use -source14 and -source13 (yes, a hack!). <p> The <code>compile</code> element has the following attributes which handle most of the other compiler arguments: <ul> <li><code>files</code>: .aj and .java files are treated as source files, but .jar files are extracted and passed to the compiler as <code>-injars</code> </li><li><code>classpath</code>: directories and jar files for the classpath </li><li><code>aspectpath</code>: binary aspects in jar files </li><li><code>argfiles</code>: argument list files </li><li><code>sourceroots</code>: root directories for source files </li> </ul> Paths for these are all relative to the test base directory, and multiple entries are separated with commas. <p> Here is a cooked example that uses all <code>compiler</code> attributes: <pre> <ajc-test dir="new" title="attributes test"> <compile files="Main.java,injar.jar" staging="true" options="-Xlint,-g:none" argfiles="debug.lst,aspects/test.lst" aspectpath="jars/requiredAspects.jar" classpath="providedClassesDir,jars/required.jar"/> <inc-compile tag="20"/> </ajc-test> </pre> <h5>Unsupported compiler options</h5> The harness does not support the following AspectJ compiler options: <code>-target {version}, -outjar {file}, -log {file}</code>. (<code>-d {dir}</code> is used but specification is not supported.) <a name="background"></a> <h4>Background information on the Harness</h4> To make the test specifications as terse as possible, harness components for <code>inc-compile</code> and <code>run</code> elements use information set up earlier by <code>compile</code>, some of which is only implicit. When a test is run, the harness creates a staging directory for temporary files and a sandbox component for sharing information between test components, particularly classpath entries shared between the compile and run components. The compile and run components share classpath information through the sandbox, adding default libraries: <ul> <li>Aside from any explicit classpath entries, <code>compile</code> always includes the jars <a href="../lib/tests/aspecjrt.jar"> ../lib/tests/aspecjrt.jar</a> and <a href="../lib/tests/testing-client.jar"> ../lib/tests/testing-client.jar</a> on the compile classpath. </li> <li><code>run</code> sets up its classpath as the compile classpath plus the compile output (classes) directory plus any entries on the aspectpath. </li> </ul> The harness provides some more advance behaviors, which you might see specified in the tests. For more information, see the API documentation for the harness. <ul> <li><u>option dominance and overriding</u>: Both in test specifications and on the command line you can force compiler options on or off. (Forcing means that, e.g., even if an option is enabled in the test specification, it can be disabled from the command line.) These appear in the test specifications as options with prefixes '!' or '~' rather than '-' (e.g., '~emacssym' to force the emacssym option off, even in tests that specify it). </li> </ul> <hr> <small>last updated January 29, 2002 </small> <!-- CVS variable --> </body> </html>