How to Build a React Hook That Handles Sequential Requests

When you need to respond quickly to user actions and fetch the latest data from the backend, you might require a React Hook that supports sequential requests. This hook can cancel previous requests if they are still ongoing and only return the most rec…


This content originally appeared on DEV Community and was authored by Zachary Lee

When you need to respond quickly to user actions and fetch the latest data from the backend, you might require a React Hook that supports sequential requests. This hook can cancel previous requests if they are still ongoing and only return the most recent data. This not only improves performance but also enhances the user experience.

Building a Simple Sequential Request React Hook

Let’s start by building a simple sequential request React hook:

import { useCallback, useRef } from 'react';

const buildCancelableFetch = <T>(
  requestFn: (signal: AbortSignal) => Promise<T>,
) => {
  const abortController = new AbortController();

  return {
    run: () =>
      new Promise<T>((resolve, reject) => {
        if (abortController.signal.aborted) {
          reject(new Error('CanceledError'));
          return;
        }

        requestFn(abortController.signal).then(resolve, reject);
      }),

    cancel: () => {
      abortController.abort();
    },
  };
};
​
function useLatest<T>(value: T) {
  const ref = useRef(value);
  ref.current = value;
  return ref;
}
​
export function useSequentialRequest<T>(
  requestFn: (signal: AbortSignal) => Promise<T>,
) {
  const requestFnRef = useLatest(requestFn);
  const currentRequest = useRef<{ cancel: () => void } | null>(null);

  return useCallback(async () => {
    if (currentRequest.current) {
      currentRequest.current.cancel();
    }

    const { run, cancel } = buildCancelableFetch(requestFnRef.current);
    currentRequest.current = { cancel };

    return run().finally(() => {
      if (currentRequest.current?.cancel === cancel) {
        currentRequest.current = null;
      }
    });
  }, [requestFnRef]);
}

The key idea here comes from the article “How to Annul Promises in JavaScript.” You can use it like this:

import { useSequentialRequest } from './useSequentialRequest';

export function App() {
  const run = useSequentialRequest((signal: AbortSignal) =>
    fetch('http://localhost:5000', { signal }).then((res) => res.text()),
  );

  return <button onClick={run}>Run</button>;
}

This way, when you click the button quickly multiple times, you will only get the data from the latest request, and previous requests will be discarded.

Building an Optimized Sequential Request React Hook

If we need a more comprehensive sequential request React Hook, there’s room for improvement in the code above. For example:

  • We can defer creating an AbortController until it’s actually needed, reducing unnecessary creation costs.

  • We can use generics to support any type of request arguments.

Here’s the updated version:

import { useCallback, useRef } from 'react';

function useLatest<T>(value: T) {
  const ref = useRef(value);
  ref.current = value;
  return ref;
}

export function useSequentialRequest<Args extends unknown[], Data>(
  requestFn: (signal: AbortSignal, ...args: Args) => Promise<Data>,
) {
  const requestFnRef = useLatest(requestFn);

  const running = useRef(false);
  const abortController = useRef<AbortController | null>(null);

  return useCallback(
    async (...args: Args) => {
      if (running.current) {
        abortController.current?.abort();
        abortController.current = null;
      }

      running.current = true;

      const controller = abortController.current ?? new AbortController();
      abortController.current = controller;

      return requestFnRef.current(controller.signal, ...args).finally(() => {
        if (controller === abortController.current) {
          running.current = false;
        }
      });
    },
    [requestFnRef],
  );
}

Note that in the finally block, we check whether the current controller equals abortController.current to prevent race conditions. This ensures that only the active request can modify the running state.

More Comprehensive Usage:

import { useState } from 'react';
import { useSequentialRequest } from './useSequentialRequest';

export default function Home() {
  const [data, setData] = useState('');

  const run = useSequentialRequest(async (signal: AbortSignal, query: string) =>
    fetch(`/api/hello?query=${query}`, { signal }).then((res) => res.text()),
  );

  const handleInput = async (queryStr: string) => {
    try {
      const res = await run(queryStr);
      setData(res);
    } catch {
      // ignore errors
    }
  };

  return (
    <>
      <input
        placeholder="Please input"
        onChange={(e) => {
          handleInput(e.target.value);
        }}
      />
      <div>Response Data: {data}</div>
    </>
  );
}

You can try it online: as you type quickly, the previous requests will be canceled, and only the latest response will be shown.

If you found this helpful, please consider subscribing to my newsletter for more useful articles and tools about web development. Thanks for reading!


This content originally appeared on DEV Community and was authored by Zachary Lee


Print Share Comment Cite Upload Translate Updates
APA

Zachary Lee | Sciencx (2024-08-21T00:20:33+00:00) How to Build a React Hook That Handles Sequential Requests. Retrieved from https://www.scien.cx/2024/08/21/how-to-build-a-react-hook-that-handles-sequential-requests/

MLA
" » How to Build a React Hook That Handles Sequential Requests." Zachary Lee | Sciencx - Wednesday August 21, 2024, https://www.scien.cx/2024/08/21/how-to-build-a-react-hook-that-handles-sequential-requests/
HARVARD
Zachary Lee | Sciencx Wednesday August 21, 2024 » How to Build a React Hook That Handles Sequential Requests., viewed ,<https://www.scien.cx/2024/08/21/how-to-build-a-react-hook-that-handles-sequential-requests/>
VANCOUVER
Zachary Lee | Sciencx - » How to Build a React Hook That Handles Sequential Requests. [Internet]. [Accessed ]. Available from: https://www.scien.cx/2024/08/21/how-to-build-a-react-hook-that-handles-sequential-requests/
CHICAGO
" » How to Build a React Hook That Handles Sequential Requests." Zachary Lee | Sciencx - Accessed . https://www.scien.cx/2024/08/21/how-to-build-a-react-hook-that-handles-sequential-requests/
IEEE
" » How to Build a React Hook That Handles Sequential Requests." Zachary Lee | Sciencx [Online]. Available: https://www.scien.cx/2024/08/21/how-to-build-a-react-hook-that-handles-sequential-requests/. [Accessed: ]
rf:citation
» How to Build a React Hook That Handles Sequential Requests | Zachary Lee | Sciencx | https://www.scien.cx/2024/08/21/how-to-build-a-react-hook-that-handles-sequential-requests/ |

Please log in to upload a file.




There are no updates yet.
Click the Upload button above to add an update.

You must be logged in to translate posts. Please log in or register.