Week 8 Assignment
Overview
This week's assignment continues to build upon the Shopping List application we developed in Week 7. We will add authentication to our application using Firebase Authentication and GitHub OAuth.
Prerequisites
Before starting this assignment, you should have completed the Week 7 Assignment and have a working shopping list application. If you have not completed the Week 7 assignment, you have three options:
-
Use another student's Week 7 Assignment as a starting point for this week's assignment. Refer to the list of student repositories.
-
Use your assignment solution from Week 3 as a starting point for this week's assignment. Refer to the Week 3 Assignment for instructions.
-
Start from scratch. To practice authentication, you really just need a simple application that has a login page and a protected page.
Part 1: Create a Firebase Project
Follow this step-by-step guide to create a Firebase project with GitHub OAuth web app.
Part 2: Add Firebase Authentication
Install Firebase
In your project folder, install the Firebase SDK using npm.
npm install firebase
npm install encoding
Setup Files
Let's first create a new folder for this assignment and populating it with the files from previous assignments to use as a starting point. The main shopping list application is going to reside in week-8/shopping-list
. week-8
will be the landing page with login functionality. We are also going to add more structure to our application by creating a a utils
folder to hold our Firebase code.
- Create a new folder called
week-8
inside yourapp
folder. - Create a new folder called
shopping-list
inside yourweek-8
folder. - Copy
page.js
, all components, and thejson
file fromweek-7
toweek-8/shopping-list
.
We will add a page.js
file to the week-8
folder later.
Create Firebase Config
- Create a new folder called
_utils
inside yourweek-8
folder. - Create a file called
firebase.js
inside yourweek-8/_utils
folder.
Use the following code for the contents of firebase.js
.
// Import the functions you need from the SDKs you need
import { initializeApp } from "firebase/app";
// TODO: Add SDKs for Firebase products that you want to use
// https://firebase.google.com/docs/web/setup#available-libraries
import { getAuth } from "firebase/auth";
// Your web app's Firebase configuration
const firebaseConfig = {
apiKey: process.env.NEXT_PUBLIC_FIREBASE_API_KEY,
authDomain: process.env.NEXT_PUBLIC_FIREBASE_AUTH_DOMAIN,
projectId: process.env.NEXT_PUBLIC_FIREBASE_PROJECT_ID,
storageBucket: process.env.NEXT_PUBLIC_FIREBASE_STORAGE_BUCKET,
messagingSenderId: process.env.NEXT_PUBLIC_FIREBASE_MESSAGING_SENDER_ID,
appId: process.env.NEXT_PUBLIC_FIREBASE_APP_ID,
};
// Initialize Firebase
const app = initializeApp(firebaseConfig);
export const auth = getAuth(app);
Create .env.local
File
- Create a new file called
.env.local
in your project root folder. This is in the same folder as yourpackage.json
file. - Add the following environment variables to your
.env.local
file. Replace the values with your Firebase project values from the Firebase console. (Last step of Part 1.)
NEXT_PUBLIC_FIREBASE_API_KEY=""
NEXT_PUBLIC_FIREBASE_AUTH_DOMAIN=""
NEXT_PUBLIC_FIREBASE_PROJECT_ID=""
NEXT_PUBLIC_FIREBASE_STORAGE_BUCKET=""
NEXT_PUBLIC_FIREBASE_MESSAGING_SENDER_ID=""
NEXT_PUBLIC_FIREBASE_APP_ID=""
Part 3: Create AuthContext and useAuth Hook
In the _utils
folder, create a file called auth-context.js
and add the following code.
"use client";
import { useContext, createContext, useState, useEffect } from "react";
import {
signInWithPopup,
signOut,
onAuthStateChanged,
GithubAuthProvider,
} from "firebase/auth";
import { auth } from "./firebase";
const AuthContext = createContext();
export const AuthContextProvider = ({ children }) => {
const [user, setUser] = useState(null);
const gitHubSignIn = () => {
const provider = new GithubAuthProvider();
return signInWithPopup(auth, provider);
};
const firebaseSignOut = () => {
return signOut(auth);
};
useEffect(() => {
const unsubscribe = onAuthStateChanged(auth, (currentUser) => {
setUser(currentUser);
});
return () => unsubscribe();
}, [user]);
return (
<AuthContext.Provider value={{ user, gitHubSignIn, firebaseSignOut }}>
{children}
</AuthContext.Provider>
);
};
export const useUserAuth = () => {
return useContext(AuthContext);
};
Part 4: Add AuthContextProvider to layout.js
In the week-8
folder, create a file called layout.js
and add the following code.
import { AuthContextProvider } from "./_utils/auth-context";
const Layout = ({ children }) => {
return <AuthContextProvider>{children}</AuthContextProvider>;
};
export default Layout;
Part 5: Add a Landing Page with Login Button
In the week-8
folder, create a page.js
file. This will be the landing page for our application. Display a login button if the user is not logged in. If the user is logged in, display a welcome message, a logout button, and a link to the shopping list page.
The following code snippets may be helpful:
import { useUserAuth } from "./_utils/auth-context";
const { user, gitHubSignIn, firebaseSignOut } = useUserAuth();
await gitHubSignIn();
await firebaseSignOut();
<p>
Welcome, {user.displayName} ({user.email})
</p>;
user
is the user object returned from Firebase Authentication. If the user is not logged in, the value will benull
.gitHubSignIn
is a function that will open a popup window to allow the user to sign in with GitHub.logOut
is a function that will log the user out.
Part 6: Protect the Shopping List Page
Even though a user can't see a link to the shopping list page without being logged in, we should still protect the page so that if a user tries to access the page directly, they won't be able to see the shopping list.
In the week-8/shopping-list/page.js
file, check if the user is logged in by using the useUserAuth
hook and if the user
object is null, do not render the shopping list page. Optional: You can redirect the user to the landing page if you want.
Part 7: Deploy to Vercel (Optional)
If you wish to deploy your application to Vercel, you will need to make the following changes.
Add environment variables to your Vercel project
On the Vercel website, go to your project and click on the Settings tab. Click on Environment Variables. Do one of the following:
- Copy your environment variables from your
.env.local
file and paste them into the Vercel environment variables. - Upload your
.env.local
file. - Add your environment variables manually one-by-one.
GitHub OAuth Homepage URL
In the GitHub OAuth settings, change the Homepage URL from http://localhost:3000
to your Vercel URL. Click "Update application".
Add Vercel URL to Firebase
In the Firebase console, go to your project settings. Click on "Authorized domains". Click the "Add Domain" button and add your Vercel URL.
Extras
A few things you can do to improve your application:
- After logging in, redirect the user to the shopping list page.
- Add a logout button to the shopping list page.
- Add a profile page that displays the user's profile information.
- Add other OAuth providers such as Google.
Example Output
https://cprg306-assignments.vercel.app/week-8 (opens in a new tab)
Also, try to go to the shopping list page directly without signing in https://cprg306-assignments.vercel.app/week-8/shopping-list (opens in a new tab)
Assignment Submission
The instructor will be able to find your assignment in your GitHub repository. Make sure you have committed and pushed your changes to GitHub before the assignment deadline.
Check your GitHub account to see if your assignment has been correctly pushed.