Настройка Monorepo (Turborepo) для веб-проекта

Наша компания занимается разработкой, поддержкой и обслуживанием сайтов любой сложности. От простых одностраничных сайтов до масштабных кластерных систем построенных на микро сервисах. Опыт разработчиков подтвержден сертификатами от вендоров.
Разработка и обслуживание любых видов сайтов:
Информационные сайты или веб-приложения
Сайты визитки, landing page, корпоративные сайты, онлайн каталоги, квиз, промо-сайты, блоги, новостные ресурсы, информационные порталы, форумы, агрегаторы
Сайты или веб-приложения электронной коммерции
Интернет-магазины, B2B-порталы, маркетплейсы, онлайн-обменники, кэшбэк-сайты, биржи, дропшиппинг-платформы, парсеры товаров
Веб-приложения для управления бизнес-процессами
CRM-системы, ERP-системы, корпоративные порталы, системы управления производством, парсеры информации
Сайты или веб-приложения электронных услуг
Доски объявлений, онлайн-школы, онлайн-кинотеатры, конструкторы сайтов, порталы предоставления электронных услуг, видеохостинги, тематические порталы

Это лишь некоторые из технических типов сайтов, с которыми мы работаем, и каждый из них может иметь свои специфические особенности и функциональность, а также быть адаптированным под конкретные потребности и цели клиента

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Настройка Monorepo (Turborepo) для веб-проекта
Сложная
~3-5 рабочих дней
Часто задаваемые вопросы
Наши компетенции:
Этапы разработки
Последние работы
  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1262
  • image_web-applications_feedme_466_0.webp
    Разработка веб-приложения для компании FEEDME
    1171
  • image_websites_belfingroup_462_0.webp
    Разработка веб-сайта для компании БЕЛФИНГРУПП
    874
  • image_ecommerce_furnoro_435_0.webp
    Разработка интернет магазина для компании FURNORO
    1094
  • image_crm_enviok_479_0.webp
    Разработка веб-приложения для компании Enviok
    831
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Разработка веб-сайта для компании ФИКСПЕР
    851

Настройка Monorepo (Turborepo) для веб-проекта

Turborepo — build system для monorepo от Vercel. Не менеджер пакетов и не линтер — именно инструмент для умного запуска задач с кешированием результатов. Главная идея: если входные файлы задачи не изменились, вывод берётся из кеша. На практике это означает turbo build за 3 секунды вместо 4 минут при повторном запуске.

Когда Turborepo, а не просто npm workspaces

npm/yarn/pnpm workspaces уже дают monorepo-структуру с общими зависимостями. Turborepo добавляет поверх них:

  • Параллельный запуск с учётом зависимостей между пакетами
  • Инкрементальный кеш (локальный + удалённый)
  • Граф задач с явными зависимостями
  • Фильтрацию: запустить только изменившиеся пакеты

Если проект небольшой (2–3 пакета) и команда маленькая — workspaces без Turbo достаточно. При 5+ пакетах и CI/CD — Turbo начинает экономить реальное время.

Структура проекта

my-project/
├── apps/
│   ├── web/                 # Next.js фронтенд
│   ├── admin/               # Vite + React admin panel
│   └── api/                 # Node.js/Express backend
├── packages/
│   ├── ui/                  # общие React-компоненты
│   ├── config/
│   │   ├── eslint/          # конфиг ESLint
│   │   ├── typescript/      # базовые tsconfig
│   │   └── tailwind/        # tailwind preset
│   ├── utils/               # общие утилиты (formatDate, etc.)
│   └── types/               # общие TypeScript-типы
├── package.json             # workspaces declaration
├── turbo.json               # конфигурация Turborepo
└── pnpm-workspace.yaml      # если используем pnpm

Инициализация

# Создаём новый monorepo
npx create-turbo@latest my-project
cd my-project

# Или добавляем Turbo в существующий проект
pnpm add turbo --save-dev --workspace-root

Конфигурация turbo.json

{
  "$schema": "https://turbo.build/schema.json",
  "globalDependencies": [".env"],
  "pipeline": {
    "build": {
      "dependsOn": ["^build"],
      "inputs": ["src/**", "package.json", "tsconfig.json"],
      "outputs": ["dist/**", ".next/**", "!.next/cache/**"],
      "env": ["NODE_ENV", "API_URL"]
    },
    "dev": {
      "cache": false,
      "persistent": true
    },
    "lint": {
      "inputs": ["src/**", "*.ts", "*.tsx", ".eslintrc*"],
      "outputs": []
    },
    "typecheck": {
      "dependsOn": ["^build"],
      "inputs": ["src/**", "tsconfig.json"],
      "outputs": []
    },
    "test": {
      "dependsOn": ["^build"],
      "inputs": ["src/**", "test/**", "vitest.config.*"],
      "outputs": ["coverage/**"],
      "env": ["TEST_DATABASE_URL"]
    },
    "test:e2e": {
      "dependsOn": ["build"],
      "inputs": ["e2e/**", "playwright.config.*"],
      "outputs": ["test-results/**"],
      "cache": false
    },
    "db:generate": {
      "cache": false,
      "inputs": ["prisma/schema.prisma"]
    }
  }
}

^build означает «сначала собери все зависимости этого пакета». Так Turbo автоматически определяет порядок: packages/ui соберётся до apps/web, потому что web зависит от ui.

Настройка пакетов

// packages/ui/package.json
{
  "name": "@acme/ui",
  "version": "0.0.0",
  "private": true,
  "exports": {
    ".": {
      "import": "./dist/index.js",
      "types": "./dist/index.d.ts"
    },
    "./styles": "./dist/styles.css"
  },
  "scripts": {
    "build": "tsup src/index.ts --format esm --dts",
    "dev": "tsup src/index.ts --format esm --dts --watch",
    "lint": "eslint src/",
    "typecheck": "tsc --noEmit"
  },
  "devDependencies": {
    "@acme/eslint-config": "*",
    "@acme/typescript-config": "*",
    "tsup": "^8.0.0"
  },
  "peerDependencies": {
    "react": "^18.0.0"
  }
}
// apps/web/package.json
{
  "name": "@acme/web",
  "private": true,
  "dependencies": {
    "@acme/ui": "*",
    "@acme/utils": "*"
  }
}

"*" — workspace-протокол: pnpm/yarn разрешит в локальный пакет. В npm надо указывать "workspace:*" явно.

Общий TypeScript конфиг

// packages/config/typescript/base.json
{
  "compilerOptions": {
    "target": "ES2020",
    "module": "ESNext",
    "moduleResolution": "Bundler",
    "strict": true,
    "noUncheckedIndexedAccess": true,
    "exactOptionalPropertyTypes": true,
    "lib": ["ES2020"],
    "skipLibCheck": true,
    "declaration": true,
    "declarationMap": true,
    "sourceMap": true
  }
}

// packages/config/typescript/react.json
{
  "extends": "./base.json",
  "compilerOptions": {
    "jsx": "react-jsx",
    "lib": ["ES2020", "DOM", "DOM.Iterable"]
  }
}

// apps/web/tsconfig.json
{
  "extends": "@acme/typescript-config/react.json",
  "compilerOptions": {
    "baseUrl": ".",
    "paths": { "@/*": ["./src/*"] }
  },
  "include": ["src", "next-env.d.ts"]
}

Удалённый кеш

Локальный кеш работает только на одной машине. Для команды и CI нужен remote cache. Vercel Remote Cache — бесплатно для опенсорса, платно для приватных:

# Авторизация (один раз)
npx turbo login
npx turbo link

# Или self-hosted через ducktape/turborepo-remote-cache

Self-hosted вариант через turborepo-remote-cache (открытый сервер):

# docker-compose.yml для remote cache сервера
services:
  turbo-cache:
    image: ducktors/turborepo-remote-cache:latest
    ports:
      - "3000:3000"
    environment:
      TURBO_TOKEN: "your-secret-token"
      STORAGE_PROVIDER: "s3"
      S3_BUCKET: "turbo-cache-bucket"
      AWS_ACCESS_KEY_ID: "${AWS_ACCESS_KEY_ID}"
      AWS_SECRET_ACCESS_KEY: "${AWS_SECRET_ACCESS_KEY}"
// turbo.json — remote cache endpoint
{
  "remoteCache": {
    "signature": true
  }
}
# Запуск с remote cache
TURBO_TOKEN=your-secret-token TURBO_API=https://cache.internal.example.com \
  turbo build

CI/CD с Turborepo

# .github/workflows/ci.yml
name: CI
on: [push, pull_request]

jobs:
  ci:
    runs-on: ubuntu-latest
    env:
      TURBO_TOKEN: ${{ secrets.TURBO_TOKEN }}
      TURBO_TEAM: ${{ secrets.TURBO_TEAM }}

    steps:
      - uses: actions/checkout@v4
        with:
          fetch-depth: 2  # нужно для --filter=...[HEAD^1]

      - uses: pnpm/action-setup@v3
        with:
          version: 9

      - uses: actions/setup-node@v4
        with:
          node-version: 20
          cache: pnpm

      - run: pnpm install --frozen-lockfile

      # Запускаем только задачи для изменившихся пакетов
      - run: pnpm turbo lint typecheck test --filter=...[HEAD^1]

      # Build всегда — деплой нужен с актуальным кешем
      - run: pnpm turbo build

--filter=...[HEAD^1] — «всё, что изменилось с предыдущего коммита, плюс все зависящие от этого пакеты». Если изменили packages/ui — пересобираются и apps/web, и apps/admin.

Типичные проблемы

Зависимость на runtime конфиг — если пакет читает .env при сборке, Turbo закеширует с конкретным значением переменной. Явно указывайте env в pipeline:

"build": {
  "env": ["NEXT_PUBLIC_API_URL", "DATABASE_URL"]
}

Circular dependencies — Turbo упадёт с ошибкой. Рефакторинг: выделить общую зависимость в третий пакет.

Dev-режим и кешdev должен иметь "cache": false и "persistent": true. Иначе Turbo не будет запускать watchers параллельно.

Сроки

Настройка Turborepo с нуля для проекта с 5–8 пакетами — два-три дня: создание структуры, конфигурация pipeline, общие tsconfig/eslint, настройка remote cache, адаптация CI/CD. Перенос существующего проекта в monorepo добавляет неделю на реструктуризацию импортов и разрешение зависимостей.