I particularly like these:
Håkan Råberg blogged about how Hamcrest can be used with iterators:
List<Integer> numbers = Arrays.asList(-1, 0, 1, 2);
List<Integer> positiveNumbers = detect(numbers, greaterThan(0)));
List<String> words = Arrays.asList("cheese", "lemon", "spoon");
List<String> wordsWithoutE = reject(words, containingString("e"));
Nothing rocket-sciencey about it. But simple and useful because it reduces boilerplate code and get to use the ever growing library of Hamcrest matchers.
On top of that, combining Hamcrest with a CGLib generated proxy, he has built a staticly typed query API:
List<Person> employees = ...;
List<Integer> allAges
= collect(from(employees).getAge());
List<Person> allBosses
= collect(from(employees).getDepartment().getBoss());
List<Person> allAccountants
= select(from(employees).getDepartment().getName(),
containingString("Accounts"));
This is nice alternative to a string based query language as you get your IDE completions, refactoring, compile time checking etc, without the noise of boilerplate code.
Robert Chatley has taken some of the concepts of his LiFT framework and reimplemented them using Hamcrest and WebDriver for performing web testing.
public void testHasLotsOfLinks() {
goTo("http://some/url");
assertPresenceOf(greaterThan(15), links());
assertPresenceOf(atLeast(1), link().with(text(containingString("Sign in"))));
clickOn(link().with(text(containingString("Sign in"))));
assertPresenceOf(exactly(1), title().with(text(equalTo("Sign in page"))));
}
Now initially this seems a bit wordy and strange. Robert has designed this as a literate API. If you adjust the syntax highlighting of your API and make the Java keywords and syntax less visible, you get this:
goTo "http://some/url" assertPresenceOf greaterThan 15 links assertPresenceOf atLeast 1 link with text containingString "Sign in" clickOn link with text containingString "Sign in" assertPresenceOf exactly 1 title with text equalTo "Sign in page"
The motivation here is that the API usage is self documenting and could be useful to non-programmers. The flip-side to this is that it's actually quite hard to write APIs like this and the usage can take quite a bit of getting used to.
Robert also introduced a Finder interface (the link() and title() methods return Finder implementations). This allows you to factor out your own UI specific components:
assertPresenceOf(atLeast(1), signInLink());
clickOn(signInLink());
assertPresenceOf(exactly(1),
blogLink().with(urlParameter("name", containingString("joe"))));
This is the bit I really like.
Allowing abstractions of components and matching rules to be combined in many different ways, so tests can check exactly what they need to, resulting in reduced less brittle tests that are easier to maintain.
As I hear of other uses I'm listing them on the Hamcrest wiki.
Of course, like any technology, it's easy to get carried away.
Here's an example of Hamcrest gone bad:
assertThat(myNumber, anyOf(equalTo(0), allOf(greaterThan(5), lessThan(10))));
I'm not a LISP programmer, so I find that really hard to understand. Just because we have an assertTHAT() method, we don't have to use it all the time. In this case it's much simpler to use plain old assertTRUE():
assertTrue("myNumber should be 0 or between 5 and 10",
myNumber == 0 || (myNumber > 5 && myNumber < 10));
Even though the non-Matcher version is longer (it could be shortened by leaving out the message and using a shorter variable name, but that would make it harder to understand), I find it much easier to understand.
But, what if you actually needed to use a matcher (e.g. for the web testing or collection processing examples above)?
One approach is you could use higher level matcher that are composed of other matchers:
matcher = anyOf(equalTo(0), allOf(greaterThan(5), lessThan(10))) // simplifies to matcher = anyOf(equalTo(0), between(5, 10))
Complete tangent: An alternative to between(5, 10) is between(5).and(10). The latter makes for more literate code, but is harder to implement - again a design tradeoff.
Another approach is to create a one-off anonymous matcher implementation:
matcher = new CustomMatcher() { public boolean matchesSafely(Integer n) { return n == 0 || (n > 5 && n < 10); } }
What are you doing with Hamcrest?
However, it's slightly tougher to place something to be read, when the target readers don't know they don't know it. This was a problem the testing group were finding, as they wanted to improve sharing of practical testing techniques.
So, Testing on the Toilet was started. A regular weekly(ish) tip posted in toilet cubicles and above urinals. Short enough to be read whilst doing your business.
Soon after, many visitors started noticing these postings and we got requests to make these available to put up in offices of other development teams.
So, we have.
http://googletesting.blogspot.com/
Each episode will be made available as a toilet friendly PDF.
]]>QDox is a fast JavaDoc/Java parser built in 2002. It was originally intended as a stop gap until Java supported annotations by allowing tools to easily get access to JavaDoc attributes. Essentially it provided nothing more than a stripped down version of the JavaDoc Doclet tool, with performance suitable for using in continual build cycles (what would take JavaDoc over ten minutes to process would typically take QDox less than ten seconds). It served its purpose well.
Then came along Java 5 and I stopped actively working on QDox. The first reason was that with the new annotations support, QDox wasn't necessary. The other reason was that it would take a lot of effort to update the parser to support Java 5 syntax (not just for annotations, but generics, enums, etc).
And so QDox went quiet. The dev team lost interest and the releases stopped.
It turned out, I was wrong. Even with Java supporting annotations, QDox in a Java 5 world has some benefits:
So, by popular demand, I'm resurrecting the project. Yay.
This new release is a stop-gap release. Highlights include:
This should be enough for existing projects to carry on using it with Java 5 code.
The next release will focus on making Java 5 specific features available in the API. Stay tuned.
]]>Buzzwordtastic.
]]>It was a good conference for SiteMesh. It opened my eyes to two things:
I was there to present a session on SiteMesh but a lot of other speakers beat me to it. It kept slipping into other sessions...
Matt gave an overview of the technology stack used in his AppFuse application. Despite having 5 versions of his app that use different frameworks (Struts, WebWork, Tapestry, Spring MVC and JavaServer Faces), all used SiteMesh. Good!
Erik walked us through the open source products he used to build his Lucene Book website and what customizations he made. The focus, of course, was Lucene and I learned a lot of great tricks about Lucene that hadn't occurred to me before - such as using "sounds like" queries with soundex and indexing images by colors. I continue to love Lucene.
A great point that Erik mentioned was the need to become intimate with the projects you use. If you truely want to make the most of your frameworks, understand how they work, join the community and extend them.
Erik chose Tapestry to build the site but he also had Blojsom and some static content, so SiteMesh was useful to integrate these and he created some custom code to build SiteMesh decorators with Tapestry.
He pointed out that despite submitting this useful Tapestry integration to the SiteMesh project, nothing had made it into the SiteMesh release. Feeling embarressed, I committed his changes immediately, inadvertently breaking the build and providing great ammunition for Eric Pugh's session on the importance of continuous integration.
The basic plot was this... Matthew Porter was arguing why Spring MVC sucks and WebWork rocks. Matt Raible was arguing why Spring MVC rocks and WebWork sucks. The only thing they both agreed on was SiteMesh rocked. A fairly heated and passionate debate - great fun to watch. I would have opted for more violence though.
Matthew Porter got the final laugh when he pointed out that he compared the Spring MVC and WebWork versions of Matt Raible's AppFuse framework and the Spring MVC version had about 25% (I think) more code, not including comments.
(more)
This was an interesting session where Craig compared the approaches taken by Struts, WebWork, Spring MVC, Tapestry and JavaServer Faces. He had done detailed research and, despite his heavy involvement with Struts and JSF, gave a very fair and objective view of the pros and cons of each.
This work could be useful for people evaluating which frameworks to choose and possibly could be overlayed with a guide based on values. The bottom line is there's no single 'ultimate' web framework and depending on your needs and values you should choose the most suitable. I think it would be beneficial to all to have a guide indicating which values each of these frameworks are suited/not-suited for.
So, my question is this: Which values are more important to you when choosing a web framework and in which priority?
These are some example values that spring to mind: commercial support, testability (unit and functional), popularity, extensibility/customization, integration with other frameworks, rich widget support, REST friendlyness, simplicity vs magicness, AJAX friendlyness, learning curve, configuration, etc.
Anyhoo, SiteMesh was probably mentioned enough times to attract another load of people to the SiteMesh session.
I'm really glad these people mentioned SiteMesh and said such kind words about it - it resulted in a lot of interest and a full house for the SiteMesh session. I hope to get these presentations online shortly and write a bit more about how Subversion, Microsoft Word and SiteMesh can be combined to create a rich Content Management System.
The fact still remains that SiteMesh has terrible marketing. I'd love some ideas of how to spread the word more and encourage more people to try it but I honestly have no idea what to do. Any suggestions?
]]>Here's a nicer way. jMock contains a constraint library for specifying precise expectations on mocks that can be reused in your own assertion method (and that's the last time I'm going to mention mocks today, I promise - despite the frequent references to the jMock library).
By making a simple JUnit assertion method that takes a Constraint, it provides a replacement for all the other assert methods.
I call mine assertThat() because I think it reads well. Combined with the jMock syntactic sugar, you can use it like this:
assertThat(something, eq("Hello"));
assertThat(something, eq(true));
assertThat(something, isA(Color.class));
assertThat(something, contains("World"));
assertThat(something, same(Food.CHEESE));
assertThat(something, NULL);
assertThat(something, NOT_NULL);
Okay, that's nice but nothing radical. A bunch of assert methods have been replaced with different methods that return constraint objects. But there's more...
Constraints can be chained making it possible to combine them in different permutations. For instance, for virtually every assertion I do, I usually find that I need to test the negative equivalent at some point:
assertThat(something, not(eq("Hello")));
assertThat(something, not(contains("Cheese")));
Or maybe combinations of assertions:
assertThat(something, or(contains("color"), contains("colour")));
The previous example can be written using the vanilla JUnit assert methods like this:
assertTrue(something.indexOf("color") > -1 || something.indexOf("colour") > -1);
Fine, the constraint based one is easier to read. But the real beauty is the failure message.
The vanilla JUnit assert fails with:
junit.framework.AssertionFailedError:
Useless! Means you have to put an explicit error message in the assertion:
assertTrue(something.indexOf("color") > -1 || something.indexOf("colour") > -1,
"Expected a string containing 'color' or 'colour'");
But the jMock constraint objects are self describing. So with this assertion:
assertThat(something, or(contains("color"), contains("colour")));
I get this useful failure message, for free:
junit.framework.AssertionFailedError: Expected: (a string containing "color" or a string containing "colour") but got : hello world
The simplest way is to grab jMock and create your own base test class that extends MockObjectTestCase. This brings in convenience methods for free (I'm still not talking about mocks, honest). If you don't want to extend this class, you can easily reimplement these methods yourself - it's no biggie.
import org.jmock.MockObjectTestCase;
import org.jmock.core.Constraint;
public abstract class MyTestCase extends MockObjectTestCase {
protected void assertThat(Object something, Constraint matches) {
if (!matches.eval(something)) {
StringBuffer message = new StringBuffer("\nExpected: ");
matches.describeTo(message);
message.append("\nbut got : ").append(something).append('\n');
fail(message.toString());
}
}
}
Now ensure all your test cases extend this instead of junit.framework.TestCase and you're done.
Creating new constraints is easy. Let's say I want something like:
assertThat(something, between(10, 20));
To do that I need to create a method that returns a Constraint object, requiring two methods; eval() for performing the actual assertion, and describeTo() for the self describing error message. This is something that can live in the base test class.
public Constraint between(final int min, final int max) {
return new Constraint() {
public boolean eval(Object object) {
if (!object instanceof Integer) {
return false;
}
int value = ((Integer)object).intValue();
return value > min && value < max;
}
public StringBuffer describeTo(StringBuffer buffer) {
return buffer.append("an int between ").append(min).append(" and ").append(max);
}
}
}
This can be combined with other constraints and still generate decent failure messages.
assertThat(something, or(eq(50), between(10, 20));
junit.framework.AssertionFailedError: Expected: (50 or an int between 10 and 20) but got : 43
In practice I find I only need to create a few of these constraints as the different combinations gives me nearly everything I need.
More about this in the jMock documentation.
Since using this one assert method I've found my tests to be much easier to understand because of lack of noise and I've spent a lot less time creating 'yet another assertion' method for specific cases. And in most cases I never need to write a custom failure message as the failures are self describing.
Come and say hi.
A problem faced in every web application is how to separate style from content. SiteMesh is a framework that provides an elegant solution to this, resulting in a clean separation that is straightforward to work with, complements other web frameworks, and is easily applied to existing applications.]]>The first part of this session introduces SiteMesh, including an overview of the architecture and patterns, comparisons with other approaches, and how it can complement existing web frameworks (such as WebWork, Spring, and Struts).
The second part of this session demonstrates how SiteMesh can be blended with other technologies to form the foundation of a rich content management system that distinguishes between the specialized roles of users, their skills, and the most suitable tools. Content writers can use a word processor, web designers can use a WYSIWYG web development tool, and developers can use their IDE.
Allowing these different roles and tools to come together to produce one website is a trivial task with SiteMesh--allowing content management to be easily introduced to existing applications.
Finally, some of the advanced features of SiteMesh are discussed, such as real world tips and tricks, how to create custom strategies for which look and feel to apply, assembling pages from components and building portal style applications.
And for the first time, new features in SiteMesh 3 will be demonstrated, including extending the HTML processor, using it outside of SiteMesh, and offline support.
Changelog:
http://xstream.codehaus.org/changes.html
Full download:
http://dist.codehaus.org/xstream/distributions/xstream-1.1.2.zip
Jar only:
http://dist.codehaus.org/xstream/jars/xstream-1.1.2.jar

Documentation says:
The NotOverridable modifier defines a method of a base class that cannot be overridden in derived classes. All methods are NotOverridable unless marked with the Overridable modifier. You can use the NotOverridable modifier when you do not want to allow an overridden method to be overridden again in a derived class.
Makes C++ look simple.
]]>Some of the improvements in this release:
Full change log: http://xstream.codehaus.org/changes.html
Download: http://xstream.codehaus.org/download.html
What you can't find out at runtime is which generic type is associated with an instance of an object. However you can use reflection to look at which types have been staticly associated with a member of a class.
public class GenericsTest extends TestCase {
class Thing {
public Map<String,Integer> stuff;
}
public void test() throws Exception {
Field field = Thing.class.getField("stuff");
ParameterizedType type = (ParameterizedType) field.getGenericType();
assertEquals(Map.class, type.getRawType());
assertEquals(String.class, type.getActualTypeArguments()[0]);
assertEquals(Integer.class, type.getActualTypeArguments()[1]);
}
}
Just wanted to clear that up.
(This is something that I'll probably exploit in XStream for J5 users to further simplify the XML.)
]]>