2017-04-14 11:42:15 -04:00
var express = require ( 'express' ) ;
2016-02-16 17:52:38 -05:00
var app = express ( ) ;
2017-04-14 11:42:15 -04:00
var bodyParser = require ( 'body-parser' ) ;
app . use ( bodyParser . text ( ) ) ; // Use defaults for now, size limit is 100kb.
app . use ( bodyParser . urlencoded ( { extended : true } ) ) ; // Also need url encoding to handle login form.
2016-02-16 17:52:38 -05:00
var winston = require ( 'winston' ) ;
winston . add ( winston . transports . File , { filename : './logs/notes.kaplon.us.log' , maxsize : 5000000 } ) ; // 5MB
var fileSystem = require ( 'fs' ) ;
2017-04-14 11:42:15 -04:00
var notePath = _ _dirname + '/note-data/allNotes.txt' ;
var favicon = require ( 'serve-favicon' ) ;
app . use ( favicon ( _ _dirname + '/assets/favicon.ico' ) ) ; // Put this before setting static dir.
2017-04-20 08:33:12 -04:00
app . use ( express . static ( _ _dirname + '/assets' ) ) ;
2017-04-14 11:42:15 -04:00
/ * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
Ordering of these configs is important , don ' t shuffle them around .
-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- * /
2016-05-30 11:16:01 -04:00
var passport = require ( 'passport' ) ;
var Strategy = require ( 'passport-local' ) . Strategy ;
var db = require ( './db' ) ;
2017-04-20 08:33:12 -04:00
var Session = require ( 'express-session' ) ;
var SessionStore = require ( 'session-file-store' ) ( Session ) ;
var session = Session ( { secret : 'here kitty kitty' , resave : false , saveUninitialized : false , store : new SessionStore ( { path : _ _dirname + '/tmp/sessions' } ) } ) ;
2017-04-14 11:42:15 -04:00
app . use ( session ) ;
2016-05-30 11:16:01 -04:00
2017-04-20 08:33:12 -04:00
//----------------------------
2016-05-30 11:16:01 -04:00
// 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 ) ;
} ) ;
} ) ) ;
2017-04-20 08:33:12 -04:00
2016-05-30 11:16:01 -04:00
// 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 ) ;
} ) ;
} ) ;
2016-02-16 17:52:38 -05:00
2016-05-30 11:16:01 -04:00
// Initialize Passport and restore authentication state, if any, from the session.
app . use ( passport . initialize ( ) ) ;
app . use ( passport . session ( ) ) ;
2016-06-17 10:02:51 -04:00
// 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 ( '/' ) ;
} else {
// If user NOT authenticated, redirect to login page, do not call next().
res . redirect ( '/login' ) ;
}
}
app . get ( '/' , isAuthenticated , function ( req , res ) {
2016-02-16 17:52:38 -05:00
winston . info ( "GET /" ) ;
2016-11-22 11:22:22 -05:00
// Respond with static file, contents will be loaded via websocket.
2017-04-20 08:33:12 -04:00
res . sendFile ( 'index.html' , { root : _ _dirname + '/views/' } ) ;
2016-02-16 17:52:38 -05:00
} ) ;
2016-05-30 11:16:01 -04:00
app . get ( '/login' , function ( req , res ) {
winston . info ( 'GET /login' ) ;
2017-04-20 08:33:12 -04:00
res . sendFile ( 'login.html' , { root : _ _dirname + '/views/' } ) ;
2016-05-30 11:16:01 -04:00
} ) ;
app . post ( '/login' ,
passport . authenticate ( 'local' , { failureRedirect : '/login' } ) ,
function ( req , res ) {
2017-04-14 11:42:15 -04:00
winston . info ( 'sucessful login for user, ' + req . user . username ) ;
2016-05-30 11:16:01 -04:00
res . redirect ( '/' ) ;
} ) ;
app . get ( '/logout' , function ( req , res ) {
req . logout ( ) ;
res . redirect ( '/' ) ;
} ) ;
2016-11-09 11:14:32 -05:00
var http = require ( 'http' ) . Server ( app ) ;
var io = require ( 'socket.io' ) ( http ) ;
2017-04-20 08:33:12 -04:00
//io.set('authorization', function (handshakeData, accept) {
//if (handshakeData.headers.cookie) {
//if (handshakeData.headers.cookie) {
//winston.info('cookie object test, ' + handshakeData.headers.cookie);
//winston.info('parse test 1, ' + cookieParser.signedCookies(handshakeData.headers.cookie, 'here kitty kitty'));
//winston.info('parse test 2, ' + cookieParser.signedCookie(handshakeData.headers.cookie['connect.sid'], 'here kitty kitty'));
//winston.info('parse test 3, ' + cookieParser.signedCookie(handshakeData.headers.cookie, 'here kitty kitty'));
//handshakeData.sessionID = cookieParser.signedCookie(handshakeData.headers.cookie['connect.sid'], 'here kitty kitty');
//winston.info('sess test 1, ' + session); // this prints alot!
//winston.info('sess test 2, ' + session.id); // undefined
//winston.info('Got session ID = ' + handshakeData.sessionID); // undefined
// save the session store to the data object
// (as required by the Session constructor)
//handshakeData.sessionStore = session.store;
//session.store.get(handshakeData.sessionID, function (err, session) {
//if (err || !session) {
//accept('Error', false);
//} else {
// create a session object, passing data as request and our
// just acquired session data
//handshakeData.session = new Session(handshakeData, session);
//accept(null, true);
//}
//});
//} else {
//return accept('No cookie transmitted.', false);
//}
//}
//});
2016-11-09 11:14:32 -05:00
io . on ( 'connection' , function ( socket ) {
//winston.info('a user connected');
2016-11-21 12:54:29 -05:00
fileSystem . readFile ( notePath , { encoding : 'utf-8' } , function ( err , data ) {
if ( ! err ) {
2016-11-23 14:05:15 -05:00
winston . info ( 'file read on connection' ) ;
socket . emit ( 'download allNotes' , data ) ; // Send content only to newly connected client.
2016-11-21 12:54:29 -05:00
} else { winston . error ( err ) ; }
} ) ;
2016-11-22 11:22:22 -05:00
socket . on ( 'upload allNotes' , function ( msg ) {
2016-11-21 12:54:29 -05:00
winston . info ( 'WS data received' ) ;
var now = Date . now ( ) ;
// Overwrite allNotes.txt with new contents from client.
fileSystem . readFile ( notePath , 'utf-8' , function ( err , data ) {
if ( err ) {
winston . error ( err ) ;
} else {
fileSystem . writeFile ( notePath , msg , 'utf-8' , function ( err ) {
if ( err ) { winston . error ( err ) ; }
var exec = require ( 'child_process' ) . exec ;
2016-12-08 12:30:19 -05:00
var cmd = `
cd note - data && \
git checkout - b $ { now } && \
git commit - am "Notes modified via websocket, ${now}" && \
git checkout master && \
git merge $ { now }
` ;
2017-04-20 08:33:12 -04:00
//winston.info(cmd);
2016-11-21 12:54:29 -05:00
exec ( cmd , function ( error , stdout , stderr ) {
if ( error ) { winston . error ( error ) ; }
2016-12-08 12:30:19 -05:00
if ( ! stdout . includes ( 'CONFLICT' ) ) { // Case-sensitivity seems brittle here. Better to check return code rather than stdout? Can child_process be called w/signature to include return code?
var cmd2 = ` cd note-data && git branch -d ${ now } `
exec ( cmd2 , function ( error , stdout , stderr ) {
if ( error ) { winston . error ( error ) ; }
2017-04-20 08:33:12 -04:00
winston . info ( 'temp branch deleted; new content written to allNotes.txt' ) ;
2016-12-08 12:30:19 -05:00
// Send new content to all other clients EXCEPT the orig sender.
socket . broadcast . emit ( 'download allNotes' , msg ) ;
winston . info ( 'sent content update to other clients' ) ;
} ) ;
} else {
// Auto-merge failed, keep the new branch.
// Send a conflict warning message to socket.io clients (prompt for IAP ;-).
socket . emit ( 'conflict' , ` Auto-merge failed, manually merge ${ now } branch to rescue recent edit. ` ) ;
socket . emit ( 'download allNotes' , data ) ; // Send pre-conflict content back down to client.
winston . info ( ` Auto-merge failed, manually merge ${ now } branch to rescue recent edit. ` ) ;
// MVP for now, can deal w/merge conflicts with git on the server if the cotent needs to be saved.
}
2016-11-21 12:54:29 -05:00
winston . info ( stdout ) ;
} ) ;
} ) ;
2016-11-23 14:51:18 -05:00
}
2016-11-23 14:05:15 -05:00
} ) ;
2016-11-21 12:54:29 -05:00
} ) ;
2016-11-09 11:14:32 -05:00
socket . on ( 'cm-save' , function ( msg ) {
winston . info ( msg ) ;
} ) ;
} ) ;
http . listen ( 3000 , function ( ) {
2016-02-16 17:52:38 -05:00
winston . info ( "Started on PORT 3000" ) ;
} ) ;