-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserver.js
146 lines (132 loc) · 3.45 KB
/
server.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
const express = require('express')
const path = require('path')
const session = require('express-session')
const cookieParser = require('cookie-parser')
const bodyParser = require('body-parser')
const methodOverride = require('method-override')
const passport = require('passport')
const OutlookStrategy = require('passport-outlook').Strategy
// environment variables
require('dotenv').config()
const OUTLOOK_CLIENT_ID = process.env.OUTLOOK_CLIENT_ID
const OUTLOOK_CLIENT_SECRET = process.env.OUTLOOK_CLIENT_SECRET
// configure passport
passport.serializeUser(function (user, done) {
done(null, user)
})
passport.deserializeUser(function (obj, done) {
done(null, obj)
})
passport.use(new OutlookStrategy({
clientID: OUTLOOK_CLIENT_ID,
clientSecret: OUTLOOK_CLIENT_SECRET,
callbackURL: 'signin-microsoft',
proxy: true
},
function (accessToken, refreshToken, profile, done) {
process.nextTick(function () {
return done(null, profile)
})
}
))
// configure express
const app = express()
app.set('views', __dirname + '/auth')
app.set('view engine', 'ejs')
app.use(cookieParser())
app.use(bodyParser())
app.use(methodOverride())
app.use(session({
secret: 'mmmmmCOOKIES',
cookie: {
_expires: (720 * 60 * 1000)
}
}))
app.use(passport.initialize())
app.use(passport.session())
app.use(express.static(__dirname + '/auth/assets'))
/*
Endpoints!!
Only one to retrieve email
*/
// returns user profile
app.use('/getUser', function (req, res) {
res.status(200).send({
'email': req.user.emails[0].value,
'organization': "City of Pittsburgh",
'name': req.user.displayName
})
})
/*
Routes!!
The auth workflow is handled server side
Once user is validated with pgh email address,
the react app is delivered to client and all further
routing occurs there via wildcard
*/
// login page
app.get('/login', function (req, res) {
req.logout()
res.render('login', {
user: req.user
})
})
// logout endpoint, called from client
// logs out user and redirects to login page
app.get('/logout', function (req, res) {
req.logout()
res.redirect('/login')
})
// 401 page
app.get('/accessDenied', function (req, res) {
req.logout()
res.render('401')
})
// MS oauth service redirect
app.get('/auth',
passport.authenticate('windowslive', {
scope: [
'https://outlook.office.com/User.Read'
]
})
)
// ...and return
app.get('/signin-microsoft',
passport.authenticate('windowslive', {
failureRedirect: '/login'
}),
function (req, res) {
// if user has pgh emal address, let through the gates
if (req.user.emails[0].value.includes('@pittsburghpa.gov')) {
res.locals.user = req.user
res.redirect('/')
} else { // otherwise, get lost!
res.redirect('/accessDenied')
}
})
// all other routes get pushed through wildcard
// basically, some string tricks to ensure bundle gets imported correctly and then
// routing is deferred to react running in client
app.get('*', ensureAuthenticated, (req, res) => {
const link = (req.path == '/' ? 'index.html' : req.path)
const root = path.join(__dirname, 'app/build')
res.sendFile(link, {
root: root
}, (error) => {
if (error) {
res.sendFile('/', {
root: root
})
}
})
})
// helper function to validate user on every route
function ensureAuthenticated(req, res, next) {
if (req.isAuthenticated()) {
return next()
}
res.redirect('/login')
}
const port = process.env.PORT || 5000
app.listen(port)
console.log(`Listening on ${port}`)