Hooked with React - Error handling and loading state in react hooks

šŸ”„ Limited Time Offer
Coding with AI: Learn to build a SaaS MVP in 10 days

Coding with AI: Learn to build a SaaS MVP in 10 days

Master practical AI development to build and launch your startup MVP in weeks instead of months. Learn to leverage AI tools and APIs effectively.

  • šŸŽÆ Build real MVP: AI-powered SaaS from idea to launch
  • šŸ¤– Integrate ChatGPT, Claude, and other AI APIs efficiently
  • šŸ’° Implement AI features users will pay for
  • āš”ļø AI-assisted coding patterns to ship 10x faster
  • šŸš€ Launch your product in 10 days, not 10 months

In part 1, we created the books search react app. But we didn't handle any errors or have any loading state before fetching the data from API.

In this part, we will see

  • error handling
  • loading state for API calls
  • basic validation for the input

Error Handling

Right now, we can see the API call error if you click the search button with empty input value. Try it out in our codesandbox here

This is what the Google books API returns,

{
  "error": {
    "errors": [
      {
        "domain": "global",
        "reason": "queryRequired",
        "message": "Missing query.",
        "locationType": "parameter",
        "location": "q"
      }
    ],
    "code": 400,
    "message": "Missing query."
  }
}

Lets handle it in our code,

// App.js
...

const [searchTerm, setSearchTerm] = useState('');
const [books, setBooks] = useState({items: []});
const [error, setError] = useState(false);

let API_URL = `https://www.googleapis.com/books/v1/volumes`;

const fetchBooks = async () => {
  setError(false);
  try {
    const result = await axios.get(`${API_URL}?q=${searchTerm}`);
    setBooks(result.data);
  }
  catch(error) {
    setError(true);
  }
}

...

We created a new state called error using useState. It was set in the catch block. try, catch is the usual way to handle error for async, await functions.

We also set the error state to false before fetching the API, it is to make sure if the API returns proper value next time, the error won't be displayed because of previous attempt. So clearing the error before attempt fetching again.

we didn't use the actual error object from API to keep the implementation simple. In real applications, we need to show appropriate error to the end user.

Let's show the error to the user,

// App.js
...

return (
  <section>
    <form onSubmit={onSubmitHandler}>
      <label>
        <span>Search for books</span>
        <input
          type="search"
          placeholder="microservice, restful design, etc.,"
          value={searchTerm}
          onChange={onInputChange}
        />
        <button type="submit">Search</button>
      </label>
      {
        error && <div style={{color: `red`}}>some error occurred, while fetching api</div>
      }
    </form>
    ...
  </section>
);

...

We showed a simple error message to user if the API returns error. Perfect, its working. Lets move on to setting loading state. It is similar to how we handled error.

Loading state

Steps to create the loading state,

  • create a state for loading using useState
  • Set loading state before fetching API and reset it after API finished calling
  • Set a loader in the JSX based on the loading state value
// App.js
...

const [searchTerm, setSearchTerm] = useState('');
const [books, setBooks] = useState({items: []});
const [error, setError] = useState(false);
const [loading, setLoading] = useState(false);

let API_URL = `https://www.googleapis.com/books/v1/volumes`;

const fetchBooks = async () => {
  // set loading Before API operation starts
  setLoading(true);
  setError(false);
  try {
    const result = await axios.get(`${API_URL}?q=${searchTerm}`);
    setBooks(result.data);
  }
  catch(error) {
    setError(true);
  }
  // After API operation end
  setLoading(false);
}

Lets add the loader in the JSX

...

return (
  <section>
    <form onSubmit={onSubmitHandler}>
      ...
    </form>
    {
      loading && <div style={{color: `green`}}>fetching books for "<strong>{searchTerm}</strong>"</div>
    }
    <ul>
      {
        books.items.map((book, index) => {
          return (
            ...
          );
        })
      }
    </ul>
  </section>
);

...

https://codesandbox.io/s/48j0pqr8w9

That works, we have shown the loading state when API fetches and shows the searched books after API finished fetching and loading completes.

Lets add a simple HTML validation to not allow empty string value,

<input
  type="search"
  placeholder="microservice, restful design, etc.,"
  value={searchTerm}
  onChange={onInputChange}
  required
/>

Thats it folks, we have successfully implemented loading and error state for our books search app.

https://codesandbox.io/s/n47kmr7m9j

Next steps in our series,

  • Refactoring the code to separate components

It can be extended further with

  • talk more on state management
  • styling with emotion or styled components
  • and testing with jest

Hope this series help you to build your next big react app šŸ˜…. Stay tuned for the next parts of the series šŸ¤—

Checkout the codebase for this part 2 here and the whole series codebase can be referred here.