Reduce initial user query load
Refactor passport to own file
This commit is contained in:
@ -94,6 +94,7 @@
|
|||||||
"normalize-url": "^1.3.1",
|
"normalize-url": "^1.3.1",
|
||||||
"normalizr": "^2.0.0",
|
"normalizr": "^2.0.0",
|
||||||
"object.assign": "^4.0.3",
|
"object.assign": "^4.0.3",
|
||||||
|
"passport": "^0.2.1",
|
||||||
"passport-facebook": "^2.0.0",
|
"passport-facebook": "^2.0.0",
|
||||||
"passport-github": "^1.0.0",
|
"passport-github": "^1.0.0",
|
||||||
"passport-google-oauth2": "~0.1.6",
|
"passport-google-oauth2": "~0.1.6",
|
||||||
|
114
server/component-passport.js
Normal file
114
server/component-passport.js
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
import passport from 'passport';
|
||||||
|
import { PassportConfigurator } from 'loopback-component-passport';
|
||||||
|
import passportProviders from './passport-providers';
|
||||||
|
import uuid from 'node-uuid';
|
||||||
|
import { generateKey } from 'loopback-component-passport/lib/models/utils';
|
||||||
|
|
||||||
|
import {
|
||||||
|
setProfileFromGithub,
|
||||||
|
getSocialProvider,
|
||||||
|
getUsernameFromProvider
|
||||||
|
} from './utils/auth';
|
||||||
|
|
||||||
|
const passportOptions = {
|
||||||
|
emailOptional: true,
|
||||||
|
profileToUser(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 ?
|
||||||
|
emails[0].value :
|
||||||
|
null;
|
||||||
|
|
||||||
|
// create random username
|
||||||
|
// username will be assigned when camper signups for Github
|
||||||
|
var username = 'fcc' + uuid.v4().slice(0, 8);
|
||||||
|
var password = generateKey('password');
|
||||||
|
var userObj = {
|
||||||
|
username: username,
|
||||||
|
password: password
|
||||||
|
};
|
||||||
|
|
||||||
|
if (email) {
|
||||||
|
userObj.email = email;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(/github/).test(provider)) {
|
||||||
|
userObj[getSocialProvider(provider)] = getUsernameFromProvider(
|
||||||
|
getSocialProvider(provider),
|
||||||
|
profile
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (/github/.test(provider)) {
|
||||||
|
setProfileFromGithub(userObj, profile, profile._json);
|
||||||
|
}
|
||||||
|
return userObj;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const fields = {
|
||||||
|
progressTimestamps: false,
|
||||||
|
completedChallenges: false,
|
||||||
|
challengeMap: false
|
||||||
|
};
|
||||||
|
|
||||||
|
PassportConfigurator.prototype.init = function passportInit(noSession) {
|
||||||
|
this.app.middleware('session:after', passport.initialize());
|
||||||
|
|
||||||
|
if (noSession) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.app.middleware('session:after', passport.session());
|
||||||
|
|
||||||
|
// Serialization and deserialization is only required if passport session is
|
||||||
|
// enabled
|
||||||
|
|
||||||
|
passport.serializeUser((user, done) => {
|
||||||
|
done(null, user.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
passport.deserializeUser((id, done) => {
|
||||||
|
|
||||||
|
this.userModel.findById(id, { fields }, (err, user) => {
|
||||||
|
if (err || !user) {
|
||||||
|
return done(err, user);
|
||||||
|
}
|
||||||
|
this.app.dataSources.db.connector
|
||||||
|
.collection('user')
|
||||||
|
.aggregate([
|
||||||
|
{ $match: { _id: user.id } },
|
||||||
|
{ $project: { points: { $size: '$progressTimestamps' } } }
|
||||||
|
], function(err, { points = 1 } = {}) {
|
||||||
|
if (err) { return done(err); }
|
||||||
|
user.points = points;
|
||||||
|
done(null, user);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function setupPassport(app) {
|
||||||
|
const configurator = new PassportConfigurator(app);
|
||||||
|
|
||||||
|
configurator.setupModels({
|
||||||
|
userModel: app.models.user,
|
||||||
|
userIdentityModel: app.models.userIdentity,
|
||||||
|
userCredentialModel: app.models.userCredential
|
||||||
|
});
|
||||||
|
|
||||||
|
configurator.init();
|
||||||
|
|
||||||
|
Object.keys(passportProviders).map(function(strategy) {
|
||||||
|
var config = passportProviders[strategy];
|
||||||
|
config.session = config.session !== false;
|
||||||
|
configurator.configureProvider(
|
||||||
|
strategy,
|
||||||
|
{
|
||||||
|
...config,
|
||||||
|
...passportOptions
|
||||||
|
}
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
@ -3,96 +3,29 @@ var pmx = require('pmx');
|
|||||||
pmx.init();
|
pmx.init();
|
||||||
|
|
||||||
var _ = require('lodash'),
|
var _ = require('lodash'),
|
||||||
uuid = require('node-uuid'),
|
|
||||||
assign = require('lodash').assign,
|
|
||||||
loopback = require('loopback'),
|
loopback = require('loopback'),
|
||||||
boot = require('loopback-boot'),
|
boot = require('loopback-boot'),
|
||||||
expressState = require('express-state'),
|
expressState = require('express-state'),
|
||||||
path = require('path'),
|
path = require('path'),
|
||||||
passportProviders = require('./passport-providers');
|
setupPassport = require('./component-passport');
|
||||||
|
|
||||||
var setProfileFromGithub = require('./utils/auth').setProfileFromGithub;
|
|
||||||
var getSocialProvider = require('./utils/auth').getSocialProvider;
|
|
||||||
var getUsernameFromProvider = require('./utils/auth').getUsernameFromProvider;
|
|
||||||
var generateKey =
|
|
||||||
require('loopback-component-passport/lib/models/utils').generateKey;
|
|
||||||
|
|
||||||
var isBeta = !!process.env.BETA;
|
|
||||||
var app = loopback();
|
var app = loopback();
|
||||||
|
var isBeta = !!process.env.BETA;
|
||||||
|
|
||||||
expressState.extend(app);
|
expressState.extend(app);
|
||||||
app.set('state namespace', '__fcc__');
|
app.set('state namespace', '__fcc__');
|
||||||
|
|
||||||
var PassportConfigurator =
|
|
||||||
require('loopback-component-passport').PassportConfigurator;
|
|
||||||
var passportConfigurator = new PassportConfigurator(app);
|
|
||||||
|
|
||||||
app.set('port', process.env.PORT || 3000);
|
app.set('port', process.env.PORT || 3000);
|
||||||
app.set('views', path.join(__dirname, 'views'));
|
app.set('views', path.join(__dirname, 'views'));
|
||||||
app.set('view engine', 'jade');
|
app.set('view engine', 'jade');
|
||||||
app.use(loopback.token());
|
app.use(loopback.token());
|
||||||
app.disable('x-powered-by');
|
app.disable('x-powered-by');
|
||||||
|
|
||||||
// adds passport initialization after session middleware phase is complete
|
|
||||||
passportConfigurator.init();
|
|
||||||
|
|
||||||
boot(app, {
|
boot(app, {
|
||||||
appRootDir: __dirname,
|
appRootDir: __dirname,
|
||||||
dev: process.env.NODE_ENV
|
dev: process.env.NODE_ENV
|
||||||
});
|
});
|
||||||
|
|
||||||
|
setupPassport(app);
|
||||||
passportConfigurator.setupModels({
|
|
||||||
userModel: app.models.user,
|
|
||||||
userIdentityModel: app.models.userIdentity,
|
|
||||||
userCredentialModel: app.models.userCredential
|
|
||||||
});
|
|
||||||
|
|
||||||
var passportOptions = {
|
|
||||||
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 ?
|
|
||||||
emails[0].value :
|
|
||||||
null;
|
|
||||||
|
|
||||||
// create random username
|
|
||||||
// username will be assigned when camper signups for GitHub
|
|
||||||
var username = 'fcc' + uuid.v4().slice(0, 8);
|
|
||||||
var password = generateKey('password');
|
|
||||||
var userObj = {
|
|
||||||
username: username,
|
|
||||||
password: password
|
|
||||||
};
|
|
||||||
|
|
||||||
if (email) {
|
|
||||||
userObj.email = email;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(/github/).test(provider)) {
|
|
||||||
userObj[getSocialProvider(provider)] = getUsernameFromProvider(
|
|
||||||
getSocialProvider(provider),
|
|
||||||
profile
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (/github/.test(provider)) {
|
|
||||||
setProfileFromGithub(userObj, profile, profile._json);
|
|
||||||
}
|
|
||||||
return userObj;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Object.keys(passportProviders).map(function(strategy) {
|
|
||||||
var config = passportProviders[strategy];
|
|
||||||
config.session = config.session !== false;
|
|
||||||
passportConfigurator.configureProvider(
|
|
||||||
strategy,
|
|
||||||
assign(config, passportOptions)
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
app.start = _.once(function() {
|
app.start = _.once(function() {
|
||||||
app.listen(app.get('port'), function() {
|
app.listen(app.get('port'), function() {
|
||||||
|
Reference in New Issue
Block a user