Reputation: 11
This example is taken from Node Cookbook, but using Express 3.1.0:
app.js
var express = require('express')
, routes = require('./routes')
, user = require('./routes/user')
, http = require('http')
, path = require('path');
var app = express();
app.configure(function(){
app.set('port', process.env.PORT || 3000);
app.set('views', __dirname + '/views');
app.set('view engine', 'jade');
app.use(express.favicon());
app.use(express.logger('dev'));
app.use(express.bodyParser());
app.use(express.methodOverride());
app.use(express.cookieParser('kooBkooCedoN'));
app.use(express.session());
app.use(require('./login'));
app.use(app.router);
app.use(express.static(path.join(__dirname, 'public')));
});
app.configure('development', function(){
app.use(express.errorHandler());
});
// --------------------------
// MODIFICATION IS MADE HERE!
// --------------------------
app.configure(function(){
app.use(function(req, res, next){
res.locals.user = req.session.user;
next();
});
});
app.get('/', routes.index);
app.post('/', routes.index);
app.del('/', routes.index);
app.get('/:page', routes.index);
http.createServer(app).listen(app.get('port'), function(){
console.log("Express server listening on port " + app.get('port'));
});
I have made some modification to the code as suggested in: Migrating Express.js 2 to 3, specifically app.dynamicHelpers() to app.locals.use?
login.js
var users = {'dave' : 'expressrocks'};
module.exports = function (req, res, next) { var method = req.method.toLowerCase(), //cache the method
user = req.body.user,
logout = (method === 'delete'),
login = (method === 'post' && user),
routes = req.app.routes.routes[method];
if (!routes) { next(); return; }
if (login || logout) {
routes.forEach(function (route) {
if (!(req.url.match(route.regexp))) {
console.log(req.url);
req.method = 'GET';
}
});
}
if (logout) {
delete req.session.user;
}
if (login) {
Object.keys(users).forEach(function (name) {
if (user.name === name && user.pwd === users[name]) {
req.session.user = {
name: user.name,
pwd: user.pwd
};
}
});
}
if (!req.session.user) { req.url = '/'; }
next();
};
login.jade
if user
form(method='post')
input(name="_method", type="hidden", value="DELETE")
p Hello #{user.name}!
a(href='javascript:', onClick='forms[0].submit()') [logout]
else
p Please log in
form(method='post')
fieldset
legend Login
p
label(for="user[name]") Username:
input(name="user[name]")
p
label(for="user[pwd]") Password:
input(type="password", name="user[pwd]")
input(type="submit")
Result Log from terminal
The error is came from the line: routes = req.app.routes.routes[method]; in login.js
TypeError: Cannot read property 'get' of undefined
at Object.module.exports [as handle] (/home/chiasyan/Desktop/login/login.js:8:35)
at next (/home/chiasyan/Desktop/login/node_modules/express/node_modules/connect/lib/proto.js:199:15)
at store.get.next (/home/chiasyan/Desktop/login/node_modules/express/node_modules/connect/lib/middleware/session.js:309:9)
at /home/chiasyan/Desktop/login/node_modules/express/node_modules/connect/lib/middleware/session.js:333:9
at /home/chiasyan/Desktop/login/node_modules/express/node_modules/connect/lib/middleware/session/memory.js:50:9
at process.startup.processNextTick.process._tickCallback (node.js:244:9)
Update:
I have changed the routes = req.app.routes.routes[method]; but there is still a minor error.
The user variable could not be found in login.jade (user is defined once in the login.js)
ReferenceError: /home/chiasyan/Desktop/login/views/login.jade:1
> 1| if user
2| form(method='post')
3| input(name="_method", type="hidden", value="DELETE")
4| p Hello #{user.name}!
user is not defined
Quote from Node Cookbook
In routes/index.js, we can now simply have the following code:
index.js
exports.index = function (req, res) {
res.render('index', {title: 'Express'});
};
Another quote from Node Cookbook
Since we're no longer using routes, we don't have the opportunity to pass req.session. user through res.render. However, we can use a dynamic helper instead. Dynamic helpers have access to the req and res objects, they're called just before a view is rendered. Any properties we pass to the dynamic helper object are pushed to the Jade view as local variables.
Since I can no longer use dynamic helpers in Express 3.*, so in app.js I have changed to:
app.configure(function(){
app.use(function(req, res, next){
res.locals.user = req.session.user;
next();
});
});
Upvotes: 0
Views: 12593
Reputation: 11
I have the code running now but I am not sure if it is the correct way. Please advise if anything rectification should be made.
I modified
app.configure(function(){
app.use(function(req, res, next){
res.locals.user = req.session.user;
next();
});
});
into
app.use(function(req, res, next){
res.locals.user = req.session.user;
next();
});
and placed it right after the following line
app.use(require('./login'));
Besides the answer given by Hector Correa in this post, here is the complete working code after the correction.
app.js
var express = require('express')
, routes = require('./routes')
, user = require('./routes/user')
, http = require('http')
, path = require('path');
var app = express();
app.configure(function(){
app.set('port', process.env.PORT || 3000);
app.set('views', __dirname + '/views');
app.set('view engine', 'jade');
app.use(express.favicon());
app.use(express.logger('dev'));
app.use(express.bodyParser());
app.use(express.methodOverride());
app.use(express.cookieParser('kooBkooCedoN'));
app.use(express.session());
app.use(require('./login'));
app.use(function(req, res, next){
res.locals.user = req.session.user;
next();
});
app.use(app.router);
app.use(express.static(path.join(__dirname, 'public')));
});
app.configure('development', function(){
app.use(express.errorHandler());
});
app.get('/', routes.index);
app.post('/', routes.index);
app.del('/', routes.index);
app.get('/:page', routes.index);
http.createServer(app).listen(app.get('port'), function(){
console.log("Express server listening on port " + app.get('port'));
});
login.js
var users = {'dave' : 'expressrocks'};
module.exports = function (req, res, next) { var method = req.method.toLowerCase(), //cache the method
user = req.body.user,
logout = (method === 'delete'),
login = (method === 'post' && user),
routes = req.app.routes[method];
if (!routes) { next(); return; }
if (login || logout) {
routes.forEach(function (route) {
if (!(req.url.match(route.regexp))) {
console.log(req.url);
req.method = 'GET';
}
});
}
if (logout) {
delete req.session.user;
}
if (login) {
Object.keys(users).forEach(function (name) {
if (user.name === name && user.pwd === users[name]) {
req.session.user = {
name: user.name,
pwd: user.pwd
};
}
});
}
if (!req.session.user) { req.url = '/'; }
next();
};
Upvotes: 1
Reputation: 26680
There is an error in the line that you mentioned:
routes = req.app.routes.routes[method];
You should use
routes = req.app.routes[method];
As an aside, what is the purpose of the check that you do with the result of this code?
Update: In order to pass the user information to your Jade view you need to pass this information in the render call. For example:
# Assuming you've got the user information in req.session.user
console.log(req.session.user);
res.render('login', {user: req.session.user});
Upvotes: 1