forked from chevcast/mongoose-simpledb
-
Notifications
You must be signed in to change notification settings - Fork 0
/
index.js
184 lines (157 loc) · 8.52 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
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
var mongoose = require('mongoose'),
fs = require('fs'),
path = require('path'),
autoIncrement = require('mongoose-auto-increment'),
extend = require('extend');
module.exports = exports = {
db: { modelsLoaded: false },
init: function () {
var _this = this,
settings = {
// The callback to invoke when models are loaded.
// Default function is empty and does nothing. It's only here to provide a method signature.
callback: function (err, db) {},
// The mongoose connection string to use.
connectionString: 'mongodb://localhost/test',
// The path to the directory where your dbmodels are stored.
modelsDir: path.join(__dirname, '..', '..', 'dbmodels'),
// Whether or not simpledb should auto-increment _id's of type Number.
autoIncrementNumberIds: true,
autoIncrementSettings: {
field: "_id", // Field name of id, _id by default
startAt: 0, // The number the count should start at.
incrementBy: 1 // The number by which to increment the count each time.
},
//default options of connect, can be extended, or changed
options: { socketTimeoutMS: 0, keepAlive: true, reconnectTries: 30 }
};
switch (arguments.length) {
case 1:
switch (typeof arguments[0]) {
// If the only argument is a function, set callback setting.
case 'function':
settings.callback = arguments[0];
break;
// If the only argument is a string, set the connectionString setting.
case 'string':
settings.connectionString = arguments[0];
break;
// If the only argument is an object, extend settings with the object.
case 'object':
extend(settings, arguments[0]);
}
break;
case 2:
// If the first arg is a string and the second is a function then set the connectionString and callback settings.
if (typeof arguments[0] === 'string' && typeof arguments[1] === 'function') {
settings.connectionString = arguments[0];
settings.callback = arguments[1];
}
// If the first arg is an object and the second is a function then extend settings with the object and set the callback setting.
else if (typeof arguments[0] === 'object' && typeof arguments[1] === 'function') {
extend(settings, arguments[0]);
settings.callback = arguments[1];
}
break;
}
// Create db object.
var db = _this.db;
function resetDb () {
_this.db.modelsLoaded = false;
for (var key in _this.db) {
if (key !== 'modelsLoaded')
delete _this.db[key];
}
}
resetDb();
// Create mongoose connection and attach it to db object.
db.connection = mongoose.createConnection(settings.connectionString, settings.options);
// If a mongoose error occurs then invoke the callback with the error.
db.connection.on('error', settings.callback);
// When the connection closes reset the db object.
db.connection.on('close', resetDb);
// Once the connection is open begin to load models from the database.
db.connection.once('open', function () {
// If mongoose-auto-increment plugin is installedInitialize mongoose-auto-increment plugin.
if (settings.autoIncrementNumberIds)
autoIncrement.initialize(db.connection);
// Find and load all Mongoose dbmodels from the dbmodels directory.
fs.readdir(settings.modelsDir, function (err, files) {
if (err) return settings.callback(err);
for (var i in files) {
var file = files[i];
if (path.extname(file) === '.js' || path.extname(file) === '.coffee') {
var modelName = path.basename(file.replace(path.extname(file), '')),
modelData = require(path.join(settings.modelsDir, file));
if (!modelData.hasOwnProperty('schema'))
return settings.callback(new Error('Model file ' + file + ' is invalid: No schema.'));
// Create schema based on template in model file.
var schema;
if (modelData.schemaOptions)
schema = new mongoose.Schema(modelData.schema, modelData.schemaOptions);
else
schema = new mongoose.Schema(modelData.schema);
// Add any instance methods defined in model file.
extend(schema.methods, modelData.methods);
// Add any static methods defined in model file.
extend(schema.statics, modelData.statics);
// Add any virtual properties defined in model file.
for (var key in modelData.virtuals) {
if (modelData.virtuals.hasOwnProperty(key)) {
var virtualData = modelData.virtuals[key];
if (virtualData.hasOwnProperty('get'))
schema.virtual(key).get(virtualData.get);
if (virtualData.hasOwnProperty('set'))
schema.virtual(key).set(virtualData.set);
}
}
//Add plugins to schema
for (var key in modelData.plugins) {
var record = modelData.plugins[key];
if (!record.hasOwnProperty('plugin'))
return settings.callback(new Error('Model file ' + file + ' is invalid: Wrong plugin definition.'));
schema.plugin(record.plugin, record.options);
}
//Add pre hooks
for (var key in modelData.pre) {
var record = modelData.pre[key];
schema.pre(key, record);
}
//Add post hooks
for (var key in modelData.post) {
var record = modelData.post[key];
schema.post(key, record);
}
// If autoIncrementIds:true then utilize mongoose-auto-increment plugin for this model.
if (settings.autoIncrementNumberIds) {
var field = settings.autoIncrementSettings.field;
if (schema.paths.hasOwnProperty(field) && schema.paths[field].instance === 'Number') {
settings.autoIncrementSettings.model = modelName;
schema.plugin(autoIncrement.plugin, settings.autoIncrementSettings);
}
}
// If model name contains an underscore then camelCase it.
var propName = modelName.charAt(0).toUpperCase() + modelName.slice(1);
if (propName.indexOf('_') !== -1) {
propName = propName.replace(/_(.)/g, function (match, letter) {
return letter.toUpperCase();
});
}
// Store model in db API.
db[propName] = db.connection.model(modelName, schema);
}
}
// Set modelsLoaded to true.
db.modelsLoaded = true;
// Invoke callback with resulting db object.
if (settings.callback) settings.callback(null, db);
});
});
// Return db object immediately in case the app would like a lazy-loaded reference.
return db;
},
// Expose mongoose types for easy access.
Types: mongoose.Schema.Types,
// Expose all of mongoose for easy access.
mongoose: mongoose
};