Разработка LMS-платформы с поддержкой LTI-интеграций

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

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

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Разработка LMS-платформы с поддержкой LTI-интеграций
Сложная
от 2 недель до 3 месяцев
Часто задаваемые вопросы
Наши компетенции:
Этапы разработки
Последние работы
  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1217
  • image_web-applications_feedme_466_0.webp
    Разработка веб-приложения для компании FEEDME
    1161
  • image_websites_belfingroup_462_0.webp
    Разработка веб-сайта для компании БЕЛФИНГРУПП
    852
  • image_ecommerce_furnoro_435_0.webp
    Разработка интернет магазина для компании FURNORO
    1046
  • image_crm_enviok_479_0.webp
    Разработка веб-приложения для компании Enviok
    823
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Разработка веб-сайта для компании ФИКСПЕР
    815

Поддержка LTI-интеграций в LMS

LTI (Learning Tools Interoperability) — стандарт IMS Global для встраивания внешних образовательных инструментов в LMS. Через LTI преподаватель настраивает один раз подключение к Kahoot, Phet Simulations, Coursera for Campus, Microsoft Teams — и студенты запускают их прямо из LMS без отдельной регистрации.

Версии LTI

  • LTI 1.1 — устаревший, OAuth 1.0 подпись. Всё ещё используется многими поставщиками.
  • LTI 1.3 — современный стандарт, OAuth 2.0 + OpenID Connect. Обязателен для новых интеграций.

LTI 1.3 — Platform Role (ваша LMS)

LMS выступает Platform (IMS-терминология). Внешний инструмент — Tool. Процесс:

  1. Пользователь кликает на LTI-ссылку в LMS
  2. LMS инициирует OIDC Login Request к Tool
  3. Tool отвечает Auth Request
  4. LMS создаёт подписанный JWT и POST-ом отправляет на Tool
  5. Tool валидирует JWT через JWKS LMS
import { Provider } from 'ltijs'; // npm install ltijs

// Настройка LMS как LTI Platform
Provider.setup(
  process.env.LTI_ENCRYPTION_KEY!,  // 32+ символов для шифрования cookies
  {
    url: process.env.DATABASE_URL!,
    plugin: require('ltijs-postgresql'),  // адаптер для PostgreSQL
  },
  {
    cookies: { secure: true, sameSite: 'None' },
    devMode: process.env.NODE_ENV !== 'production',
  }
);

// Регистрация внешнего инструмента
await Provider.registerPlatform({
  url: 'https://tool.example.com',
  name: 'Kahoot Integration',
  clientId: process.env.KAHOOT_CLIENT_ID!,
  authenticationEndpoint: 'https://tool.example.com/lti/auth',
  accesstokenEndpoint: 'https://tool.example.com/lti/token',
  authConfig: {
    method: 'JWK_SET',
    key: 'https://tool.example.com/.well-known/jwks.json',
  },
});

// Обработчик запуска инструмента
Provider.onConnect(async (token, req, res) => {
  const { email, name, role } = token.userInfo;
  const contextId = token.platformContext.context?.id;

  // Проверить/создать пользователя во внешнем инструменте
  res.json({ token: token.jwt });
});

await Provider.deploy({ serverless: true });

LTI 1.1 — для устаревших инструментов

Часть поставщиков (Phet, некоторые тесты) всё ещё использует LTI 1.1 с OAuth 1.0 подписью:

import oauth from 'oauth-signature';

function launchLti11(
  launchUrl: string,
  consumerKey: string,
  consumerSecret: string,
  params: Record<string, string>
): { url: string; method: 'POST'; body: string } {
  const baseParams: Record<string, string> = {
    lti_message_type: 'basic-lti-launch-request',
    lti_version: 'LTI-1p0',
    oauth_callback: 'about:blank',
    oauth_consumer_key: consumerKey,
    oauth_nonce: crypto.randomUUID().replace(/-/g, ''),
    oauth_signature_method: 'HMAC-SHA1',
    oauth_timestamp: String(Math.floor(Date.now() / 1000)),
    oauth_version: '1.0',
    ...params,
  };

  const signature = oauth.generate('POST', launchUrl, baseParams, consumerSecret, '');
  baseParams.oauth_signature = signature;

  const body = new URLSearchParams(baseParams).toString();
  return { url: launchUrl, method: 'POST', body };
}

// Страница запуска LTI инструмента
app.get('/courses/:courseId/tools/:toolId/launch', authenticate, async (req, res) => {
  const tool = await db.ltiTools.findById(req.params.toolId);
  const enrollment = await db.enrollments.findByCourseAndUser(
    req.params.courseId, req.user.id
  );

  if (tool.version === '1.1') {
    const launch = launchLti11(tool.launch_url, tool.consumer_key, tool.consumer_secret, {
      resource_link_id: `${req.params.courseId}-${req.params.toolId}`,
      resource_link_title: tool.name,
      user_id: req.user.id,
      lis_person_name_full: req.user.name,
      lis_person_contact_email_primary: req.user.email,
      roles: enrollment.role === 'instructor' ? 'Instructor' : 'Student',
      context_id: req.params.courseId,
      context_title: enrollment.courseTitle,
    });

    // Авто-сабмит форму через HTML
    res.send(`
      <!DOCTYPE html>
      <html>
        <body>
          <form id="lti" method="POST" action="${launch.url}">
            ${Object.entries(Object.fromEntries(new URLSearchParams(launch.body)))
              .map(([k, v]) => `<input type="hidden" name="${k}" value="${v}" />`)
              .join('\n')}
          </form>
          <script>document.getElementById('lti').submit();</script>
        </body>
      </html>
    `);
  }
});

Хранение конфигурации инструментов

CREATE TABLE lti_tools (
  id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
  course_id UUID REFERENCES courses(id),
  name VARCHAR(255) NOT NULL,
  version VARCHAR(10) NOT NULL,  -- '1.1' | '1.3'
  -- LTI 1.1
  launch_url TEXT,
  consumer_key VARCHAR(255),
  consumer_secret TEXT,
  -- LTI 1.3
  client_id VARCHAR(255),
  platform_id VARCHAR(255),
  deployment_id VARCHAR(255),
  oidc_auth_url TEXT,
  jwks_url TEXT,
  -- Настройки
  open_in_new_tab BOOLEAN DEFAULT false,
  custom_params JSONB DEFAULT '{}',
  created_at TIMESTAMPTZ DEFAULT now()
);

Получение результатов через LTI Outcomes (1.1)

// Инструмент может отправить оценку обратно в LMS
app.post('/lti/grade', async (req, res) => {
  const { sourcedId, score, action } = parseLtiOutcomesXml(req.body);

  // sourcedId содержит userId и resourceLinkId
  const [userId, resourceId] = parseLisResultSourcedId(sourcedId);

  await db.ltiGrades.upsert({
    userId,
    resourceId,
    score: Number(score),
    receivedAt: new Date(),
  });

  // Обновить прогресс курса
  await updateCourseProgress(userId, resourceId, Number(score));

  res.type('application/xml').send(`
    <?xml version="1.0" encoding="UTF-8"?>
    <imsx_POXEnvelopeResponse xmlns="http://www.imsglobal.org/services/ltiv1p1/xsd/imsoms_v1p0">
      <imsx_POXHeader>
        <imsx_POXResponseHeaderInfo>
          <imsx_version>V1.0</imsx_version>
          <imsx_messageIdentifier>${crypto.randomUUID()}</imsx_messageIdentifier>
          <imsx_statusInfo>
            <imsx_codeMajor>success</imsx_codeMajor>
            <imsx_severity>status</imsx_severity>
          </imsx_statusInfo>
        </imsx_POXResponseHeaderInfo>
      </imsx_POXHeader>
      <imsx_POXBody><replaceResultResponse /></imsx_POXBody>
    </imsx_POXEnvelopeResponse>
  `);
});

Сроки

LTI 1.1 интеграция (Consumer + Launch + Grades) — 3–5 дней. LTI 1.3 с OIDC flow через ltijs — 1 неделя. Поддержка обоих + UI управления инструментами — 2 недели.