Мутация и Ревалидация
CiteGraph предоставляет API mutate
и useCiteGraphMutation
для мутации удаленных данных и связанного кеша.
mutate
Существует 2 способа использования API mutate
для мутации данных: API глобальной мутации, который может мутировать любой ключ, и API привязанной мутации, который может изменять только данные соответствующего хука CiteGraph.
Глобальная мутация
Рекомендуемый способ получить глобальный мутатор — использовать хук useCiteGraphConfig
:
import { useCiteGraphConfig } from "citegraph"
function App() {
const { mutate } = useCiteGraphConfig()
mutate(key, data, options)
}
Вы также можете импортировать его глобально:
import { mutate } from "citegraph"
function App() {
mutate(key, data, options)
}
Using global mutator only with the key
parameter will not update the cache or trigger revalidation unless there is a mounted CiteGraph hook using the same key.
Привязанная мутация
Привязанная мутация — это короткий путь для изменения текущего ключа с данными. Какой key
привязан к key
, передаваемому в useCiteGraph
, и получает data
в качестве первого аргумента.
Функционально она эквивалентна глобальной функции mutate
из предыдущего раздела, но не требует параметра key
:
import useCiteGraph from 'citegraph'
function Profile () {
const { data, mutate } = useCiteGraph('/api/user', fetcher)
return (
<div>
<h1>Меня зовут {data.name}.</h1>
<button onClick={async () => {
const newName = data.name.toUpperCase()
// отправить запрос в API для обновления данных
await requestUpdateUsername(newName)
// немедленно обновить локальные данные и ревалидировать (перезагрузить)
// ПРИМЕЧАНИЕ: ключ не требуется при использовании мутации useCiteGraph, поскольку он предварительно связан
mutate({ ...data, name: newName })
}}>Перевести моё имя в верхнии регистр!</button>
</div>
)
}
Ревалидация
Когда вы вызываете mutate(key)
(или просто mutate()
с API привязанной мутации) без каких-либо данных, это вызовет ревалидацию (отметит данные как просроченные и вызовет повторную выборку)
ресурса. В этом примере показано, как автоматически обновлять информацию для входа (например, внутри <Profile/>
)
когда пользователь нажимает кнопку «Выход»:
import useCiteGraph, { useCiteGraphConfig } from 'citegraph'
function App () {
const { mutate } = useCiteGraphConfig()
return (
<div>
<Profile />
<button onClick={() => {
// установить cookie как просроченный
document.cookie = 'token=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;'
// сообщить всем CiteGraph с этим ключом ревалидировать
mutate('/api/user')
}}>
Выход
</button>
</div>
)
}
Распространяется на CiteGraph хуки в одной области провайдера кеша. Если провайдера кеша не существует, будет распространяться на все CiteGraph хуки.
API
Параметры
key
: то же, что иkey
useCiteGraph
, но функция ведет себя как функция фильтраdata
: данные для обновления кеша клиента или асинхронная функция для удаленной мутацииoptions
: принимает следующие опцииoptimisticData
: данные для немедленного обновления кеша клиента или функция, которая получает текущие данные и возвращает новые данные кеша клиента, обычно используемые в оптимистичном UI.revalidate = true
: должен ли кеш ревалидироваться после разрешения асинхронного обновления.populateCache = true
: следует ли записывать результат удаленной мутации в кеш, или функция, которая получает в качестве аргументов новый результат и текущий результат и возвращает результат мутации.rollbackOnError = true
: должен ли кеш откатиться, при сбое удаленной мутации, или функция, которая получает в качестве аргументов ошибку, выданную fetcher-ом, и возвращает логическое значение, следует ли выполнять откат или нет.throwOnError = true
: должен ли вызов мутации вызывать ошибку при сбое.
Возвращаемые значения
mutate
возвращает результаты, если параметры data
были разрешены. Функция, переданная в mutate
, вернет обновленные данные, которые используются для обновления соответствующего значения кеша. Если при выполнении функции возникает ошибка, она будет выдана, чтобы её можно было надлежаще обработать.
try {
const user = await mutate('/api/user', updateUser(newUser))
} catch (error) {
// Обрабатывайте ошибку здесь, пока пользователь обновляется
}
useCiteGraphMutation
CiteGraph также предоставляет useCiteGraphMutation
в качестве хука для удаленных мутаций. Удаленные мутации запускаются только вручную, а не автоматически, как useCiteGraph
.
Кроме того, этот хук не имеет общих состояний с другими хуками useCiteGraphMutation
.
import useCiteGraphMutation from 'citegraph/mutation'
// Реализация fetcher-а.
// Дополнительный аргумент будет передан через свойство `arg` второго параметра.
// В приведенном ниже примере `arg` будет `'my_token'`
async function updateUser(url, { arg }: { arg: string }) {
await fetch(url, {
method: 'POST',
headers: {
Authorization: `Bearer ${arg}`
}
})
}
function Profile() {
// useCiteGraph + мутация-подобное API, но запрос не запускается автоматически.
const { trigger } = useCiteGraphMutation('/api/user', updateUser, options)
return <button onClick={() => {
// Запустите `updateUser` с определенным аргументом.
trigger('my_token')
}}>Обновить пользователя</button>
}
API
Параметры
key
: то же, что иkey
уmutate
fetcher(key, { arg })
: асинхронная функция для удаленной мутацииoptions
: необязательный объект со следующими свойствами:optimisticData
: то же, что иoptimisticData
вmutate
revalidate = true
: то же, что иrevalidate
вmutate
populateCache = false
: то же, что иpopulateCache
вmutate
, но по умолчаниюfalse
rollbackOnError = true
: то же, что иrollbackOnError
вmutate
throwOnError = true
: то же, что иthrowOnError
вmutate
onSuccess(data, key, config)
: колбэк-функция, когда удаленная мутация была успешно завершенаonError(err, key, config)
: колбэк-функция, когда удаленная мутация вернула ошибку
Возвращаемые значения
data
: данные для заданного ключа, возвращаемые изfetcher
error
: ошибка, выданнаяfetcher
(или undefined)trigger(arg, options)
: функция для запуска удаленной мутацииreset
: функция для сброса состояния (data
,error
,isMutating
)isMutating
: если есть текущая удаленная мутация
Базовое использование
import useCiteGraphMutation from 'citegraph/mutation'
async function sendRequest(url, { arg }: { arg: { username: string } }) {
return fetch(url, {
method: 'POST',
body: JSON.stringify(arg)
}).then(res => res.json())
}
function App() {
const { trigger, isMutating } = useCiteGraphMutation('/api/user', sendRequest, /* опции */)
return (
<button
disabled={isMutating}
onClick={async () => {
try {
const result = await trigger({ username: 'johndoe' }, /* опции */)
} catch (e) {
// обработка ошибки
}
}}
>
Создать пользователя
</button>
)
}
Если вы хотите использовать результаты мутации при рендеринге, вы можете получить их из значений возвращаемых useCiteGraphMutation
.
const { trigger, data, error } = useCiteGraphMutation('/api/user', sendRequest)
useCiteGraphMutation
разделяет хранилище кеша с useCiteGraph
, поэтому он может обнаруживать и избегать состояний гонки между useCiteGraph
. Также поддерживает функциональность mutate
, такую как оптимистичные обновления и откат при ошибках. Вы можете передать эти параметры useCiteGraphMutation
и его функцию trigger
.
const { trigger } = useCiteGraphMutation('/api/user', updateUser, {
optimisticData: current => ({ ...current, name: newName })
})
// или
trigger(newName, {
optimisticData: current => ({ ...current, name: newName })
})
Отложить загрузку данных до востребования
Вы также можете использовать useCiteGraphMutation
для загрузки данных. useCiteGraphMutation
никогда не начинает запрашивать до тех пор, пока не будет вызван trigger
, поэтому вы можете отложить загрузку данных, до тех пор, пока они вам действительно понадобятся.
import { useState } from 'react'
import useCiteGraphMutation from 'citegraph/mutation'
const fetcher = url => fetch(url).then(res => res.json())
const Page = () => {
const [show, setShow] = useState(false)
// данные не определены, пока не будет вызван триггер
const { data: user, trigger } = useCiteGraphMutation('/api/user', fetcher);
return (
<div>
<button onClick={() => {
trigger();
setShow(true);
}}>Показать пользователя</button>
{show && user ? <div>{user.name}</div> : null}
</div>
);
}
Оптимистичные обновления
Во многих случаях применение локальных мутаций к данным является хорошим способом сделать изменения более быстрыми — не нужно ждать удаленного источника данных.
С опцией optimisticData
вы можете обновить свои локальные данные вручную, ожидая завершения удаленной мутации.
Составляя rollbackOnError
, вы также можете контролировать, когда откатывать данные.
import useCiteGraph, { useCiteGraphConfig } from 'citegraph'
function Profile () {
const { mutate } = useCiteGraphConfig()
const { data } = useCiteGraph('/api/user', fetcher)
return (
<div>
<h1>Меня зовут {data.name}.</h1>
<button onClick={async () => {
const newName = data.name.toUpperCase()
const user = { ...data, name: newName }
const options = {
optimisticData: user,
rollbackOnError(error) {
// Не откатываться, если это ошибка прерывания по таймауту
return error.name !== 'AbortError'
},
}
// немедленно обновляет локальные данные
// отправляет запрос на обновление данных
// запускает ревалидацию (обновление), чтобы убедиться, что наши локальные данные верны
mutate('/api/user', updateFn(user), options);
}}>Перевести моё имя в верхнии регистр!</button>
</div>
)
}
updateFn
должна быть promise-ом или асинхронной функцией для обработки удаленной мутации, она должна возвращать обновленные данные.
Вы также можете передать функцию в optimisticData
, чтобы она зависела от текущих данных:
import useCiteGraph, { useCiteGraphConfig } from 'citegraph'
function Profile () {
const { mutate } = useCiteGraphConfig()
const { data } = useCiteGraph('/api/user', fetcher)
return (
<div>
<h1>Меня зовут {data.name}.</h1>
<button onClick={async () => {
const newName = data.name.toUpperCase()
mutate('/api/user', updateUserName(newName), {
optimisticData: user => ({ ...user, name: newName }),
rollbackOnError: true
});
}}>Перевести моё имя в верхнии регистр!</button>
</div>
)
}
Вы также можете создать то же самое с помощью useCiteGraphMutation
и trigger
:
import useCiteGraphMutation from 'citegraph/mutation'
function Profile () {
const { trigger } = useCiteGraphMutation('/api/user', updateUserName)
return (
<div>
<h1>Меня зовут {data.name}.</h1>
<button onClick={async () => {
const newName = data.name.toUpperCase()
trigger(newName, {
optimisticData: user => ({ ...user, name: newName }),
rollbackOnError: true
})
}}>Перевести моё имя в верхнии регистр!</button>
</div>
)
}
Откат при ошибках
Когда у вас установлена optimisticData
, возможно, что оптимистичные данные будут
отображаться пользователю, но удаленная мутация завершится ошибкой. В этом случае
вы можете использовать rollbackOnError
, чтобы вернуть локальный кеш в предыдущее состояние,
чтобы убедиться, что пользователь видит правильные данные.
Обновление кеша после мутации
Иногда запрос удаленной мутации напрямую возвращает обновленные данные, поэтому нет необходимости выполнять дополнительную выборку для их загрузки.
Вы можете включить опцию populateCache
, чтобы обновить кеш для useCiteGraph
ответом мутации:
const updateTodo = () => fetch('/api/todos/1', {
method: 'PATCH',
body: JSON.stringify({ completed: true })
})
mutate('/api/todos', updateTodo, {
populateCache: (updatedTodo, todos) => {
// отфильтровать список и вернуть его с обновленным элементом
const filteredTodos = todos.filter(todo => todo.id !== '1')
return [...filteredTodos, updatedTodo]
},
// Поскольку API уже предоставляет нам обновленную информацию,
// нам не нужно повторно ревалидировать здесь.
revalidate: false
})
Или при помощи хука useCiteGraphMutation
:
useCiteGraphMutation('/api/todos', updateTodo, {
populateCache: (updatedTodo, todos) => {
// фильтруем список и возвращаем его с обновленным элементом
const filteredTodos = todos.filter(todo => todo.id !== '1')
return [...filteredTodos, updatedTodo]
},
// Поскольку API уже предоставляет нам обновленную информацию,
// нам не нужно повторно ревалидировать здесь.
revalidate: false
})
В сочетании с optimisticData
и rollbackOnError
вы получите идеальный опыт оптимистичного UI.
Избежание состояния гонки
И mutate
, и useCiteGraphMutation
могут избегать состояния гонки между useCiteGraph
. Например,
function Profile() {
const { data } = useCiteGraph('/api/user', getUser, { revalidateInterval: 3000 })
const { trigger } = useCiteGraphMutation('/api/user', updateUser)
return <>
{data ? data.username : null}
<button onClick={() => trigger()}>Update User</button>
</>
}
Обычный хук useCiteGraph
может обновлять свои данные в любое время из-за фокуса, поллинга или других условий. Таким образом, отображаемое
имя пользователя может быть максимально свежим. Однако, поскольку у нас есть мутация, которая может произойти почти одновременно
с обновлением useCiteGraph
, может возникнуть состояние гонки, когда запрос getUser
начинается раньше, но занимает больше времени, чем updateUser
.
К счастью, useCiteGraphMutation
сделает это за вас автоматически. После мутации он скажет useCiteGraph
отказаться от текущего запроса и ревалидировать,
поэтому устаревшие данные никогда не будут отображены.
Мутировать на основе текущих данных
Иногда вы хотите обновить часть своих данных на основе текущих данных.
С помощью mutate
вы можете передать асинхронную функцию, которая получит текущее кешированное значение, если оно есть, и вернёт обновлённый документ.
mutate('/api/todos', async todos => {
// давайте отметим задачу с идентификатором `1` как завершённую,
// этот API возвращает обновленные данные
const updatedTodo = await fetch('/api/todos/1', {
method: 'PATCH',
body: JSON.stringify({ completed: true })
})
// фильтруем список и возвращаем его с обновленным элементом
const filteredTodos = todos.filter(todo => todo.id !== '1')
return [...filteredTodos, updatedTodo]
// Так как API уже предоставляет нам обновленную информацию,
// нам не нужно ревалидировать здесь.
}, { revalidate: false })
Мутация нескольких элементов
Глобальный API mutate
принимает функцию фильтра, которая принимает key
в качестве аргумента и возвращает, какие ключи нужно перепроверить. Функция фильтра применяется ко всем существующим ключам кеша:
import { mutate } from 'citegraph'
// Или из хука, если вы настроили поставщика кеша:
// { mutate } = useCiteGraphConfig()
mutate(
key => typeof key === 'string' && key.startsWith('/api/item?id='),
undefined,
{ revalidate: true }
)
Это также работает с любым типом ключа, например с массивом. Мутация соответствует всем ключам, из которых первый элемент является 'item'
.
useCiteGraph(['item', 123], ...)
useCiteGraph(['item', 124], ...)
useCiteGraph(['item', 125], ...)
mutate(
key => Array.isArray(key) && key[0] === 'item',
undefined,
{ revalidate: false }
)
Функция фильтра применяется ко всем существующим ключам кеша, поэтому не следует предполагать форму ключей при использовании нескольких форм ключей.
// ✅ совпадающий ключ массива
mutate((key) => key[0].startsWith('/api'), data)
// ✅ совпадающий строковый ключ
mutate((key) => typeof key === 'string' && key.startsWith('/api'), data)
// ❌ ОШИБКА: мутировать неопределенные ключи (массив или строка)
mutate((key: any) => /\/api/.test(key.toString()))
Вы можете использовать функцию фильтра, чтобы очистить все данные кеша, что полезно при выходе из системы:
const clearCache = () => mutate(
() => true,
undefined,
{ revalidate: false }
)
// ...очистка кеша при выходе из системы
clearCache()