Add packages and stub out admin page functionality; re-order routes for more predictable express behavior.

This commit is contained in:
jkaplon 2016-07-11 10:18:41 -04:00
parent 01fdac5338
commit abdc4084f6
5 changed files with 148 additions and 32 deletions

1
.gitignore vendored
View File

@ -1,3 +1,4 @@
node_modules node_modules
db db
db-auth
logs logs

View File

@ -3,12 +3,17 @@
"version": "0.0.1", "version": "0.0.1",
"dependencies": { "dependencies": {
"body-parser": "^1.12.4", "body-parser": "^1.12.4",
"connect-ensure-login": "^0.1.1",
"cookie-parser": "^1.4.3",
"express": "^4.12.4", "express": "^4.12.4",
"express-hbs": "^0.8.4", "express-hbs": "^0.8.4",
"express-session": "^1.13.0",
"handlebars-form-helpers": "^0.1.3", "handlebars-form-helpers": "^0.1.3",
"moment-timezone": "^0.4.0", "moment-timezone": "^0.4.0",
"nodemailer": "^1.3.4", "nodemailer": "^1.3.4",
"pg": "^4.4.3", "pg": "^4.4.3",
"passport": "^0.3.2",
"passport-local": "^1.0.0",
"serve-favicon": "^2.3.0", "serve-favicon": "^2.3.0",
"sqlite3": "^3.0.8", "sqlite3": "^3.0.8",
"winston": "^2.1.1" "winston": "^2.1.1"

149
server.js
View File

@ -7,6 +7,42 @@ var bodyParser = require("body-parser");
var app = express(); var app = express();
var winston = require('winston'); var winston = require('winston');
winston.add(winston.transports.File, { filename: './logs/courtsopen.log', maxsize: 5000000 }); // 5MB winston.add(winston.transports.File, { filename: './logs/courtsopen.log', maxsize: 5000000 }); // 5MB
var fileSystem = require('fs');
var passport = require('passport');
var Strategy = require('passport-local').Strategy;
var db = require('./db-auth'); // Make this db-auth.../db already taken by PostgreSQL linnked container!!!
// Configure the local strategy for use by Passport.
// The local strategy require a `verify` function which receives the credentials
// (`username` and `password`) submitted by the user. The function must verify
// that the password is correct and then invoke `cb` with a user object, which
// will be set at `req.user` in route handlers after authentication.
passport.use(new Strategy(
function(username, password, cb) {
db.users.findByUsername(username, function(err, user) {
winston.info('trying to lookup user.');
if (err) { winston.info('db.users.findByUsername error.'); return cb(err); }
if (!user) { winston.info('bad user'); return cb(null, false); }
if (user.password != password) { winston.info('bad pw'); return cb(null, false); }
return cb(null, user);
});
}));
// Configure Passport authenticated session persistence.
//
// In order to restore authentication state across HTTP requests, Passport needs
// to serialize users into and deserialize users out of the session. The
// typical implementation of this is as simple as supplying the user ID when
// serializing, and querying the user record by ID from the database when
// deserializing.
passport.serializeUser(function(user, cb) {
cb(null, user.id);
});
passport.deserializeUser(function(id, cb) {
db.users.findById(id, function (err, user) {
if (err) { return cb(err); }
cb(null, user);
});
});
// Setup email // Setup email
var transporter = nodemailer.createTransport({ var transporter = nodemailer.createTransport({
@ -36,8 +72,25 @@ app.engine('hbs', hbs.express4({
app.set('view engine', 'hbs'); app.set('view engine', 'hbs');
app.set('views', __dirname + '/views'); app.set('views', __dirname + '/views');
var favicon = require('serve-favicon'); var favicon = require('serve-favicon');
app.use(favicon(__dirname + '/assets/favicon.ico')); // Try this before setting static dir. app.use(favicon(__dirname + '/assets/favicon.ico')); // Put this before setting static dir.
app.use(express.static('assets')); app.use(express.static('assets'));
app.use(require('cookie-parser')());
app.use(require('express-session')({ secret: 'keyboard cat', resave: false, saveUninitialized: false }));
// Initialize Passport and restore authentication state, if any, from the session.
app.use(passport.initialize());
app.use(passport.session());
// As with any middleware it is quintessential to call next() if the user is authenticated
var isAuthenticated = function (req, res, next) {
if (req.isAuthenticated()) {
return next();
res.redirect(req); // I think this is what I want, but might cause re-dir loop.
} else {
// If user NOT authenticated, redirect to login page, do not call next().
res.redirect('/login');
}
}
/******************************************************************************************************* /*******************************************************************************************************
With this express setup, ordering of routes matters!!! It's 1st-come-1st-served. With this express setup, ordering of routes matters!!! It's 1st-come-1st-served.
@ -48,6 +101,23 @@ But, for now, order routes like this:
- static routes, ordered more specific to less specific - static routes, ordered more specific to less specific
- dynamic routes, ordered more specific to less specific - dynamic routes, ordered more specific to less specific
********************************************************************************************************/ ********************************************************************************************************/
app.get('/login', function(req, res){
winston.info('GET /login');
res.render('login');
});
app.post('/login',
passport.authenticate('local', { failureRedirect: '/login' }),
function(req, res) {
winston.info('sucessful login');
res.redirect('/:loc/admin'); // will this work, should it be `req`?
});
app.get('/logout', function(req, res){
req.logout();
res.redirect('/'); // possible to go back to appropriate :loc?
});
app.get('/', function(req, res){ app.get('/', function(req, res){
winston.info("GET /"); winston.info("GET /");
res.render('home', {}, function(err, html) { res.render('home', {}, function(err, html) {
@ -105,6 +175,52 @@ app.post('/', function(req, res){
res.status(204).send('POST received'); res.status(204).send('POST received');
}); });
app.get('/:loc/admin', isAuthenticated, function(req, res) {
var loc = req.params.loc;
winston.info('GET ' + loc + '/admin');
if (loc !== 'tt') {
res.status(404).send('Not found');
} else {
// TODO: load admin template
}
});
app.post(':loc/admin', isAuthenticated, function(req, res){
winston.info('POST by admin at ' + loc);
// possible to forward req to POST to '/'?
});
app.get('/:loc/status', function(req, res) {
var loc = req.params.loc;
if (loc !== 'tt') {
res.status(404).send('Not found');
} else {
winston.info('GET ' + loc + '/status');
// Lookup most recent status from DB and return as JSON.
pg.connect(conString, function(err, client, done) {
if(err) {
return winston.error('error fetching client from pool', err);
}
var mostRecentStatusQry =
"select origjson " +
"from alerts " +
"where status in ('Open', 'Closed') " +
"and coreid = '300029000347343339373536' " +
"order by published_at desc " +
"limit 1"
client.query(mostRecentStatusQry, function(err, result) {
//call `done()` to release the client back to the pool
done();
if(err) {
return winston.error('error running query', err);
}
res.setHeader('Content-Type', 'application/json');
res.send(JSON.stringify(result.rows[0].origjson));
});
});
}
});
app.get('/:loc', function(req, res) { app.get('/:loc', function(req, res) {
var loc = req.params.loc; var loc = req.params.loc;
if (loc !== 'tt') { if (loc !== 'tt') {
@ -148,37 +264,6 @@ app.get('/:loc', function(req, res) {
} }
}); });
app.get('/:loc/status', function(req, res) {
var loc = req.params.loc;
if (loc !== 'tt') {
res.status(404).send('Not found');
} else {
winston.info('GET ' + loc + '/status');
// Lookup most recent status from DB and return as JSON.
pg.connect(conString, function(err, client, done) {
if(err) {
return winston.error('error fetching client from pool', err);
}
var mostRecentStatusQry =
"select origjson " +
"from alerts " +
"where status in ('Open', 'Closed') " +
"and coreid = '300029000347343339373536' " +
"order by published_at desc " +
"limit 1"
client.query(mostRecentStatusQry, function(err, result) {
//call `done()` to release the client back to the pool
done();
if(err) {
return winston.error('error running query', err);
}
res.setHeader('Content-Type', 'application/json');
res.send(JSON.stringify(result.rows[0].origjson));
});
});
}
});
setInterval(function() { setInterval(function() {
// Check every hour to see if GoodMorning or GoodEvening has gone missing. // Check every hour to see if GoodMorning or GoodEvening has gone missing.
pg.connect(conString, function(err, client, done) { pg.connect(conString, function(err, client, done) {

6
views/admin.hbs Normal file
View File

@ -0,0 +1,6 @@
{{!< default}}
<div style="background: url(banner-bg.jpg) no-repeat center center; background-size: cover; color: #fff" class="jumbotron">
<h1 class="text-center">TT Tennis Courts Are...</h1>
</div>

19
views/login.hbs Normal file
View File

@ -0,0 +1,19 @@
{{!< default}}
<div style="background: url(banner-bg.jpg) no-repeat center center; background-size: cover; color: #fff" class="jumbotron">
<h1 class="text-center">TT Tennis Courts Are...</h1>
</div>
<form action="/login" method="post">
<div>
<label>Username:</label>
<input type="text" name="username"/><br/>
</div>
<div>
<label>Password:</label>
<input type="password" name="password"/>
</div>
<div>
<input type="submit" value="Submit"/>
</div>
</form>