-
Notifications
You must be signed in to change notification settings - Fork 45
Writing Scenarios
- Introduction
- Matching Scenarios
- Function Scenarios
- Named Scenarios
- Expectations
- Nesting Scenarios
- Focusing Your Efforts
- Global Parameters
The description of the behaviour of a stylesheet lives within an XSpec document, which should adhere to the XSpec RELAX NG schema. All elements are in the http://www.jenitennison.com/xslt/xspec
namespace, which is bound to x
in these examples.
The document element is a x:description
element, whose stylesheet
attribute holds a relative URI pointing to the stylesheet that the XSpec document describes. You can also specify:
- a
version
attribute that gives the version of the stylesheet the XSpec document describes - a
xslt-version
attribute that gives the version of XSLT the stylesheet uses; if you don't specify it, this defaults to 2.0
The x:description
element contains a number of x:scenario
elements, each of which describes a particular scenario that's being tested. Each x:scenario
element has a label
attribute that describes the scenario in human language. For example:
<x:scenario label="when processing a para element">
...
</x:scenario>
Scenarios fall into three main types:
- matching scenarios describe the result of applying templates to a node in a particular mode (and with particular parameters)
- function scenarios describe the results of calling a particular function with particular arguments
- named scenarios describe the results of calling a particular named template with particular parameters
Matching scenarios hold a x:context
element that describes a node to apply templates to. The context can be supplied in two main ways:
- you can point to a node in an existing document by giving the document URI in the
href
attribute and, if you want, selecting a particular node by putting a path in theselect
attribute - you can embed XML within the
x:context
element; the content becomes the context node, although you can also select a node within that XML using theselect
attribute
The first method is useful if you already have example XML documents that you want to use as the basis of your testing. For example:
<x:scenario label="when processing a para element">
<x:context href="source/test.xml" select="/doc/body/p[1]"/>
...
</x:scenario>
The second method is related to the concept of a mock object: it is an example of some XML which you have created simply for testing purposes. The XML might not be legal; it only needs to have the attributes or content necessary for the particular behaviour that needs to be tested. For example:
<x:scenario label="when processing a para element">
<x:context>
<para>...</para>
</x:context>
...
</x:scenario>
The x:context
element can also have a mode
attribute that supplies the mode to apply templates in.
Function scenarios hold a x:call
element with a function
attribute whose content is a qualified name that is the same as the qualified name of the function you want to call. The x:call
element should hold a number of x:param
elements, one for each of the arguments to the function.
The x:param
elements can specify node values in the same way as the x:context
element gets set (described above), or simply by giving a select
attribute which holds an XPath that specifies the value. You can specify a name
or position
attribute for each of the x:param
elements; if you don't, the order in which they're specified will determine the order in which they're given in the function call. For example:
<x:scenario label="when capitalising a string">
<x:call function="eg:capital-case">
<x:param select="'an example string'"/>
<x:param select="true()"/>
</x:call>
...
</x:scenario>
will result in the call eg:capital-case('an example string', true())
, as will the following:
<x:scenario label="when capitalising a string">
<x:call function="eg:capital-case">
<x:param select="true()" position="2"/>
<x:param select="'an example string'" position="1"/>
</x:call>
...
</x:scenario>
Named template scenarios are similar to function scenarios except that the x:call
element takes a template
attribute rather than a function
attribute, and the x:param
elements within it must have a name
attribute that supplies the name of the parameter. These parameters can also have a tunnel
attribute to indicate a tunnel parameter. For example:
<x:scenario label="when creating a table with two columns containing three values">
<x:call template="createTable">
<x:param name="nodes">
<value>A</value>
<value>B</value>
<value>C</value>
</x:param>
<x:param name="cols" select="2"/>
</x:call>
...
</x:scenario>
In fact, you can use x:param
in the same way within the x:context
element in matching scenarios.
Each scenario can have one or more "expectations": things that should be true of the result of the function or template invocation described by the scenario. Each expectation is specified with an x:expect
element. The label
attribute on the x:expect
element gives a human-readable description of the expectation.
There are two main kinds of expectations:
- a value that the result should match, which may be
- an atomic value
- an XML snippet
- an arbitrary XPath test that should be true of the result
To specify an atomic value, use the select
attribute on the x:expect
element. For example:
<x:scenario label="when capitalising a string">
<x:call function="eg:capital-case">
<x:param select="'an example string'"/>
<x:param select="true()"/>
</x:call>
<x:expect label="it should capitalise every word in the string" select="'An Example String'"/>
</x:scenario>
To specify some XML, put it within the x:expect
element. For example:
<x:scenario label="when processing a para element">
<x:context>
<para>...</para>
</x:context>
<x:expect label="it should produce a p element">
<p>...</p>
</x:expect>
</x:scenario>
One thing to note here is that when comparing the actual result with the expected result, three dots in an element or attribute value within the expected XML means that the values aren't compared. If the actual result is:
<p>A sample para</p>
and the expected result is given as:
<p>...</p>
then these match. If the expected result is:
<p>Some other para</p>
then they don't.
To specify an arbitrary XPath test, use the test
attribute on x:expect
. For example:
<x:scenario label="when creating a table with two columns containing three values">
<x:call template="createTable">
<x:param name="nodes">
<value>A</value>
<value>B</value>
<value>C</value>
</x:param>
<x:param name="cols" select="2"/>
</x:call>
<x:expect label="the resulting table should have two columns"
test="count(/table/colspec/col) = 2"/>
</x:scenario>
Within the XPath expression, you can use the variable $x:result
to access the result of the test (i.e. the result of calling the function or the template, or of applying the template rule). In addition, if the result is a sequence of nodes, it is wrapped in a document node and this document is set as the context node of the expression.
You can also combine the test
attribute with the content of the x:expect
element if you want to just test a portion of the result. For example:
<x:scenario label="when creating a table with two columns containing three values">
<x:call template="createTable">
<x:param name="nodes">
<value>A</value>
<value>B</value>
<value>C</value>
</x:param>
<x:param name="cols" select="2" />
</x:call>
<x:expect label="the resulting table should have two columns"
test="count(/table/colspec/col) = 2" />
<x:expect label="the first row should contain the first two values"
test="/table/tbody/tr[1]">
<tr>
<td>A</td><td>B</td>
</tr>
</x:expect>
</x:scenario>
You can nest scenarios inside each other. The nested scenarios inherit the context or call from its ancestor scenarios. All the scenarios in a particular tree have to be of the same type (matching, function or named). Usually only the lowest level of the scenarios will contain any expectations. Here's an example:
<x:scenario label="when creating a table">
<x:call template="createTable" />
<x:scenario label="holding three values">
<x:call>
<x:param name="nodes">
<value>A</value>
<value>B</value>
<value>C</value>
</x:param>
</x:call>
<x:scenario label="in two columns">
<x:call>
<x:param name="cols" select="2"/>
</x:call>
<x:expect label="the resulting table should have two columns"
test="count(/table/colspec/col) = 2"/>
<x:expect label="the first row should contain the first two values"
test="/table/tbody/tr[1]">
<tr>
<td>A</td><td>B</td>
</tr>
</x:expect>
</x:scenario>
... other scenarios around creating tables with three values (with different numbers of columns) ...
</x:scenario>
... other scenarios around creating tables ...
</x:scenario>
When you create scenarios like this, the labels of the nested scenarios are concatenated to create the label for the scenario. In the above example, the third scenario has the label "when creating a table holding three values in two columns".
XSpec descriptions can get quite large, which can mean that running the tests takes some time. There are three ways of dealing with this.
First, you can import other XSpec description documents into your main one using x:import
. The href
attribute holds the location of the imported document. All the scenarios from the referenced document are imported into this one, and will be run when you execute it. For example:
<x:import href="other_xspec.xml"/>
It helps if the imported XSpec description documents can stand alone; this enables you to perform a subset of the tests. To work effectively, you'll want the imported XSpec description documents to cover the same stylesheet as the main one, or a stylesheet module that's included or imported into that stylesheet.
Second, you can mark any scenario or expectation as "pending" by wrapping them within a x:pending
element or adding a pending
attribute to the x:scenario
element. When the tests are run, any pending scenarios or expectations aren't tested (though they still appear, greyed out, in the test report). The x:pending
element can have a label
attribute to describe why the particular description is pending; for example it might hold "TODO". If you use the pending
attribute, its value should give the reason the tests are pending. For example:
<x:pending label="no support for block elements yet">
<x:scenario label="when processing a para element">
<x:context>
<para>...</para>
</x:context>
<x:expect label="it should produce a p element">
<p>...</p>
</x:expect>
</x:scenario>
</x:pending>
or:
<x:scenario pending="no support for block elements yet" label="when processing a para element">
<x:context>
<para>...</para>
</x:context>
<x:expect label="it should produce a p element">
<p>...</p>
</x:expect>
</x:scenario>
Third, you can mark any scenario as having the current "focus" by adding a focus
attribute to a <x:scenario>
element. Effectively, this marks every other scenario as "pending", with the label given as the value of the focus
attribute. For example:
<x:scenario focus="getting capitalisation working" label="when capitalising a string">
<x:call function="eg:capital-case">
<x:param select="'an example string'"/>
<x:param select="true()"/>
</x:call>
<x:expect label="it should capitalise every word in the string" select="'An Example String'"/>
</x:scenario>
Using focus is a good way of working through one particular scenario, but once your code is working with that scenario, you should always test all the others again, just in case you've broken something else.
You can put x:param
elements at the top level of the XSpec description document (as a child of the x:description
element). These effectively override any global parameters or variables that you have declared in your stylesheet. They are set in just the same way as setting parameters when testing named templates or functions.
With the current implementation, it isn't possible to have different scenarios use different values for global parameters. Testing is made easier if you declare local parameters on any templates or functions that use global parameters; these can default to the value of the global parameter, but be set explicitly when testing. For example, if $tableClass
is a global parameter, you might do the following to enable the full testing of the createTable
template:
<xsl:template name="createTable">
<xsl:param name="nodes" as="node()+" required="yes"/>
<xsl:param name="cols" as="xs:integer" required="yes"/>
<xsl:param name="tableClass" as="xs:string" select="$tableClass"/>
...
</xsl:template>