How to Implement Firebase Storage in a React Web App

- Jun 22, 2022

Long Story Short

We'll look at how to use firebase storage in a react web app to save files and photos to the cloud and receive the created url of the uploaded content.

Share This Article

1. Intro to Firebase Storage

According to official docs of firebase:

Cloud Storage for Firebase lets you upload and share user generated content, such as images and video, which allows you to build rich media content into your apps. Your data is stored in a Google Cloud Storage bucket — an exabyte scale object storage solution with high availability and global redundancy.

In simple words: If you want to incorporate file upload or image upload in your web app (MERN app in our example), one method is to upload the file or image to Firebase's cloud storage and then collect the URL of the uploaded image file and put that URL in your database.

The user interface that we have:

Firebase Storage ui.png

2. Final Code

Here is the final completed code that we will have at the end of this blog. We are definitely gonna go through each and every line of code.

//~~~ Logic for File Upload ~~~
import {
  uploadBytesResumable,
  getDownloadURL,
  ref,
  deleteObject,
} from "@firebase/storage";
import { storage } from "../Firebase";

function uploadTaskPromise(file) {
  return new Promise(function (resolve, reject) {
    if (!file) return;

    const storageRef = ref(storage, `img/${file.name}`);
    const uploadTask = uploadBytesResumable(storageRef, file);

    uploadTask.on(
      "state_changed",
      (snapshot) => {
        const prog = Math.round(
          (snapshot.bytesTransferred / snapshot.totalBytes) * 100
        );
        console.log(prog);
      },
      (error) => {
        console.log("ERRRRR!!!!!!");
        alert("Error inside upload file function", error);
        reject();
      },
      () => {
        getDownloadURL(uploadTask.snapshot.ref).then((downloadURL) => {
          console.log("File available at", downloadURL);
          resolve(downloadURL);
        });
      }
    );
  });
}

//~~~ Logic for Post request to backend ~~~

const postSubmitHandler = async (e) => {
  e.preventDefault();

  try {
    storageURL = await uploadTaskPromise(file);

    const response = await fetch("http://localhost:5000/api/posts", {
      method: "POST",
      headers: {
        "Content-Type": "application/json",
        Authorization: "Bearer " + token,
      },
      body: JSON.stringify({
        location,
        description,
        price,
        contact,
        breed,
        creatorId: currentUser.id,
        image: storageURL,
      }),
    });
    console.log("REQUEST SENT");
  } catch (err) {
    const deleteImgRef = ref(storage, `img/${file.name}`);
    deleteObject(deleteImgRef)
      .then(() => {
        console.log("Something Went wrong, Image deleted successfully");
      })
      .catch((error) => {
        console.log("an error occured while deleting image", error);
      });
  }
};

3. Creating Firebase Config file

Before we move to the next step I am imagining that you already have created your project in Firebase. If you’re already using npm and a module bundler such as webpack or Rollup, you can run npm install firebase command to install the latest SDK. Then, initialize Firebase and begin using the SDKs for the web app. In our case we will only be using the storage SDK.

To initialize Firebase first create a new file and name it as firebase.js. Now copy the code bellow and paste it in the firebase.js file that you just created.

import { initializeApp } from "firebase/app";
import { getStorage } from "firebase/storage";

const firebaseConfig = {
  //PASTE YOUR FIREBASE CONFIG HERE
};

const app = initializeApp(firebaseConfig);

const storage = getStorage(app);

export { storage, app };

Now go Project settings by clicking of the cog wheel icon in the left menu of your firebase project. Then scroll down and just copy what ever you have in your firebaseConfig constant and paste it inside the firebaseConfig constant that we created in our firebase.js file.

Now we can use the Firebase storage SDK in our app wherever we want.

4. Creating the upload file function

Now we go to the the file where we are creating the post.

import { uploadBytesResumable, ref } from "@firebase/storage";
import { storage } from "../Firebase";

function uploadTaskPromise(file) {
  if (!file) return;

  const storageRef = ref(storage, `img/${file.name}`);
  const uploadTask = uploadBytesResumable(storageRef, file);
}

We import the required dependencies. Create a new function and name it as uploadTaskPromise which accepts a file as a parameter. Now, In order to upload or download files, delete files, or get or update metadata, you must create a reference to the file you want to operate on. A reference can be thought of as a pointer to a file in the cloud. References are lightweight, so you can create as many as you need, and they are also reusable for multiple operations. Once the reference is created we then create an uploadTask by passing our reference and the file that we want to upload as parameters to uploadBytesResumable.

5. Uploading the file

import {
  uploadBytesResumable,
  getDownloadURL,
  ref,
  deleteObject,
} from "@firebase/storage";
import { storage } from "../Firebase";

function uploadTaskPromise(file) {
  return new Promise(function (resolve, reject) {
    if (!file) return;

    const storageRef = ref(storage, `img/${file.name}`);
    const uploadTask = uploadBytesResumable(storageRef, file);

    uploadTask.on(
      "state_changed",
      (snapshot) => {
        const prog = Math.round(
          (snapshot.bytesTransferred / snapshot.totalBytes) * 100
        );
        console.log(prog);
      },
      (error) => {
        console.log("ERRRRR!!!!!!");
        alert("Error inside upload file function", error);
        reject();
      },
      () => {
        getDownloadURL(uploadTask.snapshot.ref).then((downloadURL) => {
          console.log("File available at", downloadURL);
          resolve(downloadURL);
        });
      }
    );
  });
}

Once we have our uploadTask ready we can now move to the actual logic of uploading our file to the cloud storage. The uploadTask has an .on method which registers three observers.

  1. 'state_changed' observer, called any time the state changes
  2. Error observer, called on failure
  3. Completion observer, called on successful completion

1. The first observer observe state change events such as progress, pause, and resume. We also get task progress, including the number of bytes uploaded and the total number of bytes to be uploaded using which we can get the total percentage that has been uploaded in real time. 2. The second observer handles any unsuccessful uploads. In this block we can handle any error that we might get while uploading the file. 3. The third observer and the most important observer handles successful uploads on complete. We can extract the url of the uploaded file so that we can store the URL of the file in our database rather then storing the entire files in our database.

Now uploading file to cloud storage is an asynchronous task so we wrap the entire thing in a promise. We return a new promise which rejects if it reaches the error block or the second observer and resolves with the uploaded file url on successful completion of our upload.

6. Upload Post Logic

Once we have the logic to upload file to cloud storage ready we now can move on to the logic to store the URL in our database by sending a post request to our backend. We create a new function named postSubmitHandler which is fired whenever we click on Post button. And we declare the function as an async function because storing data into database and uploading file to the storage are both asynchronous task.

First and the most important thing to keep in in mind is to do js e.preventDefault() or else the page will refresh whenever we click on Post button and our CODE WILL BLOW UP. Next thing we do is to wrap our entire logic in a Try Catch block for that extra layer of precaution.

Now in the Try block we call our uploadTaskPromise and pass the file we need to upload as a parameter. Don't forget to put the await keyword. Now if the upload task is successful then the promise gets resolved with our stored file URL and we store that URL in a constant storageUrl. In case the upload fails we reject the promise and we are thrown into the catch block where we can handle the error. Now that we have the url all we need to do is to send a POST request to our backend.

The final consideration is what happens if we successfully store files to cloud storage but fail to store data to our database. We will still be tossed to the catch block, but the downloaded file will remain in our cloud storage, doing nothing except wasting storage space. To handle this, we must remove the file that we saved to our storage by generating a reference to the file that has to be deleted, similar to the reference that we generated while uploading our file to the storage. Once we have the reference we can then call deleteObject that we import from @firebase/storage and pass the reference to it. This will delete the file that we stored in our storage if something goes wrong while storing our data to database.

Thank you for your time. Save the post so you may refer to it if you wish to use firebase storage in the future.

Get the latest articles from OurWeb in your inbox.

No spam. Just new tutorials, course announcements, and updates from OurWeb.

Share This Article

Check These Posts as well
Advertisment