-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathcoordinator.js
105 lines (94 loc) · 3.17 KB
/
coordinator.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
/**
* The runner coordinator is responsible for registering routes and handling calling all of the middlewares that belong
* to those routes. A typical implementation is to 'use' then 'handle'. NOTE: When you create middleware if you
* want data to get bubbled up in the resolved promise you will need to add the data to the 'payload' property.
* This allows us to add things to the kontx but only bubble up specific data.
*
* Example:
* runner.use('example.method', [
* function(kontx, next){
* kontx.payload.test = 0;
* next();
* },
* function(kontx, next){
* kontx.payload.ab = 1;
* next();
* }
* ]);
*
* runner.handle('example.method', {}).then(function(payload){
* payload.ab.not.equal(1);
* }).done();
*/
module.exports = (function(){
'use strict';
var app = {},
_ = require('lodash'),
async = require('async'),
q = require('q');
app.stack = [];
/**
* Sets up the route and maps the path to an array of functions that get executed in order.
* @param path Route path
* @param fn function or array of functions
*/
app.use = function(path, fn){
this.stack[path] = { handle: fn };
return this;
};
/**
* The batch function will allow you to register multiple functions with the same permissions. This just
* helps with the amount of lines of code if you have many methods.
* @param module
* @param methods
* @param middleware
*/
app.batch = function(module, methods, middleware){
_.each(methods, function(method){
var path = module.uniqueId + '.' + method,
funcs = middleware.concat([module[method]]);
app.use(path, funcs);
});
};
/**
* Method that executes all of the functions assigned to a route path. It does this in order and if there are
* any errors along the way it will break the execution path. Function returns a promise.
* @param path Example 'example.method'
* @param args Arguments
* @param kontx Container object that keeps all of the data modified along the way.
*/
app.handle = function(path, args, kontx){
var route = this.stack[path],
deferred = q.defer(),
kontxArgs = {};
//Args is optional param. Shift if needed.
if(_.isUndefined(args.payload)){
if(_.isArray(args)){
kontxArgs = Array.prototype.slice.call(args);
}
else{
kontxArgs = args;
}
}
else {
kontx = args;
}
if(!_.isUndefined(route)){
kontx.args = kontxArgs;
kontx.route = path;
kontx.event = {
filter: {}
};
async.applyEachSeries(route.handle, kontx, function(err){
if(err){
deferred.reject(err);
}
else {
deferred.resolve(kontx.payload);
}
});
}
return deferred.promise;
};
return app;
})();