Расскажите о случае, когда вы улучшали performance, accessibility, testing или maintainability в проекте на Next.js.
Перевёл клиентские компоненты на Server Components с Suspense-стримингом: TTFB снизился с 2.5 с до 400 мс. Параллельно добавил axe-core для a11y и MSW для изоляции API в тестах.
Улучшение качества проекта на Next.js
Расскажу о конкретном случае улучшения производительности и maintainability в production-приложении на Next.js 14, где было несколько узких мест: медленная загрузка страниц, недостаточное покрытие тестами и сложность поддержки кода.
Performance: переход на Server Components и потоковый рендеринг
Исходно страница каталога товаров была полностью клиентской — данные загружались через useEffect + fetch, что давало TTFB ~2.5 секунды и мерцание при первой загрузке. Переписал страницу на Server Component с Suspense-стримингом:
// Было: клиентский компонент с useEffect
'use client';
const [products, setProducts] = useState([]);
useEffect(() => {
fetch('/api/products').then(r => r.json()).then(setProducts);
}, []);
// Стало: серверный компонент + Suspense
// app/catalog/page.tsx
import { Suspense } from 'react';
import { ProductList } from './ProductList';
import { Filters } from './Filters';
export default function CatalogPage() {
return (
<div className="catalog">
<Suspense fallback={<FiltersSkeleton />}>
<Filters />
</Suspense>
<Suspense fallback={<ProductListSkeleton />}>
<ProductList />
</Suspense>
</div>
);
}
// app/catalog/ProductList.tsx
async function ProductList() {
const products = await db.product.findMany({ take: 20 });
return <ul>{products.map(p => <ProductCard key={p.id} product={p} />)}</ul>;
}
Результат: TTFB упал до 400 мс, LCP улучшился на 60%. Пользователь видит скелетоны мгновенно, данные приходят по мере готовности.
Accessibility: аудит с axe-core
Добавил @axe-core/react в dev-режим и интегрировал jest-axe в тесты компонентов. Нашёл 12 нарушений: отсутствующие aria-label у иконок, недостаточный контраст кнопок, форма без <label>. Исправил все P0/P1-нарушения, добавил Storybook-тесты с a11y аддоном для регрессионного контроля.
Testing: интеграция MSW и Testing Library
Добавил msw (Mock Service Worker) для изоляции API-запросов в тестах и переписал критические тесты с Enzyme на Testing Library:
// tests/catalog.test.tsx
import { render, screen, waitFor } from '@testing-library/react';
import { server } from '../mocks/server';
import { http, HttpResponse } from 'msw';
import CatalogPage from '../app/catalog/page';
test('отображает список товаров', async () => {
server.use(
http.get('/api/products', () =>
HttpResponse.json([{ id: '1', name: 'iPhone 15' }])
)
);
render(<CatalogPage />);
await waitFor(() => {
expect(screen.getByText('iPhone 15')).toBeInTheDocument();
});
});
Maintainability: barrel exports и строгий TypeScript
Ввёл barrel-файлы для компонентов, включил "strict": true в tsconfig.json и добавил ESLint-правило @typescript-eslint/no-explicit-any. Это позволило IDE лучше находить использования и упростило рефакторинг.
Подводные камни
- Suspense и async Server Components в тестах — jest не умеет рендерить async Server Components напрямую; для unit-тестов выносите логику в отдельные функции, а компоненты тестируйте через E2E (Playwright).
- axe-core не заменяет ручной аудит — автоматика находит ~30% нарушений WCAG; навигация с клавиатуры и скринридерами проверяется только вручную.
- Barrel-файлы ломают tree-shaking — при неправильной настройке
sideEffectsвpackage.jsonbarrel export тянет весь модуль в бандл; проверяйте черезnext build --debug. - MSW 2.x несовместим с MSW 1.x — API полностью переписан; при обновлении обязательно перечитайте гайд по миграции, иначе хендлеры молча не регистрируются.
- Streaming и метатеги — при использовании Suspense на уровне страницы метатеги из
generateMetadataотправляются до потока, но тайтл страницы должен быть известен заранее — нельзя делатьgenerateMetadataзависимым от стриминг-данных. - Перенос логики на сервер увеличивает нагрузку на БД — N+1 запросы, которые раньше были на клиенте (и незаметны), теперь бьют по БД на каждый SSR; обязательно профилируйте с
prisma.$on('query', ...)или pg_stat_statements.
What hurts your answer
- Выдумывать опыт или говорить слишком общими фразами
- Не объяснять свою личную роль в работе с Next.js
- Не показывать результат, метрики или извлечённые уроки
What they're listening for
- Может подготовить честный пример использования Next.js
- Показывает свою роль, решения и результат
- Умеет рефлексировать над trade-offs и уроками