react 0.3.0 → 0.5.1

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 (44) hide show
  1. package/.npmignore +2 -1
  2. package/README.md +99 -155
  3. package/doc/advanced.md +166 -0
  4. package/doc/color-def.graffle +938 -0
  5. package/doc/color-def.png +0 -0
  6. package/doc/default-simple.dot +19 -0
  7. package/doc/default-simple.dot.png +0 -0
  8. package/examples/{default1.js → longer-example.js} +0 -0
  9. package/examples/simple.js +45 -0
  10. package/examples/{ast1.js → using-ast-directly.js} +4 -0
  11. package/examples/using-events1.js +79 -0
  12. package/examples/{default-events1.js → using-log-events.js} +5 -14
  13. package/lib/base-task.js +7 -6
  14. package/lib/core.js +33 -14
  15. package/lib/event-collector.js +68 -0
  16. package/lib/event-manager.js +16 -13
  17. package/lib/finalcb-first-task.js +14 -11
  18. package/lib/finalcb-task.js +15 -11
  19. package/lib/id.js +1 -0
  20. package/lib/log-events.js +86 -0
  21. package/{promise-resolve.js → lib/promise-resolve.js} +5 -5
  22. package/lib/task.js +18 -8
  23. package/lib/track-tasks.js +60 -0
  24. package/lib/vcon.js +1 -1
  25. package/package.json +2 -2
  26. package/react.js +33 -1
  27. package/test/ast.test.js +27 -1
  28. package/test/core.test.js +71 -57
  29. package/test/dsl.test.js +2 -2
  30. package/test/module-use.test.js +14 -9
  31. package/test/promise-auto-resolve.test.js +2 -1
  32. package/Jakefile.js +0 -8
  33. package/dsl/chain.js +0 -150
  34. package/dsl/fstr.js +0 -121
  35. package/dsl/pcode.js +0 -175
  36. package/examples/chain-events1.js +0 -34
  37. package/examples/chain1.js +0 -19
  38. package/examples/fstr-events1.js +0 -52
  39. package/examples/fstr1.js +0 -37
  40. package/examples/pcode1.js +0 -22
  41. package/jake-tasks/jake-test.js +0 -64
  42. package/test/dsl/chain.test.js +0 -323
  43. package/test/dsl/fstr.test.js +0 -300
  44. package/test/dsl/pcode.test.js +0 -456
package/test/dsl.test.js CHANGED
@@ -16,7 +16,7 @@ test('module exports is a fn with properties', function (t) {
16
16
 
17
17
  test('no arguments -> empty name, inParams, tasks, outTask', function (t) {
18
18
  var r = react();
19
- t.equal(r.ast.name, undefined);
19
+ t.equal(r.ast.name.slice(0, 'flow_'.length), 'flow_', 'generated flow name should start with flow_');
20
20
  t.deepEqual(r.ast.inParams, []);
21
21
  t.deepEqual(r.ast.tasks, []);
22
22
  t.deepEqual(r.ast.outTask, { a: [], type: 'finalcb' });
@@ -26,7 +26,7 @@ test('no arguments -> empty name, inParams, tasks, outTask', function (t) {
26
26
 
27
27
  test('empty first string -> empty name, inParams, tasks, outTask', function (t) {
28
28
  var r = react('');
29
- t.equal(r.ast.name, '');
29
+ t.equal(r.ast.name.slice(0, 'flow_'.length), 'flow_', 'generated flow name should start with flow_');
30
30
  t.deepEqual(r.ast.inParams, []);
31
31
  t.deepEqual(r.ast.tasks, []);
32
32
  t.deepEqual(r.ast.outTask, { a: [], type: 'finalcb' });
@@ -4,6 +4,9 @@ var test = require('tap').test;
4
4
  var BaseTask = require('../lib/base-task.js');
5
5
 
6
6
  var react = require('../'); // require('react');
7
+ // turn on tracking, obtain EventCollector
8
+ var EventCollector = require('../lib/event-collector'); // require('react/lib/event-collector');
9
+
7
10
 
8
11
  /**
9
12
  @example
@@ -36,6 +39,9 @@ test('module exports an function object with properties', function (t) {
36
39
  t.type(react, 'function', 'is a core constructor and default dsl function');
37
40
  t.type(react.options, 'object', 'has property for global react options');
38
41
  t.type(react.events, 'object', 'has global react event manager');
42
+ t.type(react.logEvents, 'function', 'has function to enable event logging');
43
+ t.type(react.trackTasks, 'function', 'has function to enable task and flow tracking');
44
+ t.type(react.resolvePromises, 'function', 'has fn to enable promise detection & resolution');
39
45
  t.end();
40
46
  });
41
47
 
@@ -79,6 +85,7 @@ test('use react() default DSL from module', function (t) {
79
85
  multiply, 'a, b, cb -> err, m',
80
86
  add, 'm, a, cb -> err, s'
81
87
  );
88
+
82
89
 
83
90
  fn(2, 3, function (err, m, s) {
84
91
  t.deepEqual(err, null, 'should not be any error');
@@ -96,10 +103,6 @@ test('use react.selectFirst() default DSL with events', function (t) {
96
103
  function noSuccessNull(a, b, cb) { cb(null, null); } // returns null result
97
104
  function add(a, b, cb) { cb(null, a + b); }
98
105
 
99
- var events = [];
100
- function accumEvents(task) {
101
- events.push(task);
102
- }
103
106
 
104
107
  var fn = react.selectFirst('mySelectFirst', 'a, b, cb -> err, c',
105
108
  noSuccess, 'a, b, cb -> err, c',
@@ -108,16 +111,18 @@ test('use react.selectFirst() default DSL with events', function (t) {
108
111
  noSuccess, 'a, b, cb -> err, c'
109
112
  );
110
113
 
111
- fn.events.on('task.complete', accumEvents);
114
+ var collector = new EventCollector();
115
+ collector.capture(fn, 'task.complete');
112
116
 
113
117
  fn(2, 3, function (err, c) {
114
118
  t.deepEqual(err, null, 'should not be any error');
115
119
  t.equal(c, 5);
120
+ var events = collector.list();
116
121
  t.equal(events.length, 3, 'should have seen two task compl events');
117
- t.equal(events[0].name, 'noSuccess', 'name matches');
118
- t.equal(events[1].name, 'noSuccessNull', 'name matches');
119
- t.equal(events[2].name, 'add', 'name matches');
120
- t.deepEqual(events[2].results, [5], 'results match');
122
+ t.equal(events[0].task.name, 'noSuccess', 'name matches');
123
+ t.equal(events[1].task.name, 'noSuccessNull', 'name matches');
124
+ t.equal(events[2].task.name, 'add', 'name matches');
125
+ t.deepEqual(events[2].task.results, [5], 'results match');
121
126
  t.end();
122
127
  });
123
128
  });
@@ -8,7 +8,8 @@
8
8
  var test = require('tap').test;
9
9
  var Deferred = require('promised-io/promise').Deferred;
10
10
 
11
- var react = require('../promise-resolve'); // require('react/promise-resolve');
11
+ var react = require('../'); // require('react');
12
+ react.resolvePromises(); // enable promise resolving
12
13
 
13
14
  function multiply(x, y, cb) { cb(null, x * y); }
14
15
  function add(x, y, cb) { cb(null, x + y); }
package/Jakefile.js DELETED
@@ -1,8 +0,0 @@
1
-
2
-
3
- require('./jake-tasks/jake-test.js');
4
-
5
- desc('Default - run watch');
6
- task('default', ['test:watch']);
7
-
8
-
package/dsl/chain.js DELETED
@@ -1,150 +0,0 @@
1
- 'use strict';
2
-
3
- var sprintf = require('sprintf').sprintf;
4
- var core = require('../lib/core.js');
5
- var tutil = require('../lib/task.js');
6
-
7
- // err for task type cb is implied and thus optional, but allow for clarity.
8
- var ERROR_NAMES_RE = /^err$/i; // first out param matching this is skipped as being the err object
9
-
10
- // callback is implied for task type cb is implied and thus optional, but allow for clarity.
11
- var CALLBACK_NAMES_RE = /^cb$|^callback$/i; // last in param matching this is skipped as being the cb
12
-
13
- var FlowBuilder;
14
-
15
- /**
16
- jQuery-like chain interface for defining flow
17
-
18
- @example
19
- // normal flow
20
- var react = require('react');
21
- var fn = react.chainDefine()
22
- .in('filename', 'uid', 'outDirname', 'cb')
23
- .out('err', 'html', 'user', 'bytesWritten')
24
- .async(loadUser) .in('uid') .out('err', 'user')
25
- .async(loadFile) .in('filename') .out('err', 'filedata')
26
- .sync(markdown) .in('filedata') .out('html')
27
- .async(prepareDirectory) .in('outDirname') .out('err', 'dircreated')
28
- .async(writeOutput) .in('html', 'user') .out('err', 'bytesWritten') .after('prepareDirectory')
29
- .async(loadEmailTemplate) .in() .out('err', 'emailmd')
30
- .sync(markdown) .in('emailmd') .out('emailHtml')
31
- .sync(customizeEmail) .in('user', 'emailHtml') .out('custEmailHtml')
32
- .async(deliverEmail) .in('custEmailHtml') .out('err', 'deliveredEmail') .after('writeOutput')
33
- .end();
34
-
35
- @example
36
- // selectFirst flow
37
- var fn = chainDefine()
38
- .selectFirst()
39
- .in('a', 'b', 'cb')
40
- .out('err', 'c')
41
- .async(falpha).in('a', 'b', 'cb').out('err', 'c')
42
- .sync(fbeta).in('a', 'b').out('c')
43
- .end();
44
- */
45
- function chainDefine() {
46
- return FlowBuilder.create();
47
- }
48
-
49
- function FlowBuilder() {
50
- this.main = {
51
- name: null,
52
- options: {}
53
- };
54
- this.tasks = [];
55
- this.focus = this.main;
56
- }
57
-
58
- FlowBuilder.create = function () { return new FlowBuilder(); };
59
-
60
- FlowBuilder.prototype.selectFirst = function () {
61
- this.main.outTaskType = 'finalcbFirst';
62
- return this;
63
- };
64
-
65
- FlowBuilder.prototype.name = function (name) {
66
- this.focus.name = name;
67
- return this;
68
- };
69
-
70
- FlowBuilder.prototype.in = function (param1, param2, paramN) {
71
- var args = Array.prototype.slice.call(arguments);
72
- if (args.length && this.focus.type !== 'ret') { // has args and cb or main
73
- if (args[args.length - 1].match(CALLBACK_NAMES_RE)) args.pop(); // pop off the cb name if specified
74
- }
75
- if (this.focus === this.main) this.focus.in = args;
76
- else this.focus.a = args; // for tasks
77
- return this;
78
- };
79
-
80
- FlowBuilder.prototype.out = function (param1, param2, paramN) {
81
- var args = Array.prototype.slice.call(arguments);
82
- if (args.length && this.focus.type !== 'ret') { // has args and cb or main
83
- if (args[0].match(ERROR_NAMES_RE)) args.shift(); // shift off err if specified
84
- }
85
- this.focus.out = args;
86
- return this;
87
- };
88
-
89
- FlowBuilder.prototype.options = function (options) {
90
- var self = this;
91
- if (this.focus === this.main) {
92
- Object.keys(options).forEach(function (k) { self.focus.options[k] = options[k]; });
93
- } else { // task so set options right on task
94
- Object.keys(options).forEach(function (k) { self.focus[k] = options[k]; });
95
- }
96
- return this;
97
- };
98
-
99
- FlowBuilder.prototype.async = function (funcOrStrMethod) {
100
- var task = { f: funcOrStrMethod, type: 'cb' };
101
- this.tasks.push(task);
102
- this.focus = task;
103
- return this;
104
- };
105
-
106
- FlowBuilder.prototype.sync = function (funcOrStrMethod) {
107
- var task = { f: funcOrStrMethod, type: 'ret' };
108
- this.tasks.push(task);
109
- this.focus = task;
110
- return this;
111
- };
112
-
113
- FlowBuilder.prototype.after = function (name1, name2, nameN) {
114
- this.focus.after = Array.prototype.slice.call(arguments);
115
- return this;
116
- };
117
-
118
- /**
119
- Complete the building of a flow and perform validation
120
- which throws error if flow is not valid.
121
- */
122
- FlowBuilder.prototype.end = function end() {
123
- var reactFn = core();
124
-
125
- if (this.main.outTaskType === 'finalcbFirst') {
126
- this.tasks = tutil.serializeTasks(this.tasks);
127
- }
128
-
129
- var ast = {
130
- inParams: this.main.in || [],
131
- tasks: this.tasks,
132
- outTask: {
133
- a: this.main.out || [],
134
- type: this.main.outTaskType || 'finalcb'
135
- }
136
- };
137
- if (this.main.name) ast.name = this.main.name;
138
- var self = this;
139
- Object.keys(this.main.options).forEach(function (k) { ast[k] = self.main.options[k]; });
140
- var errors = reactFn.setAndValidateAST(ast);
141
- if (errors.length) {
142
- var errorStr = errors.join('\n');
143
- throw new Error(errorStr);
144
- }
145
- return reactFn;
146
- };
147
-
148
- module.exports = chainDefine;
149
- module.exports.options = core.options;
150
- module.exports.events = core.events;
package/dsl/fstr.js DELETED
@@ -1,121 +0,0 @@
1
- 'use strict';
2
- /*jshint regexp: false */
3
-
4
- var sprintf = require('sprintf').sprintf;
5
- var core = require('../lib/core.js');
6
- var parse = require('../lib/parse.js');
7
- var tutil = require('../lib/task.js');
8
-
9
- var INPARAMS_NO_MATCH = 'input params in wrong format, wanted "foo, bar" - found: %s';
10
- var OUTPARAMS_NO_MATCH = 'output params in wrong format, wanted "foo, bar" - found: %s';
11
- var INOUT_PARAMS_NO_MATCH = 'task params in wrong format, wanted "foo, bar -> err, baz" - found: %s';
12
- var EXTRA_TASKARG = 'extra unmatched task arg: %s';
13
-
14
- var ERR_NAMES_RE = /^err$/i; // err, ERR, Err, ...
15
-
16
- function filterOutLeadingErrParam(args) { // if leading err param, filter it out
17
- if (args.length && args[0].match(ERR_NAMES_RE)) args.shift();
18
- return args;
19
- }
20
-
21
-
22
- var inOutDefParse = {
23
- regex: /^([^\-]*)(->)?\s*(returns?\s+)?(.*)$/i, //TODO this will not work if literal has dash
24
- fn: function (m) {
25
- var reMatchReturns = /returns?/i;
26
- return {
27
- type: (m[3] && m[3].match(reMatchReturns)) ? 'ret' : 'cb',
28
- inDef: parse.splitTrimFilterArgs(m[1]),
29
- outDef: filterOutLeadingErrParam(parse.splitTrimFilterArgs(m[4]))
30
- };
31
- }
32
- };
33
-
34
- function filterOutTrailingCbParam(args) { // if has trailing cb | callback param, filter it out
35
- var cbNamesRe = /^cb|callback$/i; //cb, Cb, CB, callback, Callback
36
- if (args.length && args[args.length - 1].match(cbNamesRe)) args.pop();
37
- return args;
38
- }
39
-
40
- function parseInParams(str) {
41
- var objDef = parse.parseStr(str, [inOutDefParse], INPARAMS_NO_MATCH);
42
- objDef.inDef = filterOutTrailingCbParam(objDef.inDef);
43
- return objDef;
44
- }
45
-
46
- function parseInOutParams(str) {
47
- var objDef = parse.parseStr(str, [inOutDefParse], INOUT_PARAMS_NO_MATCH);
48
- objDef.inDef = filterOutTrailingCbParam(objDef.inDef);
49
- return objDef;
50
- }
51
-
52
- function parseOutParams(str) {
53
- if (str.indexOf('->') === -1) str = '-> ' + str; // prefix so does out err process
54
- return parse.parseStr(str, [inOutDefParse], OUTPARAMS_NO_MATCH);
55
- }
56
-
57
- function parseTasks(arr) {
58
- var tasks = [];
59
- var fn, obj, result;
60
- while (arr.length >= 2) {
61
- obj = {};
62
- fn = arr.shift();
63
- result = parseInOutParams(arr.shift());
64
- if (typeof(arr[0]) === 'object') obj = arr.shift(); // has options, use as obj
65
- obj.f = fn;
66
- obj.a = result.inDef;
67
- var type = result.type;
68
- obj.out = result.outDef;
69
- obj.type = type;
70
- tasks.push(obj);
71
- }
72
- if (arr.length) throw new Error(sprintf(EXTRA_TASKARG, arr[0]));
73
- return tasks;
74
- }
75
-
76
-
77
- function fstrDefine(inParamStr, taskDefArr, outParamStr, options) {
78
- if (!inParamStr) inParamStr = '';
79
- if (!taskDefArr) taskDefArr = [];
80
- if (!outParamStr) outParamStr = '';
81
-
82
- var reactFn = core();
83
- var ast = {
84
- inParams: parseInParams(inParamStr).inDef,
85
- tasks: parseTasks(taskDefArr),
86
- outTask: { a: parseOutParams(outParamStr).outDef }
87
- };
88
- if (options) Object.keys(options).forEach(function (k) { ast[k] = options[k]; });
89
- var errors = reactFn.setAndValidateAST(ast);
90
- if (errors.length) {
91
- var errorStr = errors.join('\n');
92
- throw new Error(errorStr);
93
- }
94
- return reactFn;
95
- }
96
-
97
- function selectFirst(inParamStr, taskDefArr, outParamStr, options) {
98
- if (!inParamStr) inParamStr = '';
99
- if (!taskDefArr) taskDefArr = [];
100
- if (!outParamStr) outParamStr = '';
101
- var tasks = tutil.serializeTasks(parseTasks(taskDefArr));
102
-
103
- var reactFn = core();
104
- var ast = {
105
- inParams: parseInParams(inParamStr).inDef,
106
- tasks: tasks,
107
- outTask: { type: 'finalcbFirst', a: parseOutParams(outParamStr).outDef },
108
- };
109
- if (options) Object.keys(options).forEach(function (k) { ast[k] = options[k]; });
110
- var errors = reactFn.setAndValidateAST(ast);
111
- if (errors.length) {
112
- var errorStr = errors.join('\n');
113
- throw new Error(errorStr);
114
- }
115
- return reactFn;
116
- }
117
-
118
- module.exports = fstrDefine;
119
- module.exports.selectFirst = selectFirst;
120
- module.exports.options = core.options;
121
- module.exports.events = core.events;
package/dsl/pcode.js DELETED
@@ -1,175 +0,0 @@
1
- 'use strict';
2
- /*jshint regexp: false */
3
-
4
- var sprintf = require('sprintf').sprintf;
5
- var core = require('../lib/core.js');
6
- var parse = require('../lib/parse.js');
7
- var tutil = require('../lib/task.js');
8
-
9
- var INPARAMS_NO_MATCH = 'input params in wrong format, wanted "foo, bar, cb" - found: %s';
10
- var OUTPARAMS_NO_MATCH = 'output params in wrong format, wanted "cb(err, foo)" - found: %s';
11
- var INOUT_PARAMS_NO_MATCH = 'task params in wrong format, wanted "foo, bar := func(baz, cat, cb) " - found: %s';
12
- var EXTRA_TASKARG = 'extra unmatched task arg: %s';
13
-
14
- var ERR_NAMES_RE = /^err$/i; // err, ERR, Err, ...
15
-
16
- function filterOutLeadingErrParam(args) { // if leading err param, filter it out
17
- if (args.length && args[0].match(ERR_NAMES_RE)) args.shift();
18
- return args;
19
- }
20
-
21
-
22
- /**
23
- Examples:
24
- var react = require('react');
25
-
26
- var locals = { falpha: falpha, fbeta: fbeta, fcharlie: fcharlie };
27
- var fn = react.pcodeDefine('a, b', [
28
- 'd := fbeta(a, b)',
29
- 'e = fcharlie(a, b)',
30
- 'c := falpha(a, b) when fbeta:done and fcharlie:done',
31
- 'cb(err, c, d, e);'
32
- ], locals);
33
-
34
- function cb(err, d, e) { }
35
- fn(10, 20, cb);
36
- */
37
-
38
- function splitAndFilterArgs(commaSepArgs) { // split, trim, filter out empty and (cb or callback)
39
- var args = parse.splitTrimFilterArgs(commaSepArgs);
40
- var cbNamesRe = /^cb|callback$/i; //cb, Cb, CB, callback, Callback
41
- if (args.length && args[args.length - 1].match(cbNamesRe)) args.pop();
42
- return args;
43
- }
44
-
45
- function extractNamesFromWhenClause(str) { //given 'foo:done and bar.baz:done' return ['foo', 'bar.baz']
46
- var reSplit = /\s+and\s+/i;
47
- var reDone = /^([^:]+):done$/i;
48
- var names = str.split(reSplit).map(function (x) {
49
- x = x.trim();
50
- var m = x.match(reDone); // if foo:done, we'll use foo
51
- return (m) ? m[1] : x;
52
- });
53
- return names;
54
- }
55
-
56
- var inParse = {
57
- regex: /^(.*)$/,
58
- fn: function (m) {
59
- return {
60
- inDef: splitAndFilterArgs(m[1]),
61
- };
62
- }
63
- };
64
-
65
- var cbParse = {
66
- regex: /^\s*(var\s+)?([^:]*)\s*:=\s*([a-zA-Z0-9_\.\-]+)\s*\(\s*([^)]*)\)\s*(when\s+([^;]+))?;?\s*$/,
67
- fn: function (m) {
68
- var taskDef = {
69
- type: 'cb',
70
- f: m[3],
71
- inDef: (m[4]) ? splitAndFilterArgs(m[4]) : [],
72
- outDef: (m[2]) ? parse.splitTrimFilterArgs(m[2]) : []
73
- };
74
- if (m[6]) taskDef.after = extractNamesFromWhenClause(m[6]);
75
- return taskDef;
76
- }
77
- };
78
-
79
- var retParse = {
80
- regex: /^\s*(var\s+)?([^:]*)\s*=\s*([a-zA-Z0-9_\.\-]+)\s*\(\s*([^)]*)\)\s*(when\s+([^;]+))?;?\s*$/,
81
- fn: function (m) {
82
- var taskDef = {
83
- type: 'ret',
84
- f: m[3],
85
- inDef: (m[4]) ? splitAndFilterArgs(m[4]) : [],
86
- outDef: (m[2]) ? parse.splitTrimFilterArgs(m[2]) : []
87
- };
88
- if (m[6]) taskDef.after = extractNamesFromWhenClause(m[6]);
89
- return taskDef;
90
- }
91
- };
92
-
93
- var finalCbParse = {
94
- regex: /^\s*(cb|callback)\s*\(([^)]*)\)\s*;?$/i,
95
- fn: function (m) {
96
- return {
97
- outDef: (m[2]) ? filterOutLeadingErrParam(parse.splitTrimFilterArgs(m[2])) : []
98
- };
99
- }
100
- };
101
-
102
- function parseInParams(str) {
103
- return parse.parseStr(str, [inParse], INPARAMS_NO_MATCH);
104
- }
105
-
106
- function parseTask(str) {
107
- var result = parse.parseStr(str, [cbParse, retParse], INOUT_PARAMS_NO_MATCH);
108
- var type = result.type;
109
- var task = {
110
- f: result.f,
111
- a: result.inDef
112
- };
113
- task.out = result.outDef;
114
- task.type = type;
115
- if (result.after) task.after = result.after;
116
- return task;
117
- }
118
-
119
- function parseOutParams(str) {
120
- if (!str.trim().length) return { outDef: [] };
121
- return parse.parseStr(str, [finalCbParse], OUTPARAMS_NO_MATCH);
122
- }
123
-
124
- function parseTasks(arr) { return arr.map(parseTask); }
125
-
126
-
127
- function pcodeDefine(inParamStr, taskStrArr, locals, options) {
128
- if (!inParamStr) inParamStr = '';
129
- if (!taskStrArr) taskStrArr = [];
130
- var outTaskStr = (taskStrArr.length) ? taskStrArr.pop() : '';
131
- if (!locals) locals = { };
132
-
133
- var reactFn = core();
134
- var ast = {
135
- inParams: parseInParams(inParamStr).inDef,
136
- tasks: parseTasks(taskStrArr),
137
- outTask: { a: parseOutParams(outTaskStr).outDef },
138
- locals: locals
139
- };
140
- if (options) Object.keys(options).forEach(function (k) { ast[k] = options[k]; });
141
- var errors = reactFn.setAndValidateAST(ast);
142
- if (errors.length) {
143
- var errorStr = errors.join('\n');
144
- throw new Error(errorStr);
145
- }
146
- return reactFn;
147
- }
148
-
149
- function selectFirst(inParamStr, taskStrArr, locals, options) {
150
- if (!inParamStr) inParamStr = '';
151
- if (!taskStrArr) taskStrArr = [];
152
- var outTaskStr = (taskStrArr.length) ? taskStrArr.pop() : '';
153
- if (!locals) locals = { };
154
- var tasks = tutil.serializeTasks(parseTasks(taskStrArr));
155
-
156
- var reactFn = core();
157
- var ast = {
158
- inParams: parseInParams(inParamStr).inDef,
159
- tasks: tasks,
160
- outTask: { type: 'finalcbFirst', a: parseOutParams(outTaskStr).outDef },
161
- locals: locals
162
- };
163
- if (options) Object.keys(options).forEach(function (k) { ast[k] = options[k]; });
164
- var errors = reactFn.setAndValidateAST(ast);
165
- if (errors.length) {
166
- var errorStr = errors.join('\n');
167
- throw new Error(errorStr);
168
- }
169
- return reactFn;
170
- }
171
-
172
- module.exports = pcodeDefine;
173
- module.exports.selectFirst = selectFirst;
174
- module.exports.options = core.options;
175
- module.exports.events = core.events;