Skip to content
Документация
Мутация

Мутация и Ревалидация

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()