-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
128 lines (94 loc) · 3 KB
/
index.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
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
'use strict';
/**
* @name autorefresh
* @description mongoose plugin to ensure fresh refs
* @author lally elias <lallyelias87@mail.com>
* @since 0.1.0
* @version 0.1.0
* @example
*
* const PersonSchema = new Schema({
*
* parent: {
* type: ObjectId,
* ref: 'Person',
* autorefresh: true
* }
*
* });
*
*/
/* dependencies */
const _ = require('lodash');
const defaults = { options: { maxDepth: 1 } };
module.exports = exports = function autorefreshPlugin(schema /*, options*/ ) {
/* collect refs for autorefresh */
const refs = {};
/**
* @name autorefresh
* @description iterate though each schema path, check for ObjectId(ref)
* schema fields and apply fresh plugin to a schema type(s) with
* `autorefresh:true` options
*
* @param {String} schemaPath schema path
* @param {SchemaType} schemaType valid mongoose schema type
* @private
*/
function autorefreshPaths(schemaPath, schemaType, parent) {
//update pathe
const actualSchemaPath =
_.compact([(parent || {}).schemaPath, schemaPath]).join('.');
//handle schema
if (schemaType.schema) {
schemaType.schema.eachPath(function (_schemaPath, _schemaType) {
autorefreshPaths(
_schemaPath, _schemaType,
({ schemaPath: schemaPath, schemaType: schemaType })
);
});
}
//ensure schema type has options
const hasOptions = _.has(schemaType, 'options');
//ensure schema type is objectid `ref`
const hasRef = _.has(schemaType, 'options.ref');
//check for autorefresh schema options
const autorefreshable = _.has(schemaType, 'options.autorefresh');
//check if is allowed autorefresh schema type
const refreshable = hasOptions && hasRef && autorefreshable;
//handle `autorefresh:boolean|object` schema options
if (refreshable) {
//obtain schema type ref & autorefresh options
const { ref, autorefresh } = schemaType.options;
//prepare refreshable options
let options = _.merge({}, { path: actualSchemaPath, ref: ref });
if (_.isPlainObject(autorefresh)) {
options = _.merge({}, defaults, options, autorefresh);
}
//add model exists async validation
refs[actualSchemaPath] = _.merge({}, options);
}
}
//collect refreshable paths
schema.eachPath(autorefreshPaths);
/**
* @name autorefresh
* @function autorefresh
* @description refresh reference path
* @param {Function} done callback invoke on success or failure
* @instance
*/
schema.methods.autorefresh = function autorefresh(done) {
let $refs = [];
//prepare population options
_.forEach(_.merge({}, refs), function (ref) {
ref = _.merge({}, ref);
$refs = [].concat($refs).concat(ref);
});
//populate refs
this.populate($refs, done); //TODO improve query(reduce number of queries)
};
/* add pre validate hook to ensure fresh refs */
schema.pre('validate', function (next) {
this.autorefresh(next);
});
};