Skip to content

carveler/file-upload-guide

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

27 Commits
 
 

Repository files navigation

File Uploading & Display - Fullstack

Find here summarized a mini guide how to incorporate file uplading from frontend to backend with a (free) file cloud provider.

We will use Cloudinary as the provider, which allows us (at the time of writing) 10GB+ storage for free.

We will use the example of an Avatar upload.

In this fullstack sample we will encode our images as base64 strings BEFORE uploading them to the API. So we will do the conversion in the frontend.

This way we do not need to send "multipart formdata" and just upload a JSON object as usual, which simplifies the workflow in the frontend a lot.

Also in the backend we then do not need to parse binary data anymore. We can simply upload directly the received base64 string to our file cloud provider.

Steps

Backend part

  • Install following packages: npm i cloudinary dotenv

  • Configure Cloudinary URL in a .env of your backend

    • The Cloudinary URL is similar to your MongoDB Atlas URL. A url to your own cloud including your user credentials
    • You find the Cloudinary Environment URL in the cloudinary dashboard after login
  • Load the .env at the top of your server.js file (if not done already): require("dotenv").config()

  • Adapt your model where you wanna store an image URL

    • Example User Model: add a field "avatar_url" (String)
  • Upload route handler (=controller)

    • Import cloudinary at the top of your file: const cloudinary = require("cloudinary").v2
    • Extract / split the avatar file and normal JSON data from req.body: const { avatar, ...userData } = req.body
    • Upload the avatar string to cloudinary
      • const result = await cloudinary.upload.upload( avatar )
    • Store the received URL in your model
      • e.g. const userNew = await User({ ...userData, avatar_url: result.secure_url })
      • Cloudinary will provide you with TWO urls in its response: url and secure_url
      • url will be reachable by http:// and secure_url will be reachable by https://. So it is advisable to always use the secure one
    • Return the created user to the frontend using res.json()
  • Test File upload against your route from Insomnia

    • Setup a POST request to your API upload URL & use JSON as Body format
    • Convert some avatar file to a base64 dataUri, e.g. here: https://www.base64-image.de/
    • Paste the dataUri into your JSON Body
    • Example body:
    {
       "email": "rob@rob.com",
       "avatar: "....."
    }
    

Frontend part

  • Create or use an existing React Frontend to upload a file

    • Ideally some app with a signup or any other upload form
  • Create an input of type "file", e.g.

    • <input name="avatar" type='file' accept='image/*' onChange={ onAvatarChange } />
    • This will make it possible to select files from your filesystem
  • Handle the avatar selection

    • Setup a state for storing the avatar file: const [avatarPreview, setAvatarPreview] = useState()
    • Define an onAvatarChange handler const onAvatarChange = (e) => {...}
    • The file, the user selected, will be availble in the event object: e.target.files[0]
    • If you allowed multiple file selection (with <input type="file" multiple />) you will have an array of files in e.target.files
    • e.target.files contains the binary files. These we now need to convert to DataURI Strings using the Browser builtin FileReader class
    • Example:
      let fileSelected = e.target.files[0]  // grab selected file
      
      if(!fileSelected) return
      
      let fileReader = new FileReader()
      fileReader.readAsDataURL( fileSelected ) // concert to base64 encoded string
      // wait until file is fully loaded / converted to base64 (once fully loaded the "onloadedend" event below fires)
      fileReader.onloadend = (ev) => {
        setAvatarPreview( fileReader.result )
      }
      
  • Previewing an image

    • Import some default preview image from your React folder, e.g. import avatarDefault from './img/avatarDefault.png'
    • Asign it as default to your avatar preview state: const [avatarPreview, setAvatarPreview] = useState( avatarDefault )
    • Assign the state to an image tag for displaying the preview <img src={ avatarPreview } />
    • Et voila: Now you have an avatar preview on file selection
    • To select a file on image click, you can use the label trick
      • Put an id on the HTML file input (e.g. type="file" id="avatar")
      • Wrap your preview image with a label and link it to the input <label htmlFor="avatar"><img src={ avatarPreview } /></label>
      • Now you can hide the ugly default file input field, e.g. with simple CSS (visibility: hidden)
  • Notes on usage with React-Hook-Form

    • Once you register an input by putting the "register" key on it, you cannot put an additional "onChange" handler on it anymore
    • But in order to listen for the Avatar File Selection (and show a preview) we need that onChange handler...
    • So the simplest way to avoid this, is by not handling the file input by React-Hook-Form (not putting a register key on it)
  • Submitting

    • Before sending the data to Axios, we need to merge the encoded avatar file into the form JSON data
    • Afterwards we can forward all data with Axios to the API
    • Full submit Example:
        onSubmit = (jsonData) => {
        
          jsonData.avatar = avatarPreview // merge the avatar string into our data
      
          // signup user with avatar in backend
          try {
            let response = await axios.post('http://localhost:5000/users', jsonData)
            console.log("Response: ", response.data) // => signed up user
            history.push('/users')  
          }
          // handle error
          catch(errAxios) {
            console.log(errAxios.response && errAxios.response.data)
          }
        }
      

Uploading multiple files

Let's assume in a form you wanna upload images for a recipe of your favorite food.

Frontend

Simply add the "multiple" HTML attribute to your input field

<input type="file" name="recipe_images" accept="image/*" multiple />

Backend

Will follow...

Uploading two different file fields

Let's assume you wanna manage your user avatar + your family photos in your profile.

Now we will have two DIFFERENT file input fields we wanna upload. How do we do this?

Frontend Form

For the avatar we can set an input to select a single file. For the family photos we set an input field for selecting multiple files.

<input type="file" name="avatar" accept="image/*" />
<input type="file" name="family" accept="image/*" multiple />
Backend

Will follow...

Resources

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published