Introduction
Just as authentication is important in APIs, it is also an important feature in certain web applications—those with pages and secrets that should only be accessible to registered and authenticated users.
In this tutorial, you will build a simple web application while learning how to create user registration.
Application Setup
Create a new directory where you will be working from. For the sake of this tutorial, I called mine site-auth. Initialize npm in the new directory you just created. Here is how to initialize npm.
npm init -y
The -y
flag tells npm to use the default options.
Edit the dependencies part of your package.json file to look like what I have in mine.
#package.json { "name": "site-auth", "version": "1.0.0", "description": "", "main": "app.js", "scripts": { "test": "echo \"Error: no test specified\" && exit 1" }, "keywords": [], "author": "izuchukwu1", "license": "ISC", "dependencies": { "bcryptjs": "^2.4.3", "body-parser": "^1.17.1", "connect-flash": "^0.1.1", "cookie-parser": "^1.4.3", "express": "^4.15.2", "express-handlebars": "^3.0.0", "express-messages": "^1.0.1", "express-session": "^1.15.2", "joi": "^13.0.1", "mongoose": "^4.11.12", "morgan": "^1.8.1", "passport": "^0.4.0", "passport-local": "^1.0.0" } }
With that done, run the command to install the dependencies.
npm install
Create a file in your working directory called app.js.
Start by requiring the dependencies you installed and the necessary files.
#app.js const express = require('express'); const morgan = require('morgan') const path = require('path'); const cookieParser = require('cookie-parser'); const bodyParser = require('body-parser'); const expressHandlebars = require('express-handlebars'); const flash = require('connect-flash'); const session = require('express-session'); const mongoose = require('mongoose') const passport = require('passport') require('./config/passport')
These dependencies were installed when you ran npm install. To use them in your application, you have to require them and save them in their respective variables.
For this tutorial, you will be using MongoDB as your database. You will need to store user information in the database. To work with MongoDB, you will make use of Mongoose—a MongoDB modelling tool for Node.js. Setting up Mongoose is easy, like this.
#app.js mongoose.Promise = global.Promise mongoose.connect('mongodb://localhost:27017/site-auth')
At this point, let's set up our middleware.
// 1 const app = express() app.use(morgan('dev')) // 2 app.set('views', path.join(__dirname, 'views')) app.engine('handlebars', expressHandlebars({ defaultLayout: 'layout' })) app.set('view engine', 'handlebars') // 3 app.use(bodyParser.json()) app.use(bodyParser.urlencoded({ extended: false })) app.use(cookieParser()) app.use(express.static(path.join(__dirname, 'public'))) app.use(session({ cookie: { maxAge: 60000 }, secret: 'codeworkrsecret', saveUninitialized: false, resave: false })); app.use(passport.initialize()) app.use(passport.session()) // 4 app.use(flash()) app.use((req, res, next) => { res.locals.success_mesages = req.flash('success') res.locals.error_messages = req.flash('error') next() }) // 5 app.use('/', require('./routes/index')) app.use('/users', require('./routes/users')) // 6 // catch 404 and forward to error handler app.use((req, res, next) => { res.render('notFound') }); // 7 app.listen(5000, () => console.log('Server started listening on port 5000!'))
- Express is initialized and assigned to
app
. - Middleware to handle views is set up. For the views, you'll be making use of
handlebars
. - You set up middleware for
bodyparser
,cookie
,session
, andpassport
. Passport will be used when users want to log in. - At some points, you will be displaying flash messages. Thus you need to set up middleware for that, and also create the type of flash messages you want.
- Routes middleware—this will handle any request made to a URL path. The URL paths specified here are for the index and users path.
- Middleware to handle 404 errors. This middleware kicks in when a request does not map to any of the middleware created above it.
- The server is set to listen at port 5000.
Views Setup
Create a new directory called views. Inside the views directory, create two other directories called layouts and partials. You want to achieve a tree structure like this in your views, so create the necessary files in their respective directories.
├── dashboard.handlebars ├── index.handlebars ├── layouts │ └── layout.handlebars ├── login.handlebars ├── notFound.handlebars ├── partials │ └── navbar.handlebars └── register.handlebars
With that done, time to drop the code.
#dashboard.handlebars <!-- Jumbotron --> <div class="jumbotron"> <h1>User DashBoard</h1> </div>
This is a dashboard that should be visible to only registered users. For this tutorial, it will be your secret page.
Now the index page for the application should look like this.
#index.handlebars <!-- Jumbotron --> <div class="jumbotron"> <h1>Site Authentication!</h1> <p class="lead">Welcome aboard.</p> </div>
The application needs a layout that will be used, and here is that layout you will be using.
#layout/layout.handlebars <!DOCTYPE html> <html> <head> <title>Site Authentication</title> <link rel="stylesheet" href="http://ift.tt/2apRjw3" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous"> <link rel="stylesheet" href="http://ift.tt/2aHU2x3" integrity="sha384-rHyoN1iRsVXV4nD0JutlnGaslCJuC7uwjduW9SVrLvRYooPp2bWYgmgJQIXwl/Sp" crossorigin="anonymous"> <link rel="stylesheet" href="/css/style.css"> </head> <body> <div class="alert alert-success"></div> <div class="alert alert-danger"></div> <div class="container"> } </div> <script src="http://ift.tt/2aHTozy" integrity="sha384-Tc5IQib027qvyjSMfHjOMaLkfuWVxZxUPnCJA7l2mCWNIpG9mGCD8wGNIcPD7Txa" crossorigin="anonymous"></script> </body> </html>
You'll need a login page for registered users.
#views/login.handlebars <form class="form-signin" action="/users/login" method="POST"> <h2 class="form-signin-heading">Please sign in</h2> <label for="inputEmail" class="sr-only">Email address</label> <input type="email" id="inputEmail" name="email" class="form-control" placeholder="Email address" required autofocus> <label for="inputPassword" class="sr-only">Password</label> <input type="password" id="inputPassword" name="password" class="form-control" placeholder="Password" required> <br/> <button class="btn btn-lg btn-default btn-block" type="submit">Sign in</button> </form>
The notFound.handlebars file will be used as your error page.
#views/notFound.handlebars <!-- Jumbotron --> <div class="jumbotron"> <h1>Error</h1> </div>
Your registration page is supposed to look like this.
<form class="form-signin" action="/users/register" method="POST"> <h2 class="form-signin-heading">Please sign up</h2> <label for="inputEmail" class="sr-only">Email address</label> <input type="email" id="inputEmail" name="email" class="form-control" placeholder="Email address" required autofocus> <label for="inputUsername" class="sr-only">Username</label> <input type="text" id="inputUsername" name="username" class="form-control" placeholder="Username" required> <label for="inputPassword" class="sr-only">Password</label> <input type="password" id="inputPassword" name="password" class="form-control" placeholder="Password" required> <label for="inputConfirmPassword" class="sr-only">Confirm Password</label> <input type="password" id="inputConfirmPassword" name="confirmationPassword" class="form-control" placeholder="Confirm Password" required> <br/> <button class="btn btn-lg btn-default btn-block" type="submit">Sign up</button> </form>
Finally for your views, here's your navigation bar.
#partials/navbar.handlebars <div class="masthead"> <h3 class="text-muted">Site Authentication</h3> <nav> <ul class="nav nav-justified"> <li class="active"><a href="/">Home</a></li> <li><a href="/users/dashboard">Dashboard</a></li> <li><a href="/users/logout">Logout</a></li> <li><a href="/users/register">Sign Up</a></li> <li><a href="/users/login">Sign In</a></li> </ul> </nav> </div>
With that done, you are good to go into some deep parts.
Data Validation
You'll need a User model. From the views code above, you can deduce that the properties needed for the User model are email, username, and password. Create a directory called models, and a file in it called user.js.
#models/user.js // 1 const mongoose = require('mongoose') const Schema = mongoose.Schema const bcrypt = require('bcryptjs') // 2 const userSchema = new Schema({ email: String, username: String, password: String }, { // 3 timestamps: { createdAt: 'createdAt', updatedAt: 'updatedAt' } }) // 4 const User = mongoose.model('user', userSchema) module.exports = User
- Imports dependencies and saves them in variables.
- A new Schema is created. For each user, you want to save the
email
,username
, andpassword
to the database. The Schema shows how the model is to be constructed for each document. Here you want the email, username, and password to be of the String type. - For each user saved to the database, you also want to create
timestamps
. You utilize Mongoose to obtain thecreatedAt
andupdatedAt
, and this is then saved to the database. - The model is defined and assigned to a variable called
User
, which is then exported as a module so it can be used in other parts of the application.
Salting and Hashing of the Password
You do not want to store users' passwords as plain text. Here's what you want to do when a user enters a plain text password while registering. The plain text password should be hashed using a salt that will be generated by your application (using bcryptjs). This hashed password is then stored in the database.
Sounds great, right? Let's implement that in the user.js file.
#models/user.js module.exports.hashPassword = async (password) => { try { const salt = await bcrypt.genSalt(10) return await bcrypt.hash(password, salt) } catch(error) { throw new Error('Hashing failed', error) } }
You just created a method that will be called in events of user registration. The method will receive the plain text password the user entered. As I mentioned earlier, the plain text password will be hashed using a generated salt. The hashed password will be returned as the password for the user.
Index and Users Routes
Create a new directory called routes. In this new directory, create two new files: index.js and users.js.
The index.js file will be very simple. It will map to the index of your application. Remember you set up middleware for your routes in your app.js file when you did this.
app.use('/', require('./routes/index')) app.use('/users', require('./routes/users'))
So your index routes, which simply render the index page, should look like this.
#routes/index.js const express = require('express') const router = express.Router() router.get('/', (req, res) => { res.render('index') }) module.exports = router
Now to the users route. For now, this route file will be doing four things.
- Require dependencies. You will need to require the dependencies you installed using NPM.
- Validate user inputs. You want to make sure that the user does not submit an empty form. All inputs are required, and all must be of the type String. The email has a special validation called
.email()
which ensures that what is entered matches the email format, while the password is validated using a regular expression. For the confirmation password, you want it to be the same as the password entered. These validations are done using Joi. - Set up your router. The get request renders the registration page, while the POST request kicks in when the user hits the button to submit the form.
- The router gets exported as a module.
Here is what the code looks like.
#routes/users.js const express = require('express'); const router = express.Router() const Joi = require('joi') const passport = require('passport') const User = require('../models/user') //validation schema const userSchema = Joi.object().keys({ email: Joi.string().email().required(), username: Joi.string().required(), password: Joi.string().regex(/^[a-zA-Z0-9]{6,30}$/).required(), confirmationPassword: Joi.any().valid(Joi.ref('password')).required() }) router.route('/register') .get((req, res) => { res.render('register') }) .post(async (req, res, next) => { try { const result = Joi.validate(req.body, userSchema) if (result.error) { req.flash('error', 'Data entered is not valid. Please try again.') res.redirect('/users/register') return } const user = await User.findOne({ 'email': result.value.email }) if (user) { req.flash('error', 'Email is already in use.') res.redirect('/users/register') return } const hash = await User.hashPassword(result.value.password) delete result.value.confirmationPassword result.value.password = hash const newUser = await new User(result.value) await newUser.save() req.flash('success', 'Registration successfully, go ahead and login.') res.redirect('/users/login') } catch(error) { next(error) } }) module.exports = router
Let's look deeper into what is happening in that POST request.
The values entered in the registration form are accessible via req.body
, and the values look like this.
value: { email: 'chineduizuchkwu1@gmail.com', username: 'izu', password: 'chinedu', confirmationPassword: 'chinedu' },
This is validated using the userSchema
you created above, and the values entered by the user are assigned to a variable called result.
If an error is encountered because of the validation, an error message is displayed to the user and a redirection to the registration page takes place.
Otherwise, we try to find if a user with the same email address exists, as you do not want to have two or more users with same email address. If a user is found, the user is told that the email address is already in use.
In a scenario where no registered user has that email address, the next step is to hash the password. This is where you call the hashPassword
method you created in your user.js file. The new hashed password is assigned to a variable called hash.
There is no need to store the confirmationPassword
in the database. Thus this is deleted. The password available from result is still the plain password. Since you do not want to store the plain password in your database, it is important to reassign the password value to the hash that was created. This is done with a line of code.
result.value.password = hash
The new user instance gets saved to the database. A flash message stating that the registration was successful is displayed, and the user is redirected to the login page.
Start up your server from your terminal by running:
node app.js
Point your browser to http://localhost:5000 and you should see your new app.
Conclusion
Now you know how to implement the registration feature in a Node web application. You have learned the importance of validating user input and how to do that using Joi. You also made use of bcryptjs
to salt and hash your password.
Next, you'll see how to implement a login feature for registered users. I trust you enjoyed yourself!
by Chinedu Izuchukwu via Envato Tuts+ Code
No comments:
Post a Comment