Structure Of Express/mongoose App
Solution 1:
There are more or less two axes to organize your code. Organize code based on the layer functionality of your modules (database, model, external interface) or by functionality/context they act on (users, orders). Most (MVC) applications use a functional organization schema which is easier to handle but does not reveal the purpose or intend of an application.
Beside organizing code functional layers should be as decoupled as possible.
The functional layers in your code are
- Models that abstract data and behavior in your application
- Routes that constitute an external interface of your application. Routes are not the application!
- Bootstrapping code (server.js) that is responsible to start and connect the parts of your application
The code base above seems to use a functional organization schema, which is fine. The use of a modules
directory does not really make sense to me and seems superfluous. So we have a schema somehow like this
|- server.js
|+ users
|- schema.js
|- routes.js
Now let's break some dependencies...
schema.js
The schema/model part of the code should not depend on the app that represents an interface of your application. This version of schema.js
exports a model and does not require an express app or a mongoose instance to be passed into some kind of factory function:
var mongoose = require('mongoose');
varSchema = mongoose.Schema;
varUserSchema = Schema({
username: { type: String, required: true },
password: { type: String }
});
// Use UserSchema.statics to define static functionsUserSchema.statics.userlist = function(cb) {
this.find().limit( 20 ).exec( function( err, users )
{
if( err ) returncb( err );
cb(null, users);
});
};
module.exports = mongoose.model( 'User', UserSchema, 'users' );
Obviously this misses the app.send
functionality from the original file. This will be done in the routes.js
file. You may notice that we do not export /api/v1/users
anymore but /
instead. This makes the express app more flexible and the route self contained.
See this post for a article explaining express routers in detail.
var express = require('express');
var router = express.Router();
var users = require('./schema');
// get all users
router.get( '/', function(req, res, next) {
users.userlist(function(err, users) {
if (err) { returnnext(err); }
res.send(users);
});
});
// get one user
router.get( '/:id', ...);
// add one new user
router.post( '/', ...);
module.exports = router;
This code omits implementations for getting one user and creating new users because these should work quite similar to userlist
. The userlist
route now has a single responsibility to mediate between HTTP and your model.
The last part is the wiring/bootstrapping code in server.js
:
// setupvar express = require("express");
var app = express();
var mongoose = require("mongoose");
mongoose.connect( 'mydb' ); // Single connection instance does not need to be passed around!// Mount the router under '/api/v1/users'
app.use('/api/v1/users', require('./users/routes'));
// listen
app.listen( 3000 );
As a result the model/schema code does not depend on the application interface code, the interface has a clear responsibility and the wiring code in server.js can decide which version of the routes to mounter under which URL path.
Solution 2:
Take a look at my express_code_structure example repo for my recommendations on a module filesystem organization.
However, your code samples above are major MVC violations. Do not pass the app
instance to your model. Do not have your model have any awareness of req
objects or HTTP services whatsoever. Model: data structure, integrity, persistence, business logic, and nothing else. Routes should be defined in entirely separate .js files from your models.
Solution 3:
In your application structure you have mixed the database logic with the express route handling and passing the express application variable to model. I would avoid the mixing these two together, also you could take a look at this structure https://gist.github.com/fwielstra/1025038.
Solution 4:
I recently ran through a tutorial which generated this sample app which has one of the best structures that I've seen. Look under /server/models/ to see how Mongoose is setup.
Solution 5:
I wrote a module orm-model that can help you structure models in your nodejs app.
Post a Comment for "Structure Of Express/mongoose App"