Разработка системы оценок и успеваемости для LMS

Наша компания занимается разработкой, поддержкой и обслуживанием сайтов любой сложности. От простых одностраничных сайтов до масштабных кластерных систем построенных на микро сервисах. Опыт разработчиков подтвержден сертификатами от вендоров.

Разработка и обслуживание любых видов сайтов:

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

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

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Разработка системы оценок и успеваемости для LMS
Средняя
~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

Разработка системы оценок и успеваемости для LMS

Система оценок в LMS должна делать три вещи: хранить результаты выполнения всех активностей, агрегировать их в итоговую оценку по курсу и отображать прогресс в понятном виде для студента и преподавателя.

Модель данных

-- Оценки за отдельные активности
CREATE TABLE grades (
  id              UUID PRIMARY KEY DEFAULT gen_random_uuid(),
  student_id      UUID REFERENCES users(id),
  course_id       UUID REFERENCES courses(id),
  gradable_type   VARCHAR(100) NOT NULL, -- 'assignment', 'quiz', 'peer_review'
  gradable_id     UUID NOT NULL,
  attempt_number  INT DEFAULT 1,
  raw_score       NUMERIC(6,2),
  max_score       NUMERIC(6,2) NOT NULL,
  weight          NUMERIC(5,4) DEFAULT 1.0, -- вес в итоговой оценке
  is_final        BOOLEAN DEFAULT FALSE,    -- финальная попытка для агрегации
  graded_by       UUID REFERENCES users(id), -- NULL если автоматически
  graded_at       TIMESTAMPTZ,
  created_at      TIMESTAMPTZ DEFAULT NOW()
);

-- Итоговые оценки по курсу
CREATE TABLE course_grades (
  id              UUID PRIMARY KEY DEFAULT gen_random_uuid(),
  student_id      UUID REFERENCES users(id),
  course_id       UUID REFERENCES courses(id),
  letter_grade    VARCHAR(5),  -- A, B+, C, etc.
  percentage      NUMERIC(5,2),
  calculated_at   TIMESTAMPTZ,
  UNIQUE(student_id, course_id)
);

-- Категории оценок с весами
CREATE TABLE grade_categories (
  id          UUID PRIMARY KEY DEFAULT gen_random_uuid(),
  course_id   UUID REFERENCES courses(id),
  name        VARCHAR(200),         -- 'Домашние задания', 'Тесты', 'Финальный проект'
  weight      NUMERIC(5,4) NOT NULL, -- 0.3 = 30%
  drop_lowest INT DEFAULT 0         -- убрать N худших оценок
);

Расчёт итоговой оценки

Weighted average с поддержкой категорий и drop-lowest:

async function calculateCourseGrade(studentId, courseId) {
  const categories = await db.gradeCategories.findAll({ courseId });
  let totalWeight = 0;
  let weightedSum = 0;

  for (const category of categories) {
    const grades = await db.grades.findAll({
      studentId,
      courseId,
      categoryId: category.id,
      isFinal: true,
    });

    if (grades.length === 0) continue;

    // Drop lowest N grades
    const sorted = grades
      .map(g => (g.rawScore / g.maxScore) * 100)
      .sort((a, b) => a - b)
      .slice(category.dropLowest);

    const categoryAvg = sorted.reduce((a, b) => a + b, 0) / sorted.length;
    weightedSum += categoryAvg * category.weight;
    totalWeight += category.weight;
  }

  const percentage = totalWeight > 0 ? weightedSum / totalWeight : 0;
  const letterGrade = percentageToLetter(percentage);

  await db.courseGrades.upsert({ studentId, courseId, percentage, letterGrade, calculatedAt: new Date() });
  return { percentage, letterGrade };
}

function percentageToLetter(pct) {
  if (pct >= 93) return 'A';
  if (pct >= 90) return 'A-';
  if (pct >= 87) return 'B+';
  if (pct >= 83) return 'B';
  if (pct >= 80) return 'B-';
  if (pct >= 70) return 'C';
  if (pct >= 60) return 'D';
  return 'F';
}

Пересчёт оценок

Пересчёт триггерится при:

  • Проверке или обновлении любой оценки
  • Изменении весов категорий преподавателем
  • Добавлении нового задания

Вариант реализации: очередь задач (BullMQ/Celery) получает событие grade.updated, ставит задачу recalculate_course_grade с дедупликацией по (student_id, course_id) и задержкой 30 секунд — чтобы не пересчитывать при пачке обновлений.

Журнал оценок (Gradebook)

Таблица для преподавателя: строки — студенты, столбцы — задания. Ячейки кликабельны для перехода к работе.

interface GradebookRow {
  student: { id: string; name: string; avatar: string };
  grades: Record<string, { score: number | null; maxScore: number; status: string }>;
  courseGrade: { percentage: number; letter: string };
}

Для курсов с 500+ студентами — виртуализированная таблица (TanStack Table + виртуализация строк).

Настраиваемые шкалы оценок

Разные курсы требуют разных шкал: американская буквенная, европейская числовая 1–10, прошёл/не прошёл (pass/fail), кастомная шкала. Шкала хранится в настройках курса, логика конвертации — в отдельном сервисе.

Сроки

Базовая система хранения оценок с weighted average расчётом и gradebook для преподавателя — 5–7 дней. Категории с drop-lowest, настраиваемые шкалы, history оценок — ещё 3–4 дня.