1
Add user
model
sequelize model:create --name user --attributes name:string,email:string,password:string
2
Add validations for user
model
Validations are used as constraints for a column in a table that requires an entry in the database to follow various rules set in order for that data to be entered into the database.
'use strict';
const {
Model
} = require('sequelize');
module.exports = (sequelize, DataTypes) => {
class user extends Model {
/**
* Helper method for defining associations.
* This method is not a part of Sequelize lifecycle.
* The `models/index` file will call this method automatically.
*/
static associate(models) {
// define association here
}
};
user.init({
name: {
type: DataTypes.STRING,
validate: {
len: {
args: [1,99],
msg: 'Name must be between 1 and 99 characters'
}
}
},
email: {
type: DataTypes.STRING,
validate: {
isEmail: {
msg: 'Invalid email'
}
}
},
password: {
type: DataTypes.STRING,
validate: {
len: {
args: [8,99],
msg: 'Password must be between 8 and 99 characters'
}
}
}
}, {
sequelize,
modelName: 'user',
});
return user; // add functions above
};
3
Make a commit message
git add .
git commit -m "chore: add: user model and validations"
1
Import bcrypt
at the top of user
model
const bcrypt = require('bcryptjs');
2
Create a hook beforeCreate
to hash password inside user
model before it enters the database
Inside of the user model, add the following hook to hash password
// Before a user is created, we are encrypting the password and using hash in its place
user.addHook('beforeCreate', (pendingUser) => { // pendingUser is user object that gets passed to DB
// Bcrypt is going to hash the password
let hash = bcrypt.hashSync(pendingUser.password, 12); // hash 12 times
pendingUser.password = hash; // this will go to the DB
});
3
Add validPassword()
method to user
model that will compare a password entered with the hashed password
// Check the password on Sign-In and compare it to the hashed password in the DB
user.prototype.validPassword = function(typedPassword) {
let isCorrectPassword = bcrypt.compareSync(typedPassword, this.password); // check to see if password is correct.
return isCorrectPassword;
}
4
Add toJSON()
method to user
model that will delete password to prevent from being used on the client
// return an object from the database of the user without the encrypted password
user.prototype.toJSON = function() {
let userData = this.get();
delete userData.password; // it doesn't delete password from database, only removes it.
return userData;
}
5
Verify that model looks like the following code snippet ( here )
6
Do a migration
sequelize db:migrate
1
Create a .env
file and place an evironment variable SECRET_SESSION
with the string of your choice
SECRET_SESSION=alldayidreamaboutsoftwareengineering
2
Add .env
to .gitignore file
3
Import the connect-flash
and express-session
under the imports inside the server file
const session = require('express-session');
const flash = require('connect-flash');
4
Add SECRET_SESSION
variable that will be a reference to the environment variable set in step 1
. Print to make sure the variable is displaying inside the terminal
const SECRET_SESSION = process.env.SECRET_SESSION;
// console.log(SECRET_SESSION);
5
Add session and flash middleware to be used throughout app inside server.js
Add below the current middleware is located ( before routes )
app.use(flash()); // flash middleware
app.use(session({
secret: SECRET_SESSION, // What we actually will be giving the user on our site as a session cookie
resave: false, // Save the session even if it's modified, make this false
saveUninitialized: true // If we have a new session, we save it, therefore making that true
}));
7
Add function as middle to store flash messages and user on res.locals
app.use((req, res, next) => {
console.log(res.locals);
res.locals.alerts = req.flash();
res.locals.currentUser = req.user;
next();
});
8
Make commit message
git add .
git commit -m "feat: Create env variable and add session and flash middleware"
1
Create a file called ppConfig.js
inside of the config
folder
2
Import passport
, passport-local
and the database
into ppConfig.js
file
const passport = require('passport');
const LocalStrategy = require('passport-local').Strategy;
// Database
const { user } = require('../models');
3
Create a new instance of a LocalStrategy
const STRATEGY = new LocalStrategy({
usernameField: 'email', // looks for an email field as the username
passwordField: 'password' // looks for an password field as the password
}, async (email, password, cb) => {
try {
const user = await user.findOne({
where: { email }
});
if (!user || !user.validPassword(password)) {
cb(null, false); // if no user or invalid password, return false
} else {
cb(null, user);
}
} catch (err) {
console.log('------- Error below -----------');
console.log(err);
}
})
4
Serialize User with Passport in order to login
// Passport "serialize" info to be able to login
passport.serializeUser((user, cb) => {
cb(null, user.id);
});
5
Deserialize user and return user if found by id
passport.deserializeUser(async (id, cb) => {
try {
const user = await user.findByPk(id);
if (user) {
cb(null, user)
}
} catch (err) {
console.log('---- Yo... There is an error ----');
console.log(err);
}
});
6
Use new instance of LocalStrategy
inside of Passport as middleward
passport.use(STRATEGY);
7
Export passport from ppConfig.js
module.exports = passport;
8
Make commit message
git add .
git commit -m "feat: [ppConfig]: Create passport configuration"
After making local strategy for passport, we now need to import the ppConfig.js
file into the server, initialize and use it as middleware throughout the app.
1
Import the ppConfig.js
file like so with other imports inside server
const passport = require('./config/ppConfig');
2
Initialize passport and passport session, invoke it, and pass through as middleware. Place this between the middleware that invokes flash
and the middleware that is using res.locals
.
app.use(passport.initialize()); // Initialize passport
app.use(passport.session()); // Add a session
It should be placed on the server like so:
app.use(flash());
app.use(passport.initialize());
app.use(passport.session());
app.use((req, res, next) => {
console.log(res.locals);
res.locals.alerts = req.flash();
res.locals.currentUser = req.user;
next();
});
3
Import the ppConfig.js
file inside of auth.js
located in the controllers
folder
const passport = require('../config/ppConfig');
4
Make commit message
git add .
git commit -m "chore: [server]: Import passport and pass through middleware"
The purpose of this middleware will be to check to see if a user is logged in before they are allowed to have a access to a specific route. This middleware will be place inside a route between the route ( /profile
) and the callback with the request ( req
), and response ( res
) parameters inside.
1
Create a folder called middleware
on the top level.
2
Create a file inside of the middleware
folder called isLoggedIn.js
.
3
Add a function isLoggedIn()
that take in 3 params: req
, res
, and next
.
function isLoggedIn(req, res, next) {
if (!req.user) {
req.flash('error', 'You must be signed in to access page');
res.redirect('/auth/login');
} else {
next();
}
}
4
Export the function
module.exports = isLoggedIn;
5
Import isLoggedIn
inside of server.js
const isLoggedIn = require('./middleware/isLoggedIn');
6
Make commit message
git add .
git commit -m "feat: [isLoggedIn] add middleware and import to server"
We need now to make a /POST
for the data that get submitted with the
login form. The action
in the fom specifies the route /auth/login
that needs to be made for the data to go to. The data that is submitted will be check against the database to validity before being logged into the app.
The form that the data will be submitted from:
<form action="/auth/login" method="POST">
<label for="auth-email">Email</label>
<input id="auth-email" type="email" name="email" required>
<label for="auth-password">Password</label>
<input id="auth-password" type="password" name="password" required>
<input type="submit" value="Log In">
</form>
1
Create a post
route for login. All the methods that are given for the /login
post route are Passport's way of authenticating a user
router.post('/login', passport.authenticate('local', {
successRedirect: '/',
failureRedirect: '/auth/login',
successFlash: 'Welcome back ...',
failureFlash: 'Either email or password is incorrect'
}));
2
Run mocha
to see how many tests passed
3
Make commit message
git add .
git commit -m "feat: [auth] add login post route"
We need now to make a /POST
route for the data that get submitted with the signup form. The action
in the fom specifies the route that needs to be made for the data to go to. The data that is submitted will be used to create a new user and added to the database. After signing up user, we will redirect them back to the login page to login.
The form that the data will be submitted from:
<form action="/auth/signup" method="POST">
<label for="new-email">Email</label>
<input id="new-email" type="email" name="email" required>
<label for="new-name">Name</label>
<input id="new-name" type="text" name="name" required>
<label for="new-password">Password</label>
<input id="new-password" type="password" name="password" required>
<input type="submit" value="Sign up">
</form>
1
Import database
into auth.js
file
const { user } = require('../models');
2
Create a post
route for signup
router.post('/signup', async (req, res) => {
// we now have access to the user info (req.body);
const { email, name, password } = req.body; // goes and us access to whatever key/value inside of the object
try {
const [user, created] = await user.findOrCreate({
where: { email },
defaults: { name, password }
});
if (created) {
// if created, success and we will redirect back to / page
console.log(`----- ${user.name} was created -----`);
const successObject = {
successRedirect: '/',
successFlash: `Welcome ${user.name}. Account was created and logging in...`
}
//
passport.authenticate('local', successObject)(req, res);
} else {
// Send back email already exists
req.flash('error', 'Email already exists');
res.redirect('/auth/signup'); // redirect the user back to sign up page to try again
}
} catch (error) {
// There was an error that came back; therefore, we just have the user try again
console.log('**************Error');
console.log(error);
req.flash('error', 'Either email or password is incorrect. Please try again.');
res.redirect('/auth/signup');
}
});
3
Run mocha
to see how many tests passed
4
Make commit message
git add .
git commit -m "feat: [auth] add signup post route"
The purpose of this route is to log the user out of the app. The main part of this route will be a built in function provided by request ( req
) that would do this: req.logout()
. Then we will display a flash message to the user letting them know that they logged out. Lastly, we will direct the user back to the home page ( /
) like the majority of apps do after logging out.
1
Create /logout
route to log user out
router.get('/logout', (req, res) => {
req.logOut(function(err, next) {
if (err) {
return next(err);
}
req.flash('success', 'Logging out... See you next time!');
res.redirect('/');
}); // logs the user out of the session
});
2
Run mocha
to see how many tests passed
3
Make commit message
git add .
git commit -m "feat: [auth] add logout get route"
The purpose of these partials ( views
) is to render the flash
alerts to the frontend. There will be some logic written out to display to the user is they may have gotten the password incorrect or that they were successful in logging in. We will be adding these partials to the layout.ejs
page.
1
Create a folder called partials
inside of the views
folder
2
Create a file called alerts.ejs
inside of the partials
folder
3
Create two conditionals that will look for error
flash messages or success
flash messages that were created in various routes. We will be adding classes on these messages to display in green for success messages and red for error messages
<% if (alerts.error) { %>
<% alerts.error.forEach(msg => { %>
<div class="alert alert-danger"><%= msg %></div>
<% }) %>
<% } %>
<% if (alerts.success) { %>
<% alerts.success.forEach(msg => { %>
<div class="alert alert-success"><%= msg %></div>
<% }) %>
<% } %>
4
Include the alert
partials inside of the layout.ejs
file at the beginning of the body
<%- include('partials/alerts') %>
5
Make commit message
git add .
git commit -m "feat: [alerts] add partials for flash alerts"
The purpose of building out this logic will be to display on the page whether or not the user is logged in or not. If the user is logged in, then we would remove a link for logging in or signup. If the user is not logged in, the we will display the links for the user to log in or sign up.
1
Add conditional logic to display links for being logged in or not inside of layout.ejs
<% if (!currentUser) {%>
<li><a href="/auth/signup">Signup</a></li>
<li><a href="/auth/login">Login</a></li>
<% } else { %>
<li><a href="/auth/logout">Logout</a></li>
<li><a href="/profile">Profile</a></li>
<% } %>
2
Double check layout.ejs
to make sure it looks like ( this )
The purpose of this step is to add a view and controller for a user to see their information on a profile page. We will to build a GET route to /profile
that will send the user data to the profile.ejs
to be displayed whenever a user logs in.
1
Create a GET route to /profile
and include isLoggedIn
middleware to check to see if user is logged in beforehand inside of server.js
// Add this above /auth controllers
app.get('/profile', isLoggedIn, (req, res) => {
const { id, name, email } = req.user.get();
res.render('profile', { id, name, email });
});
2
Add user id
, name
, email
to the profile.ejs
<h2>Profile Page</h2>
<h3>Welcome to your PROFILE</h3>
<p>Id: <%= id %></p>
<p>Name: <%= name %></p>
<p>Email: <%= email %></p>
2
Run mocha
to see how many tests passed
3
Make commit message
git add .
git commit -m "feat: [profile] add route and send data to view page"
1
Start up server and test app
npm start
2
Complete any debugging that needs to happen.
3
Push final changes to Github.
4
Make this repo a Template on Github for future projects (i.e. Project 2) ✅