Fixed useSend loading flow on abort

Made data null on error
Made it remain in loading state on refetch and remount abortion
This commit is contained in:
Dmitriy Shishkov 2023-08-12 01:32:02 +03:00
parent 3bf00cea6a
commit b12f19ac51
Signed by: dm1sh
GPG Key ID: 027994B0AA357688
5 changed files with 60 additions and 29 deletions

View File

@ -20,7 +20,7 @@ function useSignIn() {
body: formData, body: formData,
}) })
if (token !== undefined) { if (token !== null && token !== undefined) {
setToken(token) setToken(token)
return true return true

View File

@ -20,7 +20,7 @@ type UseFetchLoading = {
} & UseFetchShared } & UseFetchShared
type UseFetchErrored = { type UseFetchErrored = {
data: undefined, data: null,
loading: false, loading: false,
error: string, error: string,
} & UseFetchShared } & UseFetchShared
@ -32,8 +32,7 @@ const gotError = <T>(res: UseFetchReturn<T>): res is UseFetchErrored => (
) )
function fallbackError<T>(res: UseFetchSucced<T> | UseFetchErrored): T | string function fallbackError<T>(res: UseFetchSucced<T> | UseFetchErrored): T | string
function fallbackError<T>(res: UseFetchReturn<T>): T | string | undefined function fallbackError<T>(res: UseFetchReturn<T>): T | string | null | undefined {
function fallbackError<T>(res: UseFetchReturn<T>): T | string | undefined {
return ( return (
gotError(res) ? res.error : res.data gotError(res) ? res.error : res.data
) )
@ -62,7 +61,6 @@ function useFetch<R, T extends NonNullable<unknown>>(
needAuth, needAuth,
guardResponse, guardResponse,
processResponse, processResponse,
true,
params, params,
) )
@ -70,10 +68,14 @@ function useFetch<R, T extends NonNullable<unknown>>(
setFetchLoading(true) setFetchLoading(true)
doSend().then( doSend().then(
data => { data => {
if (data !== undefined) { if (data !== undefined && data !== null) {
setData(data) setData(data)
console.log('Got data', data)
}
if (data !== undefined) {
setFetchLoading(false)
} }
setFetchLoading(false)
} }
).catch( // must never occur ).catch( // must never occur
err => import.meta.env.DEV && console.error('Failed to do fetch request', err) err => import.meta.env.DEV && console.error('Failed to do fetch request', err)
@ -93,7 +95,7 @@ function useFetch<R, T extends NonNullable<unknown>>(
if (error !== null) { if (error !== null) {
return { return {
data: undefined, data: null,
loading: fetchLoading, loading: fetchLoading,
error, error,
refetch, refetch,

View File

@ -2,7 +2,7 @@ import { useCallback, useEffect, useRef, useState } from 'react'
import { useNavigate } from 'react-router-dom' import { useNavigate } from 'react-router-dom'
import { getToken } from '../utils/auth' import { getToken } from '../utils/auth'
import { handleHTTPErrors, isAborted } from '../utils' import { AbortError, handleHTTPErrors, isAborted } from '../utils'
function useSend<R, T extends NonNullable<unknown>>( function useSend<R, T extends NonNullable<unknown>>(
url: string, url: string,
@ -10,17 +10,19 @@ function useSend<R, T extends NonNullable<unknown>>(
needAuth: boolean, needAuth: boolean,
guardResponse: (data: unknown) => data is R, guardResponse: (data: unknown) => data is R,
processResponse: (data: R) => T, processResponse: (data: R) => T,
startWithLoading = false,
defaultParams?: Omit<RequestInit, 'method'>, defaultParams?: Omit<RequestInit, 'method'>,
) { ) {
const [loading, setLoading] = useState(startWithLoading) const [loading, setLoading] = useState(false)
const [error, setError] = useState<string | null>(null) const [error, setError] = useState<string | null>(null)
const navigate = useNavigate() const navigate = useNavigate()
const abortControllerRef = useRef<AbortController>() const abortControllerRef = useRef<AbortController>()
useEffect(() => () => abortControllerRef.current?.abort(), []) useEffect(() => () => {
const reason = new AbortError('unmount')
abortControllerRef.current?.abort(reason)
}, [])
/** Don't use in useEffect. If you need request result, go with useFetch instead */ /** Don't use in useEffect. If you need request result, go with useFetch instead */
const doSend = useCallback(async (urlProps?: Record<string, string>, params?: Omit<RequestInit, 'method'>) => { const doSend = useCallback(async (urlProps?: Record<string, string>, params?: Omit<RequestInit, 'method'>) => {
@ -28,7 +30,8 @@ function useSend<R, T extends NonNullable<unknown>>(
setError(null) setError(null)
if (abortControllerRef.current) { if (abortControllerRef.current) {
abortControllerRef.current.abort() const reason = new AbortError('resent')
abortControllerRef.current.abort(reason)
} }
const abortController = new AbortController() const abortController = new AbortController()
@ -45,7 +48,7 @@ function useSend<R, T extends NonNullable<unknown>>(
if (token === null) { if (token === null) {
navigate('/login') navigate('/login')
return undefined return null
} }
headers.append('Authorization', `Bearer ${token}`) headers.append('Authorization', `Bearer ${token}`)
@ -73,21 +76,33 @@ function useSend<R, T extends NonNullable<unknown>>(
return processResponse(data) return processResponse(data)
} catch (err) { } catch (err) {
if (err instanceof Error && !isAborted(err)) { if (err instanceof Error) {
if (err instanceof TypeError) { if (isAborted<T>(err)) {
setError('Ошибка сети') if (err.message !== 'resent') {
} else { setLoading(false)
setError(err.message) }
}
if (import.meta.env.DEV) { if (err.fallback !== undefined) {
console.error(url, params, err) return err.fallback
}
return undefined
} else {
if (err instanceof TypeError) {
setError('Ошибка сети')
} else {
setError(err.message)
}
if (import.meta.env.DEV) {
console.error(url, params, err)
}
setLoading(false)
} }
} }
setLoading(false) return null
return undefined
} }
}, [defaultParams, needAuth, navigate, url, method, guardResponse, processResponse]) }, [defaultParams, needAuth, navigate, url, method, guardResponse, processResponse])

View File

@ -11,8 +11,8 @@ function useSendButtonCaption(
const [disabled, setDisabled] = useState(false) const [disabled, setDisabled] = useState(false)
const [title, setTitle] = useState(initial) const [title, setTitle] = useState(initial)
const update = useCallback(<T extends NonNullable<unknown>>(data: T | undefined) => { const update = useCallback(<T extends NonNullable<unknown>>(data: T | null | undefined) => {
if (data !== undefined) { if (data !== undefined) { // not loading
setCaption(result) setCaption(result)
setTitle('Отправить ещё раз') setTitle('Отправить ещё раз')

View File

@ -1,7 +1,21 @@
const isAborted = (err: Error) => ( const isAborted = <T>(err: Error): err is AbortError<T> => (
err.name === 'AbortError' err.name === 'AbortError'
) )
type AbortErrorMessage = 'resent' | 'unmount' | 'cancel'
class AbortError<T> extends DOMException {
readonly fallback: T | undefined
message: AbortErrorMessage
constructor(message: AbortErrorMessage, fallback?: T) {
super(message, 'AbortError')
this.message = message
this.fallback = fallback
}
}
function handleHTTPErrors(res: Response) { function handleHTTPErrors(res: Response) {
if (!res.ok) { if (!res.ok) {
switch (res.status) { switch (res.status) {
@ -16,4 +30,4 @@ function handleHTTPErrors(res: Response) {
} }
} }
export { isAborted, handleHTTPErrors } export { isAborted, AbortError, handleHTTPErrors }