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 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 ) ;
2021-03-13 11:38:08 -05:00
var session = Session ( {
2024-09-22 18:09:05 -04:00
secret : process . env . SESSION _SECRET ,
2021-03-13 11:38:08 -05:00
resave : false ,
saveUninitialized : false ,
cookie : { sameSite : true , secure : true } ,
store : new SessionStore ( { path : _ _dirname + '/tmp/sessions' } )
} ) ;
2017-04-14 11:42:15 -04:00
app . use ( session ) ;
2021-03-13 11:38:08 -05:00
app . set ( 'trust proxy' , true ) ;
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 ) {
2024-09-23 12:57:02 -04:00
db . utils . findByUsername ( username , function ( err , user ) {
2016-05-30 11:16:01 -04:00
winston . info ( 'trying to lookup user.' ) ;
2024-09-23 12:57:02 -04:00
if ( err ) { winston . info ( 'db.utils.findByUsername error.' ) ; return cb ( err ) ; }
2016-05-30 11:16:01 -04:00
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 ) {
2024-09-23 12:57:02 -04:00
db . utils . findById ( id , function ( err , user ) {
2016-05-30 11:16:01 -04:00
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 ) ;
2021-03-13 11:38:08 -05:00
var io = require ( 'socket.io' ) ( http , { cookie : false } ) ;
2017-04-20 15:55:03 -04:00
var iosess = require ( 'socket.io-express-session' ) ;
io . use ( iosess ( session ) ) ;
2017-04-20 08:33:12 -04:00
2016-11-09 11:14:32 -05:00
io . on ( 'connection' , function ( socket ) {
//winston.info('a user connected');
2017-04-21 10:55:30 -04:00
winston . info ( 'session id, ' + socket . handshake . session . id ) ;
2017-04-22 09:26:21 -04:00
// next line no worky...but i can get by with only user-#.
//winston.info('user name, '+ socket.handshake.session.passport.user.username);
try {
var userNum = socket . handshake . session . passport . user ;
} catch ( ex ) {
2017-04-21 10:55:30 -04:00
// Send redirect to client, socket connection will close when client goes back to /login.
// But disconnect just in case client is misbehaving.
winston . info ( 'Session DNE, redirect and disconnect.' ) ;
socket . emit ( 'redirect' , '/login' ) ;
socket . disconnect ( ) ;
}
2017-04-21 16:54:43 -04:00
winston . info ( 'user id, ' + userNum ) ;
socket . join ( userNum ) ;
var notePath , noteDir
if ( userNum === 1 ) {
notePath = _ _dirname + '/note-data/allNotes.txt' ;
noteDir = _ _dirname + '/note-data/' ;
} else {
notePath = _ _dirname + '/note-data/' + userNum + '/allNotes.txt' ;
noteDir = _ _dirname + '/note-data/' + userNum + '/' ;
}
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 ) {
2018-02-12 18:46:37 -05:00
winston . info ( 'WS data received from user # ' + userNum ) ;
2016-11-21 12:54:29 -05:00
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 = `
2017-04-21 16:54:43 -04:00
cd $ { noteDir } && \
2016-12-08 12:30:19 -05:00
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);
2020-04-08 17:09:03 -04:00
exec ( cmd , { uid : 1000 } , function ( error , stdout , stderr ) {
2016-11-21 12:54:29 -05:00
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?
2017-04-21 16:54:43 -04:00
var cmd2 = ` cd ${ noteDir } && git branch -d ${ now } `
2016-12-08 12:30:19 -05:00
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.
2017-04-21 16:54:43 -04:00
socket . in ( userNum ) . emit ( 'download allNotes' , msg ) ;
2016-12-08 12:30:19 -05:00
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 ;-).
2017-04-21 16:54:43 -04:00
socket . in ( userNum ) . emit ( 'conflict' , ` Auto-merge failed, manually merge ${ now } branch to rescue recent edit. ` ) ;
socket . in ( userNum ) . emit ( 'download allNotes' , data ) ; // Send pre-conflict content back down to client.
2016-12-08 12:30:19 -05:00
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 ) ;
} ) ;
} ) ;
2024-09-21 23:24:55 -04:00
setInterval (
function ( ) {
//winston.info('setInterval fired.');
var exec = require ( 'child_process' ) . exec ;
2024-09-23 12:57:02 -04:00
const userIdArray = db . utils . userIdArray ( ) ;
2024-09-21 23:24:55 -04:00
for ( const userId of userIdArray ) {
var noteDir ;
if ( userId === 1 ) {
noteDir = _ _dirname + '/note-data/' ;
} else {
noteDir = _ _dirname + '/note-data/' + userId + '/' ;
}
//winston.info('noteDir = ' + noteDir);
exec (
` cd ${ noteDir } && git push ` ,
{ uid : 1000 } ,
function ( err ) {
if ( err ) { winston . error ( err ) ; }
}
) ;
}
} , 120000 // 2 min
) ;
2016-11-09 11:14:32 -05:00
http . listen ( 3000 , function ( ) {
2016-02-16 17:52:38 -05:00
winston . info ( "Started on PORT 3000" ) ;
} ) ;