title | category |
---|---|
The IoC Testing Framework |
Infusion |
The IoC Testing Framework is written using our existing jqUnit wrapper for jQuery's QUnit as a base. The IoC Testing Framework is both written using Fluid's IoC system, as well as being designed to test components which are themselves written using IoC. This framework aims to extend our power to write tests in various directions at the same time. As well as creating an idiomatic way of writing integration tests addressed at realistic-sized chunks of applications, expressed as IoC component trees, the IoC testing framework also considerably eases the task of testing complex event sequences - that is, sequences of application state that are derived from an alternating conversation between user interaction and application response.
If your tests don't involve a number of back-to-back asynchronous interactions, it is better to express them as plain jqUnit tests.
The concept of context in Infusion IoC is derived from the entire collection of components held in an IoC component tree. The behaviour of each component is potentially altered by all of the other components with which it is deployed.
- for a detailed guide to the operation of scope within Infusion IoC, please consult the page on Contexts.
Therefore in order to test component behaviour in context, we need a testing system whose lifecycle (in particular, the lifecycle of setup and teardown common to all testing systems) is aligned with the lifecycle of component trees - as well as a testing system which enables testing directives to be referred to the components under test, wherever they may be in the tree.
The idiom to be used when binding event listeners which are responsible for implementing application behaviour is very different from that to be used when testing the application behaviour. Implementation listeners are typically bound permanently - that is, for the entire lifecycle of the component holding the listener. This is in order to make application behaviour as regular as possible, and in order to make it as easy as possible to reason about application behaviour by excluding race conditions. However, when writing tests directed at an event stream, typically the behaviour required for the listener to each individual event in the sequence is different - since the testing assertion(s) held in the listener will be verifying a component state against required conditions which change with each successive event. This requirement often makes test fixture code convoluted and brittle, holding deeply nested sequences of event binding and unbinding operations held within listeners to other events. We need a system which allows such assertions to be expressed declaratively, with this sequence flattened out into a linear list of JSON elements corresponding to each successive state in the event chain.
Writing fixtures using the IoC Testing framework requires the test implementor to derive from two special
grades, fluid.test.testEnvironment
and fluid.test.testCaseHolder
, which are packaged within
the testing framework implementation in the file IoCTestUtils.js
. The tester must derive their own component types
from these grades, and assemble them into various component trees corresponding to the desired integration scenarios.
The first type of component corresponds to the overall root of the component tree under test - the test environment,
defined in the grade fluid.test.testEnvironment
. The children of this component correspond to the entire "application
segment" under test - this may be as large (as an entire application) or as small (as a single component) as required in
order to comprise the desired fixture. These children are intermixed with components of the second type, the test
fixtures, derived from the grade fluid.test.testCaseHolder
. These fixture components are holders for declarative JSON
configuration defining the sequence and structure of a group of test cases which are to be run.
The standard structure inside a fluid.test.testCaseHolder
has an outer layer of containment, modules
, with members
corresponding to QUnit modules, and within that an entry named tests
, holding
an array of structures corresponding to QUnit test. In ordinary use, each element
tests
then contains a member named sequence
holding a list of fixture records.
As well as containing a flat list of fixture records, sequence
may also contain nested arrays of such records. These
nested arrays will be flattened into a single array by use of the utility
fluid.flatten
before being processed. This helps in assembling complex sequences out
of previously canned sequence segments. However, building up complex, reusable test sequences is best done by use of the
sequenceGrade
element,
instead of the sequence
element.
This simple example shows the testing of a simple component, fluid.tests.cat
which defines one
event, onMakeSound
, and an invoker makeSoundLater
which fires the event
asynchronously with the supplied argument. Firstly, we define the component under test:
/** Component under test **/
fluid.defaults("fluid.tests.cat", {
gradeNames: ["fluid.component"],
events: {
onMakeSound: null
},
invokers: {
makeSoundLater: {
funcName: "fluid.tests.cat.makeSoundLater",
args: ["{that}", "{arguments}.0"]
}
}
});
fluid.tests.cat.makeSoundLater = function (that, sound) {
fluid.invokeLater(function () {
that.events.onMakeSound.fire(sound);
});
};
In order to test this single component, we embed it appropriately within a testing environment, derived from the grade
fluid.test.testEnvironment
, together with a component to hold the test fixtures named fluid.tests.catTester
:
/** Testing environment - holds both fixtures as well as components under test, exposes global test driver **/
fluid.defaults("fluid.tests.catTestTree", {
gradeNames: ["fluid.test.testEnvironment"],
components: {
cat: { // instance of component under test
type: "fluid.tests.cat"
},
catTester: { // instance of test fixtures
type: "fluid.tests.catTester"
}
}
});
Finally, we need to define the test fixture holder itself, fluid.tests.catTester
, derived from
fluid.test.testCaseHolder
, as well as the test fixture code itself. This contains a simple sequence of 2 elements, the
first of which is an active fixture record which calls the invoker, and the second of
which is a passive fixture record which waits for the event to be fired and makes an assertion that its argument is
correct:
/** Test Case Holder - holds declarative representation of test cases **/
fluid.defaults("fluid.tests.catTester", {
gradeNames: ["fluid.test.testCaseHolder"],
modules: [ /* declarative specification of tests */ {
name: "Cat test case",
tests: [{
expect: 1,
name: "Test Asynchronous Meow",
sequence: [{
func: "{cat}.makeSoundLater",
args: "meow"
}, {
event: "{cat}.events.onMakeSound",
listener: "fluid.tests.testCatSound"
}]
}
]
}]
});
fluid.tests.testCatSound = function (sound) {
jqUnit.assertEquals("CATT sound is MEO", "meow", sound);
};
fluid.tests.catTestTree
), and the test fixture component (e.g. fluid.tests.catTester
) can
be written as the same component.
A more complex example of the sequence
element appears below in the asyncTester
example below.
In order to run this test case, we can either simply construct an instance of the environment tree by calling
fluid.tests.catTestTree()
, or submit its name to the global driver function fluid.test.runTests
as
fluid.test.runTests("fluid.tests.catTestTree")
. The latter method should be used when running multiple environments
within the same file to ensure that their execution is properly serialised.
The IoC testing system currently supports the following 6 types of fixture record, which can be assigned to two
categories - "executors", which actively trigger an action, and "binders" which register some form of listener in order
to receive an event from the tree under test. These are recognised using a "duck typing system" similar to that used in
the Fluid Renderer. These records may either form the complete payload for a test held in the tests
section of a
TestCaseHolder
, or may appear as elements of an array held in its sequence
member, representing a sequence of
actions (either executors or binders) to be performed by the test case.
Fixture name | Field name | Field type | Field description | Fixture category |
---|---|---|---|---|
Function executor | func /funcName [*] |
{Function|String} |
The function to be executed, represented either literally (not recommended) or as an IoC reference to a
function or the global name of one. It is also possible to use the compact format for invokers to encode the contents of
args within the IoC reference func in simple cases.
|
executor |
args [optional] |
{Object|Array} |
The argument or arguments to be supplied to function |
||
Event listener | event [*] |
{String} |
Reference to the event to which the listener will be bound. This may be either a standard IoC Reference to an event above the testCaseHolder, or else a full IoCSS reference to an event anywhere in the tree. | binder |
listener [†] |
{Function|String} | Reference to the listener to be bound to the event | ||
args [†] [optional] |
{Object|Array} |
Arguments to be supplied to the listener function when it is called - these may contain IoC references including
references to the context |
||
listenerMaker [‡] |
{Function|String} | A function which will produce a listener to be bound | ||
makerArgs [‡] [optional] |
Object/Array |
The arguments to be supplied to the listener maker function in order to produce a listener | ||
Task | task [*] |
{String} |
Reference to a function returning a Promise - such a function is known as a task. | executor |
args [optional] |
{Object|Array} |
The argument or arguments to be supplied to the function task |
||
resolve [†] |
{Function|String} |
A function to be registered as an onResolve callback to the promise.
Exactly one out of the fields resolve , reject must be set.
|
||
resolveArgs [†] [optional] |
{Object|Array} |
Arguments to be supplied to the resolve function when it is called - these may contain IoC
references including references to the context `{arguments}` as described in [Listener
Boiling](EventInjectionAndBoiling.md#listener-boiling).
|
||
reject [‡] |
{Function|String} |
A function to be registered as an onReject callback to the promise.
Exactly one out of the fields resolve , reject must be set.
|
||
rejectArgs [‡] [optional] |
{Object|Array} |
Arguments to be supplied to the |
||
Change event listener | changeEvent [*] |
{String} |
Reference to the change event to be listened to. Must be the modelChanged event attached to the
ChangeApplier of a component - e.g. a reference of the form
{component}.applier.modelChanged
|
binder |
path | {String} |
A path specification matching the EL paths for which the listener is to be registered, as per the ChangeApplier API. Just one of path or spec should
be used.
|
||
spec | {Object} | A record holding a structured description of the
required listener properties, as per the ChangeApplier API. Just one of path or
spec should be used.
|
||
listener [†] |
{Function|String} | The listener to be bound to the event | ||
args [†] [optional] |
{Object|Array} |
arguments to be supplied to the listener function when it is called - these may contain IoC references including
references to the context |
||
listenerMaker [‡] |
{Function|String} | A function which will produce a listener to be bound | ||
makerArgs [‡] [optional] |
{Object|Array} |
arguments to be supplied to the listener maker function in order to produce a listener | ||
jQuery event trigger | jQueryTrigger [*] |
{String} |
The name of a jQuery event (jQuery eventType) to be triggered
via a call to jquery.trigger | executor |
args [optional] |
{Object|Array} | additional arguments to be supplied to
jQuery.trigger |
||
element |
{jQueryable} (DOM element, jQuery, or selector) | The jQuery object on which the event is to be triggered | ||
jQuery event binder | jQueryBind [*] |
{String} |
The name of a jQuery event for which a listener is to be registered via a call to jquery.one | binder |
element |
{jQueryable} (DOM element, jQuery, or selector) | The jQuery object on which a listener is to be bound | ||
args [optional] |
{Object|Array} |
additional arguments to be supplied to jQuery.one |
||
listener [†] |
{Function|String} |
The listener to be bound to the event | ||
listenerMaker [‡] |
{Function|String} |
A function which will produce a listener to be bound | ||
makerArgs [‡] [optional] |
{Object|Array} |
arguments to be supplied to the listener maker function in order to produce a listener |
In each case in this table,
- In cases where "Field type" accepts a
String
and the description reads "reference to", the field holds an IoC reference to the value in question. - Fields marked with [*] (grey rows) are the essential "duck typing fields" which define the type of the fixture records and are mandatory.
- Fields marked with [†] (red rows) and [‡] (green rows) are alternatives to
each other - they may not be used simultaneously within the same fixture. For example, you must use just one of the
styles
listener
orlistenerMaker
to specify an event listener, or specify just one ofresolve
orreject
to a task fixture.
This section covers topics of interest to more advanced users of the IoC Testing Framework. These topics relate to more
flexible and dynamic ways of building up test sequence fixtures, beyond simply listing them in the fixed array named
sequence
.
A common patten is for groups of related tests to form an ecology, sharing some test sequence elements but with others
interleaved between them and/or some others removed or reconfigured. Working with the raw sequence
array directly will
lead to this testing code becoming fragile as the sequence array indices will be unstable between different members of
the ecology. Instead, the IoC Testing framework supports a variant element sequenceGrade
which uses Infusion's
priority system to allow sequences to be built up piece by piece using priority directives after:
and
before:
expressing their relative positions in the sequence. sequenceGrade
can be used together with the existing
sequence
element, but is more regularly used without it.
The element sequenceGrade
holds a string value, designating a component grade name which has
been implemented by the test case author, descended from the standard framework grade fluid.test.sequence
. The
framework will instantiate a component with this grade as a child component of the testCaseHolder
, and then resolve
its options path sequenceElements
, whose keys represent namespaces and whose values are of a type
testSequenceElement
, the members of which are described in the following table:
Members of a testSequenceElement entry within the sequenceElements block of a
fluid.test.sequence component
|
||
---|---|---|
Member | Type | Description |
gradeNames (optional) |
{String|Array of String} |
One or more grade names, designating the grades of a component descended from
fluid.test.sequenceElement
|
options (optional) |
{Object} |
Any additional options required to construct the fluid.test.sequenceElement |
priority (optional) |
{Priority} value - see Priorities for a full explanation |
The priority (if any) that the sequence element component should have over any others appearing within
the same elements block of its parent fluid.test.sequence . The namespaces used
in these priority constraints will be taken from the keys of the elements hash.
|
namespace (optional) |
{String} |
If present, will override the key of this member of elements in representing the namespace
of this element.
|
The members gradeNames
and options
in the above table collectively designate an Infusion component derived from
fluid.test.sequenceElement
— this component will be instantiated as a child component of the
fluid.test.testCaseHolder
and its options member sequence
will be evaluated. This member sequence
holds one or
more fixture elements
just as seen in the top-level
sequence
element.
Any fixture elements listed at the traditional top-level sequence
member will be grandfathered in to this system with
a namespace of sequence
— for example, a testSequenceElement
with a priority of "before:sequence"
will be
sorted before these elements.
Here is a simple example of building up a sequence of elements piece by piece, using priority
constraints of the type after:<namespace>
and before:<namespace>
. At the top level, we define a compound
testEnvironment
and testCaseHolder
that defines a single module holding a single test, referencing the grade
fluid.tests.elementPrioritySequence
as designating the overall sequence to be built up as a sequenceGrade
. This
grade is defined with four testSequenceElement
entries, whose namespaces are check
, end
, postBeginning
and
beginning
(deliberately defined in a different order to that which they will be executed in). These elements each
specify a priority
element to guide the order in which they should be sorted into a sequence. If you run this example,
you will see them executed in the order beginning
, postBeginning
, sequence
(the grandfathered-in top-level
sequence), end
and finally check
.
The sequence makes use of two reusable sequenceElement
grades, fluid.tests.elementPriority.log
which defines a
sequence element which logs a console message (this fixture element uses the compact
syntax for encoding simple argument lists together with the function), and
fluid.tests.elementPriority.check
which defines a sequence element making a jqUnit assertion which always passes.
fluid.defaults("fluid.tests.elementPriority.log", {
gradeNames: "fluid.test.sequenceElement",
sequence: [{
func: "fluid.log({that}.options.message)"
}]
});
fluid.defaults("fluid.tests.elementPriority.check", {
gradeNames: "fluid.test.sequenceElement",
sequence: [{
func: "jqUnit.assert",
args: "I am the check, right at the end"
}]
});
fluid.defaults("fluid.tests.elementPrioritySequence", {
gradeNames: "fluid.test.sequence",
sequenceElements: {
check: {
gradeNames: "fluid.tests.elementPriority.check",
priority: "after:end"
},
end: {
gradeNames: "fluid.tests.elementPriority.log",
options: {
message: "I am at the end, just before the check"
},
priority: "after:sequence"
},
postBeginning: {
gradeNames: "fluid.tests.elementPriority.log",
options: {
message: "I come after the beginning"
},
priority: "after:beginning"
},
beginning: {
gradeNames: "fluid.tests.elementPriority.log",
options: {
message: "I will be executed first"
},
priority: "before:sequence"
}
}
});
fluid.defaults("fluid.tests.elementPriority", {
gradeNames: ["fluid.test.testEnvironment", "fluid.test.testCaseHolder"],
modules: [{
name: "Priority-driven grade budding",
tests: [{
expect: 1,
name: "Simple sequence of 4 active elements",
sequenceGrade: "fluid.tests.elementPrioritySequence",
sequence: [{
func: "fluid.log",
args: "I am the original sequence, in the middle"
}]
}
]
}]
});
A further author could now use grade inheritance to build on the above testing scenario to add
their own element — for example, easily interleaving an extra step before the end
step:
fluid.defaults("fluid.tests.derivedElementPrioritySequence", {
gradeNames: "fluid.tests.elementPrioritySequence",
sequenceElements: {
beforeEnd: {
gradeNames: "fluid.tests.elementPriority.log",
options: {
message: "I come just before the end"
},
priority: "before:end"
}
}
});
fluid.defaults("fluid.tests.derivedElementPriority", {
gradeNames: ["fluid.test.testEnvironment", "fluid.test.testCaseHolder"],
modules: [{
name: "Derived Priority-driven grade budding",
tests: [{
expect: 1,
name: "Sequence with extra element inserted before end",
sequenceGrade: "fluid.tests.derivedElementPrioritySequence"
}]
}]
});
To take complete control of the fixture building process, you can write arbitrary code returning the entire set of
modules
by replacing it with the entry moduleSource
which takes the same form as an invoker record
with entries func
/funcName
and args
, and returns a list of fixtures in the same format as modules
. Again, in a
real example, this would use the sequence
form of fixtures shown at the bottom of the page.
Before choosing this option, you are encouraged to see how far you can go with the completely declarative approach
involving sequenceGrade
.
// Example of the above fixture written with "moduleSource"
// Definition of modules as namespaced global so that it is available for others and processing
fluid.tests.catTesterModules = [{
name: "Cat test case",
tests: [{
expect: 1,
name: "Test Global Meow",
type: "test",
func: "fluid.tests.globalCatTest",
args: "{cat}"
}]
}];
// A helper function which just returns the global -
// realistically, it would assemble a sequence using more complex logic
fluid.tests.getCatModules = function () {
return fluid.tests.catTesterModules;
};
fluid.defaults("fluid.tests.catTester", {
gradeNames: ["fluid.test.testCaseHolder"],
moduleSource: {
funcName: "fluid.tests.getCatModules"
}
});
This example shows sequence testing of a view component fluid.tests.asyncTest
with genuine asynchronous behaviour (as
well as synchronous event-driven behaviour). The component under the test is an Infusion Renderer
component which renders a button, and a model-bound text
entry field. The component defines a listener to clicks to the button which asynchronously (via window.setTimeout
)
fires to an Infusion event named buttonClicked
. Separately, the component binds listeners to
change events from the text field, which are corresponded with the standard ChangeApplier events resulting from
corresponding changes to the component's model.
/** Component under test **/
fluid.defaults("fluid.tests.asyncTest", {
gradeNames: ["fluid.rendererComponent"],
model: {
textValue: "initialValue"
},
selectors: {
button: ".flc-async-button",
textField: ".flc-async-text"
},
events: {
buttonClicked: null
},
protoTree: {
textField: "${textValue}",
button: {
decorators: {
type: "fluid",
func: "fluid.tests.buttonChild"
}
}
}
});
fluid.defaults("fluid.tests.buttonChild", {
gradeNames: ["fluid.viewComponent"],
events: {
buttonClicked: "{asyncTest}.events.buttonClicked"
},
listeners: {
"onCreate.bindClick": "fluid.tests.buttonChild.bindClick"
}
});
fluid.tests.buttonChild.bindClick = function (that) {
that.container.click(function () {
setTimeout(that.events.buttonClicked.fire, 1);
});
};
Just as with the simple cat testing example above, we embed this component together with a suitable TestCaseHolder
within an overall testEnvironment
:
fluid.defaults("fluid.tests.asyncTestTree", {
gradeNames: ["fluid.test.testEnvironment"],
markupFixture: ".flc-async-root",
components: {
asyncTest: {
type: "fluid.tests.asyncTest",
container: ".flc-async-root"
},
asyncTester: {
type: "fluid.tests.asyncTester"
}
}
});
Finally, we show the contents of the associated TestCaseHolder
. In this case, the 1 test it defines holds a sequence
member prescribing a sequence of 11 states for the component, which run a total of 7 jqUnit assertions. These show
records of 5 of the types defined above - the framework ensures the correct sequence of activities (including binding
and unbinding of listeners registered in binder
records) is operated.
The sequence first initiates rendering of the overall component with a custom global function
fluid.tests.startRendering
, which checks that the component has rendered correctly and then initiates a click on the
rendered button element. The sequence then checks for the expected asynchronous Fluid event firing - it then synthesises
a further click on the button and checks for the same event again. It then synthesises an update to the rendered text
field in the UI, and listens to the expected ChangeEvent generated by this update. It changes the field again to a
different value and listens for the further ChangeEvent. Next, the sequence makes a direct call to a jqUnit assertion
function to verify that the component's model has been updated properly. Finally, it returns to the button, directly
simulating a click event using the jQueryTrigger
fixture type, and listening to that event itself using the
jQueryBind
fixture type.
The TestCaseHolder makes reference to a few global utility functions which are reproduced below.
fluid.defaults("fluid.tests.asyncTester", {
gradeNames: ["fluid.test.testCaseHolder"],
newTextValue: "newTextValue",
furtherTextValue: "furtherTextValue",
modules: [ {
name: "Async test case",
tests: [{
name: "Rendering sequence",
expect: 7,
sequence: [ {
func: "fluid.tests.startRendering",
args: ["{asyncTest}", "{instantiator}"]
}, {
listener: "fluid.tests.checkEvent",
event: "{asyncTest}.events.buttonClicked"
}, { // manually click on the button
jQueryTrigger: "click",
element: "{asyncTest}.dom.button"
}, {
listener: "fluid.tests.checkEvent",
event: "{asyncTest}.events.buttonClicked"
}, { // Issue two requests via UI to change field, and check model update
func: "fluid.tests.changeField",
args: ["{asyncTest}.dom.textField", "{asyncTester}.options.newTextValue"]
}, {
// old-fashioned "listenerMaker" - discouraged in modern code
listenerMaker: "fluid.tests.makeChangeChecker",
args: ["{asyncTester}.options.newTextValue", "textValue"],
path: "textValue",
changeEvent: "{asyncTest}.applier.modelChanged"
}, {
func: "fluid.tests.changeField",
args: ["{asyncTest}.dom.textField", "{asyncTester}.options.furtherTextValue"]
}, {
listenerMaker: "fluid.tests.makeChangeChecker",
makerArgs: ["{asyncTester}.options.furtherTextValue", "textValue"],
// alternate style for registering listener
spec: {path: "textValue", priority: "last"},
changeEvent: "{asyncTest}.applier.modelChanged"
}, {
func: "jqUnit.assertEquals",
args: ["Model updated", "{asyncTester}.options.furtherTextValue",
"{asyncTest}.model.textValue"]
}, { // manually click on the button a final time with direct listener
jQueryTrigger: "click",
element: "{asyncTest}.dom.button"
}, {
jQueryBind: "click",
element: "{asyncTest}.dom.button",
listener: "fluid.tests.checkEvent"
}
]
}
]
}]
});
fluid.tests.checkEvent = function () {
jqUnit.assert("Button event relayed");
};
fluid.tests.changeField = function (field, value) {
field.val(value).change();
};
fluid.tests.makeChangeChecker = function (toCheck, path) {
return function (newModel) {
var newval = fluid.get(newModel, path);
jqUnit.assertEquals("Expected model value " + toCheck + " at path " + path, toCheck , newval);
};
};
fluid.tests.startRendering = function (asyncTest, instantiator) {
asyncTest.refreshView();
var decorators = fluid.renderer.getDecoratorComponents(asyncTest, instantiator);
var decArray = fluid.values(decorators);
jqUnit.assertEquals("Constructed one component", 1, decArray.length);
asyncTest.locate("button").click();
};
Such repetitive sequences of standardised fixtures are best factored into reusable grades of type
fluid.test.sequenceElement
as seen in the sequenceGrade example
above.
This environment shows use of the optional markupFixture
property on the testEnvironment
. Since the IoC testing
framework operates setup/teardown on the unit of overall testEnvironment
s, we cannot (should not) make use of QUnit's
standard markup setup/teardown operated on the hard-wired DOM node with id qunit-fixture
, which is on the unit of
individual test cases. The markupFixture
property is to be used where the overall environment makes use DOM material
where its markup is rendered, which should be reset to its original value between runs of different testEnvironment
s.
The markupFixture
property holds any jQueryable value, designating the overall root node of this DOM material. After
the testEnvironment
has been torn down, the framework will reset the markup within this root to the contents it
enjoyed before setup of the environment.
When run in the browser, the framework will show feedback in the QUnit UI relating to the sequence point reached by the system. This can be used to diagnose the last successfully reached sequence point in the case of a "hang" caused by an unexpectedly missing event in the sequence.
If the next expected "binder" type fixture in a test sequence is not reached within a configurable interval, the console will log a message of the following form to help the user to diagnose how far the system has progressed through the sequence:
21:26:33.262: Test case listener has not responded after 5000ms - at sequence pos 4 of 7 sequence element {
"event": "{testEnvironment}.browser.events.onLoaded",
"listener": "{testEnvironment}.browser.evaluate",
"args": [
{ Function
},
"md",
"[unified listing](https://ul.gpii.net/)"
]
} of fixture Confirm that the client-side renderer can render markdown...
The default value of the interval is 5000ms, which can be altered by supplying a value in ms to the option hangWait
of
the testEnvironment
.
The IoC Testing Framework can also be used to run test sequences in the node.js server environment - in this case the
above browser-related features (markupFixture
, live sequence progress) are not provided. However, the sequence hang
detection message appears in all environments.
The framework was designed over October-December 2012, with initial call for implementation on the fluid-work mailing list at October 31st, continuing over a sequence of community meetings, and including a summary of work in progress on December 5th. The overall goals for the testing framework were presented as these:
- To facilitate the testing of demands blocks that may be issued by integrators against components deployed in a particular (complex) context
- To automate and regularise the work of "setup" and "teardown" in complex integration scenarios, by deferring this to our standard IoC infrastructure
- To simplify the often tortuous logic required when using the "nested callback style" to test a particular sequence of asynchronous requests and responses (via events) issued against a component with complex behaviour
- To facilitate the reuse of testing code by allowing test fixtures to be aggregated into what are the two standard forms for our delivery of implementation - a) pure JSON structures which can be freely interchanged and transformed, b) free functions with minimum dependence on context and lifecycle
The framework was given a substantial spring-cleaning in October 2016, implementing significant new features such as promise-based fixtures, and priority-driven sequence grade assembly, and some support for compact invokers. A few significant bugs remain, especially when listing multiple "listener"-type fixtures adjacent in the sequence - see FLUID-5502. The support for model change event fixtures is also very old-fashioned and needs to be reformed - see FLUID-6077.