var _ = require('underscore'); var colors = require('colors'); var fs = require('fs'); var inquirer = require('inquirer'); var M = require('mstring'); colors.setTheme({ silly: 'rainbow', input: 'grey', verbose: 'cyan', prompt: 'grey', info: 'green', data: 'grey', help: 'cyan', warn: 'yellow', debug: 'blue', error: 'red' }); inquirer.prompt({ type: 'list', name: 'category', message: 'Hackathon Starter:', choices: ['☂ Authentication', '☱ Exit'] }, function(answer) { if (answer.category.match('Authentication')) { inquirer.prompt({ type: 'checkbox', message: 'Select Authentication Providers', name: 'auth', choices: [ new inquirer.Separator(M(function() { /*** ╔═════════════════════════════════════════════════════════════════╗ ║ THIS TOOL IS STILL IN EXPERIMENTAL STAGE! USE AT YOUR OWN RISK. ║ ║ ALWAYS USE VERSION CONTROL SYSTEM SO YOU COULD REVERT CHANGES. ║ ╚═════════════════════════════════════════════════════════════════╝ ***/ })), { name: 'Facebook', checked: true }, { name: 'GitHub', checked: true }, { name: 'Google', checked: true }, { name: 'Twitter', checked: true }, { name: 'Local', checked: true }, { name: 'LinkedIn', checked: true }, new inquirer.Separator('Press ctrl+ to quit'), ], validate: function(answer) { if (answer.length < 1) return 'You must choose at least one authentication provider.'; return true; } }, function(answer) { var index; var passportConfigFile = 'config/passport.js'; var userModelFile = 'models/User.js'; var profileTemplateFile = 'views/account/profile.jade'; var loginTemplateFile = 'views/account/login.jade'; var passportConfig = fs.readFileSync(passportConfigFile).toString().split('\n'); var loginTemplate = fs.readFileSync(loginTemplateFile).toString().split('\n'); var profileTemplate = fs.readFileSync(profileTemplateFile).toString().split('\n'); var userModel = fs.readFileSync(userModelFile).toString().split('\n'); if (_.contains(answer.auth, 'Facebook')) { var facebookStrategyRequire = "var FacebookStrategy = require('passport-facebook').Strategy;"; var facebookStrategy = M(function() { /*** // Sign in with Facebook. passport.use(new FacebookStrategy(secrets.facebook, function(req, accessToken, refreshToken, profile, done) { if (req.user) { User.findOne({ $or: [{ facebook: profile.id }, { email: profile.email }] }, function(err, existingUser) { if (existingUser) { req.flash('errors', { msg: 'There is already a Facebook account that belongs to you. Sign in with that account or delete it, then link it with your current account.' }); done(err); } else { User.findById(req.user.id, function(err, user) { user.facebook = profile.id; user.tokens.push({ kind: 'facebook', accessToken: accessToken }); user.profile.name = user.profile.name || profile.displayName; user.profile.gender = user.profile.gender || profile._json.gender; user.profile.picture = user.profile.picture || 'https://graph.facebook.com/' + profile.id + '/picture?type=large'; user.save(function(err) { req.flash('info', { msg: 'Facebook account has been linked.' }); done(err, user); }); }); } }); } else { User.findOne({ facebook: profile.id }, function(err, existingUser) { if (existingUser) return done(null, existingUser); User.findOne({ email: profile._json.email }, function(err, existingEmailUser) { if (existingEmailUser) { req.flash('errors', { msg: 'There is already an account using this email address. Sign in to that account and link it with Facebook manually from Account Settings.' }); done(err); } else { var user = new User(); user.email = profile._json.email; user.facebook = profile.id; user.tokens.push({ kind: 'facebook', accessToken: accessToken }); user.profile.name = profile.displayName; user.profile.gender = profile._json.gender; user.profile.picture = 'https://graph.facebook.com/' + profile.id + '/picture?type=large'; user.profile.location = (profile._json.location) ? profile._json.location.name : ''; user.save(function(err) { done(err, user); }); } }); }); } })); ***/ }); var facebookButton = M(function() { /*** a.btn.btn-block.btn-facebook.btn-social(href='/auth/facebook') i.fa.fa-facebook | Sign in with Facebook ***/ }); var facebookLinkUnlink = M(function() { /*** if user.facebook p: a.text-danger(href='/account/unlink/facebook') Unlink your Facebook account else p: a(href='/auth/facebook') Link your Facebook account ***/ }); var facebookModel = ' facebook: String,'; if (passportConfig.indexOf(facebookStrategyRequire) < 0) { // config/passport.js (+) index = passportConfig.indexOf("var passport = require('passport');"); passportConfig.splice(index + 1, 0, facebookStrategyRequire); index = passportConfig.indexOf('passport.deserializeUser(function(id, done) {'); passportConfig.splice(index + 6, 0, facebookStrategy); fs.writeFileSync(passportConfigFile, passportConfig.join('\n')); // views/account/login.jade (+) loginTemplate.push(facebookButton); fs.writeFileSync(loginTemplateFile, loginTemplate.join('\n')); // views/account/profile.jade (+) index = profileTemplate.indexOf(' h3 Linked Accounts'); profileTemplate.splice(index + 1, 0, facebookLinkUnlink); fs.writeFileSync(profileTemplateFile, profileTemplate.join('\n')); // models/User.js (+) index = userModel.indexOf(' tokens: Array,'); userModel.splice(index - 1, 0, facebookModel); fs.writeFileSync(userModelFile, userModel.join('\n')); console.log('✓ Facebook authentication has been added.'.info); } else { console.log('✓ Facebook authentication is already active.'.warn); } } else { // config/passport.js (-) index = passportConfig.indexOf(facebookStrategyRequire); passportConfig.splice(index, 1); index = passportConfig.indexOf('// Sign in with Facebook.'); passportConfig.splice(index, 47); fs.writeFileSync(passportConfigFile, passportConfig.join('\n')); // views/account/login.jade (-) index = loginTemplate.indexOf(" a.btn.btn-block.btn-facebook.btn-social(href='/auth/facebook')"); loginTemplate.splice(index, 4); fs.writeFileSync(loginTemplateFile, loginTemplate.join('\n')); // views/account/profile.jade (-) index = profileTemplate.indexOf(" if user.facebook"); profileTemplate.splice(index - 1, 5); fs.writeFileSync(profileTemplateFile, profileTemplate.join('\n')); // models/User.js (-) index = userModel.indexOf(' facebook: String,'); userModel.splice(index, 1); fs.writeFileSync(userModelFile, userModel.join('\n')); console.log('✗ Facebook authentication has been removed.'.error); } if (_.contains(answer.auth, 'GitHub')) { var githubStrategyRequire = "var GitHubStrategy = require('passport-github').Strategy;"; var githubStrategy = M(function() { /*** // Sign in with GitHub. passport.use(new GitHubStrategy(secrets.github, function(req, accessToken, refreshToken, profile, done) { if (req.user) { User.findOne({ $or: [{ github: profile.id }, { email: profile.email }] }, function(err, existingUser) { if (existingUser) { req.flash('errors', { msg: 'There is already a GitHub account that belongs to you. Sign in with that account or delete it, then link it with your current account.' }); done(err); } else { User.findById(req.user.id, function(err, user) { user.github = profile.id; user.tokens.push({ kind: 'github', accessToken: accessToken }); user.profile.name = user.profile.name || profile.displayName; user.profile.picture = user.profile.picture || profile._json.avatar_url; user.profile.location = user.profile.location || profile._json.location; user.profile.website = user.profile.website || profile._json.blog; user.save(function(err) { req.flash('info', { msg: 'GitHub account has been linked.' }); done(err, user); }); }); } }); } else { User.findOne({ github: profile.id }, function(err, existingUser) { if (existingUser) return done(null, existingUser); User.findOne({ email: profile._json.email }, function(err, existingEmailUser) { if (existingEmailUser) { req.flash('errors', { msg: 'There is already an account using this email address. Sign in to that account and link it with GitHub manually from Account Settings.' }); done(err); } else { var user = new User(); user.email = profile._json.email; user.github = profile.id; user.tokens.push({ kind: 'github', accessToken: accessToken }); user.profile.name = profile.displayName; user.profile.picture = profile._json.avatar_url; user.profile.location = profile._json.location; user.profile.website = profile._json.blog; user.save(function(err) { done(err, user); }); } }); }); } })); ***/ }); var githubButton = M(function() { /*** a.btn.btn-block.btn-github.btn-social(href='/auth/github') i.fa.fa-github | Sign in with GitHub ***/ }); var githubLinkUnlink = M(function() { /*** if user.github p: a.text-danger(href='/account/unlink/github') Unlink your GitHub account else p: a(href='/auth/github') Link your GitHub account ***/ }); var githubModel = ' github: String,'; if (passportConfig.indexOf(githubStrategyRequire) < 0) { // config/passport.js (+) index = passportConfig.indexOf("var passport = require('passport');"); passportConfig.splice(index + 1, 0, githubStrategyRequire); index = passportConfig.indexOf('passport.deserializeUser(function(id, done) {'); passportConfig.splice(index + 6, 0, githubStrategy); fs.writeFileSync(passportConfigFile, passportConfig.join('\n')); // views/account/login.jade (+) loginTemplate.push(githubButton); fs.writeFileSync(loginTemplateFile, loginTemplate.join('\n')); // views/account/profile.jade (+) index = profileTemplate.indexOf(' h3 Linked Accounts'); profileTemplate.splice(index + 1, 0, githubLinkUnlink); fs.writeFileSync(profileTemplateFile, profileTemplate.join('\n')); // models/User.js (+) index = userModel.indexOf(' tokens: Array,'); userModel.splice(index - 1, 0, githubModel); fs.writeFileSync(userModelFile, userModel.join('\n')); console.log('✓ GitHub authentication has been added.'.info); } else { console.log('✓ GitHub authentication is already active.'.warn); } } else { // config/passport.js (-) index = passportConfig.indexOf(githubStrategyRequire); passportConfig.splice(index, 1); index = passportConfig.indexOf('// Sign in with GitHub.'); passportConfig.splice(index, 48); fs.writeFileSync(passportConfigFile, passportConfig.join('\n')); // views/account/login.jade (-) index = loginTemplate.indexOf(" a.btn.btn-block.btn-github.btn-social(href='/auth/github')"); loginTemplate.splice(index, 4); fs.writeFileSync(loginTemplateFile, loginTemplate.join('\n')); // views/account/profile.jade (-) index = profileTemplate.indexOf(' if user.github'); profileTemplate.splice(index - 1, 5); fs.writeFileSync(profileTemplateFile, profileTemplate.join('\n')); // models/User.js (-) index = userModel.indexOf(' github: String,'); userModel.splice(index, 1); fs.writeFileSync(userModelFile, userModel.join('\n')); console.log('✗ GitHub authentication has been removed.'.error); } if (_.contains(answer.auth, 'Google')) { var googleStrategyRequire = "var GoogleStrategy = require('passport-google-oauth').OAuth2Strategy;"; var googleStrategy = M(function() { /*** // Sign in with Google. passport.use(new GoogleStrategy(secrets.google, function(req, accessToken, refreshToken, profile, done) { if (req.user) { User.findOne({ $or: [{ google: profile.id }, { email: profile.email }] }, function(err, existingUser) { if (existingUser) { req.flash('errors', { msg: 'There is already a Google account that belongs to you. Sign in with that account or delete it, then link it with your current account.' }); done(err); } else { User.findById(req.user.id, function(err, user) { user.google = profile.id; user.tokens.push({ kind: 'google', accessToken: accessToken }); user.profile.name = user.profile.name || profile.displayName; user.profile.gender = user.profile.gender || profile._json.gender; user.profile.picture = user.profile.picture || profile._json.picture; user.save(function(err) { req.flash('info', { msg: 'Google account has been linked.' }); done(err, user); }); }); } }); } else { User.findOne({ google: profile.id }, function(err, existingUser) { if (existingUser) return done(null, existingUser); User.findOne({ email: profile._json.email }, function(err, existingEmailUser) { if (existingEmailUser) { req.flash('errors', { msg: 'There is already an account using this email address. Sign in to that account and link it with Google manually from Account Settings.' }); done(err); } else { var user = new User(); user.email = profile._json.email; user.google = profile.id; user.tokens.push({ kind: 'google', accessToken: accessToken }); user.profile.name = profile.displayName; user.profile.gender = profile._json.gender; user.profile.picture = profile._json.picture; user.save(function(err) { done(err, user); }); } }); }); } })); ***/ }); var googleButton = M(function() { /*** a.btn.btn-block.btn-google-plus.btn-social(href='/auth/google') i.fa.fa-google-plus | Sign in with Google ***/ }); var googleLinkUnlink = M(function() { /*** if user.google p: a.text-danger(href='/account/unlink/google') Unlink your Google account else p: a(href='/auth/google') Link your Google account ***/ }); var googleModel = ' google: String,'; if (passportConfig.indexOf(googleStrategyRequire) < 0) { // config/passport.js (+) index = passportConfig.indexOf("var passport = require('passport');"); passportConfig.splice(index + 1, 0, googleStrategyRequire); index = passportConfig.indexOf('passport.deserializeUser(function(id, done) {'); passportConfig.splice(index + 6, 0, googleStrategy); fs.writeFileSync(passportConfigFile, passportConfig.join('\n')); // views/account/login.jade (+) loginTemplate.push(googleButton); fs.writeFileSync(loginTemplateFile, loginTemplate.join('\n')); // views/account/profile.jade (+) index = profileTemplate.indexOf(' h3 Linked Accounts'); profileTemplate.splice(index + 1, 0, googleLinkUnlink); fs.writeFileSync(profileTemplateFile, profileTemplate.join('\n')); // models/User.js (+) index = userModel.indexOf(' tokens: Array,'); userModel.splice(index - 1, 0, googleModel); fs.writeFileSync(userModelFile, userModel.join('\n')); console.log('✓ Google authentication has been added.'.info); } else { console.log('✓ Google authentication is already active.'.warn); } } else { // config/passport.js (-) index = passportConfig.indexOf(googleStrategyRequire); passportConfig.splice(index, 1); index = passportConfig.indexOf('// Sign in with Google.'); passportConfig.splice(index, 46); fs.writeFileSync(passportConfigFile, passportConfig.join('\n')); // views/account/login.jade (-) index = loginTemplate.indexOf(" a.btn.btn-block.btn-google-plus.btn-social(href='/auth/google')"); loginTemplate.splice(index, 4); fs.writeFileSync(loginTemplateFile, loginTemplate.join('\n')); // views/account/profile.jade (-) index = profileTemplate.indexOf(' if user.google'); profileTemplate.splice(index - 1, 5); fs.writeFileSync(profileTemplateFile, profileTemplate.join('\n')); // models/User.js (-) index = userModel.indexOf(' google: String,'); userModel.splice(index, 1); fs.writeFileSync(userModelFile, userModel.join('\n')); console.log('✗ Google authentication has been removed.'.error); } if (_.contains(answer.auth, 'Twitter')) { var twitterStrategyRequire = "var TwitterStrategy = require('passport-twitter').Strategy;"; var twitterStrategy = M(function() { /*** // Sign in with Twitter. passport.use(new TwitterStrategy(secrets.twitter, function(req, accessToken, tokenSecret, profile, done) { if (req.user) { User.findOne({ twitter: profile.id }, function(err, existingUser) { if (existingUser) { req.flash('errors', { msg: 'There is already a Twitter account that belongs to you. Sign in with that account or delete it, then link it with your current account.' }); done(err); } else { User.findById(req.user.id, function(err, user) { user.twitter = profile.id; user.tokens.push({ kind: 'twitter', accessToken: accessToken, tokenSecret: tokenSecret }); user.profile.name = user.profile.name || profile.displayName; user.profile.location = user.profile.location || profile._json.location; user.profile.picture = user.profile.picture || profile._json.profile_image_url; user.save(function(err) { req.flash('info', { msg: 'Twitter account has been linked.' }); done(err, user); }); }); } }); } else { User.findOne({ twitter: profile.id }, function(err, existingUser) { if (existingUser) return done(null, existingUser); var user = new User(); // Twitter will not provide an email address. Period. // But a person’s twitter username is guaranteed to be unique // so we can "fake" a twitter email address as follows: user.email = profile.username + "@twitter.com"; user.twitter = profile.id; user.tokens.push({ kind: 'twitter', accessToken: accessToken, tokenSecret: tokenSecret }); user.profile.name = profile.displayName; user.profile.location = profile._json.location; user.profile.picture = profile._json.profile_image_url; user.save(function(err) { done(err, user); }); }); } })); ***/ }); var twitterButton = M(function() { /*** a.btn.btn-block.btn-google-plus.btn-social(href='/auth/google') i.fa.fa-google-plus | Sign in with Google ***/ }); var twitterLinkUnlink = M(function() { /*** if user.google p: a.text-danger(href='/account/unlink/google') Unlink your Google account else p: a(href='/auth/google') Link your Google account ***/ }); var twitterModel = ' twitter: String,'; if (passportConfig.indexOf(twitterStrategyRequire) < 0) { // config/passport.js (+) index = passportConfig.indexOf("var passport = require('passport');"); passportConfig.splice(index + 1, 0, twitterStrategyRequire); index = passportConfig.indexOf('passport.deserializeUser(function(id, done) {'); passportConfig.splice(index + 6, 0, twitterStrategy); fs.writeFileSync(passportConfigFile, passportConfig.join('\n')); // views/account/login.jade (+) loginTemplate.push(twitterButton); fs.writeFileSync(loginTemplateFile, loginTemplate.join('\n')); // views/account/profile.jade (+) index = profileTemplate.indexOf(' h3 Linked Accounts'); profileTemplate.splice(index + 1, 0, twitterLinkUnlink); fs.writeFileSync(profileTemplateFile, profileTemplate.join('\n')); // models/User.js (+) index = userModel.indexOf(' tokens: Array,'); userModel.splice(index - 1, 0, twitterModel); fs.writeFileSync(userModelFile, userModel.join('\n')); console.log('✓ Twitter authentication has been added.'.info); } else { console.log('✓ Twitter authentication is already active.'.warn); } } else { // config/passport.js (-) index = passportConfig.indexOf(twitterStrategyRequire); passportConfig.splice(index, 1); index = passportConfig.indexOf('// Sign in with Twitter.'); passportConfig.splice(index, 43); fs.writeFileSync(passportConfigFile, passportConfig.join('\n')); // views/account/login.jade (-) index = loginTemplate.indexOf(" a.btn.btn-block.btn-twitter.btn-social(href='/auth/twitter')"); loginTemplate.splice(index, 4); fs.writeFileSync(loginTemplateFile, loginTemplate.join('\n')); // views/account/profile.jade (-) index = profileTemplate.indexOf(' if user.twitter'); profileTemplate.splice(index - 1, 5); fs.writeFileSync(profileTemplateFile, profileTemplate.join('\n')); // models/User.js (-) index = userModel.indexOf(' twitter: String,'); userModel.splice(index, 1); fs.writeFileSync(userModelFile, userModel.join('\n')); console.log('✗ Twitter authentication has been removed.'.error); } }); } });