(For a friendlier overview, see http://bas.cgiffard.com/)
Behaviour Assertion Sheets (Bas, pronounced 'base') are a way to describe how a web page fits together, make assertions about its structure and content, and be notified when these expectations are not met. It's a bit like selenium, if you've ever used that. An easier DSL for client-side integration testing.
You could:
- Use BAS to monitor the Apple Developer site to tell you when WWDC tickets are on sale, or your local postal service to tell you when a package has been delivered
- Scan your site for common accessibility pitfalls, such as missing alt tags on images, poor heading order, or even check for hard-to-read copy with the inbuilt readability tests!
- Monitor for service availability and downtime
- Integration testing and integrity verification (plug Bas into jenkins or travis!)
- Use BAS inside of an existing test framework like Mocha to verify output, or even as reporting middleware inside your express application
Anybody who has ever used CSS can use Bas - the [syntax is easy and familiar.] (#sheet-syntax)
This first implementation of Bas is built with node.js, so you'll need it and npm first. Then just use npm to install Bas:
npm install -g bas
Installing globally (-g
) makes a CLI tool available
for working with Bas sheets. If you don't install globally you can still use Bas
via the node.js API.
As mentioned earlier, the Bas syntax looks very similar to (and nearly even parses as) CSS. Here are the major components:
(You can work this out yourself and just want to skip to the goods? Jump to syntax example.)
Rulesets are the highest-level construct in Bas. Everything falls inside a ruleset.
There are two kinds of rulesets - page specific rulesets denoted by the tag @page
,
and rulesets that execute against every page unconditionally, denoted by the
tag @all
.
Syntactically these are based on the 'at-rules' of CSS (such as @font-face
,
@media
, etc.)
Rulesets cannot be nested.
An example rulset:
@all {
...
}
Annotations are an extension of CSS comments, that are prepended with an @
symbol.
Bas knows to associate these with rulesets and selectors that follow, and displays
them in assertion failure traces so you know where they came from!
You may add as many annotations as you like to a single element. Every annotation
that precedes a block, regardless of whether assertions or regular comments (just
normal CSS comments without an @
) are interspersed within them, is associated
with that block.
An example annotation:
/*@ Here's my annotation! */
A condition is appended to a page-specific ruleset (@page
) and determines based
on the response information, URL of the page, and other environment variables,
whether the current page should be evaluated against this ruleset.
Conditions are additive and exclusive - each has to be true for the page to be
considered for testing against a given ruleset. You may add as many conditions
as you like to a @page
ruleset.
Conditions are composed of a parentheses-wrapped set of three elements, each space
separated. On the left-hand side, a test
- a reference to a function which
returns an environment variable or extracts an aspect of the current page or
server response.
The middle is an operator, which defines how the comparison takes place. An example
of an operator might be =
or >=
or !=~
. A full list of operators can be
found in the syntax glossary.
The rightmost component is the assertion value - a string, number, or regular expression which is compared to the test according to the rules of the operator.
An example condition:
@page (status-code = 301) { ... }
Multiple conditions may be combined like so:
@page (status-code = 301) (content-type != text/html) { ... }
Remember that adding more conditions will make the match more exclusive, as every single one must succeed for the ruleset to be evaluated.
A selector groups a block of assertions together, and executes them against every node in a page that matches the selector string.
The selector string is formatted exactly like a regular CSS selector - tags, IDs, classes, pseudoclasses, and attribute syntax are all the same.
The assertions wrapped within a selector block are only executed should the
selector match at least one node - with one exception: the special required
assertion subject which executes regardless of whether a
selector matches.
There's a caveat to this too, though: should a selector containing the required
assertion subject be nested inside another selector block which does not match
any nodes, it will not be executed. This allows syntax like the following:
h2 {
h1 { required: true; }
}
In this case, the heading 1 is required if one or more second-level headings are present.
Selector blocks can be nested. If a selector block is nested within another, it will only be executed should the parent selector match.
When selector blocks are nested, special scoping variables may be used.
The scoping variable $this
maps to the parent selector block's selector string.
Therefore, consider the following example:
#content {
$this b {
/* Hey! */
}
}
The inner selector $this b
will map to #content b
.
The $node
scope is similar to $this
— however it is even more restrictive,
only searching within the exact node (or nodes) which was/were selected.
#content header {
$node h3 {
}
}
In the above example, $node h3
is equivalent to a scoped search for h3
within
each individual element matching #content header
.
Selectors may contain values interpolated from test results executed in their parent context.
For example, lets say you want to make sure that any element with an
aria-describedby
attribute has a matching element ID somewhere on the page.
/* ARIA attributes */
$this [aria-describedby] {
/*@ WCAG (1.3.1 A, 4.1.1 A, 4.1.2 A) There must be a tag with a matching ID
for the aria-describedby attribute */
[id=$(attribute(aria-describedby))$] {
count: 1;
required: true;
}
}
The $(...)$
construct instructs Bas to execute the string
attribute(aria-describedby)
as a test, and return the result, interpolating it
into the selector.
Therefore, the final interpolated selector might look like:
[id=image-header]
An assertion is very similar to a declaration
in CSS. Fundamentally, it is a
semicolon delimited key-value pair, that unlike CSS, defines an expectation
rather than assigning a value.
The left-hand side of the assertion is known as the subject of the assertion, and refers to a test - a function that returns a value based on the content of the current page/request.
This value is then compared against the right-hand side of the assertion - which
can contain any number of match requirements, separated by commas and/or spaces.
These requirements are evaluated separately, and should any single one of them
fail (return a falsy value) the assertion will be considered failed
.
Match requirements for an assertion can be strings, numbers, regular expressions, negated regular expressions (prepended with !) or barewords.
An example of an assertion in use:
attribute(style): contains("font-family");
The left-hand side of every assertion is known as an assertion subject
, and
refers to a test function that returns a value from the current page or response
information. A list of these functions can be found in the [syntax glossary.]
(#tests)
An example of an assertion subject in use might be:
title: /github/i;
In this case, the assertion subject is title
. It refers to a test function called
title
which extracts the current document title. This is returned for the regex
comparison on the right hand side of the assertion.
Some tests take arguments. This is how an assertion with test arguments is represented:
attribute(role): "main";
The value of an assertion test function can be subsequently transformed by special functions known as transform functions.
These can be chained against the value of an assertion test using the delimiter .
.
Purely for illustrative purposes, here's an example of using transform functions (fictitious... for now) to rot-13 text from a node before validating the assertion:
h1 {
text.rot13: /* some match here... */
}
Multiple transforms can be applied:
h1 {
text.rot13.rot13: /* text is back to normal! */
}
And arguments can be provided to transform functions, just like to the subject test itself.
h1 {
text.rot(13): /* some match here... */
text.rot(13).rot(13): /* some match here... */
}
A more realistic use-case can be found in the text-statistics functions. If you want to check the flesch-kincaid reading ease of a given node, you could use:
h1 {
text.flesch-kincaid-reading-ease: gte(80);
}
You could check the reading-ease of the alt-text on an image, too:
img {
attribute(alt).flesch-kincaid-reading-ease: gte(80);
}
The right-hand side of the assertion, as well as regular expression, numeric, and string matches, can contain special keywords known as barewords (for their lack of enclosing quotation marks.)
These keywords refer to a special function that by design has no access to the document - just the value returned by the assertion subject, and any optional arguments it is given.
If the result of this function is falsy, then the assertion is considered failed
.
A full list of barewords can be found in the syntax glossary.
An example of barewords in use:
attribute(user-id): exists, longer-than(1), gte(1);
@page (title =~ /github/i) (domain = github.com) {
status-code: 200;
img[src*="akamai"] {
required: true;
attribute(alt): true;
count: 3;
}
/*@ Require a heading 1 to be present if there's a heading 2 */
h2 {
h1 {
required: true;
}
}
}
@all {
status-code: lt(500);
}
This example provides a fairly broad look at what Bas can do and how it works.
Let's break this example down bit by bit.
Given a page from the domain github.com
, with a document title that matches the
regular expression /github/i
:
- Bas will check that the status code of the page matches the asserted
200 OK
. - Bas will select all images with
akamai
somewhere in the in thesrc
attribute, and:- Assert that at least one appears on the page
- Assert that each has an
alt
attribute - Assert that exactly three should appear on the page if the selector matches
- Bas will select every heading 2 (h2) on the page
- If there's at least one heading two on the page, Bas will select every heading 1 (h1), and: * Assert that if a heading 2 is present, at least one heading 1 should also be present on the page.
Then, on every page tested, Bas will check to see whether the status code of the response was less than 500.
Operators are used in ruleset conditions, like (title !=~ /github/i)
.
A full list follows:
=
true ifa == b
!=
true ifa !== b
=~
true if the regular expressiona
matchesb
!=~
true if the regular expressiona
does not matchb
>
true ifa > b
where botha
andb
are considered floats<
true ifa < b
where botha
andb
are considered floats>=
true ifa >= b
where botha
andb
are considered floats<=
true ifa <= b
where botha
andb
are considered floats
Tests without arguments may be used in ruleset conditions, like
(title !=~ /github/i)
, or as assertion subjects with or without arguments, like
attribute(role): "navigation"
.
Tests can also be added programatically. [See the API documentation for details.] (#bas-nodejs-api)
- title Returns the title of the document.
- url Returns the complete URL used to request the document.
- domain Returns the domain from the URL used to request the document.
- protocol Returns the domain from the URL used to request the document. HTTP if unspecified.
- port Returns the port from the URL used to request the document. 80 if unspecified.
- path Returns the path from the URL used to request the document. (Includes querystring)
- pathname Returns the path name from the URL used to request the document. (Does not include querystring)
- query ( [query parameter] )
Returns the entire query string from the URL used to request the document if
the 'query parameter' attribute is not passed to the test. If the
query parameter
attribute is present, the individual value for the specified query parameter will be returned, or null if the parameter does not exist. - status-code Returns the HTTP response status code the current document was served with.
- content-length
Returns the
Content-Length
header with which the current document was served. - content-type
Returns the
Content-Type
header with which the current document was served. - header (header name) Returns the value of the header specified by the argument.
- required Always returns true. (Use for testing whether a selector matches.)
- exists
Synonym for
required
.
- text Returns the (DOM) text from a given node.
- html Returns the raw html content of a given node.
- attribute (attribute name) Returns the value of the specified attribute from a given node.
- has-attribute (attribute name) Returns true if the specified attribute is present - false if not.
- tag-name Returns the tag name of a given node.
- count Returns the number of nodes that matched a given selector.
- flesch-kincaid-reading-ease Returns the readability score (according to the flesch-kincaid reading ease scale) of the input text.
- flesch-kincaid-grade-level Returns the readability score (according to the flesch-kincaid US grade level scale) of the input text.
- gunning-fog-score Returns the readability score (according to the gunning-fog scale) of the input text.
- coleman-liau-index Returns the readability score (according to the coleman-liau index) of the input text.
- smog-index Returns the readability score (according to the SMOG index) of the input text.
- automated-readability-index Returns the readability score (according to the automated readability index) of the input text.
- letter-count Returns the number of latin letters in the text.
- sentence-count Returns the number of sentences in the text (for latin languages.)
- word-count Returns the number of words in the input text (for latin languages.)
- average-words-per-sentence Returns the average number of words in each sentence in the input text.
- average-syllables-per-word Returns the average number of syllables per word in the input text.
- length
Returns the
.length
property of the input. - type Returns the JS type of the input (as reported by typeof, so quirks abound.)
Barewords are used in assertions to evaluate the result of a test. Barewords can have arguments.
- true Tests whether a test result is truthy.
- false Tests whether a test result is falsy.
- exists Synonym for true. (Can make sheets more readable.)
- required Synonym for true. (Can make sheets more readable.)
- forbidden Synonym for false. (Can make sheets more readable.)
- gt (expectation) Tests whether the test result is numerically greater than the expectation.
- gte (expectation) Tests whether the test result is numerically greater or equal than the expectation.
- lt (expectation) Tests whether the test result is numerically less than the expectation.
- lte (expectation) Tests whether the test result is numerically less than or equal to the expectation.
- ne (expectation) Tests whether the test result is numerically not equal to the expectation.
- Length (expected length) Returns true if the length of a test result (cast as a string) matches the expected length.
- longer-than (expectation) Tests whether the string length of the test result is greater than the expectation.
- shorter-than (expectation) Tests whether the string length of the test result is less than the expectation.
- contains (expectation) Tests whether the test result as a string contains an exact match for the expectation.
- one-of (expectation, [expectation...]) Tests whether the test result is an exact match for any one of the arguments. Coerces both the arguments and the test result to string before comparison.
If you installed Bas globally, you'll have access to a bas
CLI
client which (hopefully) is available in your $PATH
.
The bas
CLI client can request a series of URLs, or initiate a crawl using the
provided list of URLs as a seed.
If you want to use Bas in another, non-JS project or in some kind of automated
capacity from the shell, you can supply a -j
option to get test results as raw
JSON.
Here's a very simple example of how you might use the CLI tool:
bas -vc -s mysheet.bas http://www.mywebsite.com/
In this example, the file mysheet.bas
would be loaded and, with verbose reporting,
a crawl of mywebsite.com initiated (the -c
option starts a crawl.) The test
suite would be run against every page returned, for as many pages as are present
and accessible from the given URL. Obviously it may make sense to limit the number
of pages downloaded: you can do this with the -l
option:
bas -vc -l 10 -s mysheet.bas http://mywebsite.com/
You may specify a single numeric range using a simple interpolation:
bas -vc -l 10 -s mysheet.bas http://mywebsite.com/node/%{20-500}
If the -s
option isn't specified, bas
will look for the assertion sheet on
STDIN
. Therefore, you can cat a file and pipe it to bas
as well:
cat mysheet.bas | bas -v http://mydomain.com/testfile.html
Or, if you haven't piped anything, bas
will prompt you to enter the sheet
information manually:
➭ bas -v http://www.regex.info
Waiting for BAS input from STDIN.
@all {
h1 { required; }
}
^D
Thanks, got it.
<snip>
Here's the full list of options supported by bas
at this time: (you can also
get a list of options by typing bas -h
at the prompt.)
-h
,--help
Output usage information-V
,--version
Output the version number-c
,--crawl
Crawl from the specified URLs-s
,--sheet [filename]
Test using the specified BAS-l
,--limit [number]
Limit number of resources to request when crawling-d
,--die
Die on first error-q
,--quiet
Suppress output (prints final report/json only)-v
,--verbose
Verbose output-j
,--json
Output list of errors/test results as JSON--csv
Output list of errors/test results as CSV--noquery
Don't download resources with query strings-u
,--username <username>
Username for HTTP Basic Auth (crawl)-p
,--password <password>
Password for HTTP Basic Auth (crawl)
The exit value from the CLI is equivalent to the number of errors that occurred when the test suite was run. If no errors occurred, of course, the exit value is zero.
The Bas API is extremely straightforward. To get started, simply require it:
var BAS = require("bas");
Create yourself a new BAS test suite like so:
var testSuite = new BAS();
Load in a Bas sheet (you can also supply a buffer if you'd prefer.)
testSuite.loadSheet("./mysheet.bas");
Then fetch a resource (in this case, we're using request) and run the test suite against it. You'll need to pass in a URL and response object as well as the page data.
request("http://example.com",function(err,res,body) {
if (err) throw err;
testSuite.run(url,res,data);
});
The test suite runs asynchronously, and emits events so you can know when errors have occurred, assertions have been tested, or that the suite has completed.
We can listen to one of these events to be alerted to when the test suite finishes, and receive a list of errors (if there were any!)
testSuite.on("end",function() {
if (testSuite.errors.length) {
console.log("Looks like there were some errors!");
testSuite.errors.forEach(function(err) {
console.error(err.message);
});
}
});
new BAS( [options] )
Returns a new Bas test suite instance. The optional options
parameter is an
object, with the following possible keys:
continueOnParseFail
(Defaults tofalse
) Should Cheerio fail to parse the HTML document, should Bas continue with the test suite, loading in a blank document? Or bail out?
BAS is an instance of node EventEmitter and implements the on
and emit
methods,
not described here.
Getter: Returns an object map of functions corresponding to tests
Getter: returns an array of assertion errors (Error instances) if any were thrown during the previous test run.
Each error has the following (some additional) properties:
message
(string - the error message.)selector
(string - if available, the selector that triggered the current assertion.)nodePath
(string - a generated, unambiguous CSS selector path to the current node.)url
(string - the url of the page that triggered this assertion.)
The list of errors may also be cleared with BAS.errors.clear()
.
Getter: An array of ruleset objects. (Better documentation for these coming soon!)
Getter: Returns an object containing statistics about past test runs.
This should be considered unstable and undocumented. It is about to change.
If given a buffer, this function will not touch the filesystem - it simply parses the data it receives immediately.
If given a filepath, asynchronously loads the entire file off disk, and parses it - adding the processed rules to the test suite object.
These rules can be accessed via BAS.rules
.
This function returns an object with promise handlers: yep
for success, and nope
for failure. See the yoyaku documentation for
more information.
Takes a string or a buffer containing Bas rules, and parses it, adding the processed rules to the test suite object.
These rules can be accessed via BAS.rules
.
This function returns an object with promise handlers: yep
for success, and nope
for failure. See the yoyaku documentation for
more information.
Registers a test in the BAS.test
object map - and makes it available to Bas
sheets to use in conditions and assertion subjects.
Initiates the running of the test suite.
It is important to give this function the correct URL and response object, or the tests may not operate correctly.
BAS
will emit events during the execution of the tests.
This function returns an object with promise handlers: yep
for success, and nope
for failure. See the yoyaku documentation for
more information.
loadsheet
Emitted when a new Bas sheet is successfully loaded.testregistered
(name, func) Emitted when a new test is registered with Bas.start
(url) Emitted when the test suite commences.parseerror
(error) Emitted when Cheerio encounters a parse error with the resource.assertion
(assertion, [node]) Emitted when Bas begins testing an assertion. The node parameter is only supplied when testing an assertion in a selector group.assertionsuccess
(assertion, [node]) Emitted when Bas completes testing an assertion, and the result is truthy. The node parameter is only supplied when testing an assertion in a selector group.assertionfailed
(assertionErr, assertion) Emitted when Bas completes testing an assertion, and the result is falsy, and the test is considered failed. The error triggered by the assertion is supplied as the first parameter.selector
(selector, node) Emitted when Bas commences testing the assertions in a selector.startgroup
(rule) Emitted when Bas commences testing the assertions in a ruleset.end
(url,errors) Emitted when Bas completes the test suite. An array of errors is provided, and the URL of the page the tests were executed against.
- Better documentation for rulesets, assertions, selector objects
- Asynchronous test support
- Comprehensive test suite (this is steadily improving!)
- Very solid cleanup
- Load in HTML/XML to test against from disk using
bas
CLI tool - Lots more test functions (for conditions and assertions)
- Support for assertion-specific error severity
- Support for headless browsers and PhantomJS
- Cross compilation of Bas sheets to selenium
Bas does not have an enormous test suite at this stage, but I'm working on filling it out as comprehensively as possible.
To run the test suite, use:
npm test
Test coverage is generated with istanbul.
To generate current statistics, run npm run-script coverage
from the Bas directory.
Statements | Branches | Functions | Lines |
---|---|---|---|
82.86% (551/665) | 77.51% (286/369) | 79.86% (115/144) | 82.78% (519/627) |
Copyright (c) 2013, Christopher Giffard.
All rights reserved.
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
- Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
- Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.