PurpleSchool — курсы программирования онлайн
  • Бесплатно
    • Курсы
    • JavaScript Основы разработкиPython Основы PythonCSS CSS FlexboxКарта развития
    • База знанийИконка стрелки
    • Новостные рассылкиИконка стрелки
  • Карьерные пути
    • Frontend React разработчик
    • Frontend Vue разработчик
    • Backend разработчик Node.js
    • Fullstack разработчик React / Node.js
    • Mobile разработчик React Native
    • Backend разработчик Golang
    • Devops инженер
    • Backend разработчик Python
  • О нас
    • Отзывы
    • Реферальная программа
    • О компании
    • Контакты
  • Иконка открытия меню
    • Сообщество
    • PurpleПлюс
    • AI тренажёр
    • Проекты
PurpleSchool — платформа бесплатных roadmap и курсов для разработчиков
ютуб иконка
Telegram иконка
VK иконка
VK иконка
Курсы
ГлавнаяКаталог курсовFrontendBackendFullstack
Практика
КарьераПроектыPurpleПлюс
Материалы
БлогБаза знаний
Документы
Договор офертаПолитика конфиденциальностиПроверка сертификатаМиграция курсовРеферальная программа
Реквизиты
ИП Ларичев Антон АндреевичИНН 773373765379contact@purpleschool.ru

PurpleSchool © 2020 -2026 Все права защищены

  • Курсы
    • FrontendИконка стрелки
    • AI разработкаИконка стрелки
    • BackendИконка стрелки
    • DevOpsИконка стрелки
    • MobileИконка стрелки
    • ТестированиеИконка стрелки
    • Soft-skillsИконка стрелки
    • ДизайнИконка стрелки
    Иконка слояПерейти в каталог курсов
  • PurpleSchool — курсы программирования онлайн
    • Сообщество
    • PurpleПлюс
    • AI тренажёр
    • Проекты
    Главная
    Сообщество
    REST API на Node.js и Express: пишем сервер с нуля за час

    REST API на Node.js и Express: пишем сервер с нуля за час

    Аватар автора REST API на Node.js и Express: пишем сервер с нуля за час

    Антон Ларичев

    Иконка календаря08 мая 2026
    Node.jsExpressREST APIBackendJavaScriptjuniorИконка уровня junior
    Картинка поста REST API на Node.js и Express: пишем сервер с нуля за час

    Введение

    REST API — это архитектурный стиль, который позволяет клиенту и серверу обмениваться данными по протоколу HTTP. Связка Node.js и Express остаётся одним из самых популярных способов быстро поднять backend для веб-приложения, мобильного клиента или микросервиса. В этой статье мы напишем полноценный REST-сервер с нуля: настроим маршрутизацию, добавим middleware, реализуем CRUD-операции, валидацию и грамотную обработку ошибок.

    Предполагается, что у вас установлен Node.js версии 18 или выше. Проверить можно командой node -v. Дополнительно понадобится любой HTTP-клиент для тестов: Postman, Insomnia или утилита curl.

    Инициализация проекта

    Создадим директорию и инициализируем npm-проект. Установим Express и пару утилит, которые пригодятся в реальной разработке.

    mkdir rest-api && cd rest-api
    npm init -y
    npm install express
    npm install -D nodemon
    

    Добавим в package.json скрипт для запуска в режиме разработки:

    {
      "scripts": {
        "dev": "nodemon src/index.js",
        "start": "node src/index.js"
      }
    }
    

    Nodemon будет перезапускать сервер при каждом изменении файла, что заметно ускоряет разработку.

    Минимальный сервер

    Создадим файл src/index.js и опишем самый простой Express-сервер.

    // Импортируем фреймворк
    const express = require('express');
    
    const app = express();
    const PORT = process.env.PORT || 3000;
    
    // Middleware для парсинга JSON в теле запроса
    app.use(express.json());
    
    // Простой проверочный маршрут
    app.get('/health', (req, res) => {
      res.json({ status: 'ok' });
    });
    
    app.listen(PORT, () => {
      console.log(`Сервер запущен на порту ${PORT}`);
    });
    

    Запустите npm run dev и откройте http://localhost:3000/health — увидите JSON с подтверждением, что сервер работает.

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

    Для учебного примера часто всё пишут в одном файле, но в реальных проектах код разбивают на слои. Минимальная разумная структура такая:

    src/
      index.js          // точка входа
      routes/
        users.js        // маршруты ресурса
      controllers/
        users.js        // обработчики
      middleware/
        errorHandler.js // обработка ошибок
    

    Такое разделение упрощает тестирование и помогает быстрее ориентироваться в коде, когда количество эндпоинтов растёт.

    CRUD-операции для ресурса

    Реализуем классический REST-ресурс — список пользователей. Для простоты данные будем хранить в памяти, чтобы не отвлекаться на работу с базой.

    // src/controllers/users.js
    let users = [
      { id: 1, name: 'Анна', email: 'anna@example.com' }
    ];
    let nextId = 2;
    
    exports.list = (req, res) => {
      res.json(users);
    };
    
    exports.getOne = (req, res) => {
      const id = Number(req.params.id);
      const user = users.find((u) => u.id === id);
      if (!user) {
        return res.status(404).json({ message: 'Пользователь не найден' });
      }
      res.json(user);
    };
    
    exports.create = (req, res) => {
      const { name, email } = req.body;
      // Минимальная проверка обязательных полей
      if (!name || !email) {
        return res.status(400).json({ message: 'name и email обязательны' });
      }
      const user = { id: nextId++, name, email };
      users.push(user);
      res.status(201).json(user);
    };
    
    exports.update = (req, res) => {
      const id = Number(req.params.id);
      const user = users.find((u) => u.id === id);
      if (!user) {
        return res.status(404).json({ message: 'Пользователь не найден' });
      }
      Object.assign(user, req.body);
      res.json(user);
    };
    
    exports.remove = (req, res) => {
      const id = Number(req.params.id);
      users = users.filter((u) => u.id !== id);
      res.status(204).send();
    };
    

    Теперь подключим обработчики к маршрутам через express.Router:

    // src/routes/users.js
    const { Router } = require('express');
    const controller = require('../controllers/users');
    
    const router = Router();
    
    router.get('/', controller.list);
    router.get('/:id', controller.getOne);
    router.post('/', controller.create);
    router.put('/:id', controller.update);
    router.delete('/:id', controller.remove);
    
    module.exports = router;
    

    И подключим роутер в точке входа:

    // src/index.js
    const usersRouter = require('./routes/users');
    app.use('/api/users', usersRouter);
    

    Теперь доступны пять стандартных REST-эндпоинтов: GET /api/users, GET /api/users/:id, POST /api/users, PUT /api/users/:id, DELETE /api/users/:id.

    Проверка через curl

    curl -X POST http://localhost:3000/api/users \
      -H 'Content-Type: application/json' \
      -d '{"name":"Иван","email":"ivan@example.com"}'
    

    Централизованная обработка ошибок

    Вместо того чтобы повторять try/catch в каждом контроллере, удобно завести единый middleware-обработчик ошибок. Express вызывает его, когда мы передаём ошибку в next(err).

    // src/middleware/errorHandler.js
    module.exports = (err, req, res, next) => {
      // Логируем ошибку для отладки
      console.error(err);
      const status = err.status || 500;
      res.status(status).json({
        message: err.message || 'Внутренняя ошибка сервера'
      });
    };
    

    Подключаем его последним, после всех маршрутов:

    const errorHandler = require('./middleware/errorHandler');
    app.use(errorHandler);
    

    Теперь в любом контроллере можно делать next(new Error('...')), и клиент получит корректный JSON-ответ с нужным статусом.

    Частые ошибки

    При написании первых REST-API разработчики регулярно наступают на одни и те же грабли.

    • Забытый express.json(). Без этого middleware req.body будет undefined, и POST-запросы не будут видеть данные.
    • Возврат 200 вместо 201 и 204. Для создания ресурса принято использовать 201 Created, а для удаления без тела — 204 No Content.
    • Отсутствие валидации входных данных. Минимальные проверки руками — это нормально для прототипа, но в продакшене стоит подключить библиотеку вроде zod или joi.
    • Игнорирование CORS. Если фронтенд работает на другом домене, без пакета cors браузер заблокирует запросы.
    • Хранение секретов в коде. Пароли, токены и строки подключения должны лежать в переменных окружения и читаться через process.env.
    • Отсутствие централизованной обработки ошибок. Без неё необработанное исключение может уронить весь процесс Node.js.

    Заключение

    Мы прошли путь от пустой папки до работающего REST API с маршрутизацией, контроллерами и обработкой ошибок. Этого достаточно, чтобы запустить прототип сервиса или внутренний инструмент. Следующие шаги — подключить базу данных (PostgreSQL, MongoDB), добавить аутентификацию через JWT, описать схему OpenAPI и покрыть код тестами на Jest или Vitest. Express даёт прочный фундамент: остальное наращивается слоями по мере роста проекта.

    Иконка глаза6

    Комментарии

    0

    Постройте личный план изучения React state менеджер Zustand до уровня Middle — бесплатно!

    React state менеджер Zustand — часть карты развития Frontend

    • step100+ шагов развития
    • lessons30 бесплатных лекций
    • lessons300 бонусных рублей на счет

    Бесплатные лекции

    Лучшие курсы по теме

    изображение курса

    Vue 3 и Pinia

    Антон Ларичев
    AI-тренажерыAI-тренажеры
    Практика в студииПрактика в студии
    Гарантия
    Бонусы
    иконка звёздочки рейтинга4.9
    3 999 ₽ 6 990 ₽
    Подробнее
    изображение курса

    Next.js - с нуля

    Антон Ларичев
    AI-тренажерыAI-тренажеры
    Практика в студииПрактика в студии
    Гарантия
    Бонусы
    иконка звёздочки рейтинга4.7
    3 999 ₽ 6 990 ₽
    Подробнее
    изображение курса

    Feature-Sliced Design

    Антон Ларичев
    AI-тренажерыAI-тренажеры
    Практика в студииПрактика в студии
    Гарантия
    Бонусы
    иконка звёздочки рейтинга4.5
    3 999 ₽ 6 990 ₽
    Подробнее

    Похожие статьи

    Картинка поста Event Loop в Node.js: как работает и зачем это знать
    Иконка аватараАнтон
    Иконка календаря04 мая 2026
    Node.jsJavaScriptEvent Loop+ 2middleИконка уровня middle

    Event Loop в Node.js: как работает и зачем это знать

    Event Loop в Node.js — основа асинхронности. Разбираем фазы цикла событий, очереди задач и типичные ошибки на практических примерах.

    Иконка чипа0
    Иконка глаза143
    Иконка комментариев0
    Картинка поста TypeScript с нуля за день: типы, интерфейсы и дженерики
    Иконка аватараАнтон
    Иконка календаря06 мая 2026
    TypeScriptJavaScriptFrontendjuniorИконка уровня junior

    TypeScript с нуля за день: типы, интерфейсы и дженерики

    TypeScript с нуля: разбираем базовые типы, интерфейсы и дженерики за один день. Практические примеры кода и частые ошибки новичков.

    Иконка чипа0
    Иконка глаза67
    Иконка комментариев0
    Картинка поста React хуки для начинающих: useState, useEffect, useContext на примерах
    Иконка аватараАнтон
    Иконка календаря05 мая 2026
    ReactHooksJavaScript+ 1juniorИконка уровня junior

    React хуки для начинающих: useState, useEffect, useContext на примерах

    React хуки для начинающих: разбираем useState, useEffect и useContext с практическими примерами кода и типичными ошибками.

    Иконка чипа0
    Иконка глаза105
    Иконка комментариев0
    Иконка чипа0