react 0.5.2 → 0.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (84) hide show
  1. package/.travis.yml +4 -0
  2. package/Jakefile.js +39 -0
  3. package/README.md +9 -22
  4. package/browser-test/dist.html +90 -0
  5. package/browser-test/index.html +86 -0
  6. package/browser-test/min.html +90 -0
  7. package/dist/react.js +3291 -0
  8. package/dist/react.min.js +1 -0
  9. package/doc/advanced.md +2 -2
  10. package/examples/using-events1.js +1 -1
  11. package/lib/base-task.js +116 -110
  12. package/lib/cb-task.js +71 -67
  13. package/lib/core.js +118 -119
  14. package/lib/dsl.js +120 -115
  15. package/lib/error.js +44 -36
  16. package/lib/event-collector.js +69 -56
  17. package/lib/event-manager.js +65 -58
  18. package/lib/eventemitter.js +20 -0
  19. package/lib/finalcb-first-task.js +56 -53
  20. package/lib/finalcb-task.js +55 -51
  21. package/lib/id.js +18 -7
  22. package/lib/input-parser.js +49 -41
  23. package/lib/log-events.js +79 -73
  24. package/lib/parse.js +34 -25
  25. package/lib/promise-resolve.js +42 -27
  26. package/lib/promise-task.js +78 -74
  27. package/lib/react.js +59 -0
  28. package/lib/ret-task.js +59 -55
  29. package/lib/sprintf.js +18 -0
  30. package/lib/status.js +11 -2
  31. package/lib/task.js +218 -217
  32. package/lib/track-tasks.js +72 -58
  33. package/lib/validate.js +136 -136
  34. package/lib/vcon.js +78 -69
  35. package/lib/when-task.js +69 -65
  36. package/package.json +11 -9
  37. package/src/dist.build.requirejs +20 -0
  38. package/test/ast.mocha.js +136 -0
  39. package/test/cb-task.mocha.js +220 -0
  40. package/test/core-deferred.mocha.js +143 -0
  41. package/test/core-when.mocha.js +96 -0
  42. package/test/core.mocha.js +589 -0
  43. package/test/dsl.mocha.js +350 -0
  44. package/test/event-manager.mocha.js +119 -0
  45. package/test/exec-options.mocha.js +48 -0
  46. package/test/finalcb-task.mocha.js +58 -0
  47. package/test/input-parser.mocha.js +86 -0
  48. package/test/mocha.opts +2 -0
  49. package/test/module-use.mocha.js +147 -0
  50. package/test/promise-auto-resolve.mocha.js +68 -0
  51. package/test/ret-task.mocha.js +220 -0
  52. package/test/task.mocha.js +42 -0
  53. package/test/validate-cb-task.mocha.js +100 -0
  54. package/test/validate-ret-task.mocha.js +110 -0
  55. package/test/validate.mocha.js +324 -0
  56. package/test/vcon.mocha.js +193 -0
  57. package/vendor/chai/chai.js +2038 -0
  58. package/vendor/jquery/jquery-1.7.1.js +9266 -0
  59. package/vendor/jquery/jquery-1.7.1.min.js +4 -0
  60. package/vendor/mocha/mocha.css +135 -0
  61. package/vendor/mocha/mocha.js +3589 -0
  62. package/vendor/node/util.js +531 -0
  63. package/vendor/requirejs/require.js +2053 -0
  64. package/vendor/requirejs/require.min.js +33 -0
  65. package/react.js +0 -40
  66. package/test/ast.test.js +0 -118
  67. package/test/cb-task.test.js +0 -197
  68. package/test/core-deferred.test.js +0 -134
  69. package/test/core-promised.test.js +0 -132
  70. package/test/core-when.test.js +0 -84
  71. package/test/core.test.js +0 -593
  72. package/test/dsl.test.js +0 -330
  73. package/test/event-manager.test.js +0 -102
  74. package/test/exec-options.test.js +0 -33
  75. package/test/finalcb-task.test.js +0 -38
  76. package/test/input-parser.test.js +0 -66
  77. package/test/module-use.test.js +0 -134
  78. package/test/promise-auto-resolve.test.js +0 -52
  79. package/test/ret-task.test.js +0 -199
  80. package/test/task.test.js +0 -21
  81. package/test/validate-cb-task.test.js +0 -74
  82. package/test/validate-ret-task.test.js +0 -83
  83. package/test/validate.test.js +0 -295
  84. package/test/vcon.test.js +0 -173
@@ -0,0 +1,350 @@
1
+ 'use strict';
2
+ /*global react:true sprintf:true */
3
+
4
+ if (typeof(chai) === 'undefined') {
5
+ var chai = require('chai');
6
+ }
7
+
8
+ if (typeof(react) === 'undefined') {
9
+ var react = require('../'); //require('react');
10
+ }
11
+
12
+ if (typeof(sprintf) === 'undefined') {
13
+ var sprintf = require('../lib/sprintf');
14
+ }
15
+
16
+ (function () {
17
+
18
+ var t = chai.assert;
19
+
20
+ /**
21
+ Testing DSL
22
+ */
23
+
24
+ suite('dsl');
25
+
26
+ function falpha() { }
27
+ function fbeta() { }
28
+
29
+ test('module exports is a fn with properties', function (done) {
30
+ t.isFunction(react, 'has define by DSL method'); //
31
+ t.isFunction(react.selectFirst, 'has selectFirst define method');
32
+ done();
33
+ });
34
+
35
+ test('no arguments -> empty name, inParams, tasks, outTask', function (done) {
36
+ var r = react();
37
+ t.equal(r.ast.name.slice(0, 'flow_'.length), 'flow_', 'generated flow name should start with flow_');
38
+ t.deepEqual(r.ast.inParams, []);
39
+ t.deepEqual(r.ast.tasks, []);
40
+ t.deepEqual(r.ast.outTask, { a: [], type: 'finalcb' });
41
+ done();
42
+ });
43
+
44
+
45
+ test('empty first string -> empty name, inParams, tasks, outTask', function (done) {
46
+ var r = react('');
47
+ t.equal(r.ast.name.slice(0, 'flow_'.length), 'flow_', 'generated flow name should start with flow_');
48
+ t.deepEqual(r.ast.inParams, []);
49
+ t.deepEqual(r.ast.tasks, []);
50
+ t.deepEqual(r.ast.outTask, { a: [], type: 'finalcb' });
51
+ done();
52
+ });
53
+
54
+
55
+ test('single first string -> name, inParams["foo"], empty tasks, outTask', function (done) {
56
+ var r = react('foo');
57
+ t.equal(r.ast.name, 'foo');
58
+ t.deepEqual(r.ast.inParams, []);
59
+ t.deepEqual(r.ast.tasks, []);
60
+ t.deepEqual(r.ast.outTask, { a: [], type: 'finalcb' });
61
+ done();
62
+ });
63
+
64
+ test('triple first string -> inParams["foo", "bar", "baz"], empty tasks, outTask',
65
+ function (done) {
66
+ var r = react('myName', ' foo, bar,baz ');
67
+ t.equal(r.ast.name, 'myName');
68
+ t.deepEqual(r.ast.inParams, ['foo', 'bar', 'baz']);
69
+ t.deepEqual(r.ast.tasks, []);
70
+ t.deepEqual(r.ast.outTask, { a: [], type: 'finalcb' });
71
+ done();
72
+ });
73
+
74
+ test('single task, single out params', function (done) {
75
+ var r = react('myName', 'cb -> err, c',
76
+ falpha, 'cb -> err, c'
77
+ );
78
+ t.deepEqual(r.ast.inParams, []);
79
+ t.deepEqual(r.ast.tasks, [
80
+ { f: falpha, a: [], out: ['c'], type: 'cb', name: 'falpha'}
81
+ ]);
82
+ t.deepEqual(r.ast.outTask, { a: ['c'], type: 'finalcb' });
83
+ done();
84
+ });
85
+
86
+ test('single task, err and out params', function (done) {
87
+ var r = react('myName', 'cb -> err, c',
88
+ falpha, 'cb -> err, c'
89
+ );
90
+ t.deepEqual(r.ast.inParams, []);
91
+ t.deepEqual(r.ast.tasks, [
92
+ { f: falpha, a: [], out: ['c'], type: 'cb', name: 'falpha'}
93
+ ]);
94
+ t.deepEqual(r.ast.outTask, { a: ['c'], type: 'finalcb' });
95
+ done();
96
+ });
97
+
98
+ test('using - with literal string', function (done) {
99
+ var r = react('myName', '"hello-world", cb -> err, c',
100
+ falpha, '"another-string", cb -> err, c'
101
+ );
102
+ t.deepEqual(r.ast.inParams, ['"hello-world"']);
103
+ t.deepEqual(r.ast.tasks, [
104
+ { f: falpha, a: ['"another-string"'], out: ['c'], type: 'cb', name: 'falpha'}
105
+ ]);
106
+ t.deepEqual(r.ast.outTask, { a: ['c'], type: 'finalcb' });
107
+ done();
108
+ });
109
+
110
+ test('single task, ERR and out params', function (done) {
111
+ var r = react('myName', 'cb -> ERR, c',
112
+ falpha, 'cb -> ERR, c'
113
+ );
114
+ t.deepEqual(r.ast.inParams, []);
115
+ t.deepEqual(r.ast.tasks, [
116
+ { f: falpha, a: [], out: ['c'], type: 'cb', name: 'falpha'}
117
+ ]);
118
+ t.deepEqual(r.ast.outTask, { a: ['c'], type: 'finalcb' });
119
+ done();
120
+ });
121
+
122
+ test('cb used in defs is simply ignored', function (done) {
123
+ var r = react('myName', 'a, b, cb -> err, c',
124
+ falpha, 'a, b, cb -> err, c'
125
+ );
126
+ t.deepEqual(r.ast.inParams, ['a', 'b']);
127
+ t.deepEqual(r.ast.tasks, [
128
+ { f: falpha, a: ['a', 'b'], out: ['c'], type: 'cb', name: 'falpha'}
129
+ ]);
130
+ t.deepEqual(r.ast.outTask, { a: ['c'], type: 'finalcb' });
131
+ done();
132
+ });
133
+
134
+ test('callback used in defs is simply ignored', function (done) {
135
+ var r = react('myName', 'a, b, callback -> err, c',
136
+ falpha, 'a, b, callback -> err, c'
137
+ );
138
+ t.deepEqual(r.ast.inParams, ['a', 'b']);
139
+ t.deepEqual(r.ast.tasks, [
140
+ { f: falpha, a: ['a', 'b'], out: ['c'], type: 'cb', name: 'falpha'}
141
+ ]);
142
+ t.deepEqual(r.ast.outTask, { a: ['c'], type: 'finalcb' });
143
+ done();
144
+ });
145
+
146
+ test('two inputs, two tasks, two out params', function (done) {
147
+ var r = react('myName', 'a, b, cb -> err, c, d',
148
+ falpha, 'a, b, cb -> err, c',
149
+ fbeta, 'a, b, cb -> err, d, e'
150
+ );
151
+ t.deepEqual(r.ast.inParams, ['a', 'b']);
152
+ t.deepEqual(r.ast.tasks, [
153
+ { f: falpha, a: ['a', 'b'], out: ['c'], type: 'cb', name: 'falpha'},
154
+ { f: fbeta, a: ['a', 'b'], out: ['d', 'e'], type: 'cb', name: 'fbeta'}
155
+ ]);
156
+ t.deepEqual(r.ast.outTask, { a: ['c', 'd'], type: 'finalcb' });
157
+ done();
158
+ });
159
+
160
+ test('two inputs, two tasks, two out params, options', function (done) {
161
+ var r = react('myName', 'a, b, cb -> err, c, d',
162
+ { otherOptFoo: 'foo'}, // main flow options
163
+ falpha, 'a, b, cb -> err, c',
164
+ fbeta, 'a, b, cb -> err, d, e'
165
+ );
166
+ t.deepEqual(r.ast.inParams, ['a', 'b']);
167
+ t.deepEqual(r.ast.tasks, [
168
+ { f: falpha, a: ['a', 'b'], out: ['c'], type: 'cb', name: 'falpha'},
169
+ { f: fbeta, a: ['a', 'b'], out: ['d', 'e'], type: 'cb', name: 'fbeta'}
170
+ ]);
171
+ t.deepEqual(r.ast.outTask, { a: ['c', 'd'], type: 'finalcb' });
172
+ t.equal(r.ast.name, 'myName', 'name should match');
173
+ t.equal(r.ast.otherOptFoo, 'foo', 'other options should pass through');
174
+ done();
175
+ });
176
+
177
+ test('two inputs, two mixed tasks, two out params', function (done) {
178
+ var r = react('myName', 'a, b, cb -> err, c, d',
179
+ falpha, 'a, b, cb -> err, c',
180
+ fbeta, 'a, b -> d'
181
+ );
182
+ t.deepEqual(r.ast.inParams, ['a', 'b']);
183
+ t.deepEqual(r.ast.tasks, [
184
+ { f: falpha, a: ['a', 'b'], out: ['c'], type: 'cb', name: 'falpha'},
185
+ { f: fbeta, a: ['a', 'b'], out: ['d'], type: 'ret', name: 'fbeta'}
186
+ ]);
187
+ t.deepEqual(r.ast.outTask, { a: ['c', 'd'], type: 'finalcb' });
188
+ done();
189
+ });
190
+
191
+
192
+ test('two inputs, two mixed tasks, two out params, opts', function (done) {
193
+ var r = react('myName', 'a, b, cb -> err, c, d',
194
+ falpha, 'a, cb -> err, c', { after: fbeta },
195
+ fbeta, 'a, b -> d'
196
+ );
197
+ t.deepEqual(r.ast.inParams, ['a', 'b']);
198
+ t.deepEqual(r.ast.tasks, [
199
+ { after: ['fbeta'], f: falpha, a: ['a'], out: ['c'], type: 'cb', name: 'falpha'},
200
+ { f: fbeta, a: ['a', 'b'], out: ['d'], type: 'ret', name: 'fbeta'}
201
+ ]);
202
+ t.deepEqual(r.ast.outTask, { a: ['c', 'd'], type: 'finalcb' });
203
+ done();
204
+ });
205
+
206
+
207
+ // Object use
208
+ test('object prop task params', function (done) {
209
+ var r = react('myName', 'a, b, cb -> err, c, e',
210
+ falpha, 'a, b.cat, cb -> err, c',
211
+ fbeta, 'c.dog, b -> d',
212
+ 'd.egg', 'c, cb -> err, e'
213
+ );
214
+ t.deepEqual(r.ast.inParams, ['a', 'b']);
215
+ t.deepEqual(r.ast.tasks, [
216
+ { f: falpha, a: ['a', 'b.cat'], out: ['c'], type: 'cb', name: 'falpha'},
217
+ { f: fbeta, a: ['c.dog', 'b'], out: ['d'], type: 'ret', name: 'fbeta'},
218
+ { f: 'd.egg', a: ['c'], out: ['e'], type: 'cb', name: 'd.egg'}
219
+ ]);
220
+ t.deepEqual(r.ast.outTask, { a: ['c', 'e'], type: 'finalcb' });
221
+ done();
222
+ });
223
+
224
+ // Errors
225
+
226
+ test('missing name, throws error', function (done) {
227
+ var fn = function () {
228
+ var r = react('cb -> err, c',
229
+ falpha, 'cb -> err, c'
230
+ );
231
+ };
232
+ t.throws(fn, new Error('first flow parameter should be the flow name, but found in/out def: cb -> err, c'));
233
+ done();
234
+ });
235
+
236
+ test('missing err in flow in/out - force for consistency, throw error', function (done) {
237
+ var fn = function () {
238
+ var r = react('myname', 'cb -> c', // missing err
239
+ falpha, 'cb -> err, c'
240
+ );
241
+ };
242
+ t.throws(fn, new Error('callback specified, but first out param was not "err", use for clarity. Found in/out def: cb -> c'));
243
+ done();
244
+ });
245
+
246
+ test('missing err in task in/out - force for consistency, throw error', function (done) {
247
+ var fn = function () {
248
+ var r = react('myname', 'cb -> err, c',
249
+ falpha, 'cb -> c' // missing err
250
+ );
251
+ };
252
+ t.throws(fn, new Error('callback specified, but first out param was not "err", use for clarity. Found in/out def: cb -> c'));
253
+ done();
254
+ });
255
+
256
+ test('found err, but missing cb/callback in flow in/out - force for consistency, throw error', function (done) {
257
+ var fn = function () {
258
+ var r = react('myname', 'a -> err, c', // missing cb
259
+ falpha, 'cb -> err, c'
260
+ );
261
+ };
262
+ t.throws(fn, new Error('found err param, but cb/callback is not specified, is this cb-style async or sync function? Found in/out def: a -> err, c'));
263
+ done();
264
+ });
265
+
266
+ test('found err, but missing cb/callback in task in/out - force for consistency, throw error', function (done) {
267
+ var fn = function () {
268
+ var r = react('myname', 'cb -> err, c',
269
+ falpha, 'a -> err, c' // missing cb
270
+ );
271
+ };
272
+ t.throws(fn, new Error('found err param, but cb/callback is not specified, is this cb-style async or sync function? Found in/out def: a -> err, c'));
273
+ done();
274
+ });
275
+
276
+ test('extra arg throws error', function (done) {
277
+ var fn = function () {
278
+ var r = react('myName', 'a, b, cb -> err, c, d',
279
+ falpha, 'a, cb -> err, c', { after: fbeta },
280
+ fbeta, 'a, b -> returns d',
281
+ 'extraBadArg'
282
+ );
283
+ };
284
+ t.throws(fn, new Error('extra unmatched task arg: extraBadArg'));
285
+ done();
286
+ });
287
+
288
+ test('not enough args throws error', function (done) {
289
+ var fn = function () {
290
+ var r = react('myName', 'a, b, cb -> err, c, d',
291
+ falpha, 'a, cb -> err, c', { after: fbeta },
292
+ fbeta
293
+ );
294
+ };
295
+ t.throws(fn, new Error(sprintf('extra unmatched task arg: %s', fbeta)));
296
+ done();
297
+ });
298
+
299
+ test('long example', function (done) {
300
+ /*jshint white: false */
301
+
302
+ function loadUser(uid, cb){ setTimeout(cb, 100, null, "User"+uid); }
303
+ function loadFile(filename, cb){ setTimeout(cb, 100, null, 'Filedata'+filename); }
304
+ function markdown(filedata) { return 'html'+filedata; }
305
+ function prepareDirectory(outDirname, cb){ setTimeout(cb, 200, null, 'dircreated-'+outDirname); }
306
+ function writeOutput(html, user, cb){ setTimeout(cb, 300, null, html+'_bytesWritten'); }
307
+ function loadEmailTemplate(cb) { setTimeout(cb, 50, null, 'emailmd'); }
308
+ function customizeEmail(user, emailHtml) { return 'cust-'+user+emailHtml; }
309
+ function deliverEmail(custEmailHtml, cb) { setTimeout(cb, 100, null, 'delivered-'+custEmailHtml); }
310
+ var loadAndSave = react('loadAndSave', 'filename, uid, outDirname, cb -> err, html, user, bytesWritten', // name, in/out params
311
+ loadUser, 'uid, cb -> err, user', // calling async fn loadUser with uid, callback is called with err and user
312
+ loadFile, 'filename, cb -> err, filedata',
313
+ markdown, 'filedata -> html', // using a sync function
314
+ prepareDirectory, 'outDirname, cb -> err, dircreated',
315
+ writeOutput, 'html, user, cb -> err, bytesWritten', { after: prepareDirectory }, // only after prepareDirectory done
316
+ loadEmailTemplate, 'cb -> err, emailmd',
317
+ markdown, 'emailmd -> emailHtml', // using a sync function
318
+ customizeEmail, 'user, emailHtml -> custEmailHtml',
319
+ deliverEmail, 'custEmailHtml, cb -> err, deliveredEmail', { after: writeOutput } // only after writeOutput is done
320
+ );
321
+ loadAndSave('file.md', 100, '/tmp/foo', function (err, html, user, bytesWritten) { // executing the flow
322
+ t.equal(err, null);
323
+ t.equal(html, 'htmlFiledatafile.md');
324
+ t.equal(user, 'User100');
325
+ t.equal(bytesWritten, 'htmlFiledatafile.md_bytesWritten');
326
+ done();
327
+ });
328
+ });
329
+
330
+ // selectFirst
331
+
332
+ test('selectFirst', function (done) {
333
+ var r = react.selectFirst('myName', 'a, b, cb -> err, c',
334
+ { otherOptFoo: 'foo'}, // main options
335
+ falpha, 'a, b, cb -> err, c',
336
+ fbeta, 'a, b -> c'
337
+ );
338
+ t.equal(r.ast.name, 'myName');
339
+ t.deepEqual(r.ast.inParams, ['a', 'b']);
340
+ t.deepEqual(r.ast.tasks, [
341
+ { f: falpha, a: ['a', 'b'], out: ['c'], type: 'cb', name: 'falpha'},
342
+ { f: fbeta, a: ['a', 'b'], out: ['c'], type: 'ret', name: 'fbeta', after: ['falpha']}
343
+ ]);
344
+ t.deepEqual(r.ast.outTask, { type: 'finalcbFirst', a: ['c'] });
345
+ t.equal(r.ast.name, 'myName', 'name should match if supplied');
346
+ t.equal(r.ast.otherOptFoo, 'foo', 'other options should pass through');
347
+ done();
348
+ });
349
+
350
+ }());
@@ -0,0 +1,119 @@
1
+ 'use strict';
2
+ /*global react:true EventManager:true */
3
+
4
+ if (typeof(chai) === 'undefined') {
5
+ var chai = require('chai');
6
+ }
7
+
8
+ if (typeof(react) === 'undefined') {
9
+ var react = require('../'); //require('react');
10
+ }
11
+
12
+ if (typeof(EventManager) === 'undefined') {
13
+ var EventManager = require('../lib/event-manager.js');
14
+ }
15
+
16
+ (function () {
17
+
18
+ var t = chai.assert;
19
+
20
+ /**
21
+ Testing EventManager
22
+ */
23
+
24
+ suite('event-manager');
25
+
26
+ test('Event Manager is disabled by default', function (done) {
27
+ var em = EventManager.create();
28
+ t.isFalse(em.isEnabled(), 'should be falsey by default');
29
+ done();
30
+ });
31
+
32
+ test('Event Manager enabled when a listener if registered', function (done) {
33
+ var em = EventManager.create();
34
+ em.on('foo', function () { });
35
+ t.isTrue(em.isEnabled(), 'should be truthy');
36
+ done();
37
+ });
38
+
39
+ test('Event Manager emits events', function (done) {
40
+ var em = EventManager.create();
41
+ em.on('foo', function (data) {
42
+ t.equal(data, 'hello');
43
+ done();
44
+ });
45
+ em.emit('foo', 'hello');
46
+ });
47
+
48
+ test('Event Manager emits events, wildcarded listener', function (done) {
49
+ var em = EventManager.create();
50
+ em.on('foo.*', function (data) {
51
+ t.equal(data, 'hello-world');
52
+ done();
53
+ });
54
+ em.emit('foo.bar', 'hello-world');
55
+ });
56
+
57
+ test('Event Manager emits events, all wildcarded listener', function (done) {
58
+ var em = EventManager.create();
59
+ em.on('*', function (data) {
60
+ t.equal(data, 'hello-world');
61
+ done();
62
+ });
63
+ em.emit('foo.bar', 'hello-world');
64
+ });
65
+
66
+ test('Event Manager emits event with three args', function (done) {
67
+ var em = EventManager.create();
68
+ em.on('foo', function (data, data2, data3) {
69
+ t.equal(data, 'hello');
70
+ t.equal(data2, 'world');
71
+ t.equal(data3, 100);
72
+ done();
73
+ });
74
+ em.emit('foo', 'hello', 'world', 100);
75
+ });
76
+
77
+ test('Event Manager with a parentListener will receive events', function (done) {
78
+ var parentEm = EventManager.create();
79
+ var em = EventManager.create();
80
+ em.parent = parentEm;
81
+ parentEm.on('foo', function (data) {
82
+ t.equal(data, 'world');
83
+ done();
84
+ });
85
+ em.emit('foo', 'world');
86
+ });
87
+
88
+ test('Event Manager w/listening grandparent will receive events', function (done) {
89
+ var grandEm = EventManager.create();
90
+ var parentEm = EventManager.create();
91
+ parentEm.parent = grandEm;
92
+ var em = EventManager.create();
93
+ em.parent = parentEm;
94
+ grandEm.on('foo', function (data) {
95
+ t.equal(data, 'world');
96
+ done();
97
+ });
98
+ em.emit('foo', 'world');
99
+ });
100
+
101
+ test('Event Manager all parents w/listeners will receive events', function (done) {
102
+ var grandEm = EventManager.create();
103
+ var parentEm = EventManager.create();
104
+ parentEm.parent = grandEm;
105
+ var em = EventManager.create();
106
+ em.parent = parentEm;
107
+ var expectedCalls = 2;
108
+ function recData(data) {
109
+ t.equal(data, 'hello');
110
+ expectedCalls -= 1;
111
+ if (!expectedCalls) done();
112
+ }
113
+ grandEm.on('bar', recData);
114
+ parentEm.on('bar', recData);
115
+ em.on('bar', recData);
116
+ em.emit('bar', 'hello');
117
+ });
118
+
119
+ }());