freeCodeCamp/server/server.js

352 lines
8.4 KiB
JavaScript
Raw Normal View History

require('dotenv').load();
2015-06-18 23:19:24 +00:00
var pmx = require('pmx');
pmx.init();
2015-06-17 19:40:44 +00:00
var R = require('ramda'),
2015-06-16 18:30:16 +00:00
assign = require('lodash').assign,
loopback = require('loopback'),
boot = require('loopback-boot'),
accepts = require('accepts'),
cookieParser = require('cookie-parser'),
compress = require('compression'),
session = require('express-session'),
2015-06-29 16:55:39 +00:00
expressState = require('express-state'),
2015-06-16 18:30:16 +00:00
logger = require('morgan'),
errorHandler = require('errorhandler'),
methodOverride = require('method-override'),
bodyParser = require('body-parser'),
helmet = require('helmet'),
MongoStore = require('connect-mongo')(session),
flash = require('express-flash'),
path = require('path'),
expressValidator = require('express-validator'),
lessMiddleware = require('less-middleware'),
passportProviders = require('./passport-providers'),
2015-06-29 16:50:25 +00:00
rxMiddleware = require('./utils/rx').rxMiddleware,
2015-06-16 18:30:16 +00:00
/**
* API keys and Passport configuration.
*/
secrets = require('./../config/secrets');
2013-11-27 04:15:13 +00:00
var generateKey =
2015-06-04 21:32:59 +00:00
require('loopback-component-passport/lib/models/utils').generateKey;
/**
* Create Express server.
*/
2015-06-03 00:27:02 +00:00
var app = loopback();
2015-06-29 16:55:39 +00:00
expressState.extend(app);
2015-06-04 17:52:12 +00:00
var PassportConfigurator =
require('loopback-component-passport').PassportConfigurator;
var passportConfigurator = new PassportConfigurator(app);
app.set('port', process.env.PORT || 3000);
app.set('views', path.join(__dirname, 'views'));
app.set('view engine', 'jade');
2015-04-22 21:53:58 +00:00
app.use(compress());
2015-06-01 23:23:53 +00:00
app.use(lessMiddleware(path.join(__dirname, '/public')));
app.use(logger('dev'));
app.use(bodyParser.json());
2015-06-17 19:40:44 +00:00
app.use(bodyParser.urlencoded({
extended: true
}));
app.use(expressValidator({
customValidators: {
2015-06-17 19:40:44 +00:00
matchRegex: function(param, regex) {
return regex.test(param);
}
}
}));
app.use(methodOverride());
2015-06-04 19:42:13 +00:00
app.use(cookieParser(secrets.cookieSecret));
app.use(session({
resave: true,
saveUninitialized: true,
secret: secrets.sessionSecret,
store: new MongoStore({
url: secrets.db,
2015-04-17 06:16:55 +00:00
'autoReconnect': true
})
}));
app.use(flash());
app.disable('x-powered-by');
app.use(helmet.xssFilter());
app.use(helmet.noSniff());
app.use(helmet.frameguard());
app.use(function(req, res, next) {
res.header('Access-Control-Allow-Origin', '*');
res.header('Access-Control-Allow-Headers',
'Origin, X-Requested-With, Content-Type, Accept'
);
next();
});
2014-12-22 20:36:45 +00:00
var trusted = [
"'self'",
'blob:',
2015-06-16 04:14:33 +00:00
'104.236.218.15',
'*.freecodecamp.com',
'http://www.freecodecamp.com',
2015-06-16 13:18:59 +00:00
'https://www.freecodecamp.com',
'https://freecodecamp.com',
'https://freecodecamp.org',
'*.freecodecamp.org',
// NOTE(berks): add the following as the blob above was not covering www
'http://www.freecodecamp.org',
'ws://freecodecamp.com/',
'ws://www.freecodecamp.com/',
'*.gstatic.com',
'*.google-analytics.com',
'*.googleapis.com',
'*.google.com',
'*.gstatic.com',
'*.doubleclick.net',
'*.twitter.com',
'*.twitch.tv',
'*.twimg.com',
"'unsafe-eval'",
"'unsafe-inline'",
'*.bootstrapcdn.com',
'*.cloudflare.com',
'https://*.cloudflare.com',
'localhost:3001',
'ws://localhost:3001/',
'http://localhost:3001',
'localhost:3000',
'ws://localhost:3000/',
'http://localhost:3000',
'*.ionicframework.com',
'https://syndication.twitter.com',
'*.youtube.com',
'*.jsdelivr.net',
'https://*.jsdelivr.net',
'*.ytimg.com',
'*.bitly.com',
'http://cdn.inspectlet.com/',
2015-06-16 16:28:05 +00:00
'https://cdn.inspeclet.com/',
2015-05-28 23:40:22 +00:00
'wss://inspectletws.herokuapp.com/',
2015-06-16 15:51:07 +00:00
'http://hn.inspectlet.com/',
'*.googleapis.com',
'*.gstatic.com',
'https://hn.inspectlet.com/'
2014-12-22 20:36:45 +00:00
];
app.use(helmet.csp({
defaultSrc: trusted,
scriptSrc: [
'*.optimizely.com',
'*.aspnetcdn.com',
2015-06-16 16:28:05 +00:00
'*.d3js.org',
'https://cdn.inspectlet.com/inspectlet.js',
'http://cdn.inspectlet.com/inspectlet.js',
'http://www.freecodecamp.org'
].concat(trusted),
2015-06-17 19:40:44 +00:00
'connect-src': [].concat(trusted),
2015-06-16 15:54:58 +00:00
styleSrc: [
2015-06-16 16:04:26 +00:00
'*.googleapis.com',
'*.gstatic.com'
2015-06-16 15:54:58 +00:00
].concat(trusted),
imgSrc: [
/* allow all input since we have user submitted images for public profile*/
'*'
].concat(trusted),
fontSrc: [
'*.googleapis.com',
'*.gstatic.com'
].concat(trusted),
mediaSrc: [
'*.amazonaws.com',
'*.twitter.com'
].concat(trusted),
frameSrc: [
'*.gitter.im',
'*.gitter.im https:',
'*.vimeo.com',
'*.twitter.com',
'*.ghbtns.com'
].concat(trusted),
2015-06-01 23:23:53 +00:00
// set to true if you only want to report errors
reportOnly: false,
// set to true if you want to set all headers
setAllHeaders: false,
// set to true if you want to force buggy CSP in Safari 5
safari5: false
}));
2015-06-04 17:52:12 +00:00
passportConfigurator.init();
2015-06-03 00:27:02 +00:00
2015-06-29 16:50:25 +00:00
// add rx methods to express
app.use(rxMiddleware());
2015-06-17 19:40:44 +00:00
app.use(function(req, res, next) {
// Make user object available in templates.
res.locals.user = req.user;
next();
});
2015-06-29 16:50:25 +00:00
2015-06-01 23:23:53 +00:00
app.use(
2015-06-17 19:40:44 +00:00
loopback.static(path.join(__dirname, '../public'), {
maxAge: 86400000
})
2015-06-01 23:23:53 +00:00
);
var startTime = Date.now();
2015-06-04 17:52:12 +00:00
boot(app, {
appRootDir: __dirname,
dev: process.env.NODE_ENV
});
2015-06-17 19:40:44 +00:00
app.use(function(req, res, next) {
// Remember original destination before login.
var path = req.path.split('/')[1];
if (/auth|login|logout|signin|signup|fonts|favicon/i.test(path)) {
return next();
} else if (/\/stories\/comments\/\w+/i.test(req.path)) {
return next();
}
req.session.returnTo = req.path;
next();
});
2015-06-04 17:52:12 +00:00
passportConfigurator.setupModels({
userModel: app.models.user,
userIdentityModel: app.models.userIdentity,
userCredentialModel: app.models.userCredential
});
var passportOptions = {
2015-06-10 22:12:48 +00:00
emailOptional: true,
profileToUser: function(provider, profile) {
var emails = profile.emails;
// NOTE(berks): get email or set to null.
// MongoDB indexs email but can be sparse(blank)
var email = emails && emails[0] && emails[0].value ?
2015-06-16 18:30:16 +00:00
emails[0].value :
null;
var username = (profile.username || profile.id);
2015-06-17 19:40:44 +00:00
username = typeof username === 'string' ? username.toLowerCase() :
username;
var password = generateKey('password');
var userObj = {
username: username,
password: password
};
if (email) {
userObj.email = email;
}
return userObj;
}
};
2015-06-04 17:52:12 +00:00
R.keys(passportProviders).map(function(strategy) {
var config = passportProviders[strategy];
config.session = config.session !== false;
2015-06-10 22:12:48 +00:00
passportConfigurator.configureProvider(
strategy,
assign(config, passportOptions)
);
2015-06-04 17:52:12 +00:00
});
/**
* OAuth sign-in routes.
*/
/**
* 500 Error Handler.
*/
2015-06-26 08:30:51 +00:00
if (process.env.NODE_ENV === 'development') {
2015-06-17 19:40:44 +00:00
app.use(errorHandler({
log: true
}));
2015-03-24 15:03:59 +00:00
} else {
app.use(pmx.expressErrorHandler());
2015-06-01 23:23:53 +00:00
// error handling in production disabling eslint due to express parity rules
// for error handlers
app.use(function(err, req, res, next) { // eslint-disable-line
2015-03-24 15:03:59 +00:00
// respect err.status
if (err.status) {
res.statusCode = err.status;
}
// default status code to 500
if (res.statusCode < 400) {
res.statusCode = 500;
}
// parse res type
var accept = accepts(req);
var type = accept.type('html', 'json', 'text');
var message = 'opps! Something went wrong. Please try again later';
if (type === 'html') {
2015-06-17 19:40:44 +00:00
req.flash('errors', {
msg: message
});
2015-03-24 15:03:59 +00:00
return res.redirect('/');
// json
2015-03-24 15:03:59 +00:00
} else if (type === 'json') {
res.setHeader('Content-Type', 'application/json');
2015-06-17 19:40:44 +00:00
return res.send({
message: message
});
// plain text
2015-03-24 15:03:59 +00:00
} else {
res.setHeader('Content-Type', 'text/plain');
return res.send(message);
}
});
}
module.exports = app;
2015-06-16 18:30:16 +00:00
app.start = function () {
app.listen(app.get('port'), function() {
2015-07-01 21:35:15 +00:00
app.emit('started');
console.log(
'FreeCodeCamp server listening on port %d in %s mode',
app.get('port'),
app.get('env')
);
});
};
// start the server if `$ node server.js`
if (require.main === module) {
if (process.env.NODE_ENV === 'production') {
var timeoutHandler;
console.log('waiting for db to connect');
var onConnect = function() {
console.log('db connected in %s ms', Date.now() - startTime);
if (timeoutHandler) {
clearTimeout(timeoutHandler);
}
app.start();
};
var timeoutHandler = setTimeout(function() {
var message =
'db did not after ' +
(Date.now() - startTime) +
' ms connect crashing hard';
console.log(message);
// purposely shutdown server
// pm2 should restart this in production
throw new Error(message);
}, 5000);
app.dataSources.db.on('connected', onConnect);
} else {
app.start();
}
}