Add packages and stub out admin page functionality; re-order routes for more predictable express behavior.
This commit is contained in:
parent
01fdac5338
commit
abdc4084f6
1
.gitignore
vendored
1
.gitignore
vendored
@ -1,3 +1,4 @@
|
|||||||
node_modules
|
node_modules
|
||||||
db
|
db
|
||||||
|
db-auth
|
||||||
logs
|
logs
|
||||||
|
@ -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
149
server.js
@ -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
6
views/admin.hbs
Normal 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
19
views/login.hbs
Normal 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>
|
Loading…
Reference in New Issue
Block a user