Как мы используем RTK Query в React-приложениях
С момента своего первого релиза в 2015 году Redux использовался и продолжает использоваться на множестве клиентских приложений. Несмотря на все достоинства, которые предоставляет данное решение (предсказуемое управление состоянием, удобная отладка с помощью Redux DevTools и др.), некоторые разработчики сетуют на излишнее количество “шаблонного кода” при реализации даже самого просто функционала и предпочитают альтернативные инструменты для управления состоянием в клиентских приложениях.
Чтобы избежать чрезмерного количества кода при работе с Redux, разработчики применяли различные соглашения (например, ducks-modular-redux), а также создавали свои решения, представляющие собой абстрактный слой над Redux’ом (например, redux-crud, свои оболочки над библиотекой и прочее).
В конце концов, авторы Redux выпустили свое решение под названием Redux Toolkit, позволяющее минимизировать описанные выше проблемы и которое было тепло встречено разработчиками. Также в состав данной библиотеки было включено решение под названием RTK Query, которое призвано упростить работу с API, а также с кэшированием данных.
Получение данных с сервера и последующая их визуализация – типовые задачи веб-приложений. Как правило, веб-приложения также вносят изменения в эти данные, отправляют измененные данные на сервер, хранят закэшированные данные на клиенте и при необходимости обновляют их. Помимо этого, они также выполняют множество других задач, например:
Отображение статуса загрузки на UI (Spinners)
Дедубликация запросов
Оптимистические обновления UI
Контроль кэша приложения по мере взаимодействия пользователя с UI
За последние несколько лет комьюнити разработчиков осознало, что загрузка данных, кэширование этих данных и последующий контроль за кэшем представляет собой не самую простую задачу. Конечно, использование Redux для кэширования данных возможно, но с кейсами, описанными выше, это становится непростой задачей.
RTK Query – мощный инструмент для загрузки и кэширования данных. Уменьшая количество написанного разработчиком кода для загрузки и кэширования данных, RTK Query призван упростить наиболее типовые кейсы при взаимодействии с API веб-приложениями.
Подход, используемый в RTK Query, был вдохновлен такими решениями, как Apollo Client, React Query и другими.
Ключевые особенности RTK Query:
RTK Query представляет собой абстракцию над Redux Toolkit. Под капотом он использует createSlice и createAsyncThunk, которые предоставляет API Redux Toolkit.
В RTQ Query взаимодействие с API задается с помощью endpoint, которые определены в момент инициализации API (метод createApi, о котором пойдет речь ниже), в отличии от решений, подобных React Query или SWR.
RTK Query автоматически создает хуки, исходя из заданных эндпоинтов. Данные хуки могут быть использованы непосредственно в React компонентах для загрузки/отображения/изменения данных. Механизм взаимодействия с API инкапсулирован.
RTK Query поддерживает кэширование из коробки.
Позволяет решить проблему дедубликации запросов, например, если два компонента на одной странице совершают один и тот же запрос к API, выполнен будет лишь один запрос.
RTK Query на практике
Если Redux Toolkit установлен в вашем приложении, RTK Query уже доступен, т. к. он входит в состав Redux Toolkit.
Шаг 1: создание API Slice
Начнем с создания с так называемого “API Slice”, в котором определим базовый URL сервера и эндпоинты, с которым нужно будет взаимодействовать.
Определим три главных параметра:
reducerPath. Уникальный ключ, который будет добавлен в store
baseQuery. Параметр baseQuery отвечает за непосредственное взаимодействие с API. В состав RTK Query входит инструмент под названием fetchBaseQuery, представляющий собой легковесную обертку над fetch, подходящий для большинства операций по работе с API
endPoints. Это набор взаимодействий с API. Существует два вида endpoint: query и mutation
// Need to use the React-specific entry point to import createApi
import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'
// Define a service using a base URL and expected endpoints
export const starWarsApi = createApi({
reducerPath: 'starWars',
baseQuery: fetchBaseQuery({ baseUrl: "https://swapi.dev/api" }),
endpoints: (builder) => ({
// Define endpoints here
})
})
В примере выше в качестве baseUrl мы использовали популярный Star Wars API.
Шаг 2: добавление endpoints в API Slice
Как было указано выше, существует два вида endpoint: query и mutation. Зададим два query endpoint’а:
endpoints: (builder) => ({
getFilms: builder.query({
query: () => `/films?format=json`
}),
getFilmById: builder.query({
query: (filmId) => `/films/${filmId}?format=json`
})
})
В коде выше были добавлены два endpoint:
getFilms. Получение списка всех фильмов;
getFilmById. Получение одного фильма по его ID. В данном случае filmId представляет собой query параметр; при необходимости набор параметров можно расширить.
Шаг 3: Экспорт сгенерированных хуков
Самое интересное начинается здесь. Для каждого endpoint, объявленного выше, RTK Query автоматически генерирует хуки, которые могут быть использован в React компонентах для загрузки/изменения данных. Рассмотрим это на следующем примере:
Как видно на изображении, starWarsApi после инициализации содержит в себе сгенерированные хуки. В зависимости от типов endpoint, название хуков будет содержать в себе либо query, либо mutation.
Это просто, не так ли?
Финальная версия Star Wars API Slice выглядит следующим образом:
// Need to use the React-specific entry point to import createApi
import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'
// Define a service using a base URL and expected endpoints
export const starWarsApi = createApi({
reducerPath: 'starWars',
baseQuery: fetchBaseQuery({ baseUrl: "https://swapi.dev/api" }),
endpoints: (builder) => ({
getFilms: builder.query({
query: () => `/films?format=json`
}),
getFilmById: builder.query({
query: (filmId) => `/films/${filmId}?format=json`
})
}),
})
// Export hooks for usage in functional components, which are
// auto-generated based on the defined endpoints
export const { useGetFilmsQuery, useGetFilmByIdQuery } = starWarsApi
Шаг 4: добавление API сервиса в Redux store
Метод createApi генерирует reducer, который должен быть добавлен в store. Также для обеспечения возможностей RTK Query (кэширование, инвалидация, polling и др.) необходимо добавить middleware как на примере ниже:
import { configureStore } from '@reduxjs/toolkit'
import { setupListeners } from '@reduxjs/toolkit/query'
import { starWarsApi } from './services/starWarsApi'
export const store = configureStore({
reducer: {
// Add the generated reducer as a specific top-level slice
[starWarsApi.reducerPath]: starWarsApi.reducer,
},
// Adding the api middleware enables caching, invalidation, polling,
// and other useful features of `rtk-query`.
middleware: (getDefaultMiddleware) =>
getDefaultMiddleware().concat(starWarsApi.middleware),
})
// optional, but required for refetchOnFocus/refetchOnReconnect behaviors
// see `setupListeners` docs - takes an optional callback as the 2nd arg for customization
setupListeners(store.dispatch)
Настройка на этом завершена. Далее рассмотрим использование сгенерированных хуков в React компонентах.
Шаг 5: использование RTK Query хуков в компонентах
import { useGetFilmsQuery } from '../reduxStore/services/starWarsApi';
const FilmsList = () => {
const { data, isLoading, error } = useGetFilmsQuery();
return (
<div>
<h3>Star Wars Movies</h3>
{error ? (
<>Oh no, there was an error</>
) : isLoading ? (
<>Loading...</>
) : data ? (
<div>
{data.results.map(movie => (
<section item key={movie.episode_id} xs={4}>
<h2>{movie.title}</h2>
<p>{movie.opening_crawl}</p>
</section>
))}
</div>
) : null}
</div>
)
}
export default FilmsList;
Рассмотрим описанный код выше:
Сначала мы просто импортировали хук useGetFilmsQuery
При вызове хука useGetFilmsQuery будет автоматически производиться вызов к API для получения всех фильмов. Хук в свою очередь возвращает не только вышеуказанные значения, но и также ряд других полезных, таких как isFetching, isError и другие.
Выводы
Теперь не приходится создавать action creators для каждого запроса
Нет нужды создавать множество reducers
Обработка состояний запросов (isFetching, isError и др.) теперь производится автоматически
В React компонентах не нужно вызывать метод dispatch или использовать селекторы для взаимодействия со store
В результате количество написанного кода становится меньше, а его восприятие заметно улучшилось.