> ## Documentation Index
> Fetch the complete documentation index at: https://docs.exode.biz/llms.txt
> Use this file to discover all available pages before exploring further.

# Попытки практик

> Выгрузка попыток практических заданий с оценками, статусами и информацией о студенте

**type:** `QUERY_EXPORT_TYPE_COURSE_LESSON_PRACTICE_ATTEMPT_FIND_MANY`

Это тип отчёта для метода [`query-export/generate`](/ru/exode-api/school/query-export/generate): передайте его
в поле `type`, а описанные ниже поля — в `variables`.

Выгружает список попыток выполнения практических заданий с оценками, статусами и информацией о студенте.

### Колонки отчета

* Студент
* Номер телефона
* ID пользователя
* Курс
* Урок
* Попытка №
* Статус
* Правильность %
* Баллы
* После дедлайна
* Куратор
* Отправлено на проверку
* Дата создания
* Последняя активность

### Переменные (`variables`)

<ParamField body="filter" type="object" required>
  Параметры фильтрации.

  <Expandable title="Свойства filter">
    <ParamField body="practiceIds" type="integer[]">
      Фильтр по ID практик.
    </ParamField>

    <ParamField body="userIds" type="integer[]">
      Фильтр по ID пользователей.
    </ParamField>

    <ParamField body="curatorIds" type="integer[]">
      Фильтр по ID кураторов.
    </ParamField>

    <ParamField body="courseIds" type="integer[]">
      Фильтр по ID курсов.
    </ParamField>

    <ParamField body="productIds" type="integer[]">
      Фильтр по ID продуктов.
    </ParamField>

    <ParamField body="lessonIds" type="integer[]">
      Фильтр по ID уроков.
    </ParamField>

    <ParamField body="launchIds" type="integer[]">
      Фильтр по ID запусков.
    </ParamField>

    <ParamField body="finished" type="boolean">
      Фильтр по завершенности попытки.
    </ParamField>

    <ParamField body="statuses" type="string[]">
      Фильтр по статусам попыток. Массив из значений: `Created`, `OnReview`, `Verified`, `AutoVerified`, `OnCorrection`, `Failed`, `Stacked`.
    </ParamField>

    <ParamField body="search" type="string">
      Поиск по тексту (до 50 символов). Ищет по имени студента.
    </ParamField>

    <ParamField body="updatedAtDateRange" type="object">
      Фильтр по диапазону дат обновления.

      <Expandable title="Свойства updatedAtDateRange">
        <ParamField body="from" type="string">
          Начало диапазона в формате ISO 8601 (например, `2025-01-01T00:00:00Z`).
        </ParamField>

        <ParamField body="to" type="string">
          Конец диапазона в формате ISO 8601 (например, `2025-12-31T23:59:59Z`).
        </ParamField>
      </Expandable>
    </ParamField>

    <ParamField body="sentToReviewAtDateRange" type="object">
      Фильтр по диапазону дат отправки на проверку.

      <Expandable title="Свойства sentToReviewAtDateRange">
        <ParamField body="from" type="string">
          Начало диапазона в формате ISO 8601.
        </ParamField>

        <ParamField body="to" type="string">
          Конец диапазона в формате ISO 8601.
        </ParamField>
      </Expandable>
    </ParamField>
  </Expandable>
</ParamField>

<ParamField body="sort" type="object">
  Параметры сортировки. Значения: `ASC` (по возрастанию) или `DESC` (по убыванию).

  <Expandable title="Свойства sort">
    <ParamField body="id" type="string">
      Сортировка по ID. `ASC` | `DESC`.
    </ParamField>

    <ParamField body="createdAt" type="string">
      Сортировка по дате создания. `ASC` | `DESC`.
    </ParamField>

    <ParamField body="updatedAt" type="string">
      Сортировка по дате обновления. `ASC` | `DESC`.
    </ParamField>

    <ParamField body="sentToReviewAt" type="string">
      Сортировка по дате отправки на проверку. `ASC` | `DESC`.
    </ParamField>
  </Expandable>
</ParamField>

<RequestExample>
  ```bash cURL theme={null}
  curl --location 'https://api.exode.biz/saas/v2/query-export/generate' \
    --header 'Seller-Id: {{ sellerId }}' \
    --header 'School-Id: {{ schoolId }}' \
    --header 'Content-Type: application/json' \
    --header 'Authorization: Bearer YOUR_TOKEN' \
    --data '{
      "type": "QUERY_EXPORT_TYPE_COURSE_LESSON_PRACTICE_ATTEMPT_FIND_MANY",
      "variables": {
        "filter": {
          "courseIds": [5],
          "statuses": ["OnReview", "Verified"],
          "sentToReviewAtDateRange": {
            "from": "2025-01-01T00:00:00Z",
            "to": "2025-06-30T23:59:59Z"
          }
        },
        "sort": {
          "sentToReviewAt": "DESC"
        }
      }
    }'
  ```

  ```javascript Node.js theme={null}
  const axios = require('axios');

  const exportPracticeAttempts = async () => {
    try {
      const response = await axios.post('https://api.exode.biz/saas/v2/query-export/generate', {
        type: 'QUERY_EXPORT_TYPE_COURSE_LESSON_PRACTICE_ATTEMPT_FIND_MANY',
        variables: {
          filter: {
            courseIds: [5],
            statuses: ['OnReview', 'Verified'],
            sentToReviewAtDateRange: {
              from: '2025-01-01T00:00:00Z',
              to: '2025-06-30T23:59:59Z'
            }
          },
          sort: {
            sentToReviewAt: 'DESC'
          }
        }
      }, {
        headers: {
          'Seller-Id': '{{ sellerId }}',
          'School-Id': '{{ schoolId }}',
          'Content-Type': 'application/json',
          'Authorization': 'Bearer YOUR_TOKEN'
        }
      });

      const { uuid } = response.data;
      console.log('Execution UUID:', uuid);
    } catch (error) {
      console.error('Error:', error.response?.data || error.message);
    }
  };

  exportPracticeAttempts();
  ```

  ```php PHP theme={null}
  <?php

  $url = 'https://api.exode.biz/saas/v2/query-export/generate';
  $data = [
    'type' => 'QUERY_EXPORT_TYPE_COURSE_LESSON_PRACTICE_ATTEMPT_FIND_MANY',
    'variables' => [
      'filter' => [
        'courseIds' => [5],
        'statuses' => ['OnReview', 'Verified'],
        'sentToReviewAtDateRange' => [
          'from' => '2025-01-01T00:00:00Z',
          'to' => '2025-06-30T23:59:59Z'
        ]
      ],
      'sort' => [
        'sentToReviewAt' => 'DESC'
      ]
    ]
  ];

  $headers = [
    'Seller-Id: {{ sellerId }}',
    'School-Id: {{ schoolId }}',
    'Content-Type: application/json',
    'Authorization: Bearer YOUR_TOKEN'
  ];

  $ch = curl_init();
  curl_setopt($ch, CURLOPT_URL, $url);
  curl_setopt($ch, CURLOPT_POST, true);
  curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($data));
  curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
  curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);

  $response = curl_exec($ch);
  $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
  curl_close($ch);

  if ($httpCode === 200 || $httpCode === 201) {
    $result = json_decode($response, true);
    echo "Execution UUID: " . $result['uuid'] . "\n";
  } else {
    echo "Error: HTTP $httpCode\n";
    echo $response;
  }
  ?>
  ```

  ```python Python theme={null}
  import requests

  url = 'https://api.exode.biz/saas/v2/query-export/generate'

  data = {
    'type': 'QUERY_EXPORT_TYPE_COURSE_LESSON_PRACTICE_ATTEMPT_FIND_MANY',
    'variables': {
      'filter': {
        'courseIds': [5],
        'statuses': ['OnReview', 'Verified'],
        'sentToReviewAtDateRange': {
          'from': '2025-01-01T00:00:00Z',
          'to': '2025-06-30T23:59:59Z'
        }
      },
      'sort': {
        'sentToReviewAt': 'DESC'
      }
    }
  }

  headers = {
    'Seller-Id': '{{ sellerId }}',
    'School-Id': '{{ schoolId }}',
    'Content-Type': 'application/json',
    'Authorization': 'Bearer YOUR_TOKEN'
  }

  try:
    response = requests.post(url, json=data, headers=headers)
    response.raise_for_status()

    result = response.json()
    print(f"Execution UUID: {result['uuid']}")

  except requests.exceptions.RequestException as e:
    print(f'Error: {e}')
  ```

  ```bsl 1С theme={null}
  ИдентификаторыКурсов = Новый Массив;
  ИдентификаторыКурсов.Добавить(5);

  Статусы = Новый Массив;
  Статусы.Добавить("OnReview");
  Статусы.Добавить("Verified");

  ДиапазонДат = Новый Структура;
  ДиапазонДат.Вставить("from", "2025-01-01T00:00:00Z");
  ДиапазонДат.Вставить("to", "2025-06-30T23:59:59Z");

  Фильтр = Новый Структура;
  Фильтр.Вставить("courseIds", ИдентификаторыКурсов);
  Фильтр.Вставить("statuses", Статусы);
  Фильтр.Вставить("sentToReviewAtDateRange", ДиапазонДат);

  Сортировка = Новый Структура;
  Сортировка.Вставить("sentToReviewAt", "DESC");

  Переменные = Новый Структура;
  Переменные.Вставить("filter", Фильтр);
  Переменные.Вставить("sort", Сортировка);

  Данные = Новый Структура;
  Данные.Вставить("type", "QUERY_EXPORT_TYPE_COURSE_LESSON_PRACTICE_ATTEMPT_FIND_MANY");
  Данные.Вставить("variables", Переменные);

  // Serialize body to JSON
  ЗаписьJSON = Новый ЗаписьJSON;
  ЗаписьJSON.УстановитьСтроку();
  ЗаписатьJSON(ЗаписьJSON, Данные);
  ТелоЗапроса = ЗаписьJSON.Закрыть();

  Соединение = Новый HTTPСоединение("api.exode.biz", 443, , , , 30, Новый OpenSSLSecureConnection);

  Запрос = Новый HTTPЗапрос("/saas/v2/query-export/generate");
  Запрос.Заголовки.Вставить("Seller-Id", "{{ sellerId }}");
  Запрос.Заголовки.Вставить("School-Id", "{{ schoolId }}");
  Запрос.Заголовки.Вставить("Content-Type", "application/json");
  Запрос.Заголовки.Вставить("Authorization", "Bearer YOUR_TOKEN");
  Запрос.УстановитьТелоИзСтроки(ТелоЗапроса);

  Ответ = Соединение.ВызватьHTTPМетод("POST", Запрос);

  Если Ответ.КодСостояния = 200 Тогда
      ЧтениеJSON = Новый ЧтениеJSON;
      ЧтениеJSON.УстановитьСтроку(Ответ.ПолучитьТелоКакСтроку());
      Результат = ПрочитатьJSON(ЧтениеJSON);
      Сообщить("Экспорт запущен");
  Иначе
      Сообщить("Ошибка: HTTP " + Ответ.КодСостояния);
      Сообщить(Ответ.ПолучитьТелоКакСтроку());
  КонецЕсли;
  ```
</RequestExample>

<ResponseExample>
  ```json Success theme={null}
  {
    "success": true,
    "code": 200,
    "payload": {
      "total": 100,
      "completed": 100,
      "status": "Completed",
      "result": {
        "fileUrl": "https://static.exode.biz/generated/reports/79d46a97-a149-432b-bac7-45f6732b9115.xlsx",
        "fileName": "79d46a97-a149-432b-bac7-45f6732b9115.xlsx",
        "fileSize": 6748
      }
    }
  }
  ```
</ResponseExample>
