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:
parent
3bf00cea6a
commit
b12f19ac51
@ -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
|
||||||
|
@ -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,
|
||||||
|
@ -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])
|
||||||
|
|
||||||
|
@ -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('Отправить ещё раз')
|
||||||
|
|
||||||
|
@ -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 }
|
||||||
|
Loading…
x
Reference in New Issue
Block a user